A fork of https://github.com/ueokande/vim-vixen
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
258 lines
6.7 KiB
258 lines
6.7 KiB
6 years ago
|
const express = require('express');
|
||
|
const lanthan = require('lanthan');
|
||
|
const path = require('path');
|
||
|
const assert = require('assert');
|
||
|
const eventually = require('./eventually');
|
||
|
|
||
|
const Key = lanthan.Key;
|
||
|
|
||
|
const newApp = () => {
|
||
|
let app = express();
|
||
|
|
||
|
app.get('/', (req, res) => {
|
||
|
res.send(`<!DOCTYPEhtml>
|
||
|
<html lang="en">
|
||
|
<body><a href="hello">hello</a></body>
|
||
|
</html">`);
|
||
|
});
|
||
|
|
||
|
app.get('/follow-input', (req, res) => {
|
||
|
res.send(`<!DOCTYPEhtml>
|
||
|
<html lang="en">
|
||
|
<body><input></body>
|
||
|
</html">`);
|
||
|
});
|
||
|
|
||
|
app.get('/area', (req, res) => {
|
||
|
res.send(`<!DOCTYPEhtml>
|
||
|
<html lang="en">
|
||
|
<body>
|
||
|
<img
|
||
|
width="256" height="256" usemap="#map"
|
||
|
src=""
|
||
|
>
|
||
|
<map name="map">
|
||
|
<area shape="rect" coords="0,0,64,64" href="/">
|
||
|
<area shape="rect" coords="64,64,64,64" href="/">
|
||
|
<area shape="rect" coords="128,128,64,64" href="/">
|
||
|
</map>
|
||
|
</body>
|
||
|
</html">`);
|
||
|
});
|
||
|
|
||
|
/*
|
||
|
* test case: link2 is out of the viewport
|
||
|
* +-----------------+
|
||
|
* | [link1] |<--- window
|
||
|
* | |
|
||
|
* |=================|<--- viewport
|
||
|
* | [link2] |
|
||
|
* | |
|
||
|
* +-----------------+
|
||
|
*/
|
||
|
app.get('/test1', (req, res) => {
|
||
|
res.send(`<!DOCTYPEhtml>
|
||
|
<html lang="en">
|
||
|
<body>
|
||
|
<div><a href="link1">link1</a></div>
|
||
|
<div style="min-height:3000px"></div>
|
||
|
<div><a href="link2">link2</a></div>
|
||
|
</body>
|
||
|
</html">`);
|
||
|
});
|
||
|
|
||
|
/*
|
||
|
* test case 2: link2 and link3 are out of window of the frame
|
||
|
* +-----------------+
|
||
|
* | +-----------+ |
|
||
|
* | | [link1] | |
|
||
|
* |=================|
|
||
|
* | | [link2] | |
|
||
|
* | +-----------+ |
|
||
|
* | |
|
||
|
* +-----------------+
|
||
|
*/
|
||
|
app.get('/test2', (req, res) => {
|
||
|
res.send(`<!DOCTYPEhtml>
|
||
|
<html lang="en">
|
||
|
<body><iframe height="5000" src='/test2-frame'></body>
|
||
|
</html">`);
|
||
|
});
|
||
|
app.get('/test2-frame', (req, res) => {
|
||
|
res.send(`<!DOCTYPEhtml>
|
||
|
<html lang="en">
|
||
|
<body>
|
||
|
<div><a href="link1">link1</a></div>
|
||
|
<div style="min-height:3000px"></div>
|
||
|
<div><a href="link2">link2</a></div>
|
||
|
</body>
|
||
|
</html">`);
|
||
|
});
|
||
|
|
||
|
/* test case 3: link2 is out of window of the frame
|
||
|
* +-----------------+
|
||
|
* | +-----------+ |
|
||
|
* | | [link1] | |
|
||
|
* | +-----------+ |
|
||
|
* | : [link2] : |
|
||
|
* | + - - - - - + |
|
||
|
* | |
|
||
|
* +-----------------+
|
||
|
*/
|
||
|
app.get('/test3', (req, res) => {
|
||
|
res.send(`<!DOCTYPEhtml>
|
||
|
<html lang="en">
|
||
|
<body><iframe src='/test3-frame'></body>
|
||
|
</html">`);
|
||
|
});
|
||
|
app.get('/test3-frame', (req, res) => {
|
||
|
res.send(`<!DOCTYPEhtml>
|
||
|
<html lang="en">
|
||
|
<body>
|
||
|
<div><a href="link1">link1</a></div>
|
||
|
<div style="min-height:3000px"></div>
|
||
|
<div><a href="link2">link2</a></div>
|
||
|
</body>
|
||
|
</html">`);
|
||
|
});
|
||
|
|
||
|
return app;
|
||
|
};
|
||
|
|
||
|
const waitForHints = async(session) => {
|
||
|
await eventually(async() => {
|
||
|
let hints = await session.findElementsByCSS('.vimvixen-hint');
|
||
|
assert(hints.length > 0);
|
||
|
});
|
||
|
};
|
||
|
|
||
|
describe('follow test', () => {
|
||
|
|
||
|
const port = 12321;
|
||
|
let http;
|
||
|
let firefox;
|
||
|
let session;
|
||
|
let browser;
|
||
|
|
||
|
before(async() => {
|
||
|
http = newApp().listen(port);
|
||
|
|
||
|
firefox = await lanthan.firefox();
|
||
|
await firefox.session.installAddonFromPath(path.join(__dirname, '..'));
|
||
|
session = firefox.session;
|
||
|
browser = firefox.browser;
|
||
|
});
|
||
|
|
||
|
after(async() => {
|
||
|
if (firefox) {
|
||
|
await firefox.close();
|
||
|
}
|
||
|
http.close();
|
||
|
});
|
||
|
|
||
|
afterEach(async() => {
|
||
|
let tabs = await browser.tabs.query({});
|
||
|
for (let tab of tabs.slice(1)) {
|
||
|
await browser.tabs.remove(tab.id);
|
||
|
}
|
||
|
});
|
||
|
|
||
|
it('should focus an input by f', async () => {
|
||
|
await session.navigateTo(`http://127.0.0.1:${port}/follow-input`);
|
||
|
|
||
|
let body = await session.findElementByCSS('body');
|
||
|
await body.sendKeys('f');
|
||
|
await waitForHints(session);
|
||
|
await body.sendKeys('a');
|
||
|
|
||
|
let tagName = await session.executeScript(() => document.activeElement.tagName);
|
||
|
assert.equal(tagName.toLowerCase(), 'input');
|
||
|
});
|
||
|
|
||
|
it('should open a link by f', async () => {
|
||
|
await session.navigateTo(`http://127.0.0.1:${port}/`);
|
||
|
|
||
|
let body = await session.findElementByCSS('body');
|
||
|
await body.sendKeys('f', 'a');
|
||
|
|
||
|
let hash = await session.executeScript('location.pathname');
|
||
|
await body.sendKeys(hash, '/hello');
|
||
|
});
|
||
|
|
||
|
it('should focus an input by F', async () => {
|
||
|
await session.navigateTo(`http://127.0.0.1:${port}/follow-input`);
|
||
|
|
||
|
let body = await session.findElementByCSS('body');
|
||
|
await body.sendKeys(Key.Shift, 'f');
|
||
|
await waitForHints(session);
|
||
|
await body.sendKeys('a');
|
||
|
|
||
|
let tagName = await session.executeScript(() => document.activeElement.tagName);
|
||
|
assert.equal(tagName.toLowerCase(), 'input');
|
||
|
});
|
||
|
|
||
|
it('should open a link to new tab by F', async () => {
|
||
|
await session.navigateTo(`http://127.0.0.1:${port}/`);
|
||
|
|
||
|
let body = await session.findElementByCSS('body');
|
||
|
await body.sendKeys(Key.Shift, 'f');
|
||
|
await waitForHints(session);
|
||
|
await body.sendKeys('a');
|
||
|
|
||
|
await eventually(async() => {
|
||
|
let tabs = await browser.tabs.query({});
|
||
|
assert.equal(tabs.length, 2);
|
||
|
assert.equal(new URL(tabs[1].url).pathname, '/hello');
|
||
|
assert.equal(tabs[1].openerTabId, tabs[0].id);
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should show hints of links in area', async () => {
|
||
|
await session.navigateTo(`http://127.0.0.1:${port}/area`);
|
||
|
|
||
|
let body = await session.findElementByCSS('body');
|
||
|
await body.sendKeys(Key.Shift, 'f');
|
||
|
await eventually(async() => {
|
||
|
let hints = await session.findElementsByCSS('.vimvixen-hint');
|
||
|
assert.equal(hints.length, 3);
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should shows hints only in viewport', async () => {
|
||
|
await session.navigateTo(`http://127.0.0.1:${port}/test1`);
|
||
|
|
||
|
let body = await session.findElementByCSS('body');
|
||
|
await body.sendKeys(Key.Shift, 'f');
|
||
|
await eventually(async() => {
|
||
|
let hints = await session.findElementsByCSS('.vimvixen-hint');
|
||
|
assert.equal(hints.length, 1);
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should shows hints only in window of the frame', async () => {
|
||
|
await session.navigateTo(`http://127.0.0.1:${port}/test2`);
|
||
|
|
||
|
let body = await session.findElementByCSS('body');
|
||
|
await body.sendKeys(Key.Shift, 'f');
|
||
|
|
||
|
await session.switchToFrame(0);
|
||
|
await eventually(async() => {
|
||
|
let hints = await session.findElementsByCSS('.vimvixen-hint');
|
||
|
assert.equal(hints.length, 1);
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should shows hints only in the frame', async () => {
|
||
|
await session.navigateTo(`http://127.0.0.1:${port}/test3`);
|
||
|
|
||
|
let body = await session.findElementByCSS('body');
|
||
|
await body.sendKeys(Key.Shift, 'f');
|
||
|
|
||
|
await session.switchToFrame(0);
|
||
|
await eventually(async() => {
|
||
|
let hints = await session.findElementsByCSS('.vimvixen-hint');
|
||
|
assert.equal(hints.length, 1);
|
||
|
});
|
||
|
});
|
||
|
});
|