When selecting a toot via keyboard, ensure it is scrolled into view (#10593)
This commit is contained in:
		
							parent
							
								
									05ef3462ba
								
							
						
					
					
						commit
						5121d9c12f
					
				
					 5 changed files with 51 additions and 22 deletions
				
			
		|  | @ -46,22 +46,28 @@ export default class StatusList extends ImmutablePureComponent { | ||||||
| 
 | 
 | ||||||
|   handleMoveUp = (id, featured) => { |   handleMoveUp = (id, featured) => { | ||||||
|     const elementIndex = this.getCurrentStatusIndex(id, featured) - 1; |     const elementIndex = this.getCurrentStatusIndex(id, featured) - 1; | ||||||
|     this._selectChild(elementIndex); |     this._selectChild(elementIndex, true); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   handleMoveDown = (id, featured) => { |   handleMoveDown = (id, featured) => { | ||||||
|     const elementIndex = this.getCurrentStatusIndex(id, featured) + 1; |     const elementIndex = this.getCurrentStatusIndex(id, featured) + 1; | ||||||
|     this._selectChild(elementIndex); |     this._selectChild(elementIndex, false); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   handleLoadOlder = debounce(() => { |   handleLoadOlder = debounce(() => { | ||||||
|     this.props.onLoadMore(this.props.statusIds.size > 0 ? this.props.statusIds.last() : undefined); |     this.props.onLoadMore(this.props.statusIds.size > 0 ? this.props.statusIds.last() : undefined); | ||||||
|   }, 300, { leading: true }) |   }, 300, { leading: true }) | ||||||
| 
 | 
 | ||||||
|   _selectChild (index) { |   _selectChild (index, align_top) { | ||||||
|     const element = this.node.node.querySelector(`article:nth-of-type(${index + 1}) .focusable`); |     const container = this.node.node; | ||||||
|  |     const element = container.querySelector(`article:nth-of-type(${index + 1}) .focusable`); | ||||||
| 
 | 
 | ||||||
|     if (element) { |     if (element) { | ||||||
|  |       if (align_top && container.scrollTop > element.offsetTop) { | ||||||
|  |         element.scrollIntoView(true); | ||||||
|  |       } else if (!align_top && container.scrollTop + container.clientHeight < element.offsetTop + element.offsetHeight) { | ||||||
|  |         element.scrollIntoView(false); | ||||||
|  |       } | ||||||
|       element.focus(); |       element.focus(); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  | @ -20,18 +20,24 @@ export default class ConversationsList extends ImmutablePureComponent { | ||||||
| 
 | 
 | ||||||
|   handleMoveUp = id => { |   handleMoveUp = id => { | ||||||
|     const elementIndex = this.getCurrentIndex(id) - 1; |     const elementIndex = this.getCurrentIndex(id) - 1; | ||||||
|     this._selectChild(elementIndex); |     this._selectChild(elementIndex, true); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   handleMoveDown = id => { |   handleMoveDown = id => { | ||||||
|     const elementIndex = this.getCurrentIndex(id) + 1; |     const elementIndex = this.getCurrentIndex(id) + 1; | ||||||
|     this._selectChild(elementIndex); |     this._selectChild(elementIndex, false); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   _selectChild (index) { |   _selectChild (index, align_top) { | ||||||
|     const element = this.node.node.querySelector(`article:nth-of-type(${index + 1}) .focusable`); |     const container = this.node.node; | ||||||
|  |     const element = container.querySelector(`article:nth-of-type(${index + 1}) .focusable`); | ||||||
| 
 | 
 | ||||||
|     if (element) { |     if (element) { | ||||||
|  |       if (align_top && container.scrollTop > element.offsetTop) { | ||||||
|  |         element.scrollIntoView(true); | ||||||
|  |       } else if (!align_top && container.scrollTop + container.clientHeight < element.offsetTop + element.offsetHeight) { | ||||||
|  |         element.scrollIntoView(false); | ||||||
|  |       } | ||||||
|       element.focus(); |       element.focus(); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  | @ -113,18 +113,24 @@ class Notifications extends React.PureComponent { | ||||||
| 
 | 
 | ||||||
|   handleMoveUp = id => { |   handleMoveUp = id => { | ||||||
|     const elementIndex = this.props.notifications.findIndex(item => item !== null && item.get('id') === id) - 1; |     const elementIndex = this.props.notifications.findIndex(item => item !== null && item.get('id') === id) - 1; | ||||||
|     this._selectChild(elementIndex); |     this._selectChild(elementIndex, true); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   handleMoveDown = id => { |   handleMoveDown = id => { | ||||||
|     const elementIndex = this.props.notifications.findIndex(item => item !== null && item.get('id') === id) + 1; |     const elementIndex = this.props.notifications.findIndex(item => item !== null && item.get('id') === id) + 1; | ||||||
|     this._selectChild(elementIndex); |     this._selectChild(elementIndex, false); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   _selectChild (index) { |   _selectChild (index, align_top) { | ||||||
|     const element = this.column.node.querySelector(`article:nth-of-type(${index + 1}) .focusable`); |     const container = this.column.node; | ||||||
|  |     const element = container.querySelector(`article:nth-of-type(${index + 1}) .focusable`); | ||||||
| 
 | 
 | ||||||
|     if (element) { |     if (element) { | ||||||
|  |       if (align_top && container.scrollTop > element.offsetTop) { | ||||||
|  |         element.scrollIntoView(true); | ||||||
|  |       } else if (!align_top && container.scrollTop + container.clientHeight < element.offsetTop + element.offsetHeight) { | ||||||
|  |         element.scrollIntoView(false); | ||||||
|  |       } | ||||||
|       element.focus(); |       element.focus(); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  | @ -316,15 +316,15 @@ class Status extends ImmutablePureComponent { | ||||||
|     const { status, ancestorsIds, descendantsIds } = this.props; |     const { status, ancestorsIds, descendantsIds } = this.props; | ||||||
| 
 | 
 | ||||||
|     if (id === status.get('id')) { |     if (id === status.get('id')) { | ||||||
|       this._selectChild(ancestorsIds.size - 1); |       this._selectChild(ancestorsIds.size - 1, true); | ||||||
|     } else { |     } else { | ||||||
|       let index = ancestorsIds.indexOf(id); |       let index = ancestorsIds.indexOf(id); | ||||||
| 
 | 
 | ||||||
|       if (index === -1) { |       if (index === -1) { | ||||||
|         index = descendantsIds.indexOf(id); |         index = descendantsIds.indexOf(id); | ||||||
|         this._selectChild(ancestorsIds.size + index); |         this._selectChild(ancestorsIds.size + index, true); | ||||||
|       } else { |       } else { | ||||||
|         this._selectChild(index - 1); |         this._selectChild(index - 1, true); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  | @ -333,23 +333,29 @@ class Status extends ImmutablePureComponent { | ||||||
|     const { status, ancestorsIds, descendantsIds } = this.props; |     const { status, ancestorsIds, descendantsIds } = this.props; | ||||||
| 
 | 
 | ||||||
|     if (id === status.get('id')) { |     if (id === status.get('id')) { | ||||||
|       this._selectChild(ancestorsIds.size + 1); |       this._selectChild(ancestorsIds.size + 1, false); | ||||||
|     } else { |     } else { | ||||||
|       let index = ancestorsIds.indexOf(id); |       let index = ancestorsIds.indexOf(id); | ||||||
| 
 | 
 | ||||||
|       if (index === -1) { |       if (index === -1) { | ||||||
|         index = descendantsIds.indexOf(id); |         index = descendantsIds.indexOf(id); | ||||||
|         this._selectChild(ancestorsIds.size + index + 2); |         this._selectChild(ancestorsIds.size + index + 2, false); | ||||||
|       } else { |       } else { | ||||||
|         this._selectChild(index + 1); |         this._selectChild(index + 1, false); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   _selectChild (index) { |   _selectChild (index, align_top) { | ||||||
|     const element = this.node.querySelectorAll('.focusable')[index]; |     const container = this.node; | ||||||
|  |     const element = container.querySelectorAll('.focusable')[index]; | ||||||
| 
 | 
 | ||||||
|     if (element) { |     if (element) { | ||||||
|  |       if (align_top && container.scrollTop > element.offsetTop) { | ||||||
|  |         element.scrollIntoView(true); | ||||||
|  |       } else if (!align_top && container.scrollTop + container.clientHeight < element.offsetTop + element.offsetHeight) { | ||||||
|  |         element.scrollIntoView(false); | ||||||
|  |       } | ||||||
|       element.focus(); |       element.focus(); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  | @ -367,11 +367,16 @@ class UI extends React.PureComponent { | ||||||
|   handleHotkeyFocusColumn = e => { |   handleHotkeyFocusColumn = e => { | ||||||
|     const index  = (e.key * 1) + 1; // First child is drawer, skip that
 |     const index  = (e.key * 1) + 1; // First child is drawer, skip that
 | ||||||
|     const column = this.node.querySelector(`.column:nth-child(${index})`); |     const column = this.node.querySelector(`.column:nth-child(${index})`); | ||||||
|  |     if (!column) return; | ||||||
|  |     const container = column.querySelector('.scrollable'); | ||||||
| 
 | 
 | ||||||
|     if (column) { |     if (container) { | ||||||
|       const status = column.querySelector('.focusable'); |       const status = container.querySelector('.focusable'); | ||||||
| 
 | 
 | ||||||
|       if (status) { |       if (status) { | ||||||
|  |         if (container.scrollTop > status.offsetTop) { | ||||||
|  |           status.scrollIntoView(true); | ||||||
|  |         } | ||||||
|         status.focus(); |         status.focus(); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue