├── .gitignore
├── .jshintrc
├── LICENSE.md
├── README.md
├── demos
├── assets
│ ├── castle-brick.png
│ ├── debugcanvas.js
│ ├── demo.gif
│ ├── ghost.png
│ ├── github.png
│ ├── mark-github-black-128.png
│ ├── mark-github-white-128.png
│ ├── plant.png
│ ├── shroom.png
│ ├── style.css
│ └── toad.png
└── index.html
├── dist
└── index.js
├── package.json
└── src
└── index.js
/.gitignore:
--------------------------------------------------------------------------------
1 | node_modules
2 | .DS_Store
3 |
--------------------------------------------------------------------------------
/.jshintrc:
--------------------------------------------------------------------------------
1 | {
2 | "esversion": 6,
3 | "asi": true
4 | }
5 |
--------------------------------------------------------------------------------
/LICENSE.md:
--------------------------------------------------------------------------------
1 | The MIT License (MIT)
2 | =====================
3 |
4 | Copyright © `2017` `Djordje Ungar`
5 |
6 | Permission is hereby granted, free of charge, to any person
7 | obtaining a copy of this software and associated documentation
8 | files (the “Software”), to deal in the Software without
9 | restriction, including without limitation the rights to use,
10 | copy, modify, merge, publish, distribute, sublicense, and/or sell
11 | copies of the Software, and to permit persons to whom the
12 | Software is furnished to do so, subject to the following
13 | conditions:
14 |
15 | The above copyright notice and this permission notice shall be
16 | included in all copies or substantial portions of the Software.
17 |
18 | THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND,
19 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
20 | OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
21 | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
22 | HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
23 | WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
24 | FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
25 | OTHER DEALINGS IN THE SOFTWARE.
26 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | [](https://github.com/ArtBIT/gravity-cursor) [](https://github.com/ArtBIT/gravity-cursor) [](https://github.com/ArtBIT/gravity-cursor/issues)
2 |
3 | # Cursor Gravity
4 |
5 | This is a small experiment that hijacks the user cursor and makes it attract to or repel from certain elements on the page.
6 |
7 | Try the live demo [here](https://artbit.github.io/gravity-cursor/demos/).
8 |
9 | [](http://github.com/artbit/gravity-cursor/)
10 |
11 | ## How it works?
12 | It makes the user cursor invisible using a simple `cursor: none;` CSS rule, and replaces it with a simple image element, which is moved around the screen to imitate original cursor, but making it react to attractors and deflectors on the page.
13 |
14 | ## Usage
15 | ```js
16 | GravityCursor
17 | .attract(document.querySelector('a.attractor'))
18 | .repel(document.querySelector('a.deflector'))
19 | .start();
20 |
21 | document.querySelector('a.stop').addEventListener('click', function() {
22 | GravityCursor.stop();
23 | });
24 | ```
25 |
26 | This will replace the real cursor with the fake one and activate the 'repel' and 'attract' behavior on the selected DOM elements.
27 |
28 | ## Local Build
29 | ```
30 | git clone https://github.com/ArtBIT/gravity-cursor.git
31 | cd gravity-cursor
32 | npm install
33 | npm start
34 | ```
35 |
36 | ## License
37 |
38 | MIT
39 |
40 |
41 | ## Credits
42 |
43 | Inspired by [javierbyte/control-user-cursor](https://github.com/javierbyte/control-user-cursor)
44 |
--------------------------------------------------------------------------------
/demos/assets/castle-brick.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ArtBIT/gravity-cursor/8ba0a2236fee5f5ff37ae81936e41ef4e72e9386/demos/assets/castle-brick.png
--------------------------------------------------------------------------------
/demos/assets/debugcanvas.js:
--------------------------------------------------------------------------------
1 | window.DebugCanvas = (function() {
2 | var canvas = document.createElement('canvas');
3 | canvas.style.position = 'fixed';
4 | canvas.style.pointerEvents = 'none';
5 | canvas.width = canvas.style.width = window.innerWidth;
6 | canvas.height = canvas.style.height = window.innerHeight
7 | document.body.appendChild(canvas);
8 | var ctx = canvas.getContext('2d');
9 | var applyContextOptions = function(options) {
10 | options = options || {};
11 | ctx.lineWidth = options.lineWidth || 1;
12 | ctx.strokeStyle = options.strokeStyle || '#F00';
13 | if (options.lineDash) {
14 | ctx.setLineDash(options.lineDash);
15 | }
16 | if (options.opacity) {
17 | ctx.globalAlpha = options.opacity;
18 | }
19 | };
20 | var api = {
21 | show: function() {
22 | canvas.style.display = 'block';
23 | },
24 | hide: function() {
25 | canvas.style.display = 'none';
26 | },
27 | clear: function() {
28 | ctx.clearRect(0, 0, canvas.width, canvas.height);
29 | },
30 | draw: {
31 | circle: function(x, y, radius, options) {
32 | options = options || {};
33 | ctx.beginPath();
34 | ctx.arc(x, y, radius, 0, 2 * Math.PI, false);
35 | ctx.closePath();
36 | api.draw.stroke(options);
37 | },
38 | rect: function(x, y, width, height, options) {
39 | ctx.beginPath();
40 | ctx.rect(x, y, width, height);
41 | ctx.closePath();
42 | api.draw.stroke(options);
43 | },
44 | line: function(x, y, x1, y1, options) {
45 | ctx.beginPath();
46 | ctx.moveTo(x, y);
47 | ctx.lineTo(x1, y1);
48 | ctx.closePath();
49 | api.draw.stroke(options);
50 | },
51 | moveTo: function(x, y) {
52 | ctx.moveTo(x, y);
53 | },
54 | stroke: function(options) {
55 | ctx.save();
56 | applyContextOptions(options);
57 | ctx.stroke();
58 | ctx.restore();
59 | },
60 | image: function(img, x, y, w, h, dx, dy, dw, dh, options) {
61 | ctx.save();
62 | switch (arguments.length) {
63 | case 2:
64 | applyContextOptions(x);
65 | ctx.drawImage(img, 0, 0);
66 | break;
67 | case 1:
68 | ctx.drawImage(img, 0, 0);
69 | break;
70 | case 4:
71 | case 3:
72 | applyContextOptions(w);
73 | ctx.drawImage(img, x, y);
74 | break;
75 | case 6:
76 | case 5:
77 | applyContextOptions(dx);
78 | ctx.drawImage(img, x, y, w, h);
79 | break;
80 | default:
81 | applyContextOptions(options);
82 | ctx.drawImage(img, x, y, w, h, dx, dy, dw, dh);
83 | }
84 | ctx.restore();
85 | }
86 | }
87 | };
88 | api.clear();
89 | return api;
90 | })();
91 |
--------------------------------------------------------------------------------
/demos/assets/demo.gif:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ArtBIT/gravity-cursor/8ba0a2236fee5f5ff37ae81936e41ef4e72e9386/demos/assets/demo.gif
--------------------------------------------------------------------------------
/demos/assets/ghost.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ArtBIT/gravity-cursor/8ba0a2236fee5f5ff37ae81936e41ef4e72e9386/demos/assets/ghost.png
--------------------------------------------------------------------------------
/demos/assets/github.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ArtBIT/gravity-cursor/8ba0a2236fee5f5ff37ae81936e41ef4e72e9386/demos/assets/github.png
--------------------------------------------------------------------------------
/demos/assets/mark-github-black-128.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ArtBIT/gravity-cursor/8ba0a2236fee5f5ff37ae81936e41ef4e72e9386/demos/assets/mark-github-black-128.png
--------------------------------------------------------------------------------
/demos/assets/mark-github-white-128.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ArtBIT/gravity-cursor/8ba0a2236fee5f5ff37ae81936e41ef4e72e9386/demos/assets/mark-github-white-128.png
--------------------------------------------------------------------------------
/demos/assets/plant.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ArtBIT/gravity-cursor/8ba0a2236fee5f5ff37ae81936e41ef4e72e9386/demos/assets/plant.png
--------------------------------------------------------------------------------
/demos/assets/shroom.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ArtBIT/gravity-cursor/8ba0a2236fee5f5ff37ae81936e41ef4e72e9386/demos/assets/shroom.png
--------------------------------------------------------------------------------
/demos/assets/style.css:
--------------------------------------------------------------------------------
1 | *,
2 | *::after,
3 | *::before {
4 | -webkit-box-sizing: border-box;
5 | box-sizing: border-box;
6 | }
7 |
8 | body {
9 | min-height: 100vh;
10 | height: auto;
11 | width: 100vw;
12 | font-family: Helvetica, Arial, sans-serif;
13 | font-size: 15px;
14 | margin: 0;
15 | padding: 0;
16 | color: #ddd;
17 | background-color: #111;
18 | -webkit-font-smoothing: antialiased;
19 | -moz-osx-font-smoothing: grayscale;
20 | overflow: hidden;
21 | }
22 |
23 | a {
24 | text-decoration: none;
25 | border-color: transparent;
26 | color: #ddd;
27 | outline: none;
28 | padding: 0 0 0.15em;
29 | }
30 |
31 | a.active,
32 | a:hover,
33 | a:focus {
34 | color: #000;
35 | }
36 |
37 | .hidden {
38 | position: absolute;
39 | overflow: hidden;
40 | width: 0;
41 | height: 0;
42 | pointer-events: none;
43 | }
44 |
45 | header {
46 | padding: 1em;
47 | position: absolute;
48 | width: 100%;
49 | }
50 |
51 | header .row {
52 | display: flex;
53 | flex-direction: row;
54 | flex-wrap: wrap;
55 | align-items: center;
56 | width: 100%;
57 | }
58 |
59 | header .title {
60 | font-size: 1.85em;
61 | font-weight: normal;
62 | margin: 0;
63 | padding: 0;
64 | }
65 |
66 | header .github-link {
67 | height: 1em;
68 | width: 1em;
69 | display: inline-block;
70 | background-image: url(mark-github-white-128.png);
71 | background-repeat: no-repeat;
72 | background-size: cover;
73 | text-decoration: none;
74 | border: none;
75 | position: relative;
76 | top: 5px;
77 | padding: 2px;
78 | }
79 |
80 | header a {
81 | border-bottom: 2px solid;
82 | }
83 |
84 | header .tagline {
85 | margin: 1em 0 0.5em;
86 | width: 100%;
87 | }
88 |
89 | header .description {
90 | margin: 0 0 1em 0;
91 | font-weight: bold;
92 | width: 100%;
93 | }
94 |
95 | .demos {
96 | margin: 0 0 0 auto;
97 | }
98 |
99 | .demo {
100 | display: inline-block;
101 | margin: 0 1em 0.5em 0;
102 | padding: 0 0 0.25em;
103 | }
104 |
105 | .not-clickable {
106 | pointer-events: none;
107 | cursor: normal;
108 | }
109 |
110 | @media screen and (max-width: 60em) {
111 | .header {
112 | flex-direction: column;
113 | align-items: flex-start;
114 | font-size: 0.85em;
115 | }
116 | .demos {
117 | width: 100%;
118 | margin: 1em 0 0;
119 | }
120 | }
121 |
122 | /* THEME */
123 |
124 | /* COLOR */
125 | body,
126 | .demos,
127 | a, a:visited {
128 | color: #DDD;
129 | }
130 |
131 | /* HOVER COLOR */
132 | a.active,
133 | a:hover,
134 | a:focus {
135 | color: #fff;
136 | }
137 |
138 | /* BACKGROUND */
139 |
--------------------------------------------------------------------------------
/demos/assets/toad.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ArtBIT/gravity-cursor/8ba0a2236fee5f5ff37ae81936e41ef4e72e9386/demos/assets/toad.png
--------------------------------------------------------------------------------
/demos/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 | Cursor Gravity Demo
12 |
13 |
14 |
15 |
16 |
17 |
68 |
69 |
70 |
80 |
81 |
82 |
83 |
352 |
353 |
--------------------------------------------------------------------------------
/dist/index.js:
--------------------------------------------------------------------------------
1 | window.GravityCursor = function () {
2 | const VIRTUAL_CURSOR_CLASSNAME = 'virtual-cursor';
3 | const VIRTUAL_CURSOR_ZINDEX = 10000;
4 | let showDebugInfo = false;
5 | let debugLevel = 0;
6 |
7 | let cursor;
8 | let forces = [];
9 | let body = document.getElementsByTagName('body')[0];
10 | let html = document.getElementsByTagName('html')[0];
11 |
12 | const CursorAssets = {
13 | mac_retina: {
14 | src: '',
15 | height: 22,
16 | width: 15
17 | },
18 | mac: {
19 | src: '',
20 | width: 15,
21 | height: 22
22 | },
23 | other: {
24 | src: '',
25 | width: 17,
26 | height: 23
27 | }
28 | };
29 |
30 | function onNextFrame(callback) {
31 | window.requestAnimationFrame(() => {
32 | return callback();
33 | });
34 | }
35 |
36 | function VirtualCursor() {
37 | let type = window.navigator.platform.indexOf('Mac') > -1 ? 'mac' : 'win';
38 | if (window.devicePixelRatio > 1) {
39 | type += '_retina';
40 | }
41 | let config = CursorAssets[type] || CursorAssets.other;
42 | let node = document.createElement('img');
43 |
44 | node.style.width = config.width + 'px';
45 | node.style.height = config.height + 'px';
46 | node.className = VIRTUAL_CURSOR_CLASSNAME;
47 | node.src = config.src;
48 | body.appendChild(node);
49 |
50 | this.image = node;
51 | var visible = false;
52 |
53 | this.show = () => {
54 | visible = true;
55 | };
56 |
57 | this.hide = () => {
58 | visible = false;
59 | node.style.visibility = "hidden";
60 | };
61 |
62 | this.activate = () => {
63 | html.classList.add(VIRTUAL_CURSOR_CLASSNAME);
64 | };
65 |
66 | this.deactivate = () => {
67 | onNextFrame(() => {
68 | html.classList.remove(VIRTUAL_CURSOR_CLASSNAME);
69 | });
70 | };
71 |
72 | this.moveTo = (x, y) => {
73 | if (visible && node.style.visibility == "hidden") {
74 | node.style.visibility = "visible";
75 | }
76 | node.style.transform = 'translate(' + x + 'px, ' + y + 'px)';
77 | };
78 |
79 | this.isVisible = () => {
80 | return node.style.display !== "none";
81 | };
82 | }
83 |
84 | function constrain(value, min, max) {
85 | if (min !== undefined) {
86 | if (value < min) {
87 | return min;
88 | }
89 | }
90 | if (max !== undefined) {
91 | if (value > max) {
92 | return max;
93 | }
94 | }
95 | return value;
96 | }
97 |
98 | function dispatchEvent(element, eventType, detail) {
99 | var event = new CustomEvent(eventType, { detail: detail });
100 | element.dispatchEvent(event);
101 | }
102 |
103 | function calculateForces(position, forces) {
104 | let force = {
105 | x: 0,
106 | y: 0
107 | };
108 | if (showDebugInfo) DebugCanvas.clear();
109 |
110 | forces.forEach(obj => {
111 | // Calculating object rectangle is more expensive, but simpler since we do not have to keep track
112 | // of object's position and window resizing.
113 |
114 | const rect = obj.node.getBoundingClientRect();
115 | const rx = rect.left + rect.width / 2;
116 | const ry = rect.top + rect.height / 2;
117 | const dx = position.x - rx;
118 | const dy = position.y - ry;
119 | const d = Math.sqrt(dx * dx + dy * dy);
120 |
121 | const minRadius = obj.radius;
122 | const maxRadius = obj.radius << 1;
123 | let fx, fy, strength;
124 |
125 | if (d <= maxRadius) {
126 | var radius = constrain(d, minRadius) - minRadius;
127 | var angle = Math.atan2(dy, dx);
128 | if (obj.direction == 1) {
129 | // REPEL
130 | strength = Math.pow(radius / minRadius, 2);
131 | radius *= strength;
132 | radius += minRadius - d;
133 | force.x += fx = Math.cos(angle) * radius;
134 | force.y += fy = Math.sin(angle) * radius;
135 | if (showDebugInfo) {
136 | if (debugLevel == 1) DebugCanvas.draw.line(rx, ry, rx + fx, ry + fy, { strokeStyle: '#F00', lineWidth: 3 });
137 | DebugCanvas.draw.circle(rx, ry, minRadius, { strokeStyle: '#F00', lineWidth: 1, lineDash: [2, 2] });
138 | }
139 | dispatchEvent(obj.node, 'repel', { strength: strength, distance: d });
140 | } else if (obj.direction == -1) {
141 | // ATTRACT
142 | strength = Math.pow(radius / minRadius, 2);
143 | radius = strength * d;
144 | force.x += fx = Math.cos(angle) * (radius - d);
145 | force.y += fy = Math.sin(angle) * (radius - d);
146 | if (showDebugInfo) {
147 | if (debugLevel == 1) DebugCanvas.draw.line(rx, ry, rx + fx, ry + fy, { strokeStyle: '#0F0', lineWidth: 3 });
148 | DebugCanvas.draw.circle(rx, ry, minRadius, { strokeStyle: '#0F0', lineWidth: 1, lineDash: [2, 2] });
149 | }
150 | dispatchEvent(obj.node, 'attract', { strength: 1 - strength, distance: d });
151 | }
152 | }
153 | });
154 | if (showDebugInfo) DebugCanvas.draw.image(cursor.image, position.x, position.y, { opacity: 0.5 });
155 | return force;
156 | }
157 |
158 | function createCursorStyles() {
159 | var style = document.createElement('style');
160 | style.type = 'text/css';
161 | style.innerHTML = `
162 | .${VIRTUAL_CURSOR_CLASSNAME},
163 | .${VIRTUAL_CURSOR_CLASSNAME} * {
164 | cursor: none;
165 | -moz-user-select: none;
166 | user-select: none;
167 | -webkit-user-select: none;
168 | }
169 | img.${VIRTUAL_CURSOR_CLASSNAME} {
170 | position: fixed;
171 | display: none;
172 | pointer-events: none;
173 | z-index: ${VIRTUAL_CURSOR_ZINDEX};
174 | }
175 | .${VIRTUAL_CURSOR_CLASSNAME} img.${VIRTUAL_CURSOR_CLASSNAME} {
176 | display: inline-block;
177 | }
178 | `;
179 | body.appendChild(style);
180 | }
181 |
182 | function bindEvents() {
183 |
184 | function onMouseOut(e) {
185 | // If relatedTarget is null, that means the mouse has left the page
186 | if (!e.relatedTarget) {
187 | cursor.deactivate();
188 | }
189 | }
190 |
191 | function onBlur() {
192 | onNextFrame(() => {
193 | cursor.deactivate();
194 | });
195 | }
196 |
197 | function onFocus() {
198 | if (!forces.length) {
199 | return;
200 | }
201 | onNextFrame(() => {
202 | cursor.activate();
203 | });
204 | }
205 |
206 | function onMouseMove(evt) {
207 | if (!forces.length) {
208 | return;
209 | }
210 | cursor.activate();
211 | let mouse = {
212 | x: evt.clientX,
213 | y: evt.clientY
214 | };
215 |
216 | let force = calculateForces(mouse, forces);
217 | cursor.moveTo(mouse.x + force.x, mouse.y + force.y);
218 | }
219 |
220 | document.addEventListener('mouseout', onMouseOut);
221 | document.addEventListener('mousemove', onMouseMove);
222 | window.addEventListener('focus', onFocus);
223 | window.addEventListener('blur', onBlur);
224 | }
225 |
226 | function addForceElement(element, radius, direction) {
227 | forces.push({
228 | node: element,
229 | radius: radius || 100,
230 | direction: direction
231 | });
232 | }
233 |
234 | function attract(element, radius) {
235 | addForceElement(element, radius, -1);
236 | return this;
237 | }
238 |
239 | function repel(element, radius) {
240 | addForceElement(element, radius, 1);
241 | return this;
242 | }
243 |
244 | function start() {
245 | show();
246 | return this;
247 | }
248 |
249 | function stop(element) {
250 | if (element) {
251 | forces = forces.filter(item => item.node !== element);
252 | } else {
253 | forces = [];
254 | }
255 | if (!forces.length) {
256 | cursor.hide();
257 | cursor.deactivate();
258 | }
259 | return this;
260 | }
261 |
262 | function debug(enable, level) {
263 | showDebugInfo = enable;
264 | debugLevel = level || 0;
265 | if (DebugCanvas) DebugCanvas.clear();
266 | return this;
267 | }
268 |
269 | function show() {
270 | if (forces.length) {
271 | cursor.show();
272 | cursor.activate();
273 | }
274 | return this;
275 | }
276 |
277 | function hide() {
278 | cursor.hide();
279 | return this;
280 | }
281 |
282 | function init() {
283 | forces = [];
284 | cursor = new VirtualCursor();
285 |
286 | createCursorStyles();
287 | bindEvents();
288 |
289 | const controller = {};
290 | controller.debug = debug;
291 | controller.repel = repel.bind(controller);
292 | controller.attract = attract.bind(controller);
293 | controller.stop = stop.bind(controller);
294 | controller.start = start.bind(controller);
295 | controller.show = show.bind(controller);
296 | controller.hide = hide.bind(controller);
297 | return controller;
298 | }
299 |
300 | return init();
301 | }();
302 |
303 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "gravity-cursor",
3 | "version": "1.0.0",
4 | "description": "Hijack user cursor and make it react to attractive/deflective forces on the page.",
5 | "main": "src/index.js",
6 | "scripts": {
7 | "build": "babel src/index.js > dist/index.js",
8 | "serve": "http-server -p 9090 .",
9 | "demo": "open http://localhost:9090/demos/",
10 | "start": "npm run build && npm run demo && npm run serve -s"
11 | },
12 | "repository": {
13 | "type": "git",
14 | "url": "git+https://github.com/ArtBIT/gravity-cursor.git"
15 | },
16 | "keywords": [
17 | "mouse",
18 | "cursor",
19 | "hijack",
20 | "repel",
21 | "attract",
22 | "deflect"
23 | ],
24 | "author": "Djordje Ungar",
25 | "license": "MIT",
26 | "bugs": {
27 | "url": "https://github.com/ArtBIT/gravity-cursor/issues"
28 | },
29 | "homepage": "https://github.com/ArtBIT/gravity-cursor#readme",
30 | "devDependencies": {
31 | "babel-cli": "^6.24.1",
32 | "http-server": "^0.9.0",
33 | "open": "0.0.5"
34 | }
35 | }
36 |
--------------------------------------------------------------------------------
/src/index.js:
--------------------------------------------------------------------------------
1 | window.GravityCursor = (function() {
2 | const VIRTUAL_CURSOR_CLASSNAME = 'virtual-cursor';
3 | const VIRTUAL_CURSOR_ZINDEX = 10000;
4 | let showDebugInfo = false;
5 | let debugLevel = 0;
6 |
7 | let cursor;
8 | let forces = [];
9 | let body = document.getElementsByTagName('body')[0];
10 | let html = document.getElementsByTagName('html')[0];
11 |
12 | const CursorAssets = {
13 | mac_retina: {
14 | src: '',
15 | height: 22,
16 | width: 15
17 | },
18 | mac: {
19 | src: '',
20 | width: 15,
21 | height: 22
22 | },
23 | other: {
24 | src: '',
25 | width: 17,
26 | height: 23
27 | }
28 | };
29 |
30 |
31 | function onNextFrame(callback) {
32 | window.requestAnimationFrame(() => {
33 | return callback();
34 | });
35 | }
36 |
37 | function VirtualCursor() {
38 | let type = window.navigator.platform.indexOf('Mac') > -1 ? 'mac' : 'win';
39 | if (window.devicePixelRatio > 1) {
40 | type += '_retina';
41 | }
42 | let config = CursorAssets[type] || CursorAssets.other;
43 | let node = document.createElement('img');
44 |
45 | node.style.width = config.width + 'px'
46 | node.style.height = config.height + 'px';
47 | node.className = VIRTUAL_CURSOR_CLASSNAME;
48 | node.src = config.src;
49 | body.appendChild(node);
50 |
51 | this.image = node;
52 | var visible = false;
53 |
54 | this.show = () => {
55 | visible = true;
56 | }
57 |
58 | this.hide = () => {
59 | visible = false;
60 | node.style.visibility = "hidden";
61 | }
62 |
63 | this.activate = () => {
64 | html.classList.add(VIRTUAL_CURSOR_CLASSNAME);
65 | }
66 |
67 | this.deactivate = () => {
68 | onNextFrame(() => {
69 | html.classList.remove(VIRTUAL_CURSOR_CLASSNAME);
70 | });
71 | }
72 |
73 | this.moveTo = (x, y) => {
74 | if (visible && node.style.visibility == "hidden") {
75 | node.style.visibility = "visible";
76 | }
77 | node.style.transform = 'translate(' + x + 'px, ' + y + 'px)';
78 | }
79 |
80 | this.isVisible = () => {
81 | return node.style.display !== "none";
82 | }
83 | }
84 |
85 | function constrain(value, min, max) {
86 | if (min !== undefined) {
87 | if (value < min) {
88 | return min;
89 | }
90 | }
91 | if (max !== undefined) {
92 | if (value > max) {
93 | return max;
94 | }
95 | }
96 | return value;
97 | }
98 |
99 | function dispatchEvent(element, eventType, detail) {
100 | var event = new CustomEvent(eventType, {detail: detail});
101 | element.dispatchEvent(event);
102 | }
103 |
104 | function calculateForces(position, forces) {
105 | let force = {
106 | x: 0,
107 | y: 0
108 | };
109 | if (showDebugInfo) DebugCanvas.clear();
110 |
111 | forces.forEach(obj => {
112 | // Calculating object rectangle is more expensive, but simpler since we do not have to keep track
113 | // of object's position and window resizing.
114 |
115 | const rect = obj.node.getBoundingClientRect();
116 | const rx = rect.left + rect.width / 2;
117 | const ry = rect.top + rect.height/2;
118 | const dx = position.x - rx;
119 | const dy = position.y - ry;
120 | const d = Math.sqrt(dx * dx + dy * dy);
121 |
122 | const minRadius = obj.radius;
123 | const maxRadius = obj.radius << 1;
124 | let fx, fy, strength;
125 |
126 | if (d <= maxRadius) {
127 | var radius = constrain(d, minRadius) - minRadius;
128 | var angle = Math.atan2(dy, dx);
129 | if (obj.direction == 1) {
130 | // REPEL
131 | strength = Math.pow(radius / minRadius, 2)
132 | radius *= strength;
133 | radius += minRadius - d;
134 | force.x += fx = Math.cos(angle) * radius;
135 | force.y += fy = Math.sin(angle) * radius;
136 | if (showDebugInfo) {
137 | if (debugLevel == 1) DebugCanvas.draw.line(rx, ry, rx + fx, ry + fy, { strokeStyle: '#F00', lineWidth: 3 });
138 | DebugCanvas.draw.circle(rx, ry, minRadius, { strokeStyle: '#F00', lineWidth: 1, lineDash: [2, 2] });
139 | }
140 | dispatchEvent(obj.node, 'repel', {strength: strength, distance: d});
141 | } else if (obj.direction == -1) {
142 | // ATTRACT
143 | strength = Math.pow(radius / minRadius, 2);
144 | radius = strength * d;
145 | force.x += fx = Math.cos(angle) * (radius - d);
146 | force.y += fy = Math.sin(angle) * (radius - d);
147 | if (showDebugInfo) {
148 | if (debugLevel == 1) DebugCanvas.draw.line(rx, ry, rx + fx, ry + fy, { strokeStyle: '#0F0', lineWidth: 3 });
149 | DebugCanvas.draw.circle(rx, ry, minRadius, { strokeStyle: '#0F0', lineWidth: 1, lineDash: [2, 2] });
150 | }
151 | dispatchEvent(obj.node, 'attract', {strength: 1-strength, distance: d});
152 | }
153 | }
154 | })
155 | if (showDebugInfo) DebugCanvas.draw.image(cursor.image, position.x, position.y, {opacity: 0.5});
156 | return force;
157 | }
158 |
159 | function createCursorStyles() {
160 | var style = document.createElement('style');
161 | style.type = 'text/css';
162 | style.innerHTML = `
163 | .${VIRTUAL_CURSOR_CLASSNAME},
164 | .${VIRTUAL_CURSOR_CLASSNAME} * {
165 | cursor: none;
166 | -moz-user-select: none;
167 | user-select: none;
168 | -webkit-user-select: none;
169 | }
170 | img.${VIRTUAL_CURSOR_CLASSNAME} {
171 | position: fixed;
172 | display: none;
173 | pointer-events: none;
174 | z-index: ${VIRTUAL_CURSOR_ZINDEX};
175 | }
176 | .${VIRTUAL_CURSOR_CLASSNAME} img.${VIRTUAL_CURSOR_CLASSNAME} {
177 | display: inline-block;
178 | }
179 | `;
180 | body.appendChild(style);
181 | }
182 |
183 | function bindEvents() {
184 |
185 | function onMouseOut(e) {
186 | // If relatedTarget is null, that means the mouse has left the page
187 | if (!e.relatedTarget) {
188 | cursor.deactivate();
189 | }
190 | }
191 |
192 | function onBlur() {
193 | onNextFrame(() => {
194 | cursor.deactivate();
195 | });
196 | }
197 |
198 | function onFocus() {
199 | if (!forces.length) {
200 | return;
201 | }
202 | onNextFrame(() => {
203 | cursor.activate();
204 | });
205 | }
206 |
207 | function onMouseMove(evt) {
208 | if (!forces.length) {
209 | return;
210 | }
211 | cursor.activate();
212 | let mouse = {
213 | x: evt.clientX,
214 | y: evt.clientY
215 | }
216 |
217 | let force = calculateForces(mouse, forces);
218 | cursor.moveTo(mouse.x + force.x, mouse.y + force.y);
219 | }
220 |
221 | document.addEventListener('mouseout', onMouseOut);
222 | document.addEventListener('mousemove', onMouseMove);
223 | window.addEventListener('focus', onFocus);
224 | window.addEventListener('blur', onBlur);
225 | }
226 |
227 | function addForceElement(element, radius, direction) {
228 | forces.push({
229 | node: element,
230 | radius: radius || 100,
231 | direction: direction,
232 | });
233 | }
234 |
235 | function attract(element, radius) {
236 | addForceElement(element, radius, -1);
237 | return this;
238 | }
239 |
240 | function repel(element, radius) {
241 | addForceElement(element, radius, 1);
242 | return this;
243 | }
244 |
245 | function start() {
246 | show();
247 | return this;
248 | }
249 |
250 | function stop(element) {
251 | if (element) {
252 | forces = forces.filter(item => item.node !== element);
253 | } else {
254 | forces = [];
255 | }
256 | if (!forces.length) {
257 | cursor.hide();
258 | cursor.deactivate();
259 | }
260 | return this;
261 | }
262 |
263 | function debug(enable, level) {
264 | showDebugInfo = enable;
265 | debugLevel = level || 0;
266 | if (DebugCanvas) DebugCanvas.clear();
267 | return this;
268 | }
269 |
270 | function show() {
271 | if (forces.length) {
272 | cursor.show();
273 | cursor.activate();
274 | }
275 | return this;
276 | }
277 |
278 | function hide() {
279 | cursor.hide();
280 | return this;
281 | }
282 |
283 | function init() {
284 | forces = [];
285 | cursor = new VirtualCursor();
286 |
287 | createCursorStyles();
288 | bindEvents();
289 |
290 | const controller = {};
291 | controller.debug = debug;
292 | controller.repel = repel.bind(controller);
293 | controller.attract = attract.bind(controller);
294 | controller.stop = stop.bind(controller);
295 | controller.start = start.bind(controller);
296 | controller.show = show.bind(controller);
297 | controller.hide = hide.bind(controller);
298 | return controller;
299 | }
300 |
301 | return init();
302 | })();
303 |
--------------------------------------------------------------------------------