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