├── .gitignore
├── .editorconfig
├── source
├── CanvasView.js
├── Color.js
├── Star.js
└── index.js
├── ReadMe.md
├── package.json
├── demo
├── Repository.html
└── Repository.css
├── index.html
└── StarMap.js
/.gitignore:
--------------------------------------------------------------------------------
1 | # https://git-scm.com/docs/gitignore
2 | # https://help.github.com/articles/ignoring-files
3 | # Example .gitignore files: https://github.com/github/gitignore
4 |
5 | node_modules/*
--------------------------------------------------------------------------------
/.editorconfig:
--------------------------------------------------------------------------------
1 | root = true
2 |
3 | [*]
4 | charset = utf-8
5 | indent_style = space
6 | indent_size = 4
7 | end_of_line = lf
8 | insert_final_newline = true
9 | trim_trailing_whitespace = true
10 |
--------------------------------------------------------------------------------
/source/CanvasView.js:
--------------------------------------------------------------------------------
1 | define(['jquery', 'jQueryKit'], function ($) {
2 |
3 | /* ----- 星空图 ----- */
4 |
5 | function CanvasView($_View) {
6 |
7 | this.$_View = $( $_View );
8 |
9 | this.init();
10 |
11 | $( self ).on('resize', $.throttle(this.init.bind( this )));
12 |
13 | return this;
14 | }
15 |
16 | $.Class.extend(CanvasView, null, {
17 | init: function () {
18 |
19 | var $_View = this.$_View.offsetParent();
20 |
21 | this.width = this.$_View[0].width = $_View.width();
22 | this.height = this.$_View[0].height = $_View.height();
23 | }
24 | });
25 |
26 | return CanvasView;
27 |
28 | });
29 |
--------------------------------------------------------------------------------
/ReadMe.md:
--------------------------------------------------------------------------------
1 | # 粒子/星图 交互式动画
2 |
3 |
4 |
5 | ## 【项目缘起】
6 |
7 | 公司同事为了适配 UI 设计图,在网上搜到 [@wangji5850](https://github.com/wangji5850) [原作](http://www.qdfuns.com/notes/26090/1fd69cab2d93ee4fc9c0b881a859abf5.html),用在一个业务系统(原型版)的登录页,效果甚佳。但内部试用后 UI 重做设计,就没最终采用。
8 |
9 | 后来发现两个同技术社区的朋友博客上都有这个动画,就想可以把 **自己在公司开发时重构的版本**分享出来,就建了这个代码仓库。
10 |
11 |
12 |
13 | ## 【参与开发】
14 |
15 | UNIX-Shell、Windows-CMD 通用脚本 ——
16 |
17 | ```Shell
18 | git clone https://github.com/TechQuery/StarMap.git
19 |
20 | npm install
21 |
22 | npm run build
23 |
24 | npm test
25 | ```
26 |
27 |
28 | ## 【同类项目】
29 |
30 | - http://git.hust.cc/canvas-nest.js/
31 |
32 | - https://github.com/hilongjw/Qarticles
33 |
34 | - http://vincentgarreau.com/particles.js/
--------------------------------------------------------------------------------
/source/Color.js:
--------------------------------------------------------------------------------
1 | define(['jquery'], function ($) {
2 |
3 | /* ----- RGB 颜色 ----- */
4 |
5 | function Color(min) {
6 |
7 | min = min || 0;
8 |
9 | var _Self_ = this.constructor;
10 |
11 | this.red = _Self_.random( min );
12 | this.green = _Self_.random( min );
13 | this.blue = _Self_.random( min );
14 |
15 | this.style = _Self_.getStyle(this.red, this.green, this.blue);
16 | }
17 |
18 | $.extend(Color, {
19 | random: function (min) {
20 |
21 | return Math.random() * 255 + min;
22 | },
23 | getStyle: function (red, green, blue) {
24 |
25 | return 'rgba(' + [
26 | Math.floor(red), Math.floor(green), Math.floor(blue), 0.8
27 | ].join(', ') + ')';
28 | }
29 | });
30 |
31 | return Color;
32 |
33 | });
34 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "star-map",
3 | "version": "0.2.0",
4 | "description": "Particle / Star Map interactive animation",
5 | "main": "StarMap.js",
6 | "engines": {
7 | "node": "^5.0.0"
8 | },
9 | "scripts": {
10 | "bundle": "cd source && amd-bundle ../StarMap.js",
11 | "compress": "uglifyjs StarMap.js -o build/StarMap.min.js --config-file build/compress.json",
12 | "build": "npm run bundle && npm run compress",
13 | "test": "opn index.html"
14 | },
15 | "repository": {
16 | "type": "git",
17 | "url": "git+https://github.com/TechQuery/StarMap.git"
18 | },
19 | "keywords": [
20 | "particle",
21 | "animation",
22 | "javascript",
23 | "canvas"
24 | ],
25 | "author": "shiy007@qq.com",
26 | "license": "ISC",
27 | "bugs": {
28 | "url": "https://github.com/TechQuery/StarMap/issues"
29 | },
30 | "homepage": "https://github.com/TechQuery/StarMap#readme",
31 | "devDependencies": {
32 | "amd-bundle": "^0.4.2",
33 | "opn-cli": "^3.1.0",
34 | "uglify-js": "^3.0.24"
35 | }
36 | }
37 |
--------------------------------------------------------------------------------
/source/Star.js:
--------------------------------------------------------------------------------
1 | define(['jquery', './Color'], function ($, Color) {
2 |
3 | /* ----- 恒星点 ----- */
4 |
5 | function Star(Max_X, Max_Y, iContext) {
6 |
7 | this.x = Math.random() * Max_X;
8 | this.y = Math.random() * Max_Y;
9 |
10 | this.vx = Math.random() - 0.5;
11 | this.vy = Math.random() - 0.5;
12 |
13 | this.radius = Math.random();
14 |
15 | this.color = new Color();
16 |
17 | this.context = iContext;
18 | }
19 |
20 | Star.mixValue = function (comp1, weight1, comp2, weight2) {
21 |
22 | return (comp1 * weight1 + comp2 * weight2) / (weight1 + weight2);
23 | };
24 |
25 | $.extend(Star.prototype, {
26 | draw: function () {
27 |
28 | this.context.beginPath();
29 |
30 | this.context.fillStyle = this.color.style;
31 |
32 | this.context.arc(
33 | this.x, this.y, this.radius, 0, Math.PI * 2, false
34 | );
35 | this.context.fill();
36 | },
37 | mixColor: function (iOther) {
38 |
39 | return Color.getStyle(
40 | Star.mixValue(
41 | this.color.red, this.radius, iOther.color.red, iOther.radius
42 | ),
43 | Star.mixValue(
44 | this.color.green, this.radius, iOther.color.red, iOther.radius
45 | ),
46 | Star.mixValue(
47 | this.color.blue, this.radius, iOther.color.red, iOther.radius
48 | )
49 | );
50 | }
51 | });
52 |
53 | return Star;
54 |
55 | });
--------------------------------------------------------------------------------
/demo/Repository.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
--------------------------------------------------------------------------------
/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 | 粒子 / 星图 交互式动画
11 |
12 |
31 |
34 |
35 |
36 |
37 |
56 |
57 |
58 |
60 |
61 |
62 |
--------------------------------------------------------------------------------
/source/index.js:
--------------------------------------------------------------------------------
1 | define([
2 | 'jquery', './Color', './Star', './CanvasView'
3 | ], function ($, Color, Star, CanvasView) {
4 |
5 | function StarMap($_View, iMax, iDistance, iRadius) {
6 |
7 | $.extend(CanvasView.call(this, $_View), {
8 | length: 0,
9 | max: iMax || 250,
10 | distance: iDistance || 100,
11 | radius: iRadius || 150
12 | });
13 |
14 | this.context = this.$_View[0].getContext('2d');
15 |
16 | this.context.lineWidth = 0.3;
17 | this.context.strokeStyle = (new Color(150)).style;
18 |
19 | var MP = this.pointer = {
20 | x: 30 * this.width / 100,
21 | y: 30 * this.height / 100
22 | };
23 |
24 | this.$_View.on('mousemove', function (iEvent) {
25 |
26 | MP.x = iEvent.pageX;
27 | MP.y = iEvent.pageY;
28 |
29 | }).on('mouseleave', function () {
30 |
31 | MP.x = -100;
32 | MP.y = -100
33 | });
34 | }
35 |
36 | CanvasView.extend(StarMap, null, {
37 | move: function () {
38 | var _This_ = this;
39 |
40 | return $.each(this, function () {
41 |
42 | if ((this.y < 0) || (this.y > _This_.height))
43 | this.vy = - this.vy;
44 | else if ((this.x < 0) || (this.x > _This_.width))
45 | this.vx = - this.vx;
46 |
47 | this.x += this.vx;
48 | this.y += this.vy;
49 | });
50 | },
51 | connect: function () {
52 | var _This_ = this;
53 |
54 | return $.each(this, function () {
55 |
56 | for (var i = 0; _This_[i]; i++)
57 | if ((
58 | ((this.x - _This_[i].x) < _This_.distance) &&
59 | ((this.y - _This_[i].y) < _This_.distance) &&
60 | ((this.x - _This_[i].x) > -_This_.distance) &&
61 | ((this.y - _This_[i].y) > -_This_.distance)
62 | ) && (
63 | ((this.x - _This_.pointer.x) < _This_.radius) &&
64 | ((this.y - _This_.pointer.y) < _This_.radius) &&
65 | ((this.x - _This_.pointer.x) > -_This_.radius) &&
66 | ((this.y - _This_.pointer.y) > -_This_.radius)
67 | )) {
68 | _This_.context.beginPath();
69 |
70 | _This_.context.strokeStyle = this.mixColor( _This_[i] );
71 |
72 | _This_.context.moveTo(this.x, this.y);
73 |
74 | _This_.context.lineTo(_This_[i].x, _This_[i].y);
75 |
76 | _This_.context.stroke();
77 |
78 | _This_.context.closePath();
79 | }
80 | });
81 | },
82 | animate: function () {
83 | this.context.clearRect(0, 0, this.width, this.height);
84 |
85 | $.each(this.move().connect(), Star.prototype.draw);
86 |
87 | self.requestAnimationFrame( arguments.callee.bind(this) );
88 | },
89 | render: function () {
90 |
91 | for (var i = 0; i < this.max; i++)
92 | this[ this.length++ ] = new Star(
93 | this.width, this.height, this.context
94 | );
95 |
96 | self.requestAnimationFrame( this.animate.bind(this) );
97 | }
98 | });
99 |
100 | return StarMap;
101 |
102 | });
103 |
--------------------------------------------------------------------------------
/StarMap.js:
--------------------------------------------------------------------------------
1 | //
2 | // Generated by https://www.npmjs.com/package/amd-bundle
3 | //
4 | (function (factory) {
5 |
6 | if ((typeof define === 'function') && define.amd)
7 | define('StarMap', ['jquery', 'jQueryKit'], factory);
8 | else if (typeof module === 'object')
9 | module.exports = factory(require('jquery'), require('jQueryKit'));
10 | else
11 | this.StarMap = factory(this['jquery'], this['jQueryKit']);
12 |
13 | })(function (jquery, jQueryKit) {
14 |
15 |
16 | var Color = (function ($) {
17 |
18 | /* ----- RGB 颜色 ----- */
19 |
20 | function Color(min) {
21 |
22 | min = min || 0;
23 |
24 | var _Self_ = this.constructor;
25 |
26 | this.red = _Self_.random( min );
27 | this.green = _Self_.random( min );
28 | this.blue = _Self_.random( min );
29 |
30 | this.style = _Self_.getStyle(this.red, this.green, this.blue);
31 | }
32 |
33 | $.extend(Color, {
34 | random: function (min) {
35 |
36 | return Math.random() * 255 + min;
37 | },
38 | getStyle: function (red, green, blue) {
39 |
40 | return 'rgba(' + [
41 | Math.floor(red), Math.floor(green), Math.floor(blue), 0.8
42 | ].join(', ') + ')';
43 | }
44 | });
45 |
46 | return Color;
47 |
48 | })(jquery);
49 |
50 |
51 | var Star = (function ($, Color) {
52 |
53 | /* ----- 恒星点 ----- */
54 |
55 | function Star(Max_X, Max_Y, iContext) {
56 |
57 | this.x = Math.random() * Max_X;
58 | this.y = Math.random() * Max_Y;
59 |
60 | this.vx = Math.random() - 0.5;
61 | this.vy = Math.random() - 0.5;
62 |
63 | this.radius = Math.random();
64 |
65 | this.color = new Color();
66 |
67 | this.context = iContext;
68 | }
69 |
70 | Star.mixValue = function (comp1, weight1, comp2, weight2) {
71 |
72 | return (comp1 * weight1 + comp2 * weight2) / (weight1 + weight2);
73 | };
74 |
75 | $.extend(Star.prototype, {
76 | draw: function () {
77 |
78 | this.context.beginPath();
79 |
80 | this.context.fillStyle = this.color.style;
81 |
82 | this.context.arc(
83 | this.x, this.y, this.radius, 0, Math.PI * 2, false
84 | );
85 | this.context.fill();
86 | },
87 | mixColor: function (iOther) {
88 |
89 | return Color.getStyle(
90 | Star.mixValue(
91 | this.color.red, this.radius, iOther.color.red, iOther.radius
92 | ),
93 | Star.mixValue(
94 | this.color.green, this.radius, iOther.color.red, iOther.radius
95 | ),
96 | Star.mixValue(
97 | this.color.blue, this.radius, iOther.color.red, iOther.radius
98 | )
99 | );
100 | }
101 | });
102 |
103 | return Star;
104 |
105 | })(jquery, Color);
106 |
107 |
108 | var CanvasView = (function ($) {
109 |
110 | /* ----- 星空图 ----- */
111 |
112 | function CanvasView($_View) {
113 |
114 | this.$_View = $( $_View );
115 |
116 | this.init();
117 |
118 | $( self ).on('resize', $.throttle(this.init.bind( this )));
119 |
120 | return this;
121 | }
122 |
123 | $.Class.extend(CanvasView, null, {
124 | init: function () {
125 |
126 | var $_View = this.$_View.offsetParent();
127 |
128 | this.width = this.$_View[0].width = $_View.width();
129 | this.height = this.$_View[0].height = $_View.height();
130 | }
131 | });
132 |
133 | return CanvasView;
134 |
135 | })(jquery);
136 |
137 |
138 | return (function ($, Color, Star, CanvasView) {
139 |
140 | function StarMap($_View, iMax, iDistance, iRadius) {
141 |
142 | $.extend(CanvasView.call(this, $_View), {
143 | length: 0,
144 | max: iMax || 250,
145 | distance: iDistance || 100,
146 | radius: iRadius || 150
147 | });
148 |
149 | this.context = this.$_View[0].getContext('2d');
150 |
151 | this.context.lineWidth = 0.3;
152 | this.context.strokeStyle = (new Color(150)).style;
153 |
154 | var MP = this.pointer = {
155 | x: 30 * this.width / 100,
156 | y: 30 * this.height / 100
157 | };
158 |
159 | this.$_View.on('mousemove', function (iEvent) {
160 |
161 | MP.x = iEvent.pageX;
162 | MP.y = iEvent.pageY;
163 |
164 | }).on('mouseleave', function () {
165 |
166 | MP.x = -100;
167 | MP.y = -100
168 | });
169 | }
170 |
171 | CanvasView.extend(StarMap, null, {
172 | move: function () {
173 | var _This_ = this;
174 |
175 | return $.each(this, function () {
176 |
177 | if ((this.y < 0) || (this.y > _This_.height))
178 | this.vy = - this.vy;
179 | else if ((this.x < 0) || (this.x > _This_.width))
180 | this.vx = - this.vx;
181 |
182 | this.x += this.vx;
183 | this.y += this.vy;
184 | });
185 | },
186 | connect: function () {
187 | var _This_ = this;
188 |
189 | return $.each(this, function () {
190 |
191 | for (var i = 0; _This_[i]; i++)
192 | if ((
193 | ((this.x - _This_[i].x) < _This_.distance) &&
194 | ((this.y - _This_[i].y) < _This_.distance) &&
195 | ((this.x - _This_[i].x) > -_This_.distance) &&
196 | ((this.y - _This_[i].y) > -_This_.distance)
197 | ) && (
198 | ((this.x - _This_.pointer.x) < _This_.radius) &&
199 | ((this.y - _This_.pointer.y) < _This_.radius) &&
200 | ((this.x - _This_.pointer.x) > -_This_.radius) &&
201 | ((this.y - _This_.pointer.y) > -_This_.radius)
202 | )) {
203 | _This_.context.beginPath();
204 |
205 | _This_.context.strokeStyle = this.mixColor( _This_[i] );
206 |
207 | _This_.context.moveTo(this.x, this.y);
208 |
209 | _This_.context.lineTo(_This_[i].x, _This_[i].y);
210 |
211 | _This_.context.stroke();
212 |
213 | _This_.context.closePath();
214 | }
215 | });
216 | },
217 | animate: function () {
218 | this.context.clearRect(0, 0, this.width, this.height);
219 |
220 | $.each(this.move().connect(), Star.prototype.draw);
221 |
222 | self.requestAnimationFrame( arguments.callee.bind(this) );
223 | },
224 | render: function () {
225 |
226 | for (var i = 0; i < this.max; i++)
227 | this[ this.length++ ] = new Star(
228 | this.width, this.height, this.context
229 | );
230 |
231 | self.requestAnimationFrame( this.animate.bind(this) );
232 | }
233 | });
234 |
235 | return StarMap;
236 |
237 | })(jquery, Color, Star, CanvasView);
238 | });
--------------------------------------------------------------------------------
/demo/Repository.css:
--------------------------------------------------------------------------------
1 | .github-box * {
2 | -webkit-box-sizing: content-box;
3 | -moz-box-sizing: content-box;
4 | box-sizing: content-box;
5 | }
6 | .github-box {
7 | font-family: helvetica, arial, sans-serif;
8 | font-size: 13px;
9 | line-height: 18px;
10 | background: #fafafa;
11 | border: 1px solid #ddd;
12 | border-radius: 3px;
13 | color: #666;
14 | }
15 | .github-box a {
16 | color: #4183c4;
17 | border: 0;
18 | text-decoration: none;
19 | }
20 | .github-box .github-box-title {
21 | position: relative;
22 | border-bottom: 1px solid #ddd;
23 | border-radius: 3px 3px 0 0;
24 | background: #fcfcfc;
25 | background: -moz-linear-gradient(#fcfcfc, #ebebeb);
26 | background:-webkit-linear-gradient(#fcfcfc, #ebebeb);
27 | }
28 | .github-box .github-box-title h3 {
29 | word-wrap: break-word;
30 | font-family: helvetica, arial, sans-serif;
31 | font-weight: normal;
32 | font-size: 16px;
33 | color: gray;
34 | margin: 0;
35 | padding: 10px 10px 10px 30px;
36 | width: auto;
37 | background-image: url("");
38 | background-position: 7px center;
39 | background-repeat: no-repeat;
40 | }
41 | .github-box .github-box-title h3 .repo {
42 | font-weight: bold;
43 | }
44 | .github-box .github-box-title .github-stats {
45 | float: right;
46 | position: absolute;
47 | top: 8px;
48 | right: 10px;
49 | font-size: 11px;
50 | font-weight: bold;
51 | line-height: 21px;
52 | height: auto;
53 | min-height: 21px;
54 | }
55 | .github-box .github-box-title .github-stats a {
56 | display: inline-block;
57 | height: 21px;
58 | color: #666;
59 | border: 1px solid #ddd;
60 | border-radius: 3px;
61 | padding: 0 5px 0 18px;
62 | background-color: white;
63 | background-image: url("");
64 | background-repeat: no-repeat;
65 | }
66 | .github-box .github-box-title .github-stats .watchers {
67 | border-right: 1px solid #ddd;
68 | }
69 | .github-box .github-box-title .github-stats .forks {
70 | background-position: -4px -21px;
71 | padding-left: 15px;
72 | }
73 | .github-box .github-box-content {
74 | padding: 10px;
75 | font-weight: 300;
76 | }
77 | .github-box .github-box-content p {
78 | margin: 0;
79 | }
80 | .github-box .github-box-content .link {
81 | font-weight: bold;
82 | }
83 | .github-box .github-box-download {
84 | position: relative;
85 | border-top: 1px solid #ddd;
86 | background: white;
87 | border-radius: 0 0 3px 3px;
88 | padding: 10px;
89 | height: auto;
90 | min-height: 24px;
91 | }
92 | .github-box .github-box-download .updated {
93 | word-wrap: break-word;
94 | margin: 0;
95 | font-size: 11px;
96 | color: #666;
97 | line-height: 24px;
98 | font-weight: 300;
99 | width: auto;
100 | }
101 | .github-box .github-box-download .updated strong {
102 | font-weight: bold;
103 | color: #000;
104 | }
105 | .github-box .github-box-download .download {
106 | float: right;
107 | position: absolute;
108 | top: 10px;
109 | right: 10px;
110 | height: 24px;
111 | line-height: 24px;
112 | font-size: 12px;
113 | color: #666;
114 | font-weight: bold;
115 | text-shadow: 0 1px 0 rgba(255, 255, 255, 0.9);
116 | padding: 0 10px;
117 | border: 1px solid #ddd;
118 | border-bottom-color: #bbb;
119 | border-radius: 3px;
120 | background: #f5f5f5;
121 | background: -moz-linear-gradient(#f5f5f5, #e5e5e5);
122 | background: -webkit-linear-gradient(#f5f5f5, #e5e5e5);
123 | }
124 | .github-box .github-box-download .download: hover {
125 | color: #527894;
126 | border-color: #cfe3ed;
127 | border-bottom-color: #9fc7db;
128 | background: #f1f7fa;
129 | background: -moz-linear-gradient(#f1f7fa, #dbeaf1);
130 | background: -webkit-linear-gradient(#f1f7fa, #dbeaf1);
131 | }
132 | @media (max-width: 767px) {
133 | .github-box .github-box-title {
134 | height: auto;
135 | min-height: 60px;
136 | }
137 | .github-box .github-box-title h3 .repo {
138 | display: block;
139 | }
140 | .github-box .github-box-title .github-stats a {
141 | display: block;
142 | clear: right;
143 | float: right;
144 | }
145 | .github-box .github-box-download {
146 | height: auto;
147 | min-height: 46px;
148 | }
149 | .github-box .github-box-download .download {
150 | top: 32px;
151 | }
152 | }
--------------------------------------------------------------------------------