pagenate by prev/next links
This commit is contained in:
		
							parent
							
								
									aeb0e0f96d
								
							
						
					
					
						commit
						ac8b40a2f3
					
				
					 6 changed files with 99 additions and 2 deletions
				
			
		|  | @ -38,7 +38,7 @@ Firefox by WebExtensions API. | ||||||
|   - [ ] find a keyword in the page |   - [ ] find a keyword in the page | ||||||
| - [ ] navigations | - [ ] navigations | ||||||
|   - [ ] yank/paste page |   - [ ] yank/paste page | ||||||
|   - [ ] pagenation |   - [x] pagenation | ||||||
|   - [ ] open parent page |   - [ ] open parent page | ||||||
| - [ ] hints | - [ ] hints | ||||||
|   - [x] open a link |   - [x] open a link | ||||||
|  |  | ||||||
|  | @ -30,6 +30,8 @@ const defaultKeymap = { | ||||||
|   'F': { type: operations.FOLLOW_START, newTab: true }, |   'F': { type: operations.FOLLOW_START, newTab: true }, | ||||||
|   'H': { type: operations.NAVIGATE_HISTORY_PREV }, |   'H': { type: operations.NAVIGATE_HISTORY_PREV }, | ||||||
|   'L': { type: operations.NAVIGATE_HISTORY_NEXT }, |   'L': { type: operations.NAVIGATE_HISTORY_NEXT }, | ||||||
|  |   '[[': { type: operations.NAVIGATE_LINK_PREV }, | ||||||
|  |   ']]': { type: operations.NAVIGATE_LINK_NEXT }, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| const asKeymapChars = (keys) => { | const asKeymapChars = (keys) => { | ||||||
|  |  | ||||||
|  | @ -39,6 +39,10 @@ const execOperation = (operation) => { | ||||||
|     return navigates.historyPrev(window); |     return navigates.historyPrev(window); | ||||||
|   case operations.NAVIGATE_HISTORY_NEXT: |   case operations.NAVIGATE_HISTORY_NEXT: | ||||||
|     return navigates.historyNext(window); |     return navigates.historyNext(window); | ||||||
|  |   case operations.NAVIGATE_LINK_PREV: | ||||||
|  |     return navigates.linkPrev(window); | ||||||
|  |   case operations.NAVIGATE_LINK_NEXT: | ||||||
|  |     return navigates.linkNext(window); | ||||||
|   } |   } | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,8 +1,47 @@ | ||||||
|  | const PREV_LINK_PATTERNS = [ | ||||||
|  |   /\bprev\b/i, /\bprevious\b/i, /\bback\b/i, | ||||||
|  |   /</, /\u2039/, /\u2190/, /\xab/, /\u226a/, /<</ | ||||||
|  | ]; | ||||||
|  | const NEXT_LINK_PATTERNS = [ | ||||||
|  |   /\bnext\b/i, | ||||||
|  |   />/, /\u203a/, /\u2192/, /\xbb/, /\u226b/, />>/ | ||||||
|  | ]; | ||||||
|  | 
 | ||||||
|  | const findLinkByPatterns = (win, patterns) => { | ||||||
|  |   let links = win.document.getElementsByTagName('a'); | ||||||
|  |   return Array.prototype.find.call(links, (link) => { | ||||||
|  |     return patterns.some(ptn => ptn.test(link.textContent)); | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| const historyPrev = (win) => { | const historyPrev = (win) => { | ||||||
|   win.history.back(); |   win.history.back(); | ||||||
| }; | }; | ||||||
|  | 
 | ||||||
| const historyNext = (win) => { | const historyNext = (win) => { | ||||||
|   win.history.forward(); |   win.history.forward(); | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| export { historyPrev, historyNext }; | const linkPrev = (win) => { | ||||||
|  |   let link = win.document.querySelector('a[rel=prev]'); | ||||||
|  |   if (link) { | ||||||
|  |     return link.click(); | ||||||
|  |   } | ||||||
|  |   link = findLinkByPatterns(win, PREV_LINK_PATTERNS); | ||||||
|  |   if (link) { | ||||||
|  |     link.click(); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const linkNext = (win) => { | ||||||
|  |   let link = win.document.querySelector('a[rel=next]'); | ||||||
|  |   if (link) { | ||||||
|  |     return link.click(); | ||||||
|  |   } | ||||||
|  |   link = findLinkByPatterns(win, NEXT_LINK_PATTERNS); | ||||||
|  |   if (link) { | ||||||
|  |     link.click(); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export { historyPrev, historyNext, linkPrev, linkNext }; | ||||||
|  |  | ||||||
|  | @ -13,6 +13,8 @@ export default { | ||||||
|   FOLLOW_START: 'follow.start', |   FOLLOW_START: 'follow.start', | ||||||
|   NAVIGATE_HISTORY_PREV: 'navigate.history.prev', |   NAVIGATE_HISTORY_PREV: 'navigate.history.prev', | ||||||
|   NAVIGATE_HISTORY_NEXT: 'navigate.history.next', |   NAVIGATE_HISTORY_NEXT: 'navigate.history.next', | ||||||
|  |   NAVIGATE_LINK_PREV: 'navigate.link.prev', | ||||||
|  |   NAVIGATE_LINK_NEXT: 'navigate.link.next', | ||||||
| 
 | 
 | ||||||
|   // Background
 |   // Background
 | ||||||
|   TABS_CLOSE: 'tabs.close', |   TABS_CLOSE: 'tabs.close', | ||||||
|  |  | ||||||
							
								
								
									
										50
									
								
								test/content/navigates.test.js
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								test/content/navigates.test.js
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,50 @@ | ||||||
|  | import { expect } from "chai"; | ||||||
|  | import * as navigates from '../../src/content/navigates'; | ||||||
|  | 
 | ||||||
|  | describe('navigates module', () => { | ||||||
|  |   beforeEach(() => { | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   describe('#linkPrev', () => { | ||||||
|  |     it('clicks prev link by text content', (done) => { | ||||||
|  |       document.body.innerHTML = '<a href="#dummy">xprevx</a>  <a href="#prev">go to prev</a>'; | ||||||
|  |       navigates.linkPrev(window); | ||||||
|  |       setTimeout(() => { | ||||||
|  |         expect(document.location.hash).to.equal('#prev'); | ||||||
|  |         done(); | ||||||
|  |       }, 0); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('clicks a[rel=prev] element preferentially', (done) => { | ||||||
|  |       document.body.innerHTML = '<a href="#dummy">prev</a>  <a rel="prev" href="#prev">rel</a>'; | ||||||
|  |       navigates.linkPrev(window); | ||||||
|  |       setTimeout(() => { | ||||||
|  |         expect(document.location.hash).to.equal('#prev'); | ||||||
|  |         done(); | ||||||
|  |       }, 0); | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   describe('#linkNext', () => { | ||||||
|  |     it('clicks next link by text content', (done) => { | ||||||
|  |       document.body.innerHTML = '<a href="#dummy">xnextx</a>  <a href="#next">go to next</a>'; | ||||||
|  |       navigates.linkNext(window); | ||||||
|  |       setTimeout(() => { | ||||||
|  |         expect(document.location.hash).to.equal('#next'); | ||||||
|  |         done(); | ||||||
|  |       }, 0); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('clicks a[rel=next] element preferentially', (done) => { | ||||||
|  |       document.body.innerHTML = '<a href="#dummy">next</a>  <a rel="next" href="#next">rel</a>'; | ||||||
|  |       navigates.linkNext(window); | ||||||
|  |       setTimeout(() => { | ||||||
|  |         expect(document.location.hash).to.equal('#next'); | ||||||
|  |         done(); | ||||||
|  |       }, 0); | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
		Reference in a new issue