Skip to content

Commit 50bdff4

Browse files
Theodore MessinezisTheodore Messinezis
Theodore Messinezis
authored and
Theodore Messinezis
committedApr 11, 2020
test: add basic tests for plugin and directive
1 parent 4908d5c commit 50bdff4

11 files changed

+7116
-1644
lines changed
 

‎.eslintignore

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,3 @@
1-
node_modules/
21
dist/
2+
coverage/
3+
node_modules/

‎.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,3 @@
11
dist/
2+
coverage/
23
node_modules/

‎jest.config.js

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
module.exports = {
2+
verbose: true,
3+
preset: 'ts-jest',
4+
collectCoverage: true,
5+
collectCoverageFrom: [
6+
'src/**/*.ts',
7+
],
8+
};

‎package-lock.json

+6,967-1,605
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

‎package.json

+9-2
Original file line numberDiff line numberDiff line change
@@ -17,21 +17,28 @@
1717
},
1818
"scripts": {
1919
"build": "rollup -c",
20-
"lint": "eslint --ext .ts ./"
20+
"lint": "eslint --ext .ts ./",
21+
"test": "jest"
2122
},
2223
"devDependencies": {
2324
"@rollup/plugin-typescript": "^4.0.0",
2425
"@semantic-release/exec": "^5.0.0",
26+
"@types/jest": "^25.2.1",
2527
"@typescript-eslint/eslint-plugin": "^2.27.0",
28+
"@vue/test-utils": "^1.0.0-beta.33",
2629
"eslint": "^6.8.0",
2730
"eslint-config-airbnb-typescript": "^7.2.0",
2831
"eslint-plugin-import": "^2.20.2",
32+
"eslint-plugin-react": "^7.19.0",
33+
"jest": "^25.3.0",
2934
"rollup": "^2.3.3",
3035
"rollup-plugin-uglify": "^6.0.4",
3136
"semantic-release": "^17.0.4",
37+
"ts-jest": "^25.3.1",
3238
"tslib": "^1.11.1",
3339
"typescript": "^3.8.3",
34-
"vue": "^2.0.0"
40+
"vue": "^2.0.0",
41+
"vue-template-compiler": "^2.6.11"
3542
},
3643
"peerDependencies": {
3744
"vue": "^2.0.0"

‎src/directive.ts

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
import { DirectiveOptions } from 'vue';
2+
import { scroll } from './scroll';
3+
4+
export const directive: DirectiveOptions = {
5+
bind: (el) => {
6+
new MutationObserver(() => {
7+
scroll(el);
8+
}).observe(el, {
9+
childList: true,
10+
subtree: true,
11+
});
12+
},
13+
14+
inserted: (el) => {
15+
scroll(el);
16+
},
17+
};
18+
19+
export default directive;

‎src/index.ts

+3-36
Original file line numberDiff line numberDiff line change
@@ -1,38 +1,5 @@
1-
import { DirectiveOptions, PluginObject } from 'vue';
2-
3-
function scrollElementToBottom(el: Element) {
4-
if (typeof el.scroll === 'function') el.scroll({ top: el.scrollHeight });
5-
else el.scrollTop = el.scrollHeight; // eslint-disable-line no-param-reassign
6-
}
7-
8-
const directive: DirectiveOptions = {
9-
bind: (el) => {
10-
const observer = new MutationObserver((e) => {
11-
const nodesWereAdded = e[e.length - 1].addedNodes.length === 1;
12-
const nodesWereRemoved = e[e.length - 1].removedNodes.length === 1;
13-
if (nodesWereAdded === false && nodesWereRemoved === false) return;
14-
scrollElementToBottom(el);
15-
});
16-
17-
observer.observe(el, {
18-
childList: true,
19-
subtree: true,
20-
});
21-
},
22-
23-
inserted: (el) => {
24-
scrollElementToBottom(el);
25-
},
26-
};
27-
28-
const plugin: PluginObject<any> = {
29-
install: (Vue) => {
30-
Vue.directive('chat-scroll', directive);
31-
},
32-
};
33-
34-
if (typeof window !== 'undefined' && window.Vue) {
35-
window.Vue.use(plugin);
36-
}
1+
/* istanbul ignore file */
2+
import { bootstrap, plugin } from './plugin';
373

4+
bootstrap();
385
export default plugin;

‎src/plugin.ts

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
import { PluginObject } from 'vue';
2+
import { directive } from './directive';
3+
4+
export const plugin: PluginObject<any> = {
5+
install: (Vue) => {
6+
Vue.directive('chat-scroll', directive);
7+
},
8+
};
9+
10+
export const bootstrap = () => {
11+
/* istanbul ignore else */
12+
if (typeof window !== 'undefined' && window.Vue) {
13+
window.Vue.use(plugin);
14+
}
15+
};

‎src/scroll.ts

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
/* istanbul ignore file */
2+
export const scroll = (el: Element): void => {
3+
if (typeof el.scroll === 'function') el.scroll({ top: el.scrollHeight });
4+
else el.scrollTop = el.scrollHeight; // eslint-disable-line no-param-reassign
5+
};
6+
7+
export default scroll;

‎tests/directive.spec.ts

+66
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,66 @@
1+
import * as vueTestUtils from '@vue/test-utils';
2+
import { directive } from '../src/directive';
3+
import { scroll } from '../src/scroll';
4+
5+
jest.mock('../src/scroll');
6+
// Clear all mocks before each test.
7+
beforeEach(() => { jest.clearAllMocks(); });
8+
9+
test('it automatically calls scrolls on init', () => {
10+
const Component: Vue.Component = {
11+
template: '<div v-chat-scroll />',
12+
};
13+
14+
const localVue = vueTestUtils.createLocalVue();
15+
localVue.directive('chat-scroll', directive);
16+
17+
const wrapper = vueTestUtils.mount(Component, { localVue });
18+
expect(scroll).toHaveBeenCalledTimes(1);
19+
expect(scroll).toHaveBeenCalledWith(wrapper.element);
20+
});
21+
22+
test('it calls scroll when an item is added', async () => {
23+
const Component: Vue.Component = {
24+
data: () => ({ items: [] }),
25+
template: '<div v-chat-scroll><div v-for="item in items">{{ item }}</div></div>',
26+
};
27+
28+
const localVue = vueTestUtils.createLocalVue();
29+
localVue.directive('chat-scroll', directive);
30+
31+
const wrapper = vueTestUtils.mount(Component, { localVue });
32+
expect(scroll).toHaveBeenCalledTimes(1);
33+
expect(scroll).toHaveBeenCalledWith(wrapper.element);
34+
35+
(wrapper.vm as any).$data.items.push('A new item');
36+
await (wrapper.vm as any).$nextTick();
37+
expect(scroll).toHaveBeenCalledTimes(2);
38+
39+
(wrapper.vm as any).$data.items.push('Here\'s two items in a row!');
40+
(wrapper.vm as any).$data.items.push('This should call scroll just once!');
41+
await (wrapper.vm as any).$nextTick();
42+
expect(scroll).toHaveBeenCalledTimes(3);
43+
});
44+
45+
test('it calls scroll when an item is removed', async () => {
46+
const Component: Vue.Component = {
47+
data: () => ({ items: [1, 2, 3] }),
48+
template: '<div v-chat-scroll><div v-for="item in items">{{ item }}</div></div>',
49+
};
50+
51+
const localVue = vueTestUtils.createLocalVue();
52+
localVue.directive('chat-scroll', directive);
53+
54+
const wrapper = vueTestUtils.mount(Component, { localVue });
55+
expect(scroll).toHaveBeenCalledTimes(1);
56+
expect(scroll).toHaveBeenCalledWith(wrapper.element);
57+
58+
(wrapper.vm as any).$data.items.pop();
59+
await (wrapper.vm as any).$nextTick();
60+
expect(scroll).toHaveBeenCalledTimes(2);
61+
62+
(wrapper.vm as any).$data.items.pop();
63+
(wrapper.vm as any).$data.items.pop();
64+
await (wrapper.vm as any).$nextTick();
65+
expect(scroll).toHaveBeenCalledTimes(3);
66+
});

‎tests/plugin.spec.ts

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
import { createLocalVue } from '@vue/test-utils';
2+
import { bootstrap, plugin } from '../src/plugin';
3+
import { directive } from '../src/directive';
4+
5+
test('it registers the directive', async () => {
6+
const localVue = createLocalVue();
7+
localVue.directive = jest.fn();
8+
localVue.use(plugin);
9+
expect(localVue.directive).toHaveBeenCalledTimes(1);
10+
expect(localVue.directive).toBeCalledWith('chat-scroll', directive);
11+
});
12+
13+
test('it auto uses itself when window.Vue is defined', () => {
14+
window.Vue = createLocalVue();
15+
window.Vue.use = jest.fn();
16+
bootstrap(); // We assume this gets called automatically.
17+
expect(window.Vue.use).toHaveBeenCalledTimes(1);
18+
expect(window.Vue.use).toBeCalledWith(plugin);
19+
});

0 commit comments

Comments
 (0)
Please sign in to comment.