├── .browserslistrc
├── babel.config.js
├── tests
└── unit
│ ├── .eslintrc.js
│ └── example.spec.js
├── public
├── favicon.ico
└── index.html
├── src
├── assets
│ └── logo.png
├── games
│ ├── wx-jump-2
│ │ ├── dot.png
│ │ ├── index.vue
│ │ ├── defaultProp.js
│ │ ├── PropCreator.js
│ │ ├── Stage.js
│ │ ├── utils.js
│ │ ├── index.js
│ │ ├── Prop.js
│ │ ├── Particle.js
│ │ └── LittleMan.js
│ └── wx-jump-1
│ │ ├── index.vue
│ │ ├── defaultProp.js
│ │ ├── PropCreator.js
│ │ ├── utils.js
│ │ ├── Prop.js
│ │ ├── Stage.js
│ │ ├── index.js
│ │ └── LittleMan.js
├── store.js
├── main.js
├── App.vue
├── components
│ ├── MenuList.vue
│ └── MenuView.vue
└── router.js
├── examples
├── favicon.ico
├── css
│ ├── wx-jump-stage1.e29205c6.css
│ ├── wx-jump-stage2.952323e3.css
│ ├── chunk-vendors.391ed6f9.css
│ └── app.7834e5a2.css
├── index.html
└── js
│ ├── app.f1e8862e.js
│ ├── wx-jump-stage1.c74513ec.js
│ └── wx-jump-stage2.87b8f792.js
├── postcss.config.js
├── vue.config.js
├── README.md
├── .gitignore
├── .eslintrc.js
├── package.json
└── LICENSE
/.browserslistrc:
--------------------------------------------------------------------------------
1 | > 1%
2 | last 2 versions
3 |
--------------------------------------------------------------------------------
/babel.config.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | presets: [
3 | '@vue/app'
4 | ]
5 | }
6 |
--------------------------------------------------------------------------------
/tests/unit/.eslintrc.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | env: {
3 | mocha: true
4 | }
5 | }
--------------------------------------------------------------------------------
/public/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dongj0316/jsgame100/HEAD/public/favicon.ico
--------------------------------------------------------------------------------
/src/assets/logo.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dongj0316/jsgame100/HEAD/src/assets/logo.png
--------------------------------------------------------------------------------
/examples/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dongj0316/jsgame100/HEAD/examples/favicon.ico
--------------------------------------------------------------------------------
/postcss.config.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | plugins: {
3 | autoprefixer: {}
4 | }
5 | }
6 |
--------------------------------------------------------------------------------
/vue.config.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | publicPath: './',
3 | productionSourceMap: false
4 | }
--------------------------------------------------------------------------------
/src/games/wx-jump-2/dot.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/dongj0316/jsgame100/HEAD/src/games/wx-jump-2/dot.png
--------------------------------------------------------------------------------
/src/store.js:
--------------------------------------------------------------------------------
1 | import Vue from 'vue'
2 | import Vuex from 'vuex'
3 |
4 | Vue.use(Vuex)
5 |
6 | export default new Vuex.Store({
7 | state: {
8 |
9 | },
10 | mutations: {
11 |
12 | },
13 | actions: {
14 |
15 | }
16 | })
17 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # jsgame100
2 |
3 | [微信跳一跳 (1)](https://dongj0316.github.io/jsgame100/examples/#/wx-jump-stage1)
4 | [微信跳一跳 (2)](https://dongj0316.github.io/jsgame100/examples/#/wx-jump-stage2)
5 |
6 | ## License
7 |
8 | [MIT](https://opensource.org/licenses/MIT)
9 |
--------------------------------------------------------------------------------
/src/main.js:
--------------------------------------------------------------------------------
1 | import Vue from 'vue'
2 | import App from './App.vue'
3 | import router from './router'
4 | import store from './store'
5 | import 'normalize.css'
6 |
7 | Vue.config.productionTip = false
8 |
9 | new Vue({
10 | router,
11 | store,
12 | render: h => h(App)
13 | }).$mount('#app')
14 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | .DS_Store
2 | node_modules
3 | /dist
4 |
5 | # local env files
6 | .env.local
7 | .env.*.local
8 |
9 | # Log files
10 | npm-debug.log*
11 | yarn-debug.log*
12 | yarn-error.log*
13 |
14 | # Editor directories and files
15 | .idea
16 | .vscode
17 | *.suo
18 | *.ntvs*
19 | *.njsproj
20 | *.sln
21 | *.sw?
22 |
--------------------------------------------------------------------------------
/examples/css/wx-jump-stage1.e29205c6.css:
--------------------------------------------------------------------------------
1 | @media (max-width:768px){.jump-world[data-v-0bf23ea4]{position:fixed;top:0;left:0;width:100%;height:100%}.jump-world canvas[data-v-0bf23ea4]{width:100%;height:100%}}@media (min-width:768px){.jump-world[data-v-0bf23ea4]{width:375px;height:667px;margin:auto}.jump-world canvas[data-v-0bf23ea4]{width:100%;height:100%}}
--------------------------------------------------------------------------------
/examples/css/wx-jump-stage2.952323e3.css:
--------------------------------------------------------------------------------
1 | @media (max-width:768px){.jump-world[data-v-9a0b2f7a]{position:fixed;top:0;left:0;width:100%;height:100%}.jump-world canvas[data-v-9a0b2f7a]{width:100%;height:100%}}@media (min-width:768px){.jump-world[data-v-9a0b2f7a]{width:375px;height:667px;margin:auto}.jump-world canvas[data-v-9a0b2f7a]{width:100%;height:100%}}
--------------------------------------------------------------------------------
/tests/unit/example.spec.js:
--------------------------------------------------------------------------------
1 | import { expect } from 'chai'
2 | import { shallowMount } from '@vue/test-utils'
3 | import HelloWorld from '@/components/HelloWorld.vue'
4 |
5 | describe('HelloWorld.vue', () => {
6 | it('renders props.msg when passed', () => {
7 | const msg = 'new message'
8 | const wrapper = shallowMount(HelloWorld, {
9 | propsData: { msg }
10 | })
11 | expect(wrapper.text()).to.include(msg)
12 | })
13 | })
14 |
--------------------------------------------------------------------------------
/src/App.vue:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
26 |
--------------------------------------------------------------------------------
/.eslintrc.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | root: true,
3 | env: {
4 | node: true
5 | },
6 | 'extends': [
7 | 'plugin:vue/essential',
8 | 'eslint:recommended'
9 | ],
10 | rules: {
11 | 'no-console': process.env.NODE_ENV === 'production' ? 'error' : 'off',
12 | 'no-debugger': process.env.NODE_ENV === 'production' ? 'error' : 'off'
13 | },
14 | parserOptions: {
15 | parser: 'babel-eslint'
16 | },
17 | overrides: [
18 | {
19 | files: [
20 | '**/__tests__/*.{j,t}s?(x)'
21 | ],
22 | env: {
23 | mocha: true
24 | }
25 | }
26 | ]
27 | }
28 |
--------------------------------------------------------------------------------
/public/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 | jsgame100
9 |
10 |
11 |
14 |
15 |
16 |
17 |
18 |
--------------------------------------------------------------------------------
/src/games/wx-jump-1/index.vue:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
20 |
21 |
--------------------------------------------------------------------------------
/src/games/wx-jump-2/index.vue:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
20 |
21 |
--------------------------------------------------------------------------------
/src/games/wx-jump-1/defaultProp.js:
--------------------------------------------------------------------------------
1 | import {
2 | randomArrayElm,
3 | rangeNumberInclusive
4 | } from './utils'
5 |
6 | let cid = 0
7 | const colors = [0x67C23A, 0xE6A23C, 0xF56C6C, 0x909399, 0x409EFF, 0xffffff]
8 |
9 | // 静态
10 | export const statics = [
11 | // ...
12 | ]
13 |
14 | // 非静态
15 | export const actives = [
16 | // 默认纯色立方体创造器
17 | function defaultCreator (THREE, helpers) {
18 | const {
19 | propSizeRange: [min, max],
20 | propHeight,
21 | baseMeshLambertMaterial,
22 | baseBoxBufferGeometry
23 | } = helpers
24 |
25 | ++cid
26 |
27 | // 随机颜色
28 | const color = randomArrayElm(colors)
29 | // 随机大小,头2个盒子固定一下大小
30 | const size = cid < 3 ? max : rangeNumberInclusive(min, max)
31 |
32 | const geometry = baseBoxBufferGeometry.clone()
33 | geometry.scale(size, propHeight, size)
34 |
35 | const material = baseMeshLambertMaterial.clone()
36 | material.setValues({ color })
37 |
38 | return new THREE.Mesh(geometry, material)
39 | },
40 | ]
--------------------------------------------------------------------------------
/src/components/MenuList.vue:
--------------------------------------------------------------------------------
1 |
2 |
16 |
17 |
18 |
30 |
31 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "jsgame100",
3 | "version": "0.1.0",
4 | "private": true,
5 | "scripts": {
6 | "serve": "vue-cli-service serve",
7 | "build": "vue-cli-service build",
8 | "build-examples": "vue-cli-service build --dest examples",
9 | "lint": "vue-cli-service lint",
10 | "test:unit": "vue-cli-service test:unit"
11 | },
12 | "dependencies": {
13 | "@tweenjs/tween.js": "^18.3.1",
14 | "core-js": "^2.6.5",
15 | "normalize.css": "^8.0.1",
16 | "three": "^0.109.0",
17 | "vue": "^2.6.10",
18 | "vue-router": "^3.0.3",
19 | "vuex": "^3.0.1"
20 | },
21 | "devDependencies": {
22 | "@vue/cli-plugin-babel": "^3.12.0",
23 | "@vue/cli-plugin-eslint": "^3.12.0",
24 | "@vue/cli-plugin-unit-mocha": "^3.12.0",
25 | "@vue/cli-service": "^3.12.0",
26 | "@vue/test-utils": "1.0.0-beta.29",
27 | "babel-eslint": "^10.0.1",
28 | "chai": "^4.1.2",
29 | "eslint": "^5.16.0",
30 | "eslint-plugin-vue": "^5.0.0",
31 | "less": "^3.0.4",
32 | "less-loader": "^5.0.0",
33 | "vue-template-compiler": "^2.6.10"
34 | }
35 | }
36 |
--------------------------------------------------------------------------------
/examples/index.html:
--------------------------------------------------------------------------------
1 | jsgame100
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2019 dongj0316
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/src/router.js:
--------------------------------------------------------------------------------
1 | import Vue from 'vue'
2 | import Router from 'vue-router'
3 | import MenuList from '@/components/MenuList'
4 | import MenuView from '@/components/MenuView'
5 |
6 | Vue.use(Router)
7 |
8 | const withNameView = (name, view, optons) => {
9 | const { component, ...otherOptions } = optons
10 |
11 | return {
12 | components: {
13 | default: component,
14 | [name || view.name]: view
15 | },
16 | ...otherOptions
17 | }
18 | }
19 |
20 | export const withMenuViewRoutes = [
21 | {
22 | path: '/wx-jump-stage1',
23 | name: 'wx-jump-stage1',
24 | meta: {
25 | title: '微信跳一跳 (1)'
26 | },
27 | component: () => import(/* webpackChunkName: "wx-jump-stage1" */ './games/wx-jump-1/index.vue')
28 | },
29 | {
30 | path: '/wx-jump-stage2',
31 | name: 'wx-jump-stage2',
32 | meta: {
33 | title: '微信跳一跳 (2)'
34 | },
35 | component: () => import(/* webpackChunkName: "wx-jump-stage2" */ './games/wx-jump-2/index.vue')
36 | }
37 | ]
38 |
39 | const defaultRoutes = [
40 | {
41 | path: '/',
42 | name: 'menu-list',
43 | component: MenuList
44 | }
45 | ]
46 |
47 | export default new Router({
48 | routes: [
49 | ...defaultRoutes,
50 | ...withMenuViewRoutes.map(withNameView.bind(null, 'MenuView', MenuView)),
51 | {
52 | path: '*',
53 | component: MenuList
54 | }
55 | ]
56 | })
57 |
--------------------------------------------------------------------------------
/src/games/wx-jump-2/defaultProp.js:
--------------------------------------------------------------------------------
1 | import {
2 | randomArrayElm,
3 | rangeNumberInclusive,
4 | propCounter,
5 | incrementPropCounter,
6 | colors
7 | } from './utils'
8 |
9 | // 静态
10 | export const statics = [
11 | // ...
12 | ]
13 |
14 | // 非静态
15 | export const actives = [
16 | // 默认纯色立方体创造器
17 | function defaultCreator (THREE, helpers) {
18 | const {
19 | propSizeRange: [min, max],
20 | propHeight,
21 | baseMeshLambertMaterial,
22 | baseBoxBufferGeometry
23 | } = helpers
24 |
25 | incrementPropCounter()
26 |
27 | // 随机颜色
28 | const color = randomArrayElm(colors)
29 | // 随机大小,头2个盒子固定一下大小
30 | const size = propCounter < 3 ? max : rangeNumberInclusive(min, max)
31 |
32 | const geometry = baseBoxBufferGeometry.clone()
33 | geometry.scale(size, propHeight, size)
34 |
35 | const material = baseMeshLambertMaterial.clone()
36 | material.setValues({ color })
37 |
38 | return new THREE.Mesh(geometry, material)
39 | },
40 | function defaultCreator (THREE, helpers) {
41 | const {
42 | propSizeRange: [min, max],
43 | propHeight,
44 | baseMeshLambertMaterial,
45 | baseCylinderBufferGeometry
46 | } = helpers
47 |
48 | incrementPropCounter()
49 |
50 | // 随机颜色
51 | const color = randomArrayElm(colors)
52 | // 随机大小,头2个盒子固定一下大小
53 | const size = propCounter < 3 ? max : rangeNumberInclusive(min, max)
54 |
55 | const geometry = baseCylinderBufferGeometry.clone()
56 | geometry.scale(Math.ceil(size / 2), propHeight, Math.ceil(size / 2))
57 |
58 | const material = baseMeshLambertMaterial.clone()
59 | material.setValues({ color })
60 |
61 | return new THREE.Mesh(geometry, material)
62 | },
63 | ]
--------------------------------------------------------------------------------
/examples/css/chunk-vendors.391ed6f9.css:
--------------------------------------------------------------------------------
1 | /*! normalize.css v8.0.1 | MIT License | github.com/necolas/normalize.css */html{line-height:1.15;-webkit-text-size-adjust:100%}body{margin:0}main{display:block}h1{font-size:2em;margin:.67em 0}hr{-webkit-box-sizing:content-box;box-sizing:content-box;height:0;overflow:visible}pre{font-family:monospace,monospace;font-size:1em}a{background-color:transparent}abbr[title]{border-bottom:none;text-decoration:underline;-webkit-text-decoration:underline dotted;text-decoration:underline dotted}b,strong{font-weight:bolder}code,kbd,samp{font-family:monospace,monospace;font-size:1em}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}img{border-style:none}button,input,optgroup,select,textarea{font-family:inherit;font-size:100%;line-height:1.15;margin:0}button,input{overflow:visible}button,select{text-transform:none}[type=button],[type=reset],[type=submit],button{-webkit-appearance:button}[type=button]::-moz-focus-inner,[type=reset]::-moz-focus-inner,[type=submit]::-moz-focus-inner,button::-moz-focus-inner{border-style:none;padding:0}[type=button]:-moz-focusring,[type=reset]:-moz-focusring,[type=submit]:-moz-focusring,button:-moz-focusring{outline:1px dotted ButtonText}fieldset{padding:.35em .75em .625em}legend{-webkit-box-sizing:border-box;box-sizing:border-box;color:inherit;display:table;max-width:100%;padding:0;white-space:normal}progress{vertical-align:baseline}textarea{overflow:auto}[type=checkbox],[type=radio]{-webkit-box-sizing:border-box;box-sizing:border-box;padding:0}[type=number]::-webkit-inner-spin-button,[type=number]::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}[type=search]::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}details{display:block}summary{display:list-item}[hidden],template{display:none}
--------------------------------------------------------------------------------
/src/games/wx-jump-1/PropCreator.js:
--------------------------------------------------------------------------------
1 | import * as THREE from 'three'
2 | import {
3 | baseMeshLambertMaterial,
4 | baseBoxBufferGeometry,
5 | randomArrayElm,
6 | rangeNumberInclusive
7 | } from './utils'
8 | import { statics, actives } from './defaultProp'
9 |
10 | class PropCreator {
11 | constructor ({
12 | propHeight,
13 | propSizeRange,
14 | needDefaultCreator
15 | }) {
16 | this.propHeight = propHeight
17 | this.propSizeRange = propSizeRange
18 |
19 | // 维护的创造器
20 | this.propCreators = []
21 |
22 | if (needDefaultCreator) {
23 | this.createPropCreator(actives, false)
24 | this.createPropCreator(statics, true)
25 | }
26 | }
27 |
28 | createProp (index) {
29 | const { propCreators } = this
30 | return index > -1
31 | ? propCreators[index] && propCreators[index]() || randomArrayElm(propCreators)()
32 | : randomArrayElm(propCreators)()
33 | }
34 |
35 | /**
36 | * 新增定制化的创造器
37 | * @param {Function} creator 创造器函数
38 | * @param {Boolean} isStatic 是否是动态创建
39 | */
40 | createPropCreator (creator, isStatic) {
41 | if (Array.isArray(creator)) {
42 | creator.forEach(crt => this.createPropCreator(crt, isStatic))
43 | return
44 | }
45 |
46 | const { propCreators, propSizeRange, propHeight } = this
47 |
48 | if (propCreators.indexOf(creator) > -1) {
49 | return
50 | }
51 |
52 | const wrappedCreator = function () {
53 | if (isStatic && wrappedCreator.box) {
54 | // 静态盒子,下次直接clone
55 | return wrappedCreator.box.clone()
56 | } else {
57 | const box = creator(THREE, {
58 | propSizeRange,
59 | propHeight,
60 | baseMeshLambertMaterial,
61 | baseBoxBufferGeometry
62 | })
63 |
64 | if (isStatic) {
65 | // 被告知是静态盒子,缓存起来
66 | wrappedCreator.box = box
67 | }
68 | return box
69 | }
70 | }
71 |
72 | propCreators.push(wrappedCreator)
73 | }
74 | }
75 |
76 | export default PropCreator
--------------------------------------------------------------------------------
/src/games/wx-jump-2/PropCreator.js:
--------------------------------------------------------------------------------
1 | import * as THREE from 'three'
2 | import {
3 | baseMeshLambertMaterial,
4 | baseBoxBufferGeometry,
5 | baseCylinderBufferGeometry,
6 | randomArrayElm,
7 | rangeNumberInclusive
8 | } from './utils'
9 | import { statics, actives } from './defaultProp'
10 |
11 | class PropCreator {
12 | constructor ({
13 | propHeight,
14 | propSizeRange,
15 | needDefaultCreator
16 | }) {
17 | this.propHeight = propHeight
18 | this.propSizeRange = propSizeRange
19 |
20 | // 维护的创造器
21 | this.propCreators = []
22 |
23 | if (needDefaultCreator) {
24 | this.createPropCreator(actives, false)
25 | this.createPropCreator(statics, true)
26 | }
27 | }
28 |
29 | createProp (index) {
30 | const { propCreators } = this
31 | return index > -1
32 | ? propCreators[index] && propCreators[index]() || randomArrayElm(propCreators)()
33 | : randomArrayElm(propCreators)()
34 | }
35 |
36 | /**
37 | * 新增定制化的创造器
38 | * @param {Function} creator 创造器函数
39 | * @param {Boolean} isStatic 是否是动态创建
40 | */
41 | createPropCreator (creator, isStatic) {
42 | if (Array.isArray(creator)) {
43 | creator.forEach(crt => this.createPropCreator(crt, isStatic))
44 | return
45 | }
46 |
47 | const { propCreators, propSizeRange, propHeight } = this
48 |
49 | if (propCreators.indexOf(creator) > -1) {
50 | return
51 | }
52 |
53 | const wrappedCreator = function () {
54 | if (isStatic && wrappedCreator.box) {
55 | // 静态盒子,下次直接clone
56 | return wrappedCreator.box.clone()
57 | } else {
58 | const box = creator(THREE, {
59 | propSizeRange,
60 | propHeight,
61 | baseMeshLambertMaterial,
62 | baseBoxBufferGeometry,
63 | baseCylinderBufferGeometry
64 | })
65 |
66 | if (isStatic) {
67 | // 被告知是静态盒子,缓存起来
68 | wrappedCreator.box = box
69 | }
70 | return box
71 | }
72 | }
73 |
74 | propCreators.push(wrappedCreator)
75 | }
76 | }
77 |
78 | export default PropCreator
--------------------------------------------------------------------------------
/examples/css/app.7834e5a2.css:
--------------------------------------------------------------------------------
1 | *{-webkit-box-sizing:border-box;box-sizing:border-box}#app,body,html{width:100%}#app{font-family:Avenir,Helvetica,Arial,sans-serif;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;color:#2c3e50;overflow-x:hidden;overflow-y:auto}.menu-list[data-v-7aa4251c]{padding:30px;margin:0;list-style:none;text-align:center}.menu-list h1[data-v-7aa4251c]{margin-top:0;font-size:24px}.menu-list li[data-v-7aa4251c]{padding:5px 0}.menu-list a[data-v-7aa4251c]{font-weight:700;color:#2c3e50}.menu-list a.router-link-exact-active[data-v-7aa4251c]{color:#42b983}.menu-view-list[data-v-9a5b6090]{position:fixed;top:0;left:0;height:100%;width:220px;-webkit-box-shadow:0 0 15px #808b96;box-shadow:0 0 15px #808b96;overflow-x:hidden;overflow-y:auto;-ms-scroll-chaining:none;overscroll-behavior:contain;-webkit-transform:translateZ(0);transform:translateZ(0);background:#fff;z-index:5000}.menu-view-mask[data-v-9a5b6090]{position:fixed;top:0;left:0;right:0;bottom:0;z-index:4999}.menu-view-trigger[data-v-9a5b6090]{position:fixed;top:20px;left:20px;z-index:5001}.menu-view-trigger button[data-v-9a5b6090]{width:44px;height:44px;border-radius:50%;-webkit-box-shadow:0 0 15px #808b96;box-shadow:0 0 15px #808b96;border:none;background:rgba(44,62,80,.5)}@media (max-width:768px){.menu-view-list[data-v-9a5b6090]{width:70vw}}.menu-view-trigger-slide-enter-active[data-v-9a5b6090]{-webkit-animation:slide-bottom-data-v-9a5b6090 .3s;animation:slide-bottom-data-v-9a5b6090 .3s}.menu-view-trigger-slide-leave-active[data-v-9a5b6090]{-webkit-animation:slide-top-data-v-9a5b6090 .3s .3s;animation:slide-top-data-v-9a5b6090 .3s .3s}.menu-view-slide-enter-active[data-v-9a5b6090]{-webkit-animation:slide-right-data-v-9a5b6090 .3s;animation:slide-right-data-v-9a5b6090 .3s}.menu-view-slide-leave-active[data-v-9a5b6090]{-webkit-animation:slide-left-data-v-9a5b6090 .3s;animation:slide-left-data-v-9a5b6090 .3s}@-webkit-keyframes slide-right-data-v-9a5b6090{0%{-webkit-transform:translate3d(-100%,0,0);transform:translate3d(-100%,0,0)}}@keyframes slide-right-data-v-9a5b6090{0%{-webkit-transform:translate3d(-100%,0,0);transform:translate3d(-100%,0,0)}}@-webkit-keyframes slide-left-data-v-9a5b6090{to{-webkit-transform:translate3d(-100%,0,0);transform:translate3d(-100%,0,0)}}@keyframes slide-left-data-v-9a5b6090{to{-webkit-transform:translate3d(-100%,0,0);transform:translate3d(-100%,0,0)}}@-webkit-keyframes slide-bottom-data-v-9a5b6090{0%{-webkit-transform:translate3d(0,-200%,0);transform:translate3d(0,-200%,0)}}@keyframes slide-bottom-data-v-9a5b6090{0%{-webkit-transform:translate3d(0,-200%,0);transform:translate3d(0,-200%,0)}}@-webkit-keyframes slide-top-data-v-9a5b6090{to{-webkit-transform:translate3d(0,-200%,0);transform:translate3d(0,-200%,0)}}@keyframes slide-top-data-v-9a5b6090{to{-webkit-transform:translate3d(0,-200%,0);transform:translate3d(0,-200%,0)}}
--------------------------------------------------------------------------------
/src/components/MenuView.vue:
--------------------------------------------------------------------------------
1 |
2 |
17 |
18 |
19 |
39 |
40 |
--------------------------------------------------------------------------------
/src/games/wx-jump-1/utils.js:
--------------------------------------------------------------------------------
1 | import * as THREE from 'three'
2 | import TWEEN from '@tweenjs/tween.js'
3 |
4 | const { random, sqrt, floor, pow, sin, cos, tan, PI } = Math
5 |
6 | /**
7 | * 根据角度计算相机初始位置
8 | * @param {Number} verticalDeg 相机和场景中心点的垂直角度
9 | * @param {Number} horizontalDeg 相机和x轴的水平角度
10 | * @param {Number} top 相机上侧面
11 | * @param {Number} bottom 相机下侧面
12 | * @param {Number} near 摄像机视锥体近端面
13 | * @param {Number} far 摄像机视锥体远端面
14 | */
15 | export function computeCameraInitalPosition (verticalDeg, horizontalDeg, top, bottom, near, far) {
16 | const verticalRadian = verticalDeg * (PI / 180)
17 | const horizontalRadian = horizontalDeg * (PI / 180)
18 | const minY = cos(verticalRadian) * bottom
19 | const maxY = sin(verticalRadian) * (far - near - top / tan(verticalRadian))
20 |
21 | if (minY > maxY) {
22 | console.warn('警告: 垂直角度太小了!')
23 | return
24 | }
25 | const y = minY + (maxY - minY) / 2
26 | const longEdge = y / tan(verticalRadian)
27 | const x = sin(horizontalRadian) * longEdge
28 | const z = cos(horizontalRadian) * longEdge
29 |
30 | return { x, y, z }
31 | }
32 |
33 | // 材质
34 | export const baseMeshLambertMaterial = new THREE.MeshLambertMaterial()
35 | // 立方体
36 | export const baseBoxBufferGeometry = new THREE.BoxBufferGeometry()
37 |
38 | export const randomArrayElm = array => array[floor(random() * array.length)]
39 |
40 | export const rangeNumberInclusive = (min, max) => floor(random() * (max - min + 1)) + min
41 |
42 | export const getPropSize = box => {
43 | const box3 = getPropSize.box3 || (getPropSize.box3 = new THREE.Box3())
44 | box3.setFromObject(box)
45 | return box3.getSize(new THREE.Vector3())
46 | }
47 |
48 | // 斜抛计算
49 | export const computeObligueThrowValue = function (v0, theta, G) {
50 | const sin2θ = sin(2 * theta)
51 | const sinθ = sin(theta)
52 |
53 | const rangeR = pow(v0, 2) * sin2θ / G
54 | const rangeH = pow(v0 * sinθ, 2) / (2 * G)
55 | // const rangeT = 2 * v0 * sinθ / G
56 |
57 | return {
58 | rangeR,
59 | rangeH,
60 | // rangeT
61 | }
62 | }
63 |
64 | /**
65 | * 根据射程算出落地点
66 | * @param {Number} range 射程
67 | * @param {Object} c1 起跳点
68 | * @param {Object} p2 目标盒子中心点
69 | */
70 | export const computePositionByRangeR = function (range, c1, p2) {
71 | const { x: c1x, z: c1z } = c1
72 | const { x: p2x, z: p2z } = p2
73 |
74 | const p2cx = p2x - c1x
75 | const p2cz = p2z - c1z
76 | const p2c = sqrt(pow(p2cz, 2) + pow(p2cx, 2))
77 |
78 | const jumpDownX = p2cx * range / p2c
79 | const jumpDownZ = p2cz * range / p2c
80 |
81 | return {
82 | jumpDownX: c1x + jumpDownX,
83 | jumpDownZ: c1z + jumpDownZ
84 | }
85 | }
86 |
87 | export const animate = (configs, onUpdate, onComplete) => {
88 | const {
89 | from, to, duration,
90 | easing = k => k,
91 | autoStart = true // 为了使用tween的chain
92 | } = configs
93 |
94 | const tween = new TWEEN.Tween(from)
95 | .to(to, duration)
96 | .easing(easing)
97 | .onUpdate(onUpdate)
98 | .onComplete(() => {
99 | onComplete && onComplete()
100 | })
101 |
102 | if (autoStart) {
103 | tween.start()
104 | }
105 |
106 | animateFrame()
107 | return tween
108 | }
109 |
110 | const animateFrame = function () {
111 | if (animateFrame.openin) {
112 | return
113 | }
114 | animateFrame.openin = true
115 |
116 | const animate = () => {
117 | const id = requestAnimationFrame(animate)
118 | if (!TWEEN.update()) {
119 | animateFrame.openin = false
120 | cancelAnimationFrame(id)
121 | }
122 | }
123 | animate()
124 | }
--------------------------------------------------------------------------------
/src/games/wx-jump-1/Prop.js:
--------------------------------------------------------------------------------
1 | import TWEEN from '@tweenjs/tween.js'
2 | import { getPropSize, rangeNumberInclusive, animate } from './utils'
3 |
4 | class Prop {
5 | constructor ({
6 | world, // 所处世界
7 | stage, // 所处舞台
8 | body, // 主体
9 | height,
10 | enterHeight,
11 | distanceRange,
12 | prev
13 | }) {
14 | this.world = world
15 | this.stage = stage
16 | this.body = body
17 | this.height = height
18 | this.enterHeight = enterHeight
19 | this.distanceRange = distanceRange
20 | this.prev = prev
21 | }
22 |
23 | // 计算位置
24 | computeMyPosition () {
25 | const {
26 | world,
27 | prev,
28 | distanceRange,
29 | enterHeight
30 | } = this
31 | const position = {
32 | x: 0,
33 | // 头2个盒子y值为0
34 | y: enterHeight,
35 | z: 0
36 | }
37 |
38 | if (!prev) {
39 | // 第1个盒子
40 | return position
41 | }
42 |
43 | if (enterHeight === 0) {
44 | // 第2个盒子,固定一个距离
45 | position.z = world.width / 2
46 | return position
47 | }
48 |
49 | const { x, z } = prev.getPosition()
50 | // 随机2个方向 x or z
51 | const direction = Math.round(Math.random()) === 0
52 | const { x: prevWidth, z: prevDepth } = prev.getSize()
53 | const { x: currentWidth, z: currentDepth } = this.getSize()
54 | // 根据区间随机一个距离
55 | const randomDistance = rangeNumberInclusive(...distanceRange)
56 |
57 | if (direction) {
58 | position.x = x + prevWidth / 2 + randomDistance + currentWidth / 2
59 | position.z = z
60 | } else {
61 | position.x = x
62 | position.z = z + prevDepth / 2 + randomDistance + currentDepth / 2
63 | }
64 |
65 | return position
66 | }
67 |
68 | // 将道具放入舞台
69 | enterStage () {
70 | const { stage, body, height } = this
71 | const { x, y, z } = this.computeMyPosition()
72 |
73 | body.castShadow = true
74 | body.receiveShadow = true
75 | body.position.set(x, y, z)
76 | // 需要将盒子放到地面
77 | body.geometry.translate(0, height / 2, 0)
78 |
79 | stage.add(body)
80 | stage.render()
81 |
82 | this.entranceTransition()
83 | }
84 |
85 | // 盒子的入场动画
86 | entranceTransition (duration = 400) {
87 | const { body, enterHeight, stage } = this
88 |
89 | if (enterHeight === 0) {
90 | return
91 | }
92 |
93 | animate(
94 | {
95 | to: { y: 0 },
96 | from: { y: enterHeight },
97 | duration,
98 | easing: TWEEN.Easing.Bounce.Out
99 | },
100 | ({ y }) => {
101 | body.position.setY(y)
102 | stage.render()
103 | }
104 | )
105 | }
106 |
107 | // 回弹动画
108 | springbackTransition (duration) {
109 | const { body, stage } = this
110 | const y = body.scale.y
111 |
112 | animate(
113 | {
114 | from: { y },
115 | to: { y: 1 },
116 | duration,
117 | easing: TWEEN.Easing.Bounce.Out
118 | },
119 | ({ y }) => {
120 | body.scale.setY(y)
121 | stage.render()
122 | }
123 | )
124 | }
125 |
126 | setNext (next) {
127 | this.next = next
128 | }
129 |
130 | getNext (next) {
131 | return this.next
132 | }
133 |
134 | getPosition () {
135 | return this.body.position
136 | }
137 |
138 | setPosition (x, y, z) {
139 | return this.body.position.set(x, y, z)
140 | }
141 |
142 | scaleY (y) {
143 | return this.body.scale.setY(y)
144 | }
145 |
146 | // 获取道具大小
147 | getSize () {
148 | return getPropSize(this.body)
149 | }
150 |
151 | // 销毁
152 | dispose () {
153 | const { body, stage, prev, next } = this
154 | // 解除关联的引用
155 | this.prev = null
156 | this.next = null
157 | if (prev) {
158 | prev.next = null
159 | }
160 | if (next) {
161 | next.prev = null
162 | }
163 |
164 | body.geometry.dispose()
165 | body.material.dispose()
166 | stage.remove(body)
167 | }
168 | }
169 |
170 | export default Prop
--------------------------------------------------------------------------------
/src/games/wx-jump-1/Stage.js:
--------------------------------------------------------------------------------
1 | import * as THREE from 'three'
2 | import TWEEN from '@tweenjs/tween.js'
3 | import { baseMeshLambertMaterial, animate } from './utils'
4 |
5 | class Stage {
6 | constructor ({
7 | width,
8 | height,
9 | canvas,
10 | axesHelper = false, // 辅助线
11 | cameraNear, // 相机近截面
12 | cameraFar, // 相机远截面
13 | cameraInitalPosition, // 相机初始位置
14 | lightInitalPosition // 光源初始位置
15 | }) {
16 | this.width = width
17 | this.height = height
18 | this.canvas = canvas
19 | this.axesHelper = axesHelper
20 | // 正交相机配置
21 | this.cameraNear = cameraNear
22 | this.cameraFar = cameraFar
23 | this.cameraInitalPosition = cameraInitalPosition
24 | this.lightInitalPosition = lightInitalPosition
25 |
26 | this.scene = null
27 | this.plane = null
28 | this.light = null
29 | this.camera = null
30 | this.renderer = null
31 |
32 | this.init()
33 | }
34 |
35 | init () {
36 | this.createScene()
37 | this.createPlane()
38 | this.createLight()
39 | this.createCamera()
40 | this.createRenterer()
41 | this.render()
42 | }
43 |
44 | // 场景
45 | createScene () {
46 | const scene = this.scene = new THREE.Scene()
47 | scene.background = new THREE.Color(0xd6dbdf)
48 |
49 | if (this.axesHelper) {
50 | scene.add(new THREE.AxesHelper(10e3))
51 | }
52 | }
53 |
54 | // 地面
55 | createPlane () {
56 | const { scene } = this
57 | const geometry = new THREE.PlaneBufferGeometry(10e2, 10e2, 1, 1)
58 | const meterial = new THREE.ShadowMaterial()
59 | meterial.opacity = 0.5
60 |
61 | const plane = this.plane = new THREE.Mesh(geometry, meterial)
62 |
63 | plane.rotation.x = -.5 * Math.PI
64 | plane.position.y = -.1
65 | // 接收阴影
66 | plane.receiveShadow = true
67 | scene.add(plane)
68 | }
69 |
70 | // 光
71 | createLight () {
72 | const { scene, lightInitalPosition: { x, y, z }, height } = this
73 | const light = this.light = new THREE.DirectionalLight(0xffffff, .5)
74 | const lightTarget = this.lightTarget = new THREE.Object3D()
75 |
76 | light.target = lightTarget
77 | light.position.set(x, y, z)
78 | // 开启阴影投射
79 | light.castShadow = true
80 | // // 定义可见域的投射阴影
81 | light.shadow.camera.left = -height
82 | light.shadow.camera.right = height
83 | light.shadow.camera.top = height
84 | light.shadow.camera.bottom = -height
85 | light.shadow.camera.near = 0
86 | light.shadow.camera.far = 2000
87 | // 定义阴影的分辨率
88 | light.shadow.mapSize.width = 1600
89 | light.shadow.mapSize.height = 1600
90 |
91 | // 环境光
92 | scene.add(new THREE.AmbientLight(0xE5E7E9, .4))
93 | scene.add(new THREE.HemisphereLight(0xffffff, 0xffffff, .2))
94 | scene.add(lightTarget)
95 | scene.add(light)
96 | }
97 |
98 | // 相机
99 | createCamera () {
100 | const {
101 | scene,
102 | width, height,
103 | cameraInitalPosition: { x, y, z },
104 | cameraNear, cameraFar
105 | } = this
106 | const camera = this.camera = new THREE.OrthographicCamera(-width / 2, width / 2, height / 2, -height / 2, cameraNear, cameraFar)
107 |
108 | camera.position.set(x, y, z)
109 | camera.lookAt(scene.position)
110 | scene.add(camera)
111 | }
112 |
113 | // 渲染器
114 | createRenterer () {
115 | const { canvas, width, height } = this
116 | const renderer = this.renderer = new THREE.WebGLRenderer({
117 | canvas,
118 | alpha: true, // 透明场景
119 | antialias:true // 抗锯齿
120 | })
121 |
122 | renderer.setSize(width, height)
123 | // 开启阴影
124 | renderer.shadowMap.enabled = true
125 | // 设置设备像素
126 | renderer.setPixelRatio(window.devicePixelRatio)
127 | }
128 |
129 | // 执行渲染
130 | render () {
131 | const { scene, camera } = this
132 | this.renderer.render(scene, camera)
133 | }
134 |
135 | // center为2个盒子的中心点
136 | moveCamera ({ cameraTo, center, lightTo }, onComplete, duration) {
137 | const {
138 | camera, plane,
139 | light, lightTarget,
140 | lightInitalPosition
141 | } = this
142 |
143 | // 移动相机
144 | animate(
145 | {
146 | from: { ...camera.position },
147 | to: cameraTo,
148 | duration
149 | },
150 | ({ x, y, z }) => {
151 | camera.position.x = x
152 | camera.position.z = z
153 | this.render()
154 | },
155 | onComplete
156 | )
157 |
158 | // 灯光和目标也需要动起来,为了保证阴影位置不变
159 | const { x: lightInitalX, z: lightInitalZ } = lightInitalPosition
160 | animate(
161 | {
162 | from: { ...light.position },
163 | to: lightTo,
164 | duration
165 | },
166 | ({ x, y, z }) => {
167 | lightTarget.position.x = x - lightInitalX
168 | lightTarget.position.z = z - lightInitalZ
169 | light.position.set(x, y, z)
170 | }
171 | )
172 |
173 | // 保证不会跑出有限大小的地面
174 | plane.position.x = center.x
175 | plane.position.z = center.z
176 | }
177 |
178 | // 场景中添加物体
179 | add (...args) {
180 | return this.scene.add(...args)
181 | }
182 | // 移除场景中的物体
183 | remove (...args) {
184 | return this.scene.remove(...args)
185 | }
186 | }
187 |
188 | export default Stage
--------------------------------------------------------------------------------
/src/games/wx-jump-2/Stage.js:
--------------------------------------------------------------------------------
1 | import * as THREE from 'three'
2 | import { animate, destroyMesh } from './utils'
3 |
4 | class Stage {
5 | constructor ({
6 | width,
7 | height,
8 | canvas,
9 | axesHelper = false, // 辅助线
10 | cameraNear, // 相机近截面
11 | cameraFar, // 相机远截面
12 | cameraInitalPosition, // 相机初始位置
13 | lightInitalPosition // 光源初始位置
14 | }) {
15 | this.width = width
16 | this.height = height
17 | this.canvas = canvas
18 | this.axesHelper = axesHelper
19 | // 正交相机配置
20 | this.cameraNear = cameraNear
21 | this.cameraFar = cameraFar
22 | this.cameraInitalPosition = cameraInitalPosition
23 | this.lightInitalPosition = lightInitalPosition
24 |
25 | this.scene = null
26 | this.plane = null
27 | this.light = null
28 | this.camera = null
29 | this.renderer = null
30 |
31 | this.init()
32 | }
33 |
34 | init () {
35 | this.createScene()
36 | this.createPlane()
37 | this.createLight()
38 | this.createCamera()
39 | this.createRenterer()
40 | this.render()
41 | }
42 |
43 | // 场景销毁
44 | destroy () {
45 | destroyMesh(this.camera)
46 | destroyMesh(this.light)
47 | destroyMesh(this.plane)
48 |
49 | this.scene.remove(this.camera)
50 | this.scene.remove(this.light)
51 | this.scene.remove(this.plane)
52 | this.scene.children.forEach(destroyMesh)
53 | this.scene.children = null
54 |
55 | this.scene.dispose()
56 | this.renderer.dispose()
57 |
58 | this.canvas = null
59 | this.scene = null
60 | this.plane = null
61 | this.light = null
62 | this.camera = null
63 | this.renderer = null
64 | }
65 |
66 | // 重置场景到初始状态
67 | reset () {
68 | const {
69 | plane,
70 | light, lightTarget, lightInitalPosition,
71 | camera, cameraInitalPosition
72 | } = this
73 | plane.position.x = 0
74 | plane.position.z = 0
75 | lightTarget.position.x = 0
76 | lightTarget.position.z = 0
77 | light.position.set(lightInitalPosition.x, lightInitalPosition.y, lightInitalPosition.z)
78 | camera.position.x = cameraInitalPosition.x
79 | camera.position.z = cameraInitalPosition.z
80 | }
81 |
82 | // 场景
83 | createScene () {
84 | const scene = this.scene = new THREE.Scene()
85 | scene.background = new THREE.Color(0xd6dbdf)
86 |
87 | if (this.axesHelper) {
88 | scene.add(new THREE.AxesHelper(10e3))
89 | }
90 | }
91 |
92 | // 地面
93 | createPlane () {
94 | const { scene } = this
95 | const geometry = new THREE.PlaneBufferGeometry(10e2, 10e2, 1, 1)
96 | const meterial = new THREE.ShadowMaterial()
97 | meterial.opacity = 0.5
98 |
99 | const plane = this.plane = new THREE.Mesh(geometry, meterial)
100 |
101 | plane.rotation.x = -.5 * Math.PI
102 | plane.position.y = -.1
103 | // 接收阴影
104 | plane.receiveShadow = true
105 | scene.add(plane)
106 | }
107 |
108 | // 光
109 | createLight () {
110 | const { scene, lightInitalPosition: { x, y, z }, height } = this
111 | const light = this.light = new THREE.DirectionalLight(0xffffff, .5)
112 | const lightTarget = this.lightTarget = new THREE.Object3D()
113 |
114 | light.target = lightTarget
115 | light.position.set(x, y, z)
116 | // 开启阴影投射
117 | light.castShadow = true
118 | // // 定义可见域的投射阴影
119 | light.shadow.camera.left = -height
120 | light.shadow.camera.right = height
121 | light.shadow.camera.top = height
122 | light.shadow.camera.bottom = -height
123 | light.shadow.camera.near = 0
124 | light.shadow.camera.far = 2000
125 | // 定义阴影的分辨率
126 | light.shadow.mapSize.width = 1600
127 | light.shadow.mapSize.height = 1600
128 |
129 | // 环境光
130 | scene.add(new THREE.AmbientLight(0xE5E7E9, .4))
131 | scene.add(new THREE.HemisphereLight(0xffffff, 0xffffff, .2))
132 | scene.add(lightTarget)
133 | scene.add(light)
134 | }
135 |
136 | // 相机
137 | createCamera () {
138 | const {
139 | scene,
140 | width, height,
141 | cameraInitalPosition: { x, y, z },
142 | cameraNear, cameraFar
143 | } = this
144 | const camera = this.camera = new THREE.OrthographicCamera(-width / 2, width / 2, height / 2, -height / 2, cameraNear, cameraFar)
145 |
146 | camera.position.set(x, y, z)
147 | camera.lookAt(scene.position)
148 | scene.add(camera)
149 | }
150 |
151 | // 渲染器
152 | createRenterer () {
153 | const { canvas, width, height } = this
154 | const renderer = this.renderer = new THREE.WebGLRenderer({
155 | canvas,
156 | alpha: true, // 透明场景
157 | antialias:true // 抗锯齿
158 | })
159 |
160 | renderer.setSize(width, height)
161 | // 开启阴影
162 | renderer.shadowMap.enabled = true
163 | // 设置设备像素
164 | renderer.setPixelRatio(window.devicePixelRatio)
165 | }
166 |
167 | // 执行渲染
168 | render () {
169 | const { scene, camera } = this
170 | this.renderer.render(scene, camera)
171 | }
172 |
173 | // center为2个盒子的中心点
174 | moveCamera ({ cameraTo, center, lightTo }, onComplete, duration) {
175 | const {
176 | camera, plane,
177 | light, lightTarget,
178 | lightInitalPosition
179 | } = this
180 |
181 | // 移动相机
182 | animate(
183 | {
184 | from: { ...camera.position },
185 | to: cameraTo,
186 | duration
187 | },
188 | ({ x, z }) => {
189 | camera.position.x = x
190 | camera.position.z = z
191 | this.render()
192 | },
193 | onComplete
194 | )
195 |
196 | // 灯光和目标也需要动起来,为了保证阴影位置不变
197 | const { x: lightInitalX, z: lightInitalZ } = lightInitalPosition
198 | animate(
199 | {
200 | from: { ...light.position },
201 | to: lightTo,
202 | duration
203 | },
204 | ({ x, y, z }) => {
205 | lightTarget.position.x = x - lightInitalX
206 | lightTarget.position.z = z - lightInitalZ
207 | light.position.set(x, y, z)
208 | }
209 | )
210 |
211 | // 保证不会跑出有限大小的地面
212 | plane.position.x = center.x
213 | plane.position.z = center.z
214 | }
215 |
216 | // 场景中添加物体
217 | add (...args) {
218 | return this.scene.add(...args)
219 | }
220 | // 移除场景中的物体
221 | remove (...args) {
222 | return this.scene.remove(...args)
223 | }
224 | }
225 |
226 | export default Stage
--------------------------------------------------------------------------------
/src/games/wx-jump-1/index.js:
--------------------------------------------------------------------------------
1 | import {
2 | computeCameraInitalPosition,
3 | getPropSize,
4 | rangeNumberInclusive,
5 | animate
6 | } from './utils'
7 | import TWEEN from '@tweenjs/tween.js'
8 | import Stage from './Stage'
9 | import PropCreator from './PropCreator'
10 | import Prop from './Prop'
11 | import LittleMan from './LittleMan'
12 |
13 | class JumpGameWorld {
14 | constructor ({
15 | container,
16 | canvas,
17 | needDefaultCreator = true,
18 | axesHelper = false
19 | }) {
20 | const { offsetWidth, offsetHeight } = container
21 | this.G = 9.8
22 | this.container = container
23 | this.canvas = canvas
24 | this.width = offsetWidth
25 | this.height = offsetHeight
26 | this.needDefaultCreator = needDefaultCreator
27 | this.axesHelper = axesHelper
28 |
29 | const [min, max] = [~~(offsetWidth / 6), ~~(offsetWidth / 3.5)]
30 | this.propSizeRange = [min, max]
31 | this.propHeight = ~~(max / 2)
32 | this.propDistanceRange = [~~(min / 2), max * 2]
33 |
34 | this.stage = null
35 | this.propCreator = null
36 |
37 | this.init()
38 | }
39 |
40 | init () {
41 | this.initStage()
42 | this.initPropCreator()
43 | this.computeSafeClearLength()
44 | // 第一个道具
45 | this.createProp(0)
46 | // 第二个道具
47 | this.createProp(0)
48 | // 首次调整相机
49 | this.moveCamera(0)
50 | this.initLittleMan()
51 |
52 | // 测试
53 | // const autoMove = () => {
54 | // setTimeout(() => {
55 | // autoMove()
56 | // this.createProp()
57 | // this.moveCamera()
58 | // }, 2000)
59 | // }
60 | // autoMove()
61 | }
62 |
63 | // 初始化舞台
64 | initStage () {
65 | const { container, canvas, axesHelper, width, height } = this
66 | const cameraNear = 0.1
67 | const cameraFar = 2000
68 | // 计算相机应该放在哪里
69 | const cameraInitalPosition = this.cameraInitalPosition = computeCameraInitalPosition(35, 225, height / 2, height / 2, cameraNear, cameraFar)
70 | const lightInitalPosition = this.lightInitalPosition = { x: -300, y: 600, z: 200 }
71 |
72 | this.stage = new Stage({
73 | width,
74 | height,
75 | canvas,
76 | axesHelper,
77 | cameraNear,
78 | cameraFar,
79 | cameraInitalPosition,
80 | lightInitalPosition
81 | })
82 | }
83 |
84 | // 初始化道具生成器
85 | initPropCreator () {
86 | const { needDefaultCreator, propSizeRange, propHeight } = this
87 |
88 | // 管理所有道具
89 | this.props = []
90 | this.propCreator = new PropCreator({
91 | propHeight,
92 | propSizeRange,
93 | needDefaultCreator
94 | })
95 | }
96 |
97 | // 对外的新增生成器的接口
98 | createPropCreator (...args) {
99 | this.propCreator.createPropCreator(...args)
100 | }
101 |
102 | // 初始化小人
103 | initLittleMan () {
104 | const { stage, propHeight, G, props } = this
105 | const littleMan = this.littleMan = new LittleMan({
106 | world: this,
107 | color: 0x386899,
108 | G
109 | })
110 | littleMan.enterStage(stage, { x: 0, y: propHeight + 80, z: 0 }, props[0])
111 | littleMan.jump()
112 | }
113 |
114 | // 创建盒子
115 | createProp (enterHeight = 100) {
116 | const {
117 | height,
118 | propCreator,
119 | propHeight,
120 | propSizeRange: [min, max],
121 | propDistanceRange,
122 | stage, props,
123 | props: { length }
124 | } = this
125 | const currentProp = props[length - 1]
126 | const prop = new Prop({
127 | world: this,
128 | stage,
129 | // 头2个盒子用第一个创造器生成
130 | body: propCreator.createProp(length < 3 ? 0 : -1),
131 | height: propHeight,
132 | prev: currentProp,
133 | enterHeight,
134 | distanceRange: propDistanceRange
135 | })
136 | const size = prop.getSize()
137 |
138 | if (size.y !== propHeight) {
139 | console.warn(`高度: ${size.y},盒子高度必须为 ${propHeight}`)
140 | }
141 | if (size.x < min || size.x > max) {
142 | console.warn(`宽度: ${size.x}, 盒子宽度必须为 ${min} - ${max}`)
143 | }
144 | if (size.z < min || size.z > max) {
145 | console.warn(`深度: ${size.z}, 盒子深度度必须为 ${min} - ${max}`)
146 | }
147 |
148 | if (currentProp) {
149 | currentProp.setNext(prop)
150 | }
151 |
152 | prop.enterStage()
153 | props.push(prop)
154 | }
155 |
156 | // 移动相机,总是看向最后2个小球的中间位置
157 | moveCamera (duration = 500) {
158 | const {
159 | stage,
160 | height,
161 | cameraInitalPosition: { x: cameraX, y: cameraY, z: cameraZ },
162 | lightInitalPosition: { x: lightX, y: lightY, z: lightZ }
163 | } = this
164 | // 将可视区向上偏移一点,这样看起来道具的位置更合理
165 | const cameraOffsetY = height / 10
166 |
167 | const { x, y, z } = this.getLastTwoCenterPosition()
168 | const cameraTo = {
169 | x: x + cameraX + cameraOffsetY,
170 | y: cameraY, // 高度是不变的
171 | z: z + cameraZ + cameraOffsetY
172 | }
173 | const lightTo = {
174 | x: x + lightX,
175 | y: lightY,
176 | z: z + lightZ
177 | }
178 |
179 | // 移动舞台相机
180 | const options = {
181 | cameraTo,
182 | lightTo,
183 | center: { x, y, z }
184 | }
185 | stage.moveCamera(
186 | options,
187 | () => {
188 | this.clearProps()
189 | },
190 | duration
191 | )
192 | }
193 |
194 | // 计算最新的2个盒子的中心点
195 | getLastTwoCenterPosition () {
196 | const { props, props: { length } } = this
197 | const { x: x1, z: z1 } = props[length - 2].getPosition()
198 | const { x: x2, z: z2 } = props[length - 1].getPosition()
199 |
200 | return {
201 | x: x1 + (x2 - x1) / 2,
202 | z: z1 + (z2 - z1) / 2
203 | }
204 | }
205 |
206 | // 销毁道具
207 | clearProps () {
208 | const {
209 | width,
210 | height,
211 | safeClearLength,
212 | props, stage,
213 | props: { length }
214 | } = this
215 | const point = 4
216 |
217 | if (length > safeClearLength) {
218 | props.slice(0, point).forEach(prop => prop.dispose())
219 | this.props = props.slice(point)
220 | }
221 | }
222 |
223 | // 估算销毁安全值
224 | computeSafeClearLength () {
225 | const { width, height, propSizeRange } = this
226 | const minS = propSizeRange[0]
227 | const hypotenuse = Math.sqrt(minS * minS + minS * minS)
228 | this.safeClearLength = Math.ceil(width / minS) + Math.ceil(height / hypotenuse / 2) + 1
229 | }
230 | }
231 |
232 | export default JumpGameWorld
--------------------------------------------------------------------------------
/src/games/wx-jump-2/utils.js:
--------------------------------------------------------------------------------
1 | import * as THREE from 'three'
2 | import TWEEN from '@tweenjs/tween.js'
3 |
4 | const { random, sqrt, floor, pow, sin, cos, tan, PI } = Math
5 |
6 | export let propCounter = 0
7 | export const incrementPropCounter = () => propCounter++
8 | export const resetPropCounter = () => (propCounter = 0)
9 |
10 | export const colors = [0x67C23A, 0xE6A23C, 0xF56C6C, 0x909399, 0x409EFF, 0xffffff]
11 |
12 | /**
13 | * 根据角度计算相机初始位置
14 | * @param {Number} verticalDeg 相机和场景中心点的垂直角度
15 | * @param {Number} horizontalDeg 相机和x轴的水平角度
16 | * @param {Number} top 相机上侧面
17 | * @param {Number} bottom 相机下侧面
18 | * @param {Number} near 摄像机视锥体近端面
19 | * @param {Number} far 摄像机视锥体远端面
20 | */
21 | export function computeCameraInitalPosition (verticalDeg, horizontalDeg, top, bottom, near, far) {
22 | const verticalRadian = verticalDeg * (PI / 180)
23 | const horizontalRadian = horizontalDeg * (PI / 180)
24 | const minY = cos(verticalRadian) * bottom
25 | const maxY = sin(verticalRadian) * (far - near - top / tan(verticalRadian))
26 |
27 | if (minY > maxY) {
28 | console.warn('警告: 垂直角度太小了!')
29 | return
30 | }
31 | const y = minY + (maxY - minY) / 2
32 | const longEdge = y / tan(verticalRadian)
33 | const x = sin(horizontalRadian) * longEdge
34 | const z = cos(horizontalRadian) * longEdge
35 |
36 | return { x, y, z }
37 | }
38 |
39 | // 材质
40 | export const baseMeshLambertMaterial = new THREE.MeshLambertMaterial()
41 | // 立方体
42 | export const baseBoxBufferGeometry = new THREE.BoxBufferGeometry(1, 1, 1, 10, 4, 10)
43 | baseBoxBufferGeometry.userData.type = 'box'
44 | // 圆柱体
45 | export const baseCylinderBufferGeometry = new THREE.CylinderBufferGeometry(1, 1, 1, 30, 5)
46 | baseCylinderBufferGeometry.userData.type = 'Cylinder'
47 |
48 | // 物体销毁
49 | export const destroyMesh = mesh => {
50 | if (mesh.geometry) {
51 | mesh.geometry.dispose()
52 | mesh.geometry = null
53 | }
54 | if (mesh.material) {
55 | mesh.material.dispose()
56 | mesh.material = null
57 | }
58 |
59 | mesh.parent.remove(mesh)
60 |
61 | mesh.parent = null
62 | mesh = null
63 | }
64 |
65 | export const randomArrayElm = array => array[floor(random() * array.length)]
66 |
67 | export const rangeNumberInclusive = (min, max) => floor(random() * (max - min + 1)) + min
68 |
69 | export const getPropSize = box => {
70 | const box3 = getPropSize.box3 || (getPropSize.box3 = new THREE.Box3())
71 | box3.setFromObject(box)
72 | return box3.getSize(new THREE.Vector3())
73 | }
74 |
75 | // 斜抛计算
76 | export const computeObligueThrowValue = function (v0, theta, G) {
77 | const sin2θ = sin(2 * theta)
78 | const sinθ = sin(theta)
79 |
80 | const rangeR = pow(v0, 2) * sin2θ / G
81 | const rangeH = pow(v0 * sinθ, 2) / (2 * G)
82 | // const rangeT = 2 * v0 * sinθ / G
83 |
84 | return {
85 | rangeR,
86 | rangeH,
87 | // rangeT
88 | }
89 | }
90 | /**
91 | * 获取静止盒子的碰撞检测器
92 | * @param {Mesh} prop 检测的盒子
93 | * @param {String} direction 物体过来的方向(世界坐标系)
94 | * @param {Boolean} isForward 基于方向的前后
95 | */
96 | export const getHitValidator = (prop, direction, isForward) => {
97 | const origin = prop.position.clone()
98 | const vertices = prop.geometry.attributes.position
99 | const length = vertices.count
100 |
101 | // 盒子是静止的,先将顶点到中心点的向量准备好,避免重复计算
102 | const directionVectors = Array.from({ length })
103 | .map((_, i) => new THREE.Vector3().fromBufferAttribute(vertices, i))
104 | .filter(vector3 => {
105 | // 过滤掉一部分盒子离小人远端的顶点
106 | if (direction === 'z' && isForward) {
107 | // 从当前盒子倒向目标盒子
108 | return vector3.z < 0
109 | } else if (direction === 'z') {
110 | // 从目标盒子倒向当前盒子
111 | return vector3.z > 0
112 | } else if (direction === 'x' && isForward) {
113 | return vector3.x < 0
114 | } else if (direction === 'x') {
115 | return vector3.x > 0
116 | }
117 | })
118 | .map(localVertex => {
119 | const globaVertex = localVertex.applyMatrix4(prop.matrix)
120 | // 先将向量准备好
121 | return globaVertex.sub(prop.position)
122 | })
123 |
124 | return littleMan => {
125 | for (let i = 0, directionVector; (directionVector = directionVectors[i]); i++) {
126 | const raycaster = new THREE.Raycaster(origin, directionVector.clone().normalize())
127 | const collisionResults = raycaster.intersectObject(littleMan, true)
128 |
129 | // 发生了碰撞
130 | if(collisionResults.length > 0 && collisionResults[0].distance < directionVector.length() + 1.2 ){
131 | return true
132 | }
133 | }
134 | return false
135 | }
136 | }
137 |
138 | export const showSegments = mesh => {
139 | const geometry = mesh.geometry.clone()
140 | const material = new THREE.PointsMaterial({
141 | size: 3,
142 | color: 0x58D68D,
143 | transparent: true
144 | })
145 | const particle = new THREE.Points(geometry, material)
146 |
147 | mesh.add(particle)
148 | }
149 |
150 | /**
151 | * 根据射程算出落地点
152 | * @param {Number} range 射程
153 | * @param {Object} c1 起跳点
154 | * @param {Object} p2 目标盒子中心点
155 | */
156 | export const computePositionByRangeR = function (range, c1, p2) {
157 | const { x: c1x, z: c1z } = c1
158 | const { x: p2x, z: p2z } = p2
159 |
160 | const p2cx = p2x - c1x
161 | const p2cz = p2z - c1z
162 | const p2c = sqrt(pow(p2cz, 2) + pow(p2cx, 2))
163 |
164 | const jumpDownX = p2cx * range / p2c
165 | const jumpDownZ = p2cz * range / p2c
166 |
167 | return {
168 | jumpDownX: c1x + jumpDownX,
169 | jumpDownZ: c1z + jumpDownZ
170 | }
171 | }
172 |
173 | export const animate = (configs, onUpdate, onComplete) => {
174 | const {
175 | from, to, duration,
176 | easing = k => k,
177 | autoStart = true // 为了使用tween的chain
178 | } = configs
179 |
180 | const tween = new TWEEN.Tween(from)
181 | .to(to, duration)
182 | .easing(easing)
183 | .onUpdate(onUpdate)
184 | .onComplete(() => {
185 | onComplete && onComplete()
186 | })
187 |
188 | if (autoStart) {
189 | tween.start()
190 | }
191 |
192 | animateFrame()
193 | return tween
194 | }
195 |
196 | const animateFrame = function () {
197 | if (animateFrame.openin) {
198 | return
199 | }
200 | animateFrame.openin = true
201 |
202 | const animate = () => {
203 | const id = requestAnimationFrame(animate)
204 | if (!TWEEN.update()) {
205 | animateFrame.openin = false
206 | cancelAnimationFrame(id)
207 | }
208 | }
209 | animate()
210 | }
--------------------------------------------------------------------------------
/src/games/wx-jump-2/index.js:
--------------------------------------------------------------------------------
1 | import {
2 | computeCameraInitalPosition,
3 | resetPropCounter
4 | } from './utils'
5 | import Stage from './Stage'
6 | import PropCreator from './PropCreator'
7 | import Prop from './Prop'
8 | import LittleMan from './LittleMan'
9 |
10 | class JumpGameWorld {
11 | constructor ({
12 | container,
13 | canvas,
14 | needDefaultCreator = true,
15 | axesHelper = false
16 | }) {
17 | const { offsetWidth, offsetHeight } = container
18 | this.G = 9.8
19 | this.container = container
20 | this.canvas = canvas
21 | this.width = offsetWidth
22 | this.height = offsetHeight
23 | this.needDefaultCreator = needDefaultCreator
24 | this.axesHelper = axesHelper
25 |
26 | const [min, max] = [~~(offsetWidth / 6), ~~(offsetWidth / 3.5)]
27 | this.propSizeRange = [min, max]
28 | this.propHeight = ~~(max / 2)
29 | this.propDistanceRange = [~~(min / 2), max * 2]
30 |
31 | this.stage = null
32 | this.propCreator = null
33 | this.littleMans = []
34 | this.props = []
35 |
36 | this.init()
37 | }
38 |
39 | init () {
40 | this.initStage()
41 | this.initPropCreator()
42 | this.computeSafeClearLength()
43 | // 第一个道具
44 | this.createProp(0)
45 | // 第二个道具
46 | this.createProp(0)
47 | // 首次调整相机
48 | this.moveCamera(0)
49 | this.initLittleMan()
50 |
51 | // 测试
52 | // const autoMove = () => {
53 | // setTimeout(() => {
54 | // autoMove()
55 | // this.createProp()
56 | // this.moveCamera()
57 | // }, 2000)
58 | // }
59 | // autoMove()
60 | }
61 |
62 | // 游戏销毁
63 | destroy () {
64 | this.clear()
65 | }
66 |
67 | // 让游戏回到初始状态
68 | clear () {
69 | this.stage && this.stage.reset()
70 |
71 | this.props.forEach(prop => prop.destroy())
72 | this.props = []
73 | resetPropCounter()
74 |
75 | this.littleMans.forEach(littleMan => littleMan.destroy())
76 | this.littleMans = []
77 | }
78 |
79 | // 初始化舞台
80 | initStage () {
81 | const { canvas, axesHelper, width, height } = this
82 | const cameraNear = 0.1
83 | const cameraFar = 2000
84 | // 计算相机应该放在哪里
85 | const cameraInitalPosition = this.cameraInitalPosition = computeCameraInitalPosition(35, 225, height / 2, height / 2, cameraNear, cameraFar)
86 | const lightInitalPosition = this.lightInitalPosition = { x: -300, y: 600, z: 200 }
87 |
88 | this.stage = new Stage({
89 | width,
90 | height,
91 | canvas,
92 | axesHelper,
93 | cameraNear,
94 | cameraFar,
95 | cameraInitalPosition,
96 | lightInitalPosition
97 | })
98 | }
99 |
100 | // 初始化道具生成器
101 | initPropCreator () {
102 | const { needDefaultCreator, propSizeRange, propHeight } = this
103 |
104 | // 管理所有道具
105 | this.propCreator = new PropCreator({
106 | propHeight,
107 | propSizeRange,
108 | needDefaultCreator
109 | })
110 | }
111 |
112 | // 对外的新增生成器的接口
113 | createPropCreator (...args) {
114 | this.propCreator.createPropCreator(...args)
115 | }
116 |
117 | // 初始化小人
118 | initLittleMan () {
119 | const { stage, propHeight, G, props } = this
120 | const littleMan = new LittleMan({
121 | world: this,
122 | color: 0x386899,
123 | G
124 | })
125 | littleMan.enterStage(stage, { x: 0, y: propHeight + 80, z: 0 }, props[0])
126 | littleMan.jump()
127 |
128 | this.littleMans.push(littleMan)
129 | }
130 |
131 | // 创建盒子
132 | createProp (enterHeight = 100, index = -1) {
133 | const {
134 | propCreator,
135 | propHeight,
136 | propSizeRange: [min, max],
137 | propDistanceRange,
138 | stage, props,
139 | props: { length }
140 | } = this
141 | const currentProp = props[length - 1]
142 | const prop = new Prop({
143 | world: this,
144 | stage,
145 | // 头2个盒子用第一个创造器生成
146 | body: propCreator.createProp(index === -1 ? length < 3 ? 0 : index : index),
147 | height: propHeight,
148 | prev: currentProp,
149 | enterHeight,
150 | distanceRange: propDistanceRange
151 | })
152 | const size = prop.getSize()
153 |
154 | if (size.y !== propHeight) {
155 | console.warn(`高度: ${size.y},盒子高度必须为 ${propHeight}`)
156 | }
157 | if (size.x < min || size.x > max) {
158 | console.warn(`宽度: ${size.x}, 盒子宽度必须为 ${min} - ${max}`)
159 | }
160 | if (size.z < min || size.z > max) {
161 | console.warn(`深度: ${size.z}, 盒子深度度必须为 ${min} - ${max}`)
162 | }
163 |
164 | if (currentProp) {
165 | currentProp.setNext(prop)
166 | }
167 |
168 | prop.enterStage()
169 | props.push(prop)
170 |
171 | return prop
172 | }
173 |
174 | // 移动相机,总是看向最后2个盒子的中间位置
175 | moveCamera (duration = 500) {
176 | const {
177 | stage,
178 | height,
179 | cameraInitalPosition: { x: cameraX, y: cameraY, z: cameraZ },
180 | lightInitalPosition: { x: lightX, y: lightY, z: lightZ }
181 | } = this
182 | // 将可视区向上偏移一点,这样看起来道具的位置更合理
183 | const cameraOffsetY = height / 10
184 |
185 | const { x, y, z } = this.getLastTwoCenterPosition()
186 | const cameraTo = {
187 | x: x + cameraX + cameraOffsetY,
188 | y: cameraY, // 高度是不变的
189 | z: z + cameraZ + cameraOffsetY
190 | }
191 | const lightTo = {
192 | x: x + lightX,
193 | y: lightY,
194 | z: z + lightZ
195 | }
196 |
197 | // 移动舞台相机
198 | const options = {
199 | cameraTo,
200 | lightTo,
201 | center: { x, y, z }
202 | }
203 | stage.moveCamera(
204 | options,
205 | () => {
206 | this.clearProps()
207 | },
208 | duration
209 | )
210 | }
211 |
212 | // 计算最新的2个盒子的中心点
213 | getLastTwoCenterPosition () {
214 | const { props, props: { length } } = this
215 | const { x: x1, z: z1 } = props[length - 2].getPosition()
216 | const { x: x2, z: z2 } = props[length - 1].getPosition()
217 |
218 | return {
219 | x: x1 + (x2 - x1) / 2,
220 | z: z1 + (z2 - z1) / 2
221 | }
222 | }
223 |
224 | // 销毁道具
225 | clearProps () {
226 | const {
227 | safeClearLength,
228 | props,
229 | props: { length }
230 | } = this
231 | const point = 4
232 |
233 | if (length > safeClearLength) {
234 | props.slice(0, point).forEach(prop => prop.destroy())
235 | this.props = props.slice(point)
236 | }
237 | }
238 |
239 | // 估算销毁安全值
240 | computeSafeClearLength () {
241 | const { width, height, propSizeRange } = this
242 | const minS = propSizeRange[0]
243 | const hypotenuse = Math.sqrt(minS * minS + minS * minS)
244 | this.safeClearLength = Math.ceil(width / minS) + Math.ceil(height / hypotenuse / 2) + 1
245 | }
246 | }
247 |
248 | export default JumpGameWorld
--------------------------------------------------------------------------------
/examples/js/app.f1e8862e.js:
--------------------------------------------------------------------------------
1 | (function(e){function t(t){for(var r,a,s=t[0],i=t[1],c=t[2],p=0,l=[];p {
110 | body.position.setY(y)
111 | stage.render()
112 | }
113 | )
114 | }
115 |
116 | // 回弹动画
117 | springbackTransition (duration) {
118 | const { body, stage } = this
119 | const y = body.scale.y
120 |
121 | animate(
122 | {
123 | from: { y },
124 | to: { y: 1 },
125 | duration,
126 | easing: TWEEN.Easing.Bounce.Out
127 | },
128 | ({ y }) => {
129 | body.scale.setY(y)
130 | stage.render()
131 | }
132 | )
133 | }
134 |
135 | // containsPoint (x, z) {
136 | // const { body } = this
137 | // const { type } = body.geometry.userData
138 | // const { x: sx, z: sz } = this.getSize()
139 | // const { x: px, z: pz } = this.getPosition()
140 |
141 | // if (type === 'box') {
142 | // const halfSx = sx / 2
143 | // const halfSz = sz / 2
144 | // const minX = px - halfSx
145 | // const maxX = px + halfSx
146 | // const minZ = pz - halfSz
147 | // const maxZ = pz + halfSz
148 |
149 | // return x >= minX && x <= maxX && z >= minZ && z <= maxZ
150 | // } else {
151 | // const radius = sx / 2
152 | // // 小人脚下中心点离圆心的距离
153 | // const distance = Math.sqrt(Math.pow(px - x, 2) + Math.pow(pz - z, 2))
154 |
155 | // return distance <= radius
156 | // }
157 | // }
158 |
159 | /**
160 | * 计算跌落数据
161 | * @param {Number} width 小人的宽度
162 | * @param {Number} x 小人脚下中心点的X值
163 | * @param {Number} z 小人脚下中心点的Z值
164 | * @return {
165 | * contains, // 小人中心点是否在盒子上
166 | * isEdge, // 是否在边缘
167 | * translateZ, // 将小人旋转部分移动 -translateZ,将网格移动translateZ
168 | * degY, // 调整小人方向,然后使用小人的本地坐标进行平移和旋转
169 | * }
170 | */
171 | computePointInfos (width, x, z) {
172 | const { body } = this
173 |
174 | if (!body) {
175 | return {}
176 | }
177 |
178 | const { type } = body.geometry.userData
179 | const { x: sx, z: sz } = this.getSize()
180 | const { x: px, z: pz } = this.getPosition()
181 | const halfWidth = width / 2
182 |
183 | // 立方体和圆柱体的计算逻辑略有差别
184 | if (type === 'box') {
185 | const halfSx = sx / 2
186 | const halfSz = sz / 2
187 | const minX = px - halfSx
188 | const maxX = px + halfSx
189 | const minZ = pz - halfSz
190 | const maxZ = pz + halfSz
191 |
192 | const contains = x >= minX && x <= maxX && z >= minZ && z <= maxZ
193 |
194 | if (contains) {
195 | return { contains }
196 | }
197 |
198 | const translateZ1 = Math.abs(z - pz) - halfSz
199 | const translateZ2 = Math.abs(x - px) - halfSx
200 | // 半空中
201 | if (translateZ1 >= halfWidth || translateZ2 >= halfWidth) {
202 | return { contains }
203 | }
204 |
205 | // 计算是否在盒子的边缘
206 | let isEdge = false
207 | let degY = 0
208 | let translateZ = 0
209 |
210 | // 四个方向上都有可能
211 | if (x < maxX && x > minX) {
212 | if (z > maxZ && z < maxZ + halfWidth) {
213 | degY = 0
214 | } else if (z < minZ && z > minZ - halfWidth) {
215 | degY = 180
216 | }
217 | isEdge = true
218 | translateZ = translateZ1
219 | } else if (z < maxZ && z > minZ) {
220 | if (x > maxX && x < maxX + halfWidth) {
221 | degY = 90
222 | } else if (x < minX && x > minX - halfWidth) {
223 | degY = 270
224 | }
225 | isEdge = true
226 | translateZ = translateZ2
227 | }
228 |
229 | return {
230 | contains,
231 | translateZ,
232 | isEdge,
233 | degY
234 | }
235 | } else {
236 | const radius = sx / 2
237 | // 小人脚下中心点离圆心的距离
238 | const distance = Math.sqrt(Math.pow(px - x, 2) + Math.pow(pz - z, 2))
239 |
240 | const contains = distance <= radius
241 |
242 | if (contains) {
243 | return { contains }
244 | }
245 |
246 | // 半空中
247 | if (distance >= radius + halfWidth) {
248 | return { contains }
249 | }
250 |
251 | // 在圆柱体的边缘
252 | const isEdge = true
253 | const translateZ = distance - radius
254 |
255 | let degY = Math.atan(Math.abs(x - px) / Math.abs(z - pz)) * 180 / Math.PI
256 |
257 | if (x === px) {
258 | degY = z > pz ? 0 : 180
259 | } else if (z === pz) {
260 | degY = x > px ? 90 : 270
261 | } else if (x > px && z > pz) {
262 | } else if (x > px && z < pz) {
263 | degY = 180 - degY
264 | } else if (z < pz) {
265 | degY = 180 + degY
266 | } else {
267 | degY = 360 - degY
268 | }
269 |
270 | return {
271 | contains,
272 | translateZ,
273 | isEdge,
274 | degY
275 | }
276 | }
277 | }
278 |
279 | setNext (next) {
280 | this.next = next
281 | }
282 |
283 | getNext () {
284 | return this.next
285 | }
286 |
287 | getPosition () {
288 | return this.body.position
289 | }
290 |
291 | setPosition (x, y, z) {
292 | return this.body.position.set(x, y, z)
293 | }
294 |
295 | scaleY (y) {
296 | return this.body.scale.setY(y)
297 | }
298 |
299 | // 获取道具大小
300 | getSize () {
301 | return getPropSize(this.body)
302 | }
303 |
304 | // 销毁道具
305 | destroy () {
306 | if (this.prev) {
307 | this.prev.next = null
308 | }
309 | if (this.next) {
310 | this.next.prev = null
311 | }
312 |
313 | destroyMesh(this.body)
314 |
315 | this.world = null
316 | this.stage = null
317 | this.body = null
318 | this.prev = null
319 | this.next = null
320 | }
321 | }
322 |
323 | export default Prop
--------------------------------------------------------------------------------
/src/games/wx-jump-2/Particle.js:
--------------------------------------------------------------------------------
1 | import * as THREE from 'three'
2 | import { getPropSize, rangeNumberInclusive, destroyMesh } from './utils'
3 |
4 | class Particle {
5 | constructor ({
6 | world,
7 | quantity = 15, // 数量
8 | triggerObject // 触发对象
9 | }) {
10 | this.world = world
11 | this.quantity = quantity
12 | this.triggerObject = triggerObject
13 | this.particleSystem = null
14 |
15 | const { x, y } = getPropSize(triggerObject)
16 |
17 | this.triggerObjectWidth = x
18 |
19 | // 粒子流,垂直方向的范围,约定从小人的上半身出现,算上粒子最大大小
20 | const flowSizeRange = this.flowSizeRange = [x / 6, x / 3]
21 | this.flowRangeY = [y / 2, y - flowSizeRange[1]]
22 | // 粒子初始的y值应该是粒子大小的最大值
23 | this.initalY = flowSizeRange[1]
24 | // 限制粒子水平方向的范围
25 | this.flowRangeX = [-x * 2, x * 2]
26 |
27 | // 粒子喷泉,垂直方向的范围,约定从小人的下半身出现,算上粒子最大大小
28 | const fountainSizeRange = this.fountainSizeRange = this.flowSizeRange.map(s => s / 2)
29 | this.fountainRangeY = [fountainSizeRange[1], y / 3]
30 | this.fountainRangeDistance = [y / 4, y / 2]
31 | // 限制粒子水平方向的范围
32 | this.fountainRangeX = [-x / 3, x / 3]
33 |
34 | this.createParticle()
35 | }
36 |
37 | // 销毁粒子
38 | destroy () {
39 | if (this.particleSystem) {
40 | this.particleSystem.children.forEach(destroyMesh)
41 | this.particleSystem.children = null
42 | destroyMesh(this.particleSystem)
43 | }
44 |
45 | this.world = null
46 | this.triggerObject = null
47 | this.particleSystem = null
48 | }
49 |
50 | // 生成粒子
51 | createParticle () {
52 | const { quantity, triggerObject } = this
53 | // 一半白色、一半绿色
54 | const white = new THREE.Color( 0xffffff )
55 | const green = new THREE.Color( 0x58D68D )
56 | const colors = Array.from({ length: quantity }).map((_, i) => i % 2 ? white : green)
57 | const particleSystem = this.particleSystem = new THREE.Group()
58 |
59 | new THREE.TextureLoader().load(require('./dot.png'), dot => {
60 | const baseGeometry = new THREE.Geometry()
61 | baseGeometry.vertices.push(new THREE.Vector3())
62 |
63 | const baseMaterial = new THREE.PointsMaterial({
64 | size: 0,
65 | map: dot,
66 | // depthTest: false, // 开启后可以透视...
67 | transparent: true
68 | })
69 |
70 | colors.forEach(color => {
71 | const geometry = baseGeometry.clone()
72 | const material = baseMaterial.clone()
73 | material.setValues({ color })
74 |
75 | const particle = new THREE.Points(geometry, material)
76 | particleSystem.add(particle)
77 | })
78 |
79 | this.resetParticle()
80 |
81 | triggerObject.add(particleSystem)
82 | })
83 | }
84 |
85 | // 将粒子放到小人脚下
86 | resetParticle () {
87 | const { particleSystem, initalY } = this
88 | particleSystem.children.forEach(particle => {
89 | particle.position.y = initalY
90 | particle.position.x = 0
91 | particle.position.z = 0
92 | })
93 | }
94 |
95 | // 粒子流粒子泵
96 | runParticleFlowPump () {
97 | const { particleSystem, quantity, initalY } = this
98 | // 粒子泵只关心脚下的粒子(水池)
99 | const particles = particleSystem.children.filter(child => child.position.y === initalY)
100 |
101 | // 脚下的粒子量不够,抽不上来
102 | if (particles.length < quantity / 3) {
103 | return
104 | }
105 |
106 | const {
107 | triggerObjectWidth,
108 | flowRangeX, flowRangeY, flowSizeRange
109 | } = this
110 | // 比如随机 x 值为0,这个值在小人的身体范围内,累加一个1/2身体宽度,这样做可能有部分区域随机不到,不过影响不大
111 | const halfWidth = triggerObjectWidth / 2
112 |
113 | particles.forEach(particle => {
114 | const { position, material } = particle
115 | const randomX = rangeNumberInclusive(...flowRangeX)
116 | const randomZ = rangeNumberInclusive(...flowRangeX)
117 | // 小人的身体内,不能成为起点,需要根据正反将身体的宽度加上
118 | const excludeX = randomX < 0 ? -halfWidth : halfWidth
119 | const excludeZ = randomZ < 0 ? -halfWidth : halfWidth
120 |
121 | position.x = excludeX + randomX
122 | position.z = excludeZ + randomZ
123 | position.y = rangeNumberInclusive(...flowRangeY)
124 |
125 | material.setValues({ size: rangeNumberInclusive(...flowSizeRange) })
126 | })
127 | }
128 |
129 | // 粒子流
130 | runParticleFlow () {
131 | if (this.runingParticleFlow) {
132 | return
133 | }
134 | this.runingParticleFlow = true
135 |
136 | const { world, triggerObjectWidth, particleSystem, initalY } = this
137 | let prevTime = 0
138 | // 约定速度,每毫秒走多远
139 | const speed = triggerObjectWidth * 3 / 1000
140 | const animate = () => {
141 | const id = requestAnimationFrame(animate)
142 |
143 | if (this.runingParticleFlow) {
144 | // 抽粒子
145 | this.runParticleFlowPump()
146 | if (prevTime) {
147 | const actives = particleSystem.children.filter(child => child.position.y !== initalY)
148 | const diffTime = Date.now() - prevTime
149 | // 粒子的行程
150 | const trip = diffTime * speed
151 |
152 | actives.forEach(particle => {
153 | const { position } = particle
154 | const { x, y, z } = position
155 |
156 | if (y < initalY) {
157 | // 只要粒子的y值超过安全值,就认为它已经到达终点
158 | position.y = initalY
159 | position.x = 0
160 | position.z = 0
161 | } else {
162 | const distance = Math.sqrt(Math.pow(x, 2) + Math.pow(z, 2) + Math.pow(y - initalY, 2))
163 | const ratio = (distance - trip) / distance
164 |
165 | position.x = ratio * x
166 | position.z = ratio * z
167 | position.y = ratio * y
168 | }
169 | })
170 | world.stage.render()
171 | }
172 | prevTime = Date.now()
173 | } else {
174 | cancelAnimationFrame(id)
175 | }
176 | }
177 | animate()
178 | }
179 |
180 | // 停止粒子流
181 | stopRunParticleFlow () {
182 | this.runingParticleFlow = false
183 | this.resetParticle()
184 | }
185 |
186 | // 粒子喷泉粒子泵
187 | runParticleFountainPump (particles, duration) {
188 | const { fountainRangeDistance, triggerObjectWidth, initalY, world } = this
189 | // 随机设置粒子的终点
190 | particles.forEach(particle => {
191 | const { position: { x, y, z } } = particle
192 |
193 | const userData = particle.userData
194 |
195 | userData.ty = y + rangeNumberInclusive(...fountainRangeDistance)
196 | // x轴和z轴 向外侧喷出
197 | const diffX = rangeNumberInclusive(0, triggerObjectWidth / 3)
198 | userData.tx = (x < 0 ? -diffX : diffX) + x
199 | const diffZ = rangeNumberInclusive(0, triggerObjectWidth / 3)
200 | userData.tz = (z < 0 ? -diffZ : diffZ) + z
201 | })
202 |
203 | let prevTime = 0
204 | const startTime = Date.now()
205 | const speed = triggerObjectWidth * 3 / 600
206 |
207 | const animate = () => {
208 | const id = requestAnimationFrame(animate)
209 | // 已经在脚下的粒子不用处理
210 | const actives = particles.filter(particle => particle.position.y !== initalY)
211 |
212 | if (actives.length && !this.runingParticleFlow && Date.now() - startTime < duration) {
213 | if (prevTime) {
214 | const diffTime = Date.now() - prevTime
215 | // 粒子的行程
216 | const trip = diffTime * speed
217 |
218 | actives.forEach(particle => {
219 | const {
220 | position,
221 | position: { x, y, z },
222 | userData: { tx, ty, tz }
223 | } = particle
224 | if (y >= ty) {
225 | // 已经到达终点的粒子,重新放到脚下去
226 | position.x = 0
227 | position.y = initalY
228 | position.z = 0
229 | // 清空ratio值
230 | } else {
231 | const diffX = tx - x
232 | const diffY = ty - y
233 | const diffZ = tz - z
234 | const distance = Math.sqrt(Math.pow(diffX, 2) + Math.pow(diffY, 2) + Math.pow(diffZ, 2))
235 | const ratio = trip / distance
236 |
237 | position.y += ratio * diffY
238 | position.x += ratio * diffX
239 | position.z += ratio * diffZ
240 | }
241 | })
242 | world.stage.render()
243 | }
244 | prevTime = Date.now()
245 | } else {
246 | this.runingParticleFountain = false
247 | cancelAnimationFrame(id)
248 | }
249 | }
250 | animate()
251 | }
252 |
253 | // 粒子喷泉
254 | runParticleFountain () {
255 | if (this.runingParticleFountain) {
256 | return
257 | }
258 | this.runingParticleFountain = true
259 |
260 | const { particleSystem, quantity, initalY } = this
261 | // 粒子泵只关心脚下的粒子(水池)
262 | const particles = particleSystem.children.filter(child => child.position.y === initalY).slice(0, quantity)
263 |
264 | if (!particles.length) {
265 | return
266 | }
267 |
268 | const {
269 | triggerObjectWidth,
270 | fountainRangeX, fountainSizeRange, fountainRangeY
271 | } = this
272 | const halfWidth = triggerObjectWidth / 2
273 |
274 | particles.forEach(particle => {
275 | const { position, material } = particle
276 | const randomX = rangeNumberInclusive(...fountainRangeX)
277 | const randomZ = rangeNumberInclusive(...fountainRangeX)
278 | // 小人的身体内,不能成为起点,需要根据正反将身体的宽度加上
279 | const excludeX = randomX < 0 ? -halfWidth : halfWidth
280 | const excludeZ = randomZ < 0 ? -halfWidth : halfWidth
281 |
282 | position.x = excludeX + randomX
283 | position.z = excludeZ + randomZ
284 | position.y = rangeNumberInclusive(...fountainRangeY)
285 |
286 | material.setValues({ size: rangeNumberInclusive(...fountainSizeRange) })
287 | })
288 |
289 | // 喷射粒子
290 | this.runParticleFountainPump(particles, 1000)
291 | }
292 | }
293 |
294 | export default Particle
--------------------------------------------------------------------------------
/src/games/wx-jump-1/LittleMan.js:
--------------------------------------------------------------------------------
1 | import * as THREE from 'three'
2 | import TWEEN from '@tweenjs/tween.js'
3 | import {
4 | baseMeshLambertMaterial,
5 | computeObligueThrowValue,
6 | computePositionByRangeR,
7 | animate
8 | } from './utils'
9 |
10 | class LittleMan {
11 | constructor ({
12 | world,
13 | color,
14 | G
15 | }) {
16 | this.world = world
17 | this.color = color
18 | this.G = G
19 | this.v0 = world.width / 10
20 | this.theta = 90
21 |
22 | this.headSegment = null
23 | this.bodyScaleSegment = null
24 | this.bodyRotateSegment = null
25 | this.body = null
26 |
27 | this.currentProp = null
28 | this.nextProp = null
29 | this.powerStorageDuration = 1500
30 |
31 | this.stage = null
32 |
33 | this.createBody()
34 | this.resetPowerStorageParameter()
35 | }
36 |
37 | bindEvent () {
38 | const { container } = this.world
39 | const isMobile = 'ontouchstart' in document
40 | const mousedownName = isMobile ? 'touchstart' : 'mousedown'
41 | const mouseupName = isMobile ? 'touchend' : 'mouseup'
42 |
43 | // 该起跳了
44 | const mouseup = () => {
45 | if (this.jumping) {
46 | return
47 | }
48 | this.jumping = true
49 | // 蓄力动作应该停止
50 | this.poweringUp = false
51 |
52 | this.jump()
53 | container.removeEventListener(mouseupName, mouseup)
54 | }
55 |
56 | // 蓄力的时候
57 | const mousedown = event => {
58 | event.preventDefault()
59 | // 跳跃没有完成不能操作
60 | if (this.poweringUp || this.jumping || !this.currentProp) {
61 | return
62 | }
63 | this.poweringUp = true
64 |
65 | this.powerStorage()
66 | container.addEventListener(mouseupName, mouseup, false)
67 | }
68 |
69 | container.addEventListener(mousedownName, mousedown, false)
70 | }
71 |
72 | // 创建身体
73 | createBody () {
74 | const { color, world: { width } } = this
75 | const material = baseMeshLambertMaterial.clone()
76 | material.setValues({ color })
77 |
78 | // 头部
79 | const headSize = this.headSize = width * .03
80 | const headTranslateY = this.headTranslateY = headSize * 4.5
81 | const headGeometry = new THREE.SphereGeometry(headSize, 40, 40)
82 | const headSegment = this.headSegment = new THREE.Mesh(headGeometry, material)
83 | headSegment.castShadow = true
84 | headSegment.translateY(headTranslateY)
85 |
86 | // 身体
87 | this.bodySize = headSize * 4.5
88 | const bodyBottomGeometry = new THREE.CylinderBufferGeometry(headSize * .9, headSize * 1.2, headSize * 2.5, 40)
89 | bodyBottomGeometry.translate(0, headSize * 1.25, 0)
90 | const bodyCenterGeometry = new THREE.CylinderBufferGeometry(headSize, headSize * .9, headSize, 40)
91 | bodyCenterGeometry.translate(0, headSize * 3, 0)
92 | const bodyTopGeometry = new THREE.SphereGeometry(headSize, 40, 40)
93 | bodyTopGeometry.translate(0, headSize * 3.5, 0)
94 |
95 | const bodyGeometry = new THREE.Geometry()
96 | bodyGeometry.merge(bodyTopGeometry)
97 | bodyGeometry.merge(new THREE.Geometry().fromBufferGeometry(bodyCenterGeometry))
98 | bodyGeometry.merge(new THREE.Geometry().fromBufferGeometry(bodyBottomGeometry))
99 |
100 | // 缩放控制
101 | const bodyScaleSegment = this.bodyScaleSegment = new THREE.Mesh(bodyGeometry, material)
102 | bodyScaleSegment.castShadow = true
103 | bodyScaleSegment.translateY(-20)
104 |
105 | // 旋转控制
106 | const bodyRotateSegment = this.bodyRotateSegment = new THREE.Group()
107 | bodyRotateSegment.add(headSegment)
108 | bodyRotateSegment.add(bodyScaleSegment)
109 | bodyRotateSegment.translateY(20)
110 |
111 | // 整体身高 = 头部位移 + 头部高度 / 2 = headSize * 5.5
112 | const body = this.body = new THREE.Group()
113 | body.add(bodyRotateSegment)
114 | }
115 |
116 | // 进入舞台
117 | enterStage (stage, { x, y, z }, nextProp) {
118 | const { body } = this
119 |
120 | body.position.set(x, y, z)
121 |
122 | this.stage = stage
123 | // 进入舞台时告诉小人目标
124 | this.nextProp = nextProp
125 |
126 | stage.add(body)
127 | stage.render()
128 | this.bindEvent()
129 | }
130 |
131 | resetPowerStorageParameter () {
132 | // 由于蓄力导致的变形,需要记录后,在空中将小人复原
133 | this.toValues = {
134 | headTranslateY: 0,
135 | bodyScaleXZ: 0,
136 | bodyScaleY: 0
137 | }
138 | this.fromValues = this.fromValues || {
139 | headTranslateY: this.headTranslateY,
140 | bodyScaleXZ: 1,
141 | bodyScaleY: 1,
142 | propScaleY: 1
143 | }
144 | }
145 |
146 | // 蓄力
147 | powerStorage () {
148 | const {
149 | stage, powerStorageDuration,
150 | body, bodyScaleSegment, headSegment,
151 | bodySize,
152 | fromValues,
153 | currentProp,
154 | world: { propHeight }
155 | } = this
156 |
157 | this.powerStorageTime = Date.now()
158 | this.resetPowerStorageParameter()
159 |
160 | const tween = animate(
161 | {
162 | from: { ...fromValues },
163 | to: {
164 | headTranslateY: bodySize - bodySize * .6,
165 | bodyScaleXZ: 1.3,
166 | bodyScaleY: .6,
167 | propScaleY: .6
168 | },
169 | duration: powerStorageDuration
170 | },
171 | ({ headTranslateY, bodyScaleXZ, bodyScaleY, propScaleY }) => {
172 | if (!this.poweringUp) {
173 | // 抬起时停止蓄力
174 | tween.stop()
175 | } else {
176 |
177 | headSegment.position.setY(headTranslateY)
178 | bodyScaleSegment.scale.set(bodyScaleXZ, bodyScaleY, bodyScaleXZ)
179 | currentProp.scaleY(propScaleY)
180 | body.position.setY(propHeight * propScaleY)
181 |
182 | // 保存此时的位置用于复原
183 | this.toValues = {
184 | headTranslateY,
185 | bodyScaleXZ,
186 | bodyScaleY
187 | }
188 |
189 | stage.render()
190 | }
191 | }
192 | )
193 | }
194 |
195 | computePowerStorageValue () {
196 | const { powerStorageDuration, powerStorageTime, v0, theta } = this
197 | const diffTime = Date.now() - powerStorageTime
198 | const time = Math.min(diffTime, powerStorageDuration)
199 | const percentage = time / powerStorageDuration
200 |
201 | return {
202 | v0: v0 + 30 * percentage,
203 | theta: theta - 40 * percentage
204 | }
205 | }
206 |
207 | // 跳跃
208 | jump () {
209 | const {
210 | stage, body,
211 | currentProp, nextProp,
212 | world: { propHeight }
213 | } = this
214 | const duration = 400
215 | const start = body.position
216 | const target = nextProp.getPosition()
217 | const { x: startX, y: startY, z: startZ } = start
218 |
219 | // 开始游戏时,小人从第一个盒子正上方入场做弹球下落
220 | if (!currentProp && startX === target.x && startZ === target.z) {
221 | animate(
222 | {
223 | from: { y: startY },
224 | to: { y: propHeight },
225 | duration,
226 | easing: TWEEN.Easing.Bounce.Out
227 | },
228 | ({ y }) => {
229 | body.position.setY(y)
230 | stage.render()
231 | },
232 | () => {
233 | this.currentProp = nextProp
234 | this.nextProp = nextProp.getNext()
235 | this.jumping = false
236 | }
237 | )
238 | } else {
239 | if (!currentProp) {
240 | return
241 | }
242 |
243 | const { bodyScaleSegment, headSegment, G, headTranslateY } = this
244 | const { v0, theta } = this.computePowerStorageValue()
245 | const { rangeR, rangeH } = computeObligueThrowValue(v0, theta * (Math.PI / 180), G)
246 |
247 | // 水平匀速
248 | const { jumpDownX, jumpDownZ } = computePositionByRangeR(rangeR, start, target)
249 | animate(
250 | {
251 | from: {
252 | x: startX,
253 | z: startZ,
254 | ...this.toValues
255 | },
256 | to: {
257 | x: jumpDownX,
258 | z: jumpDownZ,
259 | ...this.fromValues
260 | },
261 | duration
262 | },
263 | ({ x, z, headTranslateY, bodyScaleXZ, bodyScaleY }) => {
264 | body.position.setX(x)
265 | body.position.setZ(z)
266 | headSegment.position.setY(headTranslateY)
267 | bodyScaleSegment.scale.set(bodyScaleXZ, bodyScaleY, bodyScaleXZ)
268 | }
269 | )
270 |
271 | // y轴上升段、下降段
272 | const rangeHeight = Math.max(this.world.width / 3, rangeH) + propHeight
273 | const yUp = animate(
274 | {
275 | from: { y: startY },
276 | to: { y: rangeHeight },
277 | duration: duration * .65,
278 | easing: TWEEN.Easing.Cubic.Out,
279 | autoStart: false
280 | },
281 | ({ y }) => {
282 | body.position.setY(y)
283 | }
284 | )
285 | const yDown = animate(
286 | {
287 | from: { y: rangeHeight },
288 | to: { y: propHeight },
289 | duration: duration * .35,
290 | easing: TWEEN.Easing.Cubic.In,
291 | autoStart: false
292 | },
293 | ({ y }) => {
294 | body.position.setY(y)
295 | }
296 | )
297 |
298 | // 落地后,生成下一个方块 -> 移动镜头 -> 更新关心的盒子 -> 结束
299 | const ended = () => {
300 | const { world } = this
301 | world.createProp()
302 | world.moveCamera()
303 |
304 | this.currentProp = nextProp
305 | this.nextProp = nextProp.getNext()
306 | // 跳跃结束了
307 | this.jumping = false
308 | }
309 | // 落地缓冲段
310 | const bufferUp = animate(
311 | {
312 | from: { s: .8 },
313 | to: { s: 1 },
314 | duration: 100,
315 | autoStart: false
316 | },
317 | ({ s }) => {
318 | bodyScaleSegment.scale.setY(s)
319 | },
320 | () => {
321 | // 以落地缓冲结束作为跳跃结束时间点
322 | ended()
323 | }
324 | )
325 |
326 | // 上升 -> 下降 -> 落地缓冲
327 | yDown.chain(bufferUp)
328 | yUp.chain(yDown).start()
329 |
330 | // 需要处理不同方向空翻
331 | const direction = currentProp.getPosition().z === nextProp.getPosition().z
332 | this.flip(duration, direction)
333 |
334 | // 从起跳开始就回弹
335 | currentProp.springbackTransition(500)
336 | }
337 |
338 | stage.render()
339 | }
340 |
341 | // 空翻
342 | flip (duration, direction) {
343 | const { bodyRotateSegment } = this
344 | let increment = 0
345 |
346 | animate(
347 | {
348 | from: { deg: 0 },
349 | to: { deg: 360 },
350 | duration,
351 | easing: TWEEN.Easing.Sinusoidal.InOut
352 | },
353 | ({ deg }) => {
354 | if (direction) {
355 | bodyRotateSegment.rotateZ(-(deg - increment) * (Math.PI/180))
356 | } else {
357 | bodyRotateSegment.rotateX((deg - increment) * (Math.PI/180))
358 | }
359 | increment = deg
360 | }
361 | )
362 | }
363 | }
364 |
365 | export default LittleMan
--------------------------------------------------------------------------------
/examples/js/wx-jump-stage1.c74513ec.js:
--------------------------------------------------------------------------------
1 | (window["webpackJsonp"]=window["webpackJsonp"]||[]).push([["wx-jump-stage1"],{"8bb1":function(e,t,i){},a56b:function(e,t,i){"use strict";var r=i("8bb1"),a=i.n(r);a.a},e424:function(e,t,i){"use strict";i.r(t);var r=function(){var e=this,t=e.$createElement;e._self._c;return e._m(0)},a=[function(){var e=this,t=e.$createElement,i=e._self._c||t;return i("div",{staticClass:"jump-world"},[i("canvas",{attrs:{id:"jump-world-canvas"}})])}],n=(i("ac6a"),i("768b")),o=i("d225"),s=i("b0b4"),h=i("5a89"),c=i("22b5"),u=Math.random,l=Math.sqrt,p=Math.floor,d=Math.pow,g=Math.sin,v=Math.cos,f=Math.tan,y=Math.PI;function m(e,t,i,r,a,n){var o=e*(y/180),s=t*(y/180),h=v(o)*r,c=g(o)*(n-a-i/f(o));if(!(h>c)){var u=h+(c-h)/2,l=u/f(o),p=g(s)*l,d=v(s)*l;return{x:p,y:u,z:d}}console.warn("警告: 垂直角度太小了!")}var w=new h["MeshLambertMaterial"],b=new h["BoxBufferGeometry"],S=function(e){return e[p(u()*e.length)]},P=function(e,t){return p(u()*(t-e+1))+e},x=function e(t){var i=e.box3||(e.box3=new h["Box3"]);return i.setFromObject(t),i.getSize(new h["Vector3"])},z=function(e,t,i){var r=g(2*t),a=g(t),n=d(e,2)*r/i,o=d(e*a,2)/(2*i);return{rangeR:n,rangeH:o}},O=function(e,t,i){var r=t.x,a=t.z,n=i.x,o=i.z,s=n-r,h=o-a,c=l(d(h,2)+d(s,2)),u=s*e/c,p=h*e/c;return{jumpDownX:r+u,jumpDownZ:a+p}},j=function(e,t,i){var r=e.from,a=e.to,n=e.duration,o=e.easing,s=void 0===o?function(e){return e}:o,h=e.autoStart,u=void 0===h||h,l=new c["a"].Tween(r).to(a,n).easing(s).onUpdate(t).onComplete((function(){i&&i()}));return u&&l.start(),k(),l},k=function e(){if(!e.openin){e.openin=!0;var t=function t(){var i=requestAnimationFrame(t);c["a"].update()||(e.openin=!1,cancelAnimationFrame(i))};t()}},C=(i("8e6e"),i("456d"),i("bd86"));function M(e,t){var i=Object.keys(e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);t&&(r=r.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),i.push.apply(i,r)}return i}function Y(e){for(var t=1;t-1&&t[e]&&t[e]()||S(t)()}},{key:"createPropCreator",value:function(e,t){var i=this;if(Array.isArray(e))e.forEach((function(e){return i.createPropCreator(e,t)}));else{var r=this.propCreators,a=this.propSizeRange,n=this.propHeight;if(!(r.indexOf(e)>-1)){var o=function i(){if(t&&i.box)return i.box.clone();var r=e(h,{propSizeRange:a,propHeight:n,baseMeshLambertMaterial:w,baseBoxBufferGeometry:b});return t&&(i.box=r),r};r.push(o)}}}}]),e}(),E=L,B=i("75fc"),V=function(){function e(t){var i=t.world,r=t.stage,a=t.body,n=t.height,s=t.enterHeight,h=t.distanceRange,c=t.prev;Object(o["a"])(this,e),this.world=i,this.stage=r,this.body=a,this.height=n,this.enterHeight=s,this.distanceRange=h,this.prev=c}return Object(s["a"])(e,[{key:"computeMyPosition",value:function(){var e=this.world,t=this.prev,i=this.distanceRange,r=this.enterHeight,a={x:0,y:r,z:0};if(!t)return a;if(0===r)return a.z=e.width/2,a;var n=t.getPosition(),o=n.x,s=n.z,h=0===Math.round(Math.random()),c=t.getSize(),u=c.x,l=c.z,p=this.getSize(),d=p.x,g=p.z,v=P.apply(void 0,Object(B["a"])(i));return h?(a.x=o+u/2+v+d/2,a.z=s):(a.x=o,a.z=s+l/2+v+g/2),a}},{key:"enterStage",value:function(){var e=this.stage,t=this.body,i=this.height,r=this.computeMyPosition(),a=r.x,n=r.y,o=r.z;t.castShadow=!0,t.receiveShadow=!0,t.position.set(a,n,o),t.geometry.translate(0,i/2,0),e.add(t),e.render(),this.entranceTransition()}},{key:"entranceTransition",value:function(){var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:400,t=this.body,i=this.enterHeight,r=this.stage;0!==i&&j({to:{y:0},from:{y:i},duration:e,easing:c["a"].Easing.Bounce.Out},(function(e){var i=e.y;t.position.setY(i),r.render()}))}},{key:"springbackTransition",value:function(e){var t=this.body,i=this.stage,r=t.scale.y;j({from:{y:r},to:{y:1},duration:e,easing:c["a"].Easing.Bounce.Out},(function(e){var r=e.y;t.scale.setY(r),i.render()}))}},{key:"setNext",value:function(e){this.next=e}},{key:"getNext",value:function(e){return this.next}},{key:"getPosition",value:function(){return this.body.position}},{key:"setPosition",value:function(e,t,i){return this.body.position.set(e,t,i)}},{key:"scaleY",value:function(e){return this.body.scale.setY(e)}},{key:"getSize",value:function(){return x(this.body)}},{key:"dispose",value:function(){var e=this.body,t=this.stage,i=this.prev,r=this.next;this.prev=null,this.next=null,i&&(i.next=null),r&&(r.prev=null),e.geometry.dispose(),e.material.dispose(),t.remove(e)}}]),e}(),X=V;function Z(e,t){var i=Object.keys(e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);t&&(r=r.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),i.push.apply(i,r)}return i}function N(e){for(var t=1;t0&&void 0!==arguments[0]?arguments[0]:100,t=(this.height,this.propCreator),i=this.propHeight,r=Object(n["a"])(this.propSizeRange,2),a=r[0],o=r[1],s=this.propDistanceRange,h=this.stage,c=this.props,u=this.props.length,l=c[u-1],p=new X({world:this,stage:h,body:t.createProp(u<3?0:-1),height:i,prev:l,enterHeight:e,distanceRange:s}),d=p.getSize();d.y!==i&&console.warn("高度: ".concat(d.y,",盒子高度必须为 ").concat(i)),(d.xo)&&console.warn("宽度: ".concat(d.x,", 盒子宽度必须为 ").concat(a," - ").concat(o)),(d.zo)&&console.warn("深度: ".concat(d.z,", 盒子深度度必须为 ").concat(a," - ").concat(o)),l&&l.setNext(p),p.enterStage(),c.push(p)}},{key:"moveCamera",value:function(){var e=this,t=arguments.length>0&&void 0!==arguments[0]?arguments[0]:500,i=this.stage,r=this.height,a=this.cameraInitalPosition,n=a.x,o=a.y,s=a.z,h=this.lightInitalPosition,c=h.x,u=h.y,l=h.z,p=r/10,d=this.getLastTwoCenterPosition(),g=d.x,v=d.y,f=d.z,y={x:g+n+p,y:o,z:f+s+p},m={x:g+c,y:u,z:f+l},w={cameraTo:y,lightTo:m,center:{x:g,y:v,z:f}};i.moveCamera(w,(function(){e.clearProps()}),t)}},{key:"getLastTwoCenterPosition",value:function(){var e=this.props,t=this.props.length,i=e[t-2].getPosition(),r=i.x,a=i.z,n=e[t-1].getPosition(),o=n.x,s=n.z;return{x:r+(o-r)/2,z:a+(s-a)/2}}},{key:"clearProps",value:function(){this.width,this.height;var e=this.safeClearLength,t=this.props,i=(this.stage,this.props.length),r=4;i>e&&(t.slice(0,r).forEach((function(e){return e.dispose()})),this.props=t.slice(r))}},{key:"computeSafeClearLength",value:function(){var e=this.width,t=this.height,i=this.propSizeRange,r=i[0],a=Math.sqrt(r*r+r*r);this.safeClearLength=Math.ceil(e/r)+Math.ceil(t/a/2)+1}}]),e}(),U=q,_={mounted:function(){new U({container:document.querySelector(".jump-world"),canvas:document.querySelector("#jump-world-canvas"),axesHelper:!0})}},J=_,W=(i("a56b"),i("2877")),$=Object(W["a"])(J,r,a,!1,null,"0bf23ea4",null);t["default"]=$.exports}}]);
--------------------------------------------------------------------------------
/src/games/wx-jump-2/LittleMan.js:
--------------------------------------------------------------------------------
1 | import * as THREE from 'three'
2 | import TWEEN from '@tweenjs/tween.js'
3 |
4 | import {
5 | baseMeshLambertMaterial,
6 | computeObligueThrowValue,
7 | computePositionByRangeR,
8 | animate,
9 | destroyMesh,
10 | getHitValidator
11 | } from './utils'
12 | import Particle from './Particle'
13 |
14 | class LittleMan {
15 | constructor ({
16 | world,
17 | color,
18 | G
19 | }) {
20 | this.world = world
21 | this.color = color
22 | this.G = G
23 | this.v0 = world.width / 10
24 | this.theta = 90
25 |
26 | this.headSegment = null
27 | this.bodyScaleSegment = null
28 | this.bodyRotateSegment = null
29 | this.body = null
30 |
31 | this.unbindFunc = null
32 | this.currentProp = null
33 | this.nextProp = null
34 | this.powerStorageDuration = 1500
35 |
36 | this.stage = null
37 |
38 | this.createBody()
39 | this.particle = new Particle({
40 | triggerObject: this.body,
41 | world
42 | })
43 | this.resetPowerStorageParameter()
44 | }
45 |
46 | // 销毁小人
47 | destroy () {
48 | this.particle.destroy()
49 | destroyMesh(this.headSegment)
50 | destroyMesh(this.bodyScaleSegment)
51 | destroyMesh(this.bodyRotateSegment)
52 | destroyMesh(this.body)
53 |
54 | this.unbindFunc && this.unbindFunc()
55 | this.world = null
56 | this.particle = null
57 | this.headSegment = null
58 | this.bodyScaleSegment = null
59 | this.bodyRotateSegment = null
60 | this.body = null
61 | this.unbindFunc = null
62 |
63 | this.currentProp = null
64 | this.nextProp = null
65 |
66 | this.stage = null
67 | }
68 |
69 | bindEvent () {
70 | const { canvas } = this.world
71 | const isMobile = 'ontouchstart' in document
72 | const mousedownName = isMobile ? 'touchstart' : 'mousedown'
73 | const mouseupName = isMobile ? 'touchend' : 'mouseup'
74 |
75 | // 该起跳了
76 | const mouseup = () => {
77 | if (this.jumping) {
78 | return
79 | }
80 | this.jumping = true
81 | // 蓄力动作应该停止
82 | this.poweringUp = false
83 | // 停止粒子流
84 | this.particle.stopRunParticleFlow()
85 |
86 | this.jump()
87 | canvas.removeEventListener(mouseupName, mouseup)
88 | }
89 |
90 | // 蓄力的时候
91 | const mousedown = event => {
92 | event.preventDefault()
93 | // 跳跃没有完成不能操作
94 | if (this.poweringUp || this.jumping || !this.currentProp || this.gameOver) {
95 | return
96 | }
97 | this.poweringUp = true
98 | // 开启粒子流
99 | this.particle.runParticleFlow()
100 | this.powerStorage()
101 | canvas.addEventListener(mouseupName, mouseup, false)
102 | }
103 |
104 | canvas.addEventListener(mousedownName, mousedown, false)
105 |
106 | return () => {
107 | canvas.removeEventListener(mousedownName, mousedown)
108 | canvas.removeEventListener(mouseupName, mouseup)
109 | }
110 | }
111 |
112 | // 创建身体
113 | createBody () {
114 | const { color, world: { width } } = this
115 | const material = baseMeshLambertMaterial.clone()
116 | material.setValues({ color })
117 |
118 | // 头部
119 | const headSize = this.headSize = width * .03
120 | const headTranslateY = this.headTranslateY = headSize * 4.5
121 | const headGeometry = new THREE.SphereGeometry(headSize, 16, 16)
122 | const headSegment = this.headSegment = new THREE.Mesh(headGeometry, material)
123 | headSegment.castShadow = true
124 | headSegment.translateY(headTranslateY)
125 |
126 | // 身体
127 | this.width = headSize * 1.2 * 2
128 | this.height = headSize * 5
129 | this.bodySize = headSize * 4
130 | const bodyBottomGeometry = new THREE.CylinderBufferGeometry(headSize * .9, this.width / 2, headSize * 2.5, 16)
131 | bodyBottomGeometry.translate(0, headSize * 1.25, 0)
132 | const bodyCenterGeometry = new THREE.CylinderBufferGeometry(headSize, headSize * .9, headSize, 16)
133 | bodyCenterGeometry.translate(0, headSize * 3, 0)
134 | const bodyTopGeometry = new THREE.SphereGeometry(headSize, 16, 16)
135 | bodyTopGeometry.translate(0, headSize * 3.5, 0)
136 |
137 | const bodyGeometry = new THREE.Geometry()
138 | bodyGeometry.merge(bodyTopGeometry)
139 | bodyGeometry.merge(new THREE.Geometry().fromBufferGeometry(bodyCenterGeometry))
140 | bodyGeometry.merge(new THREE.Geometry().fromBufferGeometry(bodyBottomGeometry))
141 |
142 | // 缩放控制
143 | const translateY = this.bodyTranslateY = headSize * 1.5
144 | const bodyScaleSegment = this.bodyScaleSegment = new THREE.Mesh(bodyGeometry, material)
145 | bodyScaleSegment.castShadow = true
146 | bodyScaleSegment.translateY(-translateY)
147 |
148 | // 旋转控制
149 | const bodyRotateSegment = this.bodyRotateSegment = new THREE.Group()
150 | bodyRotateSegment.add(headSegment)
151 | bodyRotateSegment.add(bodyScaleSegment)
152 | bodyRotateSegment.translateY(translateY)
153 |
154 | // 整体身高 = 头部位移 + 头部高度 / 2 = headSize * 5
155 | const body = this.body = new THREE.Group()
156 | body.add(bodyRotateSegment)
157 | }
158 |
159 | // 进入舞台
160 | enterStage (stage, { x, y, z }, nextProp) {
161 | const { body } = this
162 |
163 | body.position.set(x, y, z)
164 |
165 | this.stage = stage
166 | // 进入舞台时告诉小人目标
167 | this.nextProp = nextProp
168 |
169 | stage.add(body)
170 | stage.render()
171 | this.unbindFunc = this.bindEvent()
172 | }
173 |
174 | resetPowerStorageParameter () {
175 | // 由于蓄力导致的变形,需要记录后,在空中将小人复原
176 | this.toValues = {
177 | headTranslateY: 0,
178 | bodyScaleXZ: 0,
179 | bodyScaleY: 0
180 | }
181 | this.fromValues = this.fromValues || {
182 | headTranslateY: this.headTranslateY,
183 | bodyScaleXZ: 1,
184 | bodyScaleY: 1,
185 | propScaleY: 1
186 | }
187 | }
188 |
189 | // 蓄力
190 | powerStorage () {
191 | const {
192 | stage, powerStorageDuration,
193 | body, bodyScaleSegment, headSegment,
194 | bodySize,
195 | fromValues,
196 | currentProp,
197 | world: { propHeight }
198 | } = this
199 |
200 | this.powerStorageTime = Date.now()
201 | this.resetPowerStorageParameter()
202 |
203 | const tween = animate(
204 | {
205 | from: { ...fromValues },
206 | to: {
207 | headTranslateY: bodySize - bodySize * .6,
208 | bodyScaleXZ: 1.3,
209 | bodyScaleY: .6,
210 | propScaleY: .6
211 | },
212 | duration: powerStorageDuration
213 | },
214 | ({ headTranslateY, bodyScaleXZ, bodyScaleY, propScaleY }) => {
215 | if (!this.poweringUp) {
216 | // 抬起时停止蓄力
217 | tween.stop()
218 | } else {
219 |
220 | headSegment.position.setY(headTranslateY)
221 | bodyScaleSegment.scale.set(bodyScaleXZ, bodyScaleY, bodyScaleXZ)
222 | currentProp.scaleY(propScaleY)
223 | body.position.setY(propHeight * propScaleY)
224 |
225 | // 保存此时的位置用于复原
226 | this.toValues = {
227 | headTranslateY,
228 | bodyScaleXZ,
229 | bodyScaleY
230 | }
231 |
232 | stage.render()
233 | }
234 | }
235 | )
236 | }
237 |
238 | computePowerStorageValue () {
239 | const { powerStorageDuration, powerStorageTime, v0, theta } = this
240 | const diffTime = Date.now() - powerStorageTime
241 | const time = Math.min(diffTime, powerStorageDuration)
242 | const percentage = time / powerStorageDuration
243 |
244 | return {
245 | v0: v0 + 30 * percentage,
246 | theta: theta - 40 * percentage
247 | }
248 | }
249 |
250 | // 跳跃
251 | jump () {
252 | const {
253 | stage, body,
254 | currentProp, nextProp,
255 | world: { propHeight }
256 | } = this
257 | const duration = 400
258 | const start = body.position
259 | const target = nextProp.getPosition()
260 | const { x: startX, y: startY, z: startZ } = start
261 |
262 | // 开始游戏时,小人从第一个盒子正上方入场做弹球下落
263 | if (!currentProp && startX === target.x && startZ === target.z) {
264 | animate(
265 | {
266 | from: { y: startY },
267 | to: { y: propHeight },
268 | duration,
269 | easing: TWEEN.Easing.Bounce.Out
270 | },
271 | ({ y }) => {
272 | body.position.setY(y)
273 | stage.render()
274 | },
275 | () => {
276 | this.currentProp = nextProp
277 | this.nextProp = nextProp.getNext()
278 | this.jumping = false
279 | }
280 | )
281 | } else {
282 | if (!currentProp) {
283 | return
284 | }
285 |
286 | const { bodyScaleSegment, headSegment, G, world, width } = this
287 | const { v0, theta } = this.computePowerStorageValue()
288 | const { rangeR, rangeH } = computeObligueThrowValue(v0, theta * (Math.PI / 180), G)
289 | const { jumpDownX, jumpDownZ } = computePositionByRangeR(rangeR, start, target)
290 |
291 | // 水平匀速
292 | animate(
293 | {
294 | from: {
295 | x: startX,
296 | z: startZ,
297 | ...this.toValues
298 | },
299 | to: {
300 | x: jumpDownX,
301 | z: jumpDownZ,
302 | ...this.fromValues
303 | },
304 | duration
305 | },
306 | ({ x, z, headTranslateY, bodyScaleXZ, bodyScaleY }) => {
307 | body.position.setX(x)
308 | body.position.setZ(z)
309 | headSegment.position.setY(headTranslateY)
310 | bodyScaleSegment.scale.set(bodyScaleXZ, bodyScaleY, bodyScaleXZ)
311 | }
312 | )
313 |
314 | // y轴上升段、下降段
315 | const rangeHeight = Math.max(world.width / 3, rangeH) + propHeight
316 | const yUp = animate(
317 | {
318 | from: { y: startY },
319 | to: { y: rangeHeight },
320 | duration: duration * .65,
321 | easing: TWEEN.Easing.Cubic.Out,
322 | autoStart: false
323 | },
324 | ({ y }) => {
325 | body.position.setY(y)
326 | }
327 | )
328 | const yDown = animate(
329 | {
330 | from: { y: rangeHeight },
331 | to: { y: propHeight },
332 | duration: duration * .35,
333 | easing: TWEEN.Easing.Cubic.In,
334 | autoStart: false
335 | },
336 | ({ y }) => {
337 | body.position.setY(y)
338 | },
339 | () => yDownCallBack()
340 | )
341 |
342 | yUp.chain(yDown).start()
343 |
344 | // 空翻
345 | this.flip(duration)
346 | // 从起跳开始就回弹
347 | currentProp.springbackTransition(500)
348 |
349 | // 落地后的回调
350 | const yDownCallBack = () => {
351 | const currentInfos = currentProp.computePointInfos(width, jumpDownX, jumpDownZ)
352 | const nextInfos = nextProp.computePointInfos(width, jumpDownX, jumpDownZ)
353 |
354 | // 没有落在任何一个盒子上方
355 | if (!currentInfos.contains && !nextInfos.contains) {
356 | // gameOver 游戏结束,跌落
357 | console.log('GameOver')
358 | this.fall(currentInfos, nextInfos)
359 | } else {
360 | bufferUp.onComplete(() => {
361 | if (nextInfos.contains) {
362 | // 落在下一个盒子才更新场景
363 | // 落地后,生成下一个方块 -> 移动镜头 -> 更新关心的盒子 -> 结束
364 | world.createProp()
365 | world.moveCamera()
366 |
367 | this.currentProp = nextProp
368 | this.nextProp = nextProp.getNext()
369 | }
370 |
371 | // 粒子喷泉
372 | this.particle.runParticleFountain()
373 | // 跳跃结束了
374 | this.jumping = false
375 | }).start()
376 | }
377 | }
378 |
379 | // 落地缓冲段
380 | const bufferUp = animate(
381 | {
382 | from: { s: .8 },
383 | to: { s: 1 },
384 | duration: 100,
385 | autoStart: false
386 | },
387 | ({ s }) => {
388 | bodyScaleSegment.scale.setY(s)
389 | }
390 | )
391 | }
392 | }
393 |
394 | // 空翻
395 | flip (duration) {
396 | const { currentProp, bodyRotateSegment } = this
397 | let increment = 0
398 |
399 | animate(
400 | {
401 | from: { deg: 0 },
402 | to: { deg: 360 },
403 | duration,
404 | easing: TWEEN.Easing.Sinusoidal.InOut
405 | },
406 | ({ deg }) => {
407 | if (currentProp.nextDirection === 'x') {
408 | bodyRotateSegment.rotateZ(-(deg - increment) * (Math.PI/180))
409 | } else {
410 | bodyRotateSegment.rotateX((deg - increment) * (Math.PI/180))
411 | }
412 | increment = deg
413 | }
414 | )
415 | }
416 |
417 | // 跌落
418 | fall (currentInfos, nextInfos) {
419 | const {
420 | stage, body, currentProp, nextProp,
421 | world: { propHeight }
422 | } = this
423 | // 跳跃方向
424 | const direction = currentProp.nextDirection
425 | let degY, translateZ,
426 | validateProp, // 需要检测的盒子
427 | isForward // 相对方向的前、后
428 |
429 | if (currentInfos.isEdge && nextInfos.isEdge) {
430 | // 同时在2个盒子边缘
431 | return
432 | } else if (currentInfos.isEdge) {
433 | // 当前盒子边缘
434 | degY = currentInfos.degY
435 | translateZ = currentInfos.translateZ
436 | validateProp = nextProp
437 | isForward = true
438 | } else if (nextInfos.isEdge) {
439 | // 目标盒子边缘
440 | degY = nextInfos.degY
441 | translateZ = nextInfos.translateZ
442 | // 目标盒子边缘可能是在盒子前方或盒子后方
443 | if (direction === 'z') {
444 | isForward = degY < 90 && degY > 270
445 | } else {
446 | isForward = degY < 180
447 | }
448 | validateProp = isForward ? null : currentProp
449 | } else {
450 | // 空中掉落
451 | return animate(
452 | {
453 | from: { y: propHeight },
454 | to: { y: 0 },
455 | duration: 400,
456 | easing: TWEEN.Easing.Bounce.Out
457 | },
458 | ({ y }) => {
459 | body.position.setY(y)
460 | stage.render()
461 | }
462 | )
463 | }
464 |
465 | // 将粒子销毁掉
466 | this.particle.destroy()
467 |
468 | const {
469 | bodyRotateSegment, bodyScaleSegment,
470 | headSegment, bodyTranslateY,
471 | width, height
472 | } = this
473 | const halfWidth = width / 2
474 |
475 | // 将旋转原点放在脚下,同时让小人面向跌落方向
476 | headSegment.translateY(bodyTranslateY)
477 | bodyScaleSegment.translateY(bodyTranslateY)
478 | bodyRotateSegment.translateY(-bodyTranslateY)
479 | bodyRotateSegment.rotateY(degY * (Math.PI / 180))
480 |
481 | // 将旋转原点移动到支撑点
482 | headSegment.translateZ(translateZ)
483 | bodyScaleSegment.translateZ(translateZ)
484 | bodyRotateSegment.translateZ(-translateZ)
485 |
486 | let incrementZ = 0
487 | let incrementDeg = 0
488 | let incrementY = 0
489 |
490 | let hitValidator = validateProp && getHitValidator(validateProp.body, direction, isForward)
491 |
492 | // 第一段 先沿着支点旋转
493 | const rotate = animate(
494 | {
495 | from: {
496 | degY: 0
497 | },
498 | to: {
499 | degY: 90
500 | },
501 | duration: 500,
502 | autoStart: false,
503 | easing: TWEEN.Easing.Quintic.In
504 | },
505 | ({ degY }) => {
506 | if (hitValidator && hitValidator(body.children[0])) {
507 | rotate.stop()
508 | hitValidator = null
509 | } else {
510 | bodyRotateSegment.rotateX((degY - incrementDeg) * (Math.PI / 180))
511 | incrementDeg = degY
512 | stage.render()
513 | }
514 | }
515 | )
516 | // 第二段 跌落,沿z轴下落,沿y轴向外侧偏移
517 | const targZ = propHeight - halfWidth - translateZ
518 | const fall = animate(
519 | {
520 | from: {
521 | y: 0,
522 | z: 0
523 | },
524 | to: {
525 | y: halfWidth - translateZ,
526 | z: targZ,
527 | },
528 | duration: 300,
529 | autoStart: false,
530 | easing: TWEEN.Easing.Bounce.Out
531 | },
532 | ({ z, y }) => {
533 | if (hitValidator && hitValidator(body.children[0])) {
534 | fall.stop()
535 |
536 | // 稍微处理一下,头撞到盒子的情况
537 | const radian = Math.atan((targZ - z) / height)
538 | if (isForward && direction === 'z') {
539 | bodyRotateSegment.translateY(-height)
540 | body.position.z += height
541 | body.rotateX(-radian)
542 | } else if (direction === 'z') {
543 | bodyRotateSegment.translateY(-height)
544 | body.position.z -= height
545 | body.rotateX(radian)
546 | } else if (isForward && direction === 'x') {
547 | bodyRotateSegment.translateY(-height)
548 | body.position.x += height
549 | body.rotateZ(radian)
550 | } else if (direction === 'x') {
551 | bodyRotateSegment.translateY(-height)
552 | body.position.x -= height
553 | body.rotateZ(-radian)
554 | }
555 | stage.render()
556 | hitValidator = null
557 | } else {
558 | headSegment.translateZ(z - incrementZ)
559 | bodyScaleSegment.translateZ(z - incrementZ)
560 | bodyRotateSegment.translateY(y - incrementY)
561 | incrementZ = z
562 | incrementY = y
563 | stage.render()
564 | }
565 | }
566 | )
567 |
568 | rotate.chain(fall).start()
569 | }
570 | }
571 |
572 | export default LittleMan
--------------------------------------------------------------------------------
/examples/js/wx-jump-stage2.87b8f792.js:
--------------------------------------------------------------------------------
1 | (window["webpackJsonp"]=window["webpackJsonp"]||[]).push([["wx-jump-stage2"],{"02f4":function(t,e,i){var n=i("4588"),r=i("be13");t.exports=function(t){return function(e,i){var a,o,s=String(r(e)),h=n(i),c=s.length;return h<0||h>=c?t?"":void 0:(a=s.charCodeAt(h),a<55296||a>56319||h+1===c||(o=s.charCodeAt(h+1))<56320||o>57343?t?s.charAt(h):a:t?s.slice(h,h+2):o-56320+(a-55296<<10)+65536)}}},"1c4c":function(t,e,i){"use strict";var n=i("9b43"),r=i("5ca1"),a=i("4bf8"),o=i("1fa8"),s=i("33a4"),h=i("9def"),c=i("f1ae"),l=i("27ee");r(r.S+r.F*!i("5cc5")((function(t){Array.from(t)})),"Array",{from:function(t){var e,i,r,u,p=a(t),g="function"==typeof this?this:Array,d=arguments.length,f=d>1?arguments[1]:void 0,v=void 0!==f,y=0,m=l(p);if(v&&(f=n(f,d>2?arguments[2]:void 0,2)),void 0==m||g==Array&&s(m))for(e=h(p.length),i=new g(e);e>y;y++)c(i,y,v?f(p[y],y):p[y]);else for(u=m.call(p),i=new g;!(r=u.next()).done;y++)c(i,y,v?o(u,f,[r.value,y],!0):r.value);return i.length=y,i}})},"386b":function(t,e,i){var n=i("5ca1"),r=i("79e5"),a=i("be13"),o=/"/g,s=function(t,e,i,n){var r=String(a(t)),s="<"+e;return""!==i&&(s+=" "+i+'="'+String(n).replace(o,""")+'"'),s+">"+r+""+e+">"};t.exports=function(t,e){var i={};i[t]=e(s),n(n.P+n.F*r((function(){var e=""[t]('"');return e!==e.toLowerCase()||e.split('"').length>3})),"String",i)}},"5df3":function(t,e,i){"use strict";var n=i("02f4")(!0);i("01f9")(String,"String",(function(t){this._t=String(t),this._i=0}),(function(){var t,e=this._t,i=this._i;return i>=e.length?{value:void 0,done:!0}:(t=n(e,i),this._i+=t.length,{value:t,done:!1})}))},"673e":function(t,e,i){"use strict";i("386b")("sub",(function(t){return function(){return t(this,"sub","","")}}))},8448:function(t,e,i){},"8fad":function(t,e,i){"use strict";var n=i("8448"),r=i.n(n);r.a},cfd0:function(t,e,i){"use strict";i.r(e);var n=function(){var t=this,e=t.$createElement;t._self._c;return t._m(0)},r=[function(){var t=this,e=t.$createElement,i=t._self._c||e;return i("div",{staticClass:"jump-world"},[i("canvas",{attrs:{id:"jump-world-canvas"}})])}],a=i("768b"),o=(i("ac6a"),i("d225")),s=i("b0b4"),h=(i("673e"),i("5df3"),i("1c4c"),i("5a89")),c=i("22b5"),l=Math.random,u=Math.sqrt,p=Math.floor,g=Math.pow,d=Math.sin,f=Math.cos,v=Math.tan,y=Math.PI,m=0,w=function(){return m++},b=function(){return m=0},S=[6799930,15114812,16084076,9474969,4235007,16777215];function x(t,e,i,n,r,a){var o=t*(y/180),s=e*(y/180),h=f(o)*n,c=d(o)*(a-r-i/v(o));if(!(h>c)){var l=h+(c-h)/2,u=l/v(o),p=d(s)*u,g=f(s)*u;return{x:p,y:l,z:g}}console.warn("警告: 垂直角度太小了!")}var P=new h["MeshLambertMaterial"],A=new h["BoxBufferGeometry"](1,1,1,10,4,10);A.userData.type="box";var z=new h["CylinderBufferGeometry"](1,1,1,30,5);z.userData.type="Cylinder";var M=function(t){t.geometry&&(t.geometry.dispose(),t.geometry=null),t.material&&(t.material.dispose(),t.material=null),t.parent.remove(t),t.parent=null,t=null},j=function(t){return t[p(l()*t.length)]},Y=function(t,e){return p(l()*(e-t+1))+t},C=function t(e){var i=t.box3||(t.box3=new h["Box3"]);return i.setFromObject(e),i.getSize(new h["Vector3"])},R=function(t,e,i){var n=d(2*e),r=d(e),a=g(t,2)*n/i,o=g(t*r,2)/(2*i);return{rangeR:a,rangeH:o}},O=function(t,e,i){var n=t.position.clone(),r=t.geometry.attributes.position,a=r.count,o=Array.from({length:a}).map((function(t,e){return(new h["Vector3"]).fromBufferAttribute(r,e)})).filter((function(t){return"z"===e&&i?t.z<0:"z"===e?t.z>0:"x"===e&&i?t.x<0:"x"===e?t.x>0:void 0})).map((function(e){var i=e.applyMatrix4(t.matrix);return i.sub(t.position)}));return function(t){for(var e,i=0;e=o[i];i++){var r=new h["Raycaster"](n,e.clone().normalize()),a=r.intersectObject(t,!0);if(a.length>0&&a[0].distance-1&&e[t]&&e[t]()||j(e)()}},{key:"createPropCreator",value:function(t,e){var i=this;if(Array.isArray(t))t.forEach((function(t){return i.createPropCreator(t,e)}));else{var n=this.propCreators,r=this.propSizeRange,a=this.propHeight;if(!(n.indexOf(t)>-1)){var o=function i(){if(e&&i.box)return i.box.clone();var n=t(h,{propSizeRange:r,propHeight:a,baseMeshLambertMaterial:P,baseBoxBufferGeometry:A,baseCylinderBufferGeometry:z});return e&&(i.box=n),n};n.push(o)}}}}]),t}(),T=L,N=i("75fc"),U=function(){function t(e){var i=e.world,n=e.stage,r=e.body,a=e.height,s=e.enterHeight,h=e.distanceRange,c=e.prev;Object(o["a"])(this,t),this.world=i,this.stage=n,this.body=r,this.height=a,this.enterHeight=s,this.distanceRange=h,this.prev=c,this.nextDirection="",this.nextDistance=0}return Object(s["a"])(t,[{key:"computeMyPosition",value:function(){var t=this.world,e=this.prev,i=this.distanceRange,n=this.enterHeight,r={x:0,y:n,z:0};if(!e)return r;if(0===n)return r.z=t.width/2,e.nextDirection="z",r;var a=e.getPosition(),o=a.x,s=a.z,h=0===Math.round(Math.random()),c=e.getSize(),l=c.x,u=c.z,p=this.getSize(),g=p.x,d=p.z,f=Y.apply(void 0,Object(N["a"])(i));return h?(r.x=o+l/2+f+g/2,r.z=s,e.nextDirection="x"):(r.x=o,r.z=s+u/2+f+d/2,e.nextDirection="z"),e.nextDistance=f,r}},{key:"enterStage",value:function(){var t=this.stage,e=this.body,i=this.height,n=this.computeMyPosition(),r=n.x,a=n.y,o=n.z;e.castShadow=!0,e.receiveShadow=!0,e.position.set(r,a,o),e.geometry.translate(0,i/2,0),t.add(e),this.entranceTransition()}},{key:"entranceTransition",value:function(){var t=arguments.length>0&&void 0!==arguments[0]?arguments[0]:400,e=this.body,i=this.enterHeight,n=this.stage;0!==i&&G({to:{y:0},from:{y:i},duration:t,easing:c["a"].Easing.Bounce.Out},(function(t){var i=t.y;e.position.setY(i),n.render()}))}},{key:"springbackTransition",value:function(t){var e=this.body,i=this.stage,n=e.scale.y;G({from:{y:n},to:{y:1},duration:t,easing:c["a"].Easing.Bounce.Out},(function(t){var n=t.y;e.scale.setY(n),i.render()}))}},{key:"computePointInfos",value:function(t,e,i){var n=this.body;if(!n)return{};var r=n.geometry.userData.type,a=this.getSize(),o=a.x,s=a.z,h=this.getPosition(),c=h.x,l=h.z,u=t/2;if("box"===r){var p=o/2,g=s/2,d=c-p,f=c+p,v=l-g,y=l+g,m=e>=d&&e<=f&&i>=v&&i<=y;if(m)return{contains:m};var w=Math.abs(i-l)-g,b=Math.abs(e-c)-p;if(w>=u||b>=u)return{contains:m};var S=!1,x=0,P=0;return ed?(i>y&&iv-u&&(x=180),S=!0,P=w):iv&&(e>f&&ed-u&&(x=270),S=!0,P=b),{contains:m,translateZ:P,isEdge:S,degY:x}}var A=o/2,z=Math.sqrt(Math.pow(c-e,2)+Math.pow(l-i,2)),M=z<=A;if(M)return{contains:M};if(z>=A+u)return{contains:M};var j=!0,Y=z-A,C=180*Math.atan(Math.abs(e-c)/Math.abs(i-l))/Math.PI;return e===c?C=i>l?0:180:i===l?C=e>c?90:270:e>c&&i>l||(C=e>c&&i=c)e.x=0,e.y=a,e.z=0;else{var u=h-n,g=c-r,d=l-o,f=Math.sqrt(Math.pow(u,2)+Math.pow(g,2)+Math.pow(d,2)),v=p/f;e.y+=v*g,e.x+=v*u,e.z+=v*d}})),o.stage.render()}s=Date.now()}else i.runingParticleFountain=!1,cancelAnimationFrame(r)};l()}},{key:"runParticleFountain",value:function(){if(!this.runingParticleFountain){this.runingParticleFountain=!0;var t=this.particleSystem,e=this.quantity,i=this.initalY,n=t.children.filter((function(t){return t.position.y===i})).slice(0,e);if(n.length){var r=this.triggerObjectWidth,a=this.fountainRangeX,o=this.fountainSizeRange,s=this.fountainRangeY,h=r/2;n.forEach((function(t){var e=t.position,i=t.material,n=Y.apply(void 0,Object(N["a"])(a)),r=Y.apply(void 0,Object(N["a"])(a)),c=n<0?-h:h,l=r<0?-h:h;e.x=c+n,e.z=l+r,e.y=Y.apply(void 0,Object(N["a"])(s)),i.setValues({size:Y.apply(void 0,Object(N["a"])(o))})})),this.runParticleFountainPump(n,1e3)}}}}]),t}(),J=X;function W(t,e){var i=Object.keys(t);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(t);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(t,e).enumerable}))),i.push.apply(i,n)}return i}function q(t){for(var e=1;e270:i<180,r=a?null:h}this.particle.destroy();var g=this.bodyRotateSegment,d=this.bodyScaleSegment,f=this.headSegment,v=this.bodyTranslateY,y=this.width,m=this.height,w=y/2;f.translateY(v),d.translateY(v),g.translateY(-v),g.rotateY(i*(Math.PI/180)),f.translateZ(n),d.translateZ(n),g.translateZ(-n);var b=0,S=0,x=0,P=r&&O(r.body,p,a),A=G({from:{degY:0},to:{degY:90},duration:500,autoStart:!1,easing:c["a"].Easing.Quintic.In},(function(t){var e=t.degY;P&&P(s.children[0])?(A.stop(),P=null):(g.rotateX((e-S)*(Math.PI/180)),S=e,o.render())})),z=u-w-n,M=G({from:{y:0,z:0},to:{y:w-n,z:z},duration:300,autoStart:!1,easing:c["a"].Easing.Bounce.Out},(function(t){var e=t.z,i=t.y;if(P&&P(s.children[0])){M.stop();var n=Math.atan((z-e)/m);a&&"z"===p?(g.translateY(-m),s.position.z+=m,s.rotateX(-n)):"z"===p?(g.translateY(-m),s.position.z-=m,s.rotateX(n)):a&&"x"===p?(g.translateY(-m),s.position.x+=m,s.rotateZ(n)):"x"===p&&(g.translateY(-m),s.position.x-=m,s.rotateZ(-n)),o.render(),P=null}else f.translateZ(e-b),d.translateZ(e-b),g.translateY(i-x),b=e,x=i,o.render()}));A.chain(M).start()}}}]),t}(),_=K,$=function(){function t(e){var i=e.container,n=e.canvas,r=e.needDefaultCreator,a=void 0===r||r,s=e.axesHelper,h=void 0!==s&&s;Object(o["a"])(this,t);var c=i.offsetWidth,l=i.offsetHeight;this.G=9.8,this.container=i,this.canvas=n,this.width=c,this.height=l,this.needDefaultCreator=a,this.axesHelper=h;var u=~~(c/6),p=~~(c/3.5);this.propSizeRange=[u,p],this.propHeight=~~(p/2),this.propDistanceRange=[~~(u/2),2*p],this.stage=null,this.propCreator=null,this.littleMans=[],this.props=[],this.init()}return Object(s["a"])(t,[{key:"init",value:function(){this.initStage(),this.initPropCreator(),this.computeSafeClearLength(),this.createProp(0),this.createProp(0),this.moveCamera(0),this.initLittleMan()}},{key:"destroy",value:function(){this.clear()}},{key:"clear",value:function(){this.stage&&this.stage.reset(),this.props.forEach((function(t){return t.destroy()})),this.props=[],b(),this.littleMans.forEach((function(t){return t.destroy()})),this.littleMans=[]}},{key:"initStage",value:function(){var t=this.canvas,e=this.axesHelper,i=this.width,n=this.height,r=.1,a=2e3,o=this.cameraInitalPosition=x(35,225,n/2,n/2,r,a),s=this.lightInitalPosition={x:-300,y:600,z:200};this.stage=new F({width:i,height:n,canvas:t,axesHelper:e,cameraNear:r,cameraFar:a,cameraInitalPosition:o,lightInitalPosition:s})}},{key:"initPropCreator",value:function(){var t=this.needDefaultCreator,e=this.propSizeRange,i=this.propHeight;this.propCreator=new T({propHeight:i,propSizeRange:e,needDefaultCreator:t})}},{key:"createPropCreator",value:function(){var t;(t=this.propCreator).createPropCreator.apply(t,arguments)}},{key:"initLittleMan",value:function(){var t=this.stage,e=this.propHeight,i=this.G,n=this.props,r=new _({world:this,color:3696793,G:i});r.enterStage(t,{x:0,y:e+80,z:0},n[0]),r.jump(),this.littleMans.push(r)}},{key:"createProp",value:function(){var t=arguments.length>0&&void 0!==arguments[0]?arguments[0]:100,e=arguments.length>1&&void 0!==arguments[1]?arguments[1]:-1,i=this.propCreator,n=this.propHeight,r=Object(a["a"])(this.propSizeRange,2),o=r[0],s=r[1],h=this.propDistanceRange,c=this.stage,l=this.props,u=this.props.length,p=l[u-1],g=new V({world:this,stage:c,body:i.createProp(-1===e&&u<3?0:e),height:n,prev:p,enterHeight:t,distanceRange:h}),d=g.getSize();return d.y!==n&&console.warn("高度: ".concat(d.y,",盒子高度必须为 ").concat(n)),(d.xs)&&console.warn("宽度: ".concat(d.x,", 盒子宽度必须为 ").concat(o," - ").concat(s)),(d.zs)&&console.warn("深度: ".concat(d.z,", 盒子深度度必须为 ").concat(o," - ").concat(s)),p&&p.setNext(g),g.enterStage(),l.push(g),g}},{key:"moveCamera",value:function(){var t=this,e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:500,i=this.stage,n=this.height,r=this.cameraInitalPosition,a=r.x,o=r.y,s=r.z,h=this.lightInitalPosition,c=h.x,l=h.y,u=h.z,p=n/10,g=this.getLastTwoCenterPosition(),d=g.x,f=g.y,v=g.z,y={x:d+a+p,y:o,z:v+s+p},m={x:d+c,y:l,z:v+u},w={cameraTo:y,lightTo:m,center:{x:d,y:f,z:v}};i.moveCamera(w,(function(){t.clearProps()}),e)}},{key:"getLastTwoCenterPosition",value:function(){var t=this.props,e=this.props.length,i=t[e-2].getPosition(),n=i.x,r=i.z,a=t[e-1].getPosition(),o=a.x,s=a.z;return{x:n+(o-n)/2,z:r+(s-r)/2}}},{key:"clearProps",value:function(){var t=this.safeClearLength,e=this.props,i=this.props.length,n=4;i>t&&(e.slice(0,n).forEach((function(t){return t.destroy()})),this.props=e.slice(n))}},{key:"computeSafeClearLength",value:function(){var t=this.width,e=this.height,i=this.propSizeRange,n=i[0],r=Math.sqrt(n*n+n*n);this.safeClearLength=Math.ceil(t/n)+Math.ceil(e/r/2)+1}}]),t}(),tt=$,et={mounted:function(){new tt({container:document.querySelector(".jump-world"),canvas:document.querySelector("#jump-world-canvas"),axesHelper:!0})}},it=et,nt=(i("8fad"),i("2877")),rt=Object(nt["a"])(it,n,r,!1,null,"9a0b2f7a",null);e["default"]=rt.exports},e8df:function(t,e){t.exports=""}}]);
--------------------------------------------------------------------------------