├── .babelrc
├── .eslintignore
├── .eslintrc.js
├── .github
└── workflows
│ ├── test-phaser-3.yml
│ └── test-phaser-ce.yml
├── .gitignore
├── LICENSE
├── README.md
├── assets
├── diagram.png
├── diagram_column.png
├── diagram_quantitybar.png
├── diagram_row.png
├── diagram_valuebar.png
└── diagram_wheel3D.png
├── docs
├── assets
│ ├── anchor.js
│ ├── bass-addons.css
│ ├── bass.css
│ ├── fonts
│ │ ├── EOT
│ │ │ ├── SourceCodePro-Bold.eot
│ │ │ └── SourceCodePro-Regular.eot
│ │ ├── LICENSE.txt
│ │ ├── OTF
│ │ │ ├── SourceCodePro-Bold.otf
│ │ │ └── SourceCodePro-Regular.otf
│ │ ├── TTF
│ │ │ ├── SourceCodePro-Bold.ttf
│ │ │ └── SourceCodePro-Regular.ttf
│ │ ├── WOFF
│ │ │ ├── OTF
│ │ │ │ ├── SourceCodePro-Bold.otf.woff
│ │ │ │ └── SourceCodePro-Regular.otf.woff
│ │ │ └── TTF
│ │ │ │ ├── SourceCodePro-Bold.ttf.woff
│ │ │ │ └── SourceCodePro-Regular.ttf.woff
│ │ ├── WOFF2
│ │ │ ├── OTF
│ │ │ │ ├── SourceCodePro-Bold.otf.woff2
│ │ │ │ └── SourceCodePro-Regular.otf.woff2
│ │ │ └── TTF
│ │ │ │ ├── SourceCodePro-Bold.ttf.woff2
│ │ │ │ └── SourceCodePro-Regular.ttf.woff2
│ │ └── source-code-pro.css
│ ├── github.css
│ ├── site.js
│ ├── split.css
│ ├── split.js
│ └── style.css
├── examples
│ ├── assets
│ │ ├── horizontal
│ │ │ ├── bar.png
│ │ │ ├── sprite.png
│ │ │ └── track.png
│ │ ├── modal
│ │ │ ├── background.png
│ │ │ └── button.png
│ │ ├── quantitybar
│ │ │ ├── add.png
│ │ │ ├── horizontal
│ │ │ │ ├── bar.png
│ │ │ │ └── track.png
│ │ │ ├── subtract.png
│ │ │ └── vertical
│ │ │ │ ├── bar.png
│ │ │ │ └── track.png
│ │ ├── textover
│ │ │ ├── button.png
│ │ │ └── header.png
│ │ ├── valuebar
│ │ │ ├── alpha.png
│ │ │ ├── background.png
│ │ │ ├── bar.png
│ │ │ ├── pointer.png
│ │ │ ├── track.png
│ │ │ └── vtrack.png
│ │ ├── vertical
│ │ │ ├── bar.png
│ │ │ ├── sprite.png
│ │ │ └── track.png
│ │ └── wheel3D
│ │ │ ├── icon.png
│ │ │ ├── mmDown.png
│ │ │ └── mmUp.png
│ ├── html
│ │ ├── phaser3
│ │ │ ├── hscrollbar.html
│ │ │ ├── modal.html
│ │ │ ├── quantitybar.html
│ │ │ ├── textover.html
│ │ │ ├── valuebar.html
│ │ │ ├── vscrollbar.html
│ │ │ └── wheel3D.html
│ │ └── phaserce
│ │ │ ├── hscrollbar.html
│ │ │ ├── modal.html
│ │ │ ├── quantitybar.html
│ │ │ ├── quantitybar_column.html
│ │ │ ├── textover.html
│ │ │ ├── valuebar.html
│ │ │ ├── valuebar_column.html
│ │ │ ├── vscrollbar.html
│ │ │ └── wheel3D.html
│ ├── js
│ │ ├── phaser3
│ │ │ ├── hscrollbar.js
│ │ │ ├── modal.js
│ │ │ ├── quantitybar.js
│ │ │ ├── textover.js
│ │ │ ├── valuebar.js
│ │ │ ├── vscrollbar.js
│ │ │ └── wheel3D.js
│ │ └── phaserce
│ │ │ ├── hscrollbar.js
│ │ │ ├── modal.js
│ │ │ ├── quantitybar.js
│ │ │ ├── quantitybar_column.js
│ │ │ ├── textover.js
│ │ │ ├── valuebar.js
│ │ │ ├── valuebar_column.js
│ │ │ ├── vscrollbar.js
│ │ │ └── wheel3D.js
│ └── lib
│ │ ├── phaser-ui-tools.js
│ │ ├── phaser.min.js
│ │ └── phaser3.min.js
└── index.html
├── package-lock.json
├── package.json
├── src
├── bars
│ ├── bar.js
│ ├── draggable_bar.js
│ ├── quantitybar.js
│ ├── ranges.js
│ ├── scrollbar.js
│ └── valuebar.js
├── buttons.js
├── const.js
├── containers
│ ├── column.js
│ ├── frame.js
│ ├── row.js
│ └── viewport.js
├── entry.js
├── keyboard_group.js
├── phaserObjects
│ ├── button.js
│ ├── easing.js
│ ├── graphics.js
│ ├── group.js
│ ├── index.js
│ ├── keyboard.js
│ ├── keycodes.js
│ ├── rectangle.js
│ ├── sprite.js
│ ├── tween.js
│ └── viewport_mask.js
├── utils.js
└── wheel3D.js
├── tests
├── column.spec.js
├── frame.spec.js
├── row.spec.js
├── testrunner.html
├── testrunner_phaser3.html
├── viewport.spec.js
└── wheel3D.spec.js
└── webpack.config.js
/.babelrc:
--------------------------------------------------------------------------------
1 | {
2 | "presets": [
3 | ["@babel/preset-env", {
4 | "modules": false
5 | }]
6 | ]
7 | }
8 |
--------------------------------------------------------------------------------
/.eslintignore:
--------------------------------------------------------------------------------
1 | /dist
2 | /docs
3 | /node_modules
4 | /tests
5 |
--------------------------------------------------------------------------------
/.eslintrc.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | "extends": "airbnb-base",
3 | "rules": {
4 | "indent": ["error", 4],
5 | "linebreak-style": 0,
6 | "no-plusplus": ["error", {"allowForLoopAfterthoughts": true}],
7 | "import/prefer-default-export": "off",
8 | "max-len": ["error", {"ignoreComments": true}],
9 | "object-curly-newline": ["error", {"ObjectPattern": {"multiline": true}}],
10 | "no-lonely-if": "off",
11 | "max-len": "off",
12 | "max-classes-per-file": "off",
13 | },
14 | "parserOptions": {
15 | "sourceType": "module",
16 | },
17 | "globals": {
18 | "Phaser": true,
19 | }
20 | };
21 |
--------------------------------------------------------------------------------
/.github/workflows/test-phaser-3.yml:
--------------------------------------------------------------------------------
1 | ---
2 |
3 | name: test-phaser-3
4 |
5 | on:
6 | push:
7 | branches:
8 | - master
9 | pull_request:
10 | branches:
11 | - master
12 |
13 | jobs:
14 | build:
15 | runs-on: ubuntu-latest
16 | name: unit-tests
17 |
18 | strategy:
19 | matrix:
20 | node-version: [14.x]
21 |
22 | steps:
23 | - name: Checkout Repository
24 | uses: actions/checkout@v2
25 | with:
26 | fetch-depth: 1
27 |
28 | - name: Setup Node ${{ matrix.node-version }}
29 | uses: actions/setup-node@v2
30 | with:
31 | always-auth: false
32 | node-version: ${{ matrix.node-version }}
33 |
34 | - name: Run npm install
35 | run: npm install
36 |
37 | - name: Build js
38 | run: npm run build:js
39 |
40 | - name: Run tests
41 | run: npm run test3
42 | env:
43 | CI: true
44 |
--------------------------------------------------------------------------------
/.github/workflows/test-phaser-ce.yml:
--------------------------------------------------------------------------------
1 | ---
2 |
3 | name: test-phaser-ce
4 |
5 | on:
6 | push:
7 | branches:
8 | - master
9 | pull_request:
10 | branches:
11 | - master
12 |
13 | jobs:
14 | build:
15 | runs-on: ubuntu-latest
16 | name: unit-tests
17 |
18 | strategy:
19 | matrix:
20 | node-version: [14.x]
21 |
22 | steps:
23 | - name: Checkout Repository
24 | uses: actions/checkout@v2
25 | with:
26 | fetch-depth: 1
27 |
28 | - name: Setup Node ${{ matrix.node-version }}
29 | uses: actions/setup-node@v2
30 | with:
31 | always-auth: false
32 | node-version: ${{ matrix.node-version }}
33 |
34 | - name: Run npm install
35 | run: npm install
36 |
37 | - name: Build js
38 | run: npm run build:js
39 |
40 | - name: Run tests
41 | run: npm run test
42 | env:
43 | CI: true
44 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | dist/*
2 | node_modules/
3 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2017 Joshua Fehler
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 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Phaser UI Tools
2 |
3 | [](https://github.com/jsfehler/phaser-ui-tools/actions/workflows/test-phaser-3.yml)
4 |
5 | [](https://github.com/jsfehler/phaser-ui-tools/actions/workflows/test-phaser-ce.yml)
6 |
7 |
8 | I really wanted a viewport with a scrollbar. Things escalated.
9 |
10 | 
11 |
12 | ### Documentation
13 | https://jsfehler.github.io/phaser-ui-tools/
14 |
15 | ### References
16 | Scrollbar math:
17 | http://csdgn.org/article/scrollbar
18 |
19 |
20 | ### Getting Started
21 |
22 | #### Using NPM
23 | On the command line, type:
24 | ```
25 | npm i phaser-ui-tools
26 | ```
27 |
28 | The objects are now available via import:
29 | ```
30 | import { Column } from 'phaser-ui-tools';
31 |
32 | var column = new Column(...)
33 |
34 | ```
35 |
36 | #### Adding the file directly to your project
37 | Get phaser-ui-tools.js from the releases and add it to your project's index.html.
38 | It should look something like:
39 | ```
40 |
41 | ```
42 |
43 | The objects can now be used directly:
44 |
45 | ```
46 | var column = new uiWidgets.Column(...)
47 | ```
48 |
49 |
50 | ### The Tools
51 |
52 | #### Text Overlays
53 |
54 | ##### TextSprite
55 | A sprite that can have text on top.
56 |
57 | Text is added with the setText() method.
58 |
59 | ```javascript
60 | var textSprite = new uiWidgets.TextSprite(
61 | game, x, y, key,
62 | );
63 | textSprite.setText(label, style);
64 | ```
65 |
66 | ##### TextButton
67 | A button that can have text on top.
68 |
69 | Text is added with the setText() method.
70 |
71 | ```javascript
72 | var textButton = new uiWidgets.TextButton(
73 | game, x, y, key, callback, callbackContext, overKey, outKey, downKey, upKey,
74 | );
75 | textButton.setText(label, style);
76 | ```
77 |
78 | ##### Examples
79 |
80 | ###### Phaser CE
81 | [Header & Buttons](https://jsfehler.github.io/phaser-ui-tools/examples/html/phaserce/textover.html) | [Code](https://jsfehler.github.io/phaser-ui-tools/examples/js/phaserce/textover.js)
82 |
83 | ###### Phaser 3
84 | [Header & Buttons](https://jsfehler.github.io/phaser-ui-tools/examples/html/phaser3/textover.html) | [Code](https://jsfehler.github.io/phaser-ui-tools/examples/js/phaser3/textover.js)
85 |
86 | #### Containers
87 |
88 | ##### Column
89 |
90 | Columns are Phaser Groups where each child added to the group is placed directly under the previous child. If an object can be a child of a Group, it can likewise be in a Column.
91 |
92 | 
93 | ```javascript
94 | var column = new uiWidgets.Column(game, 8, 8);
95 | column.addNode(sprite_a, 8, 8);
96 | column.addNode(sprite_b, 8, 8);
97 | column.addNode(sprite_c, 8, 8);
98 | ```
99 |
100 | ##### Row
101 |
102 | Rows are Phaser Groups where each child added to the group is placed directly next to the previous child. If an object can be a child of a Group, it can likewise be in a Row.
103 |
104 | 
105 | ```javascript
106 | var row = new uiWidgets.Row(game, 8, 8);
107 | row.addNode(sprite_a, 8, 8);
108 | row.addNode(sprite_b, 8, 8);
109 | row.addNode(sprite_c, 8, 8);
110 | ```
111 |
112 | ##### Viewport
113 | Viewports are Phaser Groups where the children in the group are only visible if they're within the viewport's area.
114 | If an object can be a child of a Group, it can likewise be in a Viewport.
115 |
116 | Viewports can be combined with a Scrollbar to create a scrollable display.
117 |
118 | Placing a Column or Row inside a Viewport is a simple way to align content.
119 |
120 | ```javascript
121 | var viewport = new uiWidgets.Viewport(game, 75, 75, 600, 260);
122 | viewport.addNode(column);
123 | ```
124 |
125 | #### Bars
126 |
127 | ##### Scrollbar
128 | Scrollbars are used to move the objects in a Viewport. They must be used with a Viewport.
129 | A tweening duration and easing can be specified. This will be triggered when moving the bar.
130 |
131 | ```javascript
132 | var scrollbar = new uiWidgets.Scrollbar(
133 | game,
134 | viewport,
135 | true,
136 | true,
137 | true,
138 | trackImage,
139 | barImage,
140 | {'duration': 300, 'ease': Phaser.Easing.Quadratic.Out}
141 | );
142 | ```
143 |
144 | ##### Examples
145 |
146 | ###### Phaser CE
147 |
148 | [Vertical](https://jsfehler.github.io/phaser-ui-tools/examples/html/phaserce/vscrollbar.html) | [Code](https://jsfehler.github.io/phaser-ui-tools/examples/js/phaserce/vscrollbar.js)
149 |
150 | [Horizontal](https://jsfehler.github.io/phaser-ui-tools/examples/html/phaserce/hscrollbar.html) | [Code](https://jsfehler.github.io/phaser-ui-tools/examples/js/phaserce/hscrollbar.js)
151 |
152 | ###### Phaser 3
153 |
154 | [Vertical](https://jsfehler.github.io/phaser-ui-tools/examples/html/phaser3/vscrollbar.html) | [Code](https://jsfehler.github.io/phaser-ui-tools/examples/js/phaser3/vscrollbar.js)
155 |
156 | [Horizontal](https://jsfehler.github.io/phaser-ui-tools/examples/html/phaser3/hscrollbar.html) | [Code](https://jsfehler.github.io/phaser-ui-tools/examples/js/phaser3/hscrollbar.js)
157 |
158 |
159 | ##### ValueBar
160 | Valuebars are like Scrollbars, but instead of moving content, they increase/decrease a number.
161 | Valuebars always have a minimum number of 0, but the starting and maximum number can be set.
162 | A tweening duration and easing can be specified. This will be triggered when moving the bar.
163 |
164 | 
165 |
166 | ```javascript
167 | var valuebar = new uiWidgets.ValueBar(
168 | game,
169 | {'x': 50, 'y': 10},
170 | {'step': 1, 'startValue': 0, 'maxValue': 100},
171 | true,
172 | false,
173 | true,
174 | trackImage,
175 | barImage,
176 | {'duration': 100, 'ease': Phaser.Easing.Quadratic.Out}
177 | );
178 | ```
179 |
180 | ##### Examples
181 |
182 | ###### Phaser CE
183 |
184 | [ValueBar](https://jsfehler.github.io/phaser-ui-tools/examples/html/phaserce/valuebar.html) | [Code](https://jsfehler.github.io/phaser-ui-tools/examples/js/phaserce/valuebar.js)
185 |
186 | [Multiple ValueBar inside a Column, with background image and keyboard events](https://jsfehler.github.io/phaser-ui-tools/examples/html/phaserce/valuebar_column.html) | [Code](https://jsfehler.github.io/phaser-ui-tools/examples/js/phaserce/valuebar_column.js)
187 |
188 | ###### Phaser 3
189 |
190 | [ValueBar](https://jsfehler.github.io/phaser-ui-tools/examples/html/phaser3/valuebar.html) | [Code](https://jsfehler.github.io/phaser-ui-tools/examples/js/phaser3/valuebar.js)
191 |
192 |
193 | ##### QuantityBar
194 | QuantityBars do not adjust a value, they get adjusted by a value. The bar grows and shrinks based on a value.
195 | They can be used for health bars, stamina bars, etc.
196 | A tweening duration and easing can be specified. This will be triggered when moving the bar.
197 |
198 | 
199 |
200 | ```javascript
201 | var quantitybar = new uiWidgets.QuantityBar(
202 | game,
203 | {'x': 50, 'y': 10},
204 | {'startValue': 50, 'maxValue': 100},
205 | false,
206 | false,
207 | trackImage,
208 | barImage,
209 | {'duration': 400, 'ease': Phaser.Easing.Quadratic.Out}
210 | );
211 | ```
212 |
213 | ##### Examples
214 |
215 | ###### Phaser CE
216 |
217 | [QuantityBar](https://jsfehler.github.io/phaser-ui-tools/examples/html/phaserce/quantitybar.html) | [Code](https://jsfehler.github.io/phaser-ui-tools/examples/js/phaserce/quantitybar.js)
218 |
219 | ###### Phaser 3
220 |
221 | [QuantityBar](https://jsfehler.github.io/phaser-ui-tools/examples/html/phaser3/quantitybar.html) | [Code](https://jsfehler.github.io/phaser-ui-tools/examples/js/phaser3/quantitybar.js)
222 |
223 |
224 | #### Wheel3D
225 | A collection of sprites that are arranged around a three dimensional wheel.
226 | The wheel can be adjusted and rotated along the x, y, or z axis.
227 |
228 | 
229 |
230 | ```javascript
231 | var wheel = new uiWidgets.Wheel3D(
232 | game,
233 | {"x": game.world.centerX - 100, "y": game.world.centerY},
234 | [sprite1, sprite2, sprite3, sprite4],
235 | 0,
236 | 90,
237 | "y",
238 | {"x":0, "y": 0, "z": 0}
239 | );
240 | ```
241 |
242 | ##### Examples
243 |
244 | ###### Phaser CE
245 |
246 | [Wheel3D](https://jsfehler.github.io/phaser-ui-tools/examples/html/phaserce/wheel3D.html) | [Code](https://jsfehler.github.io/phaser-ui-tools/examples/js/phaserce/wheel3D.js)
247 |
248 | ###### Phaser 3
249 |
250 | [Wheel3D](https://jsfehler.github.io/phaser-ui-tools/examples/html/phaser3/wheel3D.html) | [Code](https://jsfehler.github.io/phaser-ui-tools/examples/js/phaser3/wheel3D.js)
251 |
--------------------------------------------------------------------------------
/assets/diagram.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/assets/diagram.png
--------------------------------------------------------------------------------
/assets/diagram_column.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/assets/diagram_column.png
--------------------------------------------------------------------------------
/assets/diagram_quantitybar.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/assets/diagram_quantitybar.png
--------------------------------------------------------------------------------
/assets/diagram_row.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/assets/diagram_row.png
--------------------------------------------------------------------------------
/assets/diagram_valuebar.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/assets/diagram_valuebar.png
--------------------------------------------------------------------------------
/assets/diagram_wheel3D.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/assets/diagram_wheel3D.png
--------------------------------------------------------------------------------
/docs/assets/bass-addons.css:
--------------------------------------------------------------------------------
1 | .input {
2 | font-family: inherit;
3 | display: block;
4 | width: 100%;
5 | height: 2rem;
6 | padding: .5rem;
7 | margin-bottom: 1rem;
8 | border: 1px solid #ccc;
9 | font-size: .875rem;
10 | border-radius: 3px;
11 | box-sizing: border-box;
12 | }
13 |
--------------------------------------------------------------------------------
/docs/assets/fonts/EOT/SourceCodePro-Bold.eot:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/assets/fonts/EOT/SourceCodePro-Bold.eot
--------------------------------------------------------------------------------
/docs/assets/fonts/EOT/SourceCodePro-Regular.eot:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/assets/fonts/EOT/SourceCodePro-Regular.eot
--------------------------------------------------------------------------------
/docs/assets/fonts/LICENSE.txt:
--------------------------------------------------------------------------------
1 | Copyright 2010, 2012 Adobe Systems Incorporated (http://www.adobe.com/), with Reserved Font Name 'Source'. All Rights Reserved. Source is a trademark of Adobe Systems Incorporated in the United States and/or other countries.
2 |
3 | This Font Software is licensed under the SIL Open Font License, Version 1.1.
4 |
5 | This license is copied below, and is also available with a FAQ at: http://scripts.sil.org/OFL
6 |
7 |
8 | -----------------------------------------------------------
9 | SIL OPEN FONT LICENSE Version 1.1 - 26 February 2007
10 | -----------------------------------------------------------
11 |
12 | PREAMBLE
13 | The goals of the Open Font License (OFL) are to stimulate worldwide
14 | development of collaborative font projects, to support the font creation
15 | efforts of academic and linguistic communities, and to provide a free and
16 | open framework in which fonts may be shared and improved in partnership
17 | with others.
18 |
19 | The OFL allows the licensed fonts to be used, studied, modified and
20 | redistributed freely as long as they are not sold by themselves. The
21 | fonts, including any derivative works, can be bundled, embedded,
22 | redistributed and/or sold with any software provided that any reserved
23 | names are not used by derivative works. The fonts and derivatives,
24 | however, cannot be released under any other type of license. The
25 | requirement for fonts to remain under this license does not apply
26 | to any document created using the fonts or their derivatives.
27 |
28 | DEFINITIONS
29 | "Font Software" refers to the set of files released by the Copyright
30 | Holder(s) under this license and clearly marked as such. This may
31 | include source files, build scripts and documentation.
32 |
33 | "Reserved Font Name" refers to any names specified as such after the
34 | copyright statement(s).
35 |
36 | "Original Version" refers to the collection of Font Software components as
37 | distributed by the Copyright Holder(s).
38 |
39 | "Modified Version" refers to any derivative made by adding to, deleting,
40 | or substituting -- in part or in whole -- any of the components of the
41 | Original Version, by changing formats or by porting the Font Software to a
42 | new environment.
43 |
44 | "Author" refers to any designer, engineer, programmer, technical
45 | writer or other person who contributed to the Font Software.
46 |
47 | PERMISSION & CONDITIONS
48 | Permission is hereby granted, free of charge, to any person obtaining
49 | a copy of the Font Software, to use, study, copy, merge, embed, modify,
50 | redistribute, and sell modified and unmodified copies of the Font
51 | Software, subject to the following conditions:
52 |
53 | 1) Neither the Font Software nor any of its individual components,
54 | in Original or Modified Versions, may be sold by itself.
55 |
56 | 2) Original or Modified Versions of the Font Software may be bundled,
57 | redistributed and/or sold with any software, provided that each copy
58 | contains the above copyright notice and this license. These can be
59 | included either as stand-alone text files, human-readable headers or
60 | in the appropriate machine-readable metadata fields within text or
61 | binary files as long as those fields can be easily viewed by the user.
62 |
63 | 3) No Modified Version of the Font Software may use the Reserved Font
64 | Name(s) unless explicit written permission is granted by the corresponding
65 | Copyright Holder. This restriction only applies to the primary font name as
66 | presented to the users.
67 |
68 | 4) The name(s) of the Copyright Holder(s) or the Author(s) of the Font
69 | Software shall not be used to promote, endorse or advertise any
70 | Modified Version, except to acknowledge the contribution(s) of the
71 | Copyright Holder(s) and the Author(s) or with their explicit written
72 | permission.
73 |
74 | 5) The Font Software, modified or unmodified, in part or in whole,
75 | must be distributed entirely under this license, and must not be
76 | distributed under any other license. The requirement for fonts to
77 | remain under this license does not apply to any document created
78 | using the Font Software.
79 |
80 | TERMINATION
81 | This license becomes null and void if any of the above conditions are
82 | not met.
83 |
84 | DISCLAIMER
85 | THE FONT SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
86 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OF
87 | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT
88 | OF COPYRIGHT, PATENT, TRADEMARK, OR OTHER RIGHT. IN NO EVENT SHALL THE
89 | COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
90 | INCLUDING ANY GENERAL, SPECIAL, INDIRECT, INCIDENTAL, OR CONSEQUENTIAL
91 | DAMAGES, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
92 | FROM, OUT OF THE USE OR INABILITY TO USE THE FONT SOFTWARE OR FROM
93 | OTHER DEALINGS IN THE FONT SOFTWARE.
94 |
--------------------------------------------------------------------------------
/docs/assets/fonts/OTF/SourceCodePro-Bold.otf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/assets/fonts/OTF/SourceCodePro-Bold.otf
--------------------------------------------------------------------------------
/docs/assets/fonts/OTF/SourceCodePro-Regular.otf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/assets/fonts/OTF/SourceCodePro-Regular.otf
--------------------------------------------------------------------------------
/docs/assets/fonts/TTF/SourceCodePro-Bold.ttf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/assets/fonts/TTF/SourceCodePro-Bold.ttf
--------------------------------------------------------------------------------
/docs/assets/fonts/TTF/SourceCodePro-Regular.ttf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/assets/fonts/TTF/SourceCodePro-Regular.ttf
--------------------------------------------------------------------------------
/docs/assets/fonts/WOFF/OTF/SourceCodePro-Bold.otf.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/assets/fonts/WOFF/OTF/SourceCodePro-Bold.otf.woff
--------------------------------------------------------------------------------
/docs/assets/fonts/WOFF/OTF/SourceCodePro-Regular.otf.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/assets/fonts/WOFF/OTF/SourceCodePro-Regular.otf.woff
--------------------------------------------------------------------------------
/docs/assets/fonts/WOFF/TTF/SourceCodePro-Bold.ttf.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/assets/fonts/WOFF/TTF/SourceCodePro-Bold.ttf.woff
--------------------------------------------------------------------------------
/docs/assets/fonts/WOFF/TTF/SourceCodePro-Regular.ttf.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/assets/fonts/WOFF/TTF/SourceCodePro-Regular.ttf.woff
--------------------------------------------------------------------------------
/docs/assets/fonts/WOFF2/OTF/SourceCodePro-Bold.otf.woff2:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/assets/fonts/WOFF2/OTF/SourceCodePro-Bold.otf.woff2
--------------------------------------------------------------------------------
/docs/assets/fonts/WOFF2/OTF/SourceCodePro-Regular.otf.woff2:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/assets/fonts/WOFF2/OTF/SourceCodePro-Regular.otf.woff2
--------------------------------------------------------------------------------
/docs/assets/fonts/WOFF2/TTF/SourceCodePro-Bold.ttf.woff2:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/assets/fonts/WOFF2/TTF/SourceCodePro-Bold.ttf.woff2
--------------------------------------------------------------------------------
/docs/assets/fonts/WOFF2/TTF/SourceCodePro-Regular.ttf.woff2:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/assets/fonts/WOFF2/TTF/SourceCodePro-Regular.ttf.woff2
--------------------------------------------------------------------------------
/docs/assets/fonts/source-code-pro.css:
--------------------------------------------------------------------------------
1 | @font-face{
2 | font-family: 'Source Code Pro';
3 | font-weight: 400;
4 | font-style: normal;
5 | font-stretch: normal;
6 | src: url('EOT/SourceCodePro-Regular.eot') format('embedded-opentype'),
7 | url('WOFF2/TTF/SourceCodePro-Regular.ttf.woff2') format('woff2'),
8 | url('WOFF/OTF/SourceCodePro-Regular.otf.woff') format('woff'),
9 | url('OTF/SourceCodePro-Regular.otf') format('opentype'),
10 | url('TTF/SourceCodePro-Regular.ttf') format('truetype');
11 | }
12 |
13 | @font-face{
14 | font-family: 'Source Code Pro';
15 | font-weight: 700;
16 | font-style: normal;
17 | font-stretch: normal;
18 | src: url('EOT/SourceCodePro-Bold.eot') format('embedded-opentype'),
19 | url('WOFF2/TTF/SourceCodePro-Bold.ttf.woff2') format('woff2'),
20 | url('WOFF/OTF/SourceCodePro-Bold.otf.woff') format('woff'),
21 | url('OTF/SourceCodePro-Bold.otf') format('opentype'),
22 | url('TTF/SourceCodePro-Bold.ttf') format('truetype');
23 | }
24 |
--------------------------------------------------------------------------------
/docs/assets/github.css:
--------------------------------------------------------------------------------
1 | /*
2 |
3 | github.com style (c) Vasily Polovnyov
4 |
5 | */
6 |
7 | .hljs {
8 | display: block;
9 | overflow-x: auto;
10 | padding: 0.5em;
11 | color: #333;
12 | background: #f8f8f8;
13 | -webkit-text-size-adjust: none;
14 | }
15 |
16 | .hljs-comment,
17 | .diff .hljs-header,
18 | .hljs-javadoc {
19 | color: #998;
20 | font-style: italic;
21 | }
22 |
23 | .hljs-keyword,
24 | .css .rule .hljs-keyword,
25 | .hljs-winutils,
26 | .nginx .hljs-title,
27 | .hljs-subst,
28 | .hljs-request,
29 | .hljs-status {
30 | color: #1184CE;
31 | }
32 |
33 | .hljs-number,
34 | .hljs-hexcolor,
35 | .ruby .hljs-constant {
36 | color: #ed225d;
37 | }
38 |
39 | .hljs-string,
40 | .hljs-tag .hljs-value,
41 | .hljs-phpdoc,
42 | .hljs-dartdoc,
43 | .tex .hljs-formula {
44 | color: #ed225d;
45 | }
46 |
47 | .hljs-title,
48 | .hljs-id,
49 | .scss .hljs-preprocessor {
50 | color: #900;
51 | font-weight: bold;
52 | }
53 |
54 | .hljs-list .hljs-keyword,
55 | .hljs-subst {
56 | font-weight: normal;
57 | }
58 |
59 | .hljs-class .hljs-title,
60 | .hljs-type,
61 | .vhdl .hljs-literal,
62 | .tex .hljs-command {
63 | color: #458;
64 | font-weight: bold;
65 | }
66 |
67 | .hljs-tag,
68 | .hljs-tag .hljs-title,
69 | .hljs-rules .hljs-property,
70 | .django .hljs-tag .hljs-keyword {
71 | color: #000080;
72 | font-weight: normal;
73 | }
74 |
75 | .hljs-attribute,
76 | .hljs-variable,
77 | .lisp .hljs-body {
78 | color: #008080;
79 | }
80 |
81 | .hljs-regexp {
82 | color: #009926;
83 | }
84 |
85 | .hljs-symbol,
86 | .ruby .hljs-symbol .hljs-string,
87 | .lisp .hljs-keyword,
88 | .clojure .hljs-keyword,
89 | .scheme .hljs-keyword,
90 | .tex .hljs-special,
91 | .hljs-prompt {
92 | color: #990073;
93 | }
94 |
95 | .hljs-built_in {
96 | color: #0086b3;
97 | }
98 |
99 | .hljs-preprocessor,
100 | .hljs-pragma,
101 | .hljs-pi,
102 | .hljs-doctype,
103 | .hljs-shebang,
104 | .hljs-cdata {
105 | color: #999;
106 | font-weight: bold;
107 | }
108 |
109 | .hljs-deletion {
110 | background: #fdd;
111 | }
112 |
113 | .hljs-addition {
114 | background: #dfd;
115 | }
116 |
117 | .diff .hljs-change {
118 | background: #0086b3;
119 | }
120 |
121 | .hljs-chunk {
122 | color: #aaa;
123 | }
124 |
--------------------------------------------------------------------------------
/docs/assets/site.js:
--------------------------------------------------------------------------------
1 | /* global anchors */
2 |
3 | // add anchor links to headers
4 | anchors.options.placement = 'left';
5 | anchors.add('h3');
6 |
7 | // Filter UI
8 | var tocElements = document.getElementById('toc').getElementsByTagName('li');
9 |
10 | document.getElementById('filter-input').addEventListener('keyup', function(e) {
11 | var i, element, children;
12 |
13 | // enter key
14 | if (e.keyCode === 13) {
15 | // go to the first displayed item in the toc
16 | for (i = 0; i < tocElements.length; i++) {
17 | element = tocElements[i];
18 | if (!element.classList.contains('display-none')) {
19 | location.replace(element.firstChild.href);
20 | return e.preventDefault();
21 | }
22 | }
23 | }
24 |
25 | var match = function() {
26 | return true;
27 | };
28 |
29 | var value = this.value.toLowerCase();
30 |
31 | if (!value.match(/^\s*$/)) {
32 | match = function(element) {
33 | var html = element.firstChild.innerHTML;
34 | return html && html.toLowerCase().indexOf(value) !== -1;
35 | };
36 | }
37 |
38 | for (i = 0; i < tocElements.length; i++) {
39 | element = tocElements[i];
40 | children = Array.from(element.getElementsByTagName('li'));
41 | if (match(element) || children.some(match)) {
42 | element.classList.remove('display-none');
43 | } else {
44 | element.classList.add('display-none');
45 | }
46 | }
47 | });
48 |
49 | var items = document.getElementsByClassName('toggle-sibling');
50 | for (var j = 0; j < items.length; j++) {
51 | items[j].addEventListener('click', toggleSibling);
52 | }
53 |
54 | function toggleSibling() {
55 | var stepSibling = this.parentNode.getElementsByClassName('toggle-target')[0];
56 | var icon = this.getElementsByClassName('icon')[0];
57 | var klass = 'display-none';
58 | if (stepSibling.classList.contains(klass)) {
59 | stepSibling.classList.remove(klass);
60 | icon.innerHTML = '▾';
61 | } else {
62 | stepSibling.classList.add(klass);
63 | icon.innerHTML = '▸';
64 | }
65 | }
66 |
67 | function showHashTarget(targetId) {
68 | if (targetId) {
69 | var hashTarget = document.getElementById(targetId);
70 | // new target is hidden
71 | if (
72 | hashTarget &&
73 | hashTarget.offsetHeight === 0 &&
74 | hashTarget.parentNode.parentNode.classList.contains('display-none')
75 | ) {
76 | hashTarget.parentNode.parentNode.classList.remove('display-none');
77 | }
78 | }
79 | }
80 |
81 | function scrollIntoView(targetId) {
82 | // Only scroll to element if we don't have a stored scroll position.
83 | if (targetId && !history.state) {
84 | var hashTarget = document.getElementById(targetId);
85 | if (hashTarget) {
86 | hashTarget.scrollIntoView();
87 | }
88 | }
89 | }
90 |
91 | function gotoCurrentTarget() {
92 | showHashTarget(location.hash.substring(1));
93 | scrollIntoView(location.hash.substring(1));
94 | }
95 |
96 | window.addEventListener('hashchange', gotoCurrentTarget);
97 | gotoCurrentTarget();
98 |
99 | var toclinks = document.getElementsByClassName('pre-open');
100 | for (var k = 0; k < toclinks.length; k++) {
101 | toclinks[k].addEventListener('mousedown', preOpen, false);
102 | }
103 |
104 | function preOpen() {
105 | showHashTarget(this.hash.substring(1));
106 | }
107 |
108 | var split_left = document.querySelector('#split-left');
109 | var split_right = document.querySelector('#split-right');
110 | var split_parent = split_left.parentNode;
111 | var cw_with_sb = split_left.clientWidth;
112 | split_left.style.overflow = 'hidden';
113 | var cw_without_sb = split_left.clientWidth;
114 | split_left.style.overflow = '';
115 |
116 | Split(['#split-left', '#split-right'], {
117 | elementStyle: function(dimension, size, gutterSize) {
118 | return {
119 | 'flex-basis': 'calc(' + size + '% - ' + gutterSize + 'px)'
120 | };
121 | },
122 | gutterStyle: function(dimension, gutterSize) {
123 | return {
124 | 'flex-basis': gutterSize + 'px'
125 | };
126 | },
127 | gutterSize: 20,
128 | sizes: [33, 67]
129 | });
130 |
131 | // Chrome doesn't remember scroll position properly so do it ourselves.
132 | // Also works on Firefox and Edge.
133 |
134 | function updateState() {
135 | history.replaceState(
136 | {
137 | left_top: split_left.scrollTop,
138 | right_top: split_right.scrollTop
139 | },
140 | document.title
141 | );
142 | }
143 |
144 | function loadState(ev) {
145 | if (ev) {
146 | // Edge doesn't replace change history.state on popstate.
147 | history.replaceState(ev.state, document.title);
148 | }
149 | if (history.state) {
150 | split_left.scrollTop = history.state.left_top;
151 | split_right.scrollTop = history.state.right_top;
152 | }
153 | }
154 |
155 | window.addEventListener('load', function() {
156 | // Restore after Firefox scrolls to hash.
157 | setTimeout(function() {
158 | loadState();
159 | // Update with initial scroll position.
160 | updateState();
161 | // Update scroll positions only after we've loaded because Firefox
162 | // emits an initial scroll event with 0.
163 | split_left.addEventListener('scroll', updateState);
164 | split_right.addEventListener('scroll', updateState);
165 | }, 1);
166 | });
167 |
168 | window.addEventListener('popstate', loadState);
169 |
--------------------------------------------------------------------------------
/docs/assets/split.css:
--------------------------------------------------------------------------------
1 | .gutter {
2 | background-color: #f5f5f5;
3 | background-repeat: no-repeat;
4 | background-position: 50%;
5 | }
6 |
7 | .gutter.gutter-vertical {
8 | background-image: url('');
9 | cursor: ns-resize;
10 | }
11 |
12 | .gutter.gutter-horizontal {
13 | background-image: url('');
14 | cursor: ew-resize;
15 | }
16 |
--------------------------------------------------------------------------------
/docs/assets/style.css:
--------------------------------------------------------------------------------
1 | .documentation {
2 | font-family: Helvetica, sans-serif;
3 | color: #666;
4 | line-height: 1.5;
5 | background: #f5f5f5;
6 | }
7 |
8 | .black {
9 | color: #666;
10 | }
11 |
12 | .bg-white {
13 | background-color: #fff;
14 | }
15 |
16 | h4 {
17 | margin: 20px 0 10px 0;
18 | }
19 |
20 | .documentation h3 {
21 | color: #000;
22 | }
23 |
24 | .border-bottom {
25 | border-color: #ddd;
26 | }
27 |
28 | a {
29 | color: #1184ce;
30 | text-decoration: none;
31 | }
32 |
33 | .documentation a[href]:hover {
34 | text-decoration: underline;
35 | }
36 |
37 | a:hover {
38 | cursor: pointer;
39 | }
40 |
41 | .py1-ul li {
42 | padding: 5px 0;
43 | }
44 |
45 | .max-height-100 {
46 | max-height: 100%;
47 | }
48 |
49 | .height-viewport-100 {
50 | height: 100vh;
51 | }
52 |
53 | section:target h3 {
54 | font-weight: 700;
55 | }
56 |
57 | .documentation td,
58 | .documentation th {
59 | padding: 0.25rem 0.25rem;
60 | }
61 |
62 | h1:hover .anchorjs-link,
63 | h2:hover .anchorjs-link,
64 | h3:hover .anchorjs-link,
65 | h4:hover .anchorjs-link {
66 | opacity: 1;
67 | }
68 |
69 | .fix-3 {
70 | width: 25%;
71 | max-width: 244px;
72 | }
73 |
74 | .fix-3 {
75 | width: 25%;
76 | max-width: 244px;
77 | }
78 |
79 | @media (min-width: 52em) {
80 | .fix-margin-3 {
81 | margin-left: 25%;
82 | }
83 | }
84 |
85 | .pre,
86 | pre,
87 | code,
88 | .code {
89 | font-family: Source Code Pro, Menlo, Consolas, Liberation Mono, monospace;
90 | font-size: 14px;
91 | }
92 |
93 | .fill-light {
94 | background: #f9f9f9;
95 | }
96 |
97 | .width2 {
98 | width: 1rem;
99 | }
100 |
101 | .input {
102 | font-family: inherit;
103 | display: block;
104 | width: 100%;
105 | height: 2rem;
106 | padding: 0.5rem;
107 | margin-bottom: 1rem;
108 | border: 1px solid #ccc;
109 | font-size: 0.875rem;
110 | border-radius: 3px;
111 | box-sizing: border-box;
112 | }
113 |
114 | table {
115 | border-collapse: collapse;
116 | }
117 |
118 | .prose table th,
119 | .prose table td {
120 | text-align: left;
121 | padding: 8px;
122 | border: 1px solid #ddd;
123 | }
124 |
125 | .prose table th:nth-child(1) {
126 | border-right: none;
127 | }
128 | .prose table th:nth-child(2) {
129 | border-left: none;
130 | }
131 |
132 | .prose table {
133 | border: 1px solid #ddd;
134 | }
135 |
136 | .prose-big {
137 | font-size: 18px;
138 | line-height: 30px;
139 | }
140 |
141 | .quiet {
142 | opacity: 0.7;
143 | }
144 |
145 | .minishadow {
146 | box-shadow: 2px 2px 10px #f3f3f3;
147 | }
148 |
--------------------------------------------------------------------------------
/docs/examples/assets/horizontal/bar.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/horizontal/bar.png
--------------------------------------------------------------------------------
/docs/examples/assets/horizontal/sprite.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/horizontal/sprite.png
--------------------------------------------------------------------------------
/docs/examples/assets/horizontal/track.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/horizontal/track.png
--------------------------------------------------------------------------------
/docs/examples/assets/modal/background.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/modal/background.png
--------------------------------------------------------------------------------
/docs/examples/assets/modal/button.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/modal/button.png
--------------------------------------------------------------------------------
/docs/examples/assets/quantitybar/add.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/quantitybar/add.png
--------------------------------------------------------------------------------
/docs/examples/assets/quantitybar/horizontal/bar.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/quantitybar/horizontal/bar.png
--------------------------------------------------------------------------------
/docs/examples/assets/quantitybar/horizontal/track.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/quantitybar/horizontal/track.png
--------------------------------------------------------------------------------
/docs/examples/assets/quantitybar/subtract.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/quantitybar/subtract.png
--------------------------------------------------------------------------------
/docs/examples/assets/quantitybar/vertical/bar.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/quantitybar/vertical/bar.png
--------------------------------------------------------------------------------
/docs/examples/assets/quantitybar/vertical/track.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/quantitybar/vertical/track.png
--------------------------------------------------------------------------------
/docs/examples/assets/textover/button.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/textover/button.png
--------------------------------------------------------------------------------
/docs/examples/assets/textover/header.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/textover/header.png
--------------------------------------------------------------------------------
/docs/examples/assets/valuebar/alpha.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/valuebar/alpha.png
--------------------------------------------------------------------------------
/docs/examples/assets/valuebar/background.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/valuebar/background.png
--------------------------------------------------------------------------------
/docs/examples/assets/valuebar/bar.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/valuebar/bar.png
--------------------------------------------------------------------------------
/docs/examples/assets/valuebar/pointer.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/valuebar/pointer.png
--------------------------------------------------------------------------------
/docs/examples/assets/valuebar/track.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/valuebar/track.png
--------------------------------------------------------------------------------
/docs/examples/assets/valuebar/vtrack.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/valuebar/vtrack.png
--------------------------------------------------------------------------------
/docs/examples/assets/vertical/bar.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/vertical/bar.png
--------------------------------------------------------------------------------
/docs/examples/assets/vertical/sprite.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/vertical/sprite.png
--------------------------------------------------------------------------------
/docs/examples/assets/vertical/track.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/vertical/track.png
--------------------------------------------------------------------------------
/docs/examples/assets/wheel3D/icon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/wheel3D/icon.png
--------------------------------------------------------------------------------
/docs/examples/assets/wheel3D/mmDown.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/wheel3D/mmDown.png
--------------------------------------------------------------------------------
/docs/examples/assets/wheel3D/mmUp.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsfehler/phaser-ui-tools/f3306cf9aa5e95ea746001e8c48448d24c9c4af4/docs/examples/assets/wheel3D/mmUp.png
--------------------------------------------------------------------------------
/docs/examples/html/phaser3/hscrollbar.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
--------------------------------------------------------------------------------
/docs/examples/html/phaser3/modal.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
--------------------------------------------------------------------------------
/docs/examples/html/phaser3/quantitybar.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
--------------------------------------------------------------------------------
/docs/examples/html/phaser3/textover.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
--------------------------------------------------------------------------------
/docs/examples/html/phaser3/valuebar.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
--------------------------------------------------------------------------------
/docs/examples/html/phaser3/vscrollbar.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
--------------------------------------------------------------------------------
/docs/examples/html/phaser3/wheel3D.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
--------------------------------------------------------------------------------
/docs/examples/html/phaserce/hscrollbar.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
--------------------------------------------------------------------------------
/docs/examples/html/phaserce/modal.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
--------------------------------------------------------------------------------
/docs/examples/html/phaserce/quantitybar.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
--------------------------------------------------------------------------------
/docs/examples/html/phaserce/quantitybar_column.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
--------------------------------------------------------------------------------
/docs/examples/html/phaserce/textover.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
--------------------------------------------------------------------------------
/docs/examples/html/phaserce/valuebar.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
--------------------------------------------------------------------------------
/docs/examples/html/phaserce/valuebar_column.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
--------------------------------------------------------------------------------
/docs/examples/html/phaserce/vscrollbar.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
--------------------------------------------------------------------------------
/docs/examples/html/phaserce/wheel3D.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
--------------------------------------------------------------------------------
/docs/examples/js/phaser3/hscrollbar.js:
--------------------------------------------------------------------------------
1 | var config = {
2 | type: Phaser.AUTO,
3 | width: 400,
4 | height: 300,
5 | scene: {
6 | preload: preload,
7 | create: create,
8 | update: update
9 | }
10 | };
11 |
12 | var game = new Phaser.Game(config);
13 |
14 | function preload() {
15 | var assetRoot = '../../assets/horizontal/';
16 | this.load.image('dummyButton', `${assetRoot}sprite.png`);
17 | this.load.image('track', `${assetRoot}track.png`);
18 | this.load.spritesheet('bar', `${assetRoot}bar.png`, { frameWidth: 44, frameHeight: 22 });
19 | }
20 |
21 | function create() {
22 | var viewport = new uiWidgets.Viewport(this, 75, 75, 260, 128);
23 | var row = new uiWidgets.Row(this, 0, 0);
24 |
25 | viewport.addNode(row);
26 |
27 | // Add things to the row.
28 | var dummy_sprite_a = this.add.image(0, 0, 'dummyButton');
29 | var dummy_sprite_b = this.add.image(0, 0, 'dummyButton');
30 | var dummy_sprite_c = this.add.image(0, 0, 'dummyButton');
31 | var dummy_sprite_d = this.add.image(0, 0, 'dummyButton');
32 | var dummy_sprite_e = this.add.image(0, 0, 'dummyButton');
33 | var dummy_sprite_f = this.add.image(0, 0, 'dummyButton');
34 | var dummy_sprite_g = this.add.image(0, 0, 'dummyButton');
35 | var dummy_sprite_h = this.add.image(0, 0, 'dummyButton');
36 | var dummy_sprite_i = this.add.image(0, 0, 'dummyButton');
37 |
38 | row.addNode(dummy_sprite_a);
39 | row.addNode(dummy_sprite_b);
40 | row.addNode(dummy_sprite_c);
41 | row.addNode(dummy_sprite_d);
42 | row.addNode(dummy_sprite_e);
43 | row.addNode(dummy_sprite_f);
44 | row.addNode(dummy_sprite_g);
45 | row.addNode(dummy_sprite_h);
46 | row.addNode(dummy_sprite_i);
47 |
48 | var scrollbar = new uiWidgets.Scrollbar(
49 | this,
50 | viewport,
51 | true,
52 | false,
53 | "track",
54 | "bar",
55 | {'duration': 300, 'ease': Phaser.Math.Easing.Quadratic.Out}
56 | );
57 |
58 | Phaser.Display.Align.To.BottomCenter(scrollbar, viewport, 0, 128 + 10);
59 |
60 | }
61 |
62 | function update() {}
63 |
--------------------------------------------------------------------------------
/docs/examples/js/phaser3/modal.js:
--------------------------------------------------------------------------------
1 | var config = {
2 | type: Phaser.AUTO,
3 | width: 400,
4 | height: 300,
5 | scene: {
6 | preload: preload,
7 | create: create,
8 | }
9 | };
10 |
11 | var game = new Phaser.Game(config);
12 |
13 | var textStyle = {'fill': '#FFF', 'font': '16px Courier New'};
14 |
15 | function preload() {
16 | var assetRoot = '../../assets/modal/';
17 | this.load.image("modalBg", assetRoot + "background.png");
18 | this.load.spritesheet('button', assetRoot + 'button.png', { frameWidth: 128, frameHeight: 48 });
19 | }
20 |
21 | function create() {
22 | var openButton = new uiWidgets.TextButton(this, 100, 50, "button", createModal, this, 1, 0, 2, 1)
23 | .setText("Open", textStyle)
24 | .eventTextYAdjustment(3);
25 | }
26 |
27 | function createModal() {
28 | var closeButton = new uiWidgets.TextButton(this, 0, 0, "button", closeCallback, this, 1, 0, 2, 1)
29 | .setText("Close", textStyle)
30 | .eventTextYAdjustment(3);
31 |
32 | this.modal = new uiWidgets.Column(this, 100, 100, bg="modalBg", modal=true);
33 | this.modal.addNode(closeButton, paddingX=0, paddingY=10);
34 | }
35 |
36 | function closeCallback() {
37 | this.modal.dismiss();
38 | }
39 |
--------------------------------------------------------------------------------
/docs/examples/js/phaser3/quantitybar.js:
--------------------------------------------------------------------------------
1 | var config = {
2 | type: Phaser.AUTO,
3 | width: 600,
4 | height: 400,
5 | scene: {
6 | preload: preload,
7 | create: create,
8 | }
9 | };
10 |
11 | var game = new Phaser.Game(config);
12 |
13 | function preload() {
14 | var assetRoot = '../../assets/quantitybar/';
15 | this.load.image("track", assetRoot + "horizontal/track.png");
16 | this.load.spritesheet('bar', assetRoot + 'horizontal/bar.png', { frameWidth: 260, frameHeight: 32 });
17 |
18 | this.load.image("vtrack", assetRoot + "vertical/track.png");
19 | this.load.spritesheet('vbar', assetRoot + 'vertical/bar.png', { frameWidth: 32, frameHeight: 260 });
20 |
21 | this.load.image("add", assetRoot + "add.png");
22 | this.load.image("subtract", assetRoot + "subtract.png");
23 | }
24 |
25 | var healthbar,
26 | healthbarText,
27 | vhealthbar,
28 | vhealthbarText,
29 | rhealthbar,
30 | rhealthbarText,
31 | rvhealthbar,
32 | rvhealthbarText,
33 | decreaseHealth,
34 | increaseHealth,
35 | vdecreaseHealth,
36 | vincreaseHealth,
37 | rdecreaseHealth,
38 | rincreaseHealth,
39 | rvdecreaseHealth,
40 | rvincreaseHealth,
41 | textStyle;
42 |
43 | function create() {
44 |
45 | textStyle = {
46 | font: "65px Arial",
47 | fill: "#ff0044",
48 | align: "center"
49 | };
50 |
51 | // Create a quantitybar starting at 50.
52 | healthbar = new uiWidgets.QuantityBar(
53 | this,
54 | {"x": 50, "y": 10},
55 | {"startValue": 50, maxValue: 100},
56 | false,
57 | false,
58 | "track",
59 | "bar",
60 | {'duration': 400, 'ease': Phaser.Math.Easing.Quadratic.Out}
61 | );
62 |
63 | // Display the quantitybar's value.
64 | healthbarText = this.add.text(50, 50, healthbar.valueRange.startValue, textStyle);
65 |
66 | // Add buttons to modify the value.
67 | var lessHealth = new uiWidgets.Button(this, 150, 10, 'subtract', decreaseHealth, this);
68 | var moreHealth = new uiWidgets.Button(this, 200, 10, 'add', increaseHealth, this);
69 |
70 | // Create a reverse quantitybar starting at 30.
71 | rhealthbar = new uiWidgets.QuantityBar(
72 | this,
73 | {"x": 350, "y": 10},
74 | {"startValue": 30, maxValue: 100},
75 | false,
76 | true,
77 | "track",
78 | "bar",
79 | {'duration': 400, 'ease': Phaser.Math.Easing.Quadratic.Out}
80 | );
81 |
82 | rhealthbarText = this.add.text(350, 50, rhealthbar.valueRange.startValue, textStyle);
83 |
84 | var rlessHealth = new uiWidgets.Button(this, 450, 10, 'subtract', rdecreaseHealth, this);
85 | var rmoreHealth = new uiWidgets.Button(this, 500, 10, 'add', rincreaseHealth, this);
86 |
87 | // Create a vertical quantitybar starting at 50.
88 | vhealthbar = new uiWidgets.QuantityBar(
89 | this,
90 | {"x": 30, "y": 130},
91 | {"startValue": 32, maxValue: 100},
92 | true,
93 | false,
94 | "vtrack",
95 | "vbar",
96 | {'duration': 400, 'ease': Phaser.Math.Easing.Quadratic.Out}
97 | );
98 |
99 | vhealthbarText = this.add.text(100, 150, vhealthbar.valueRange.startValue, textStyle);
100 |
101 | var vlessHealth = new uiWidgets.Button(this, 100, 250, 'subtract', vdecreaseHealth, this);
102 | var vmoreHealth = new uiWidgets.Button(this, 150, 250, 'add', vincreaseHealth, this);
103 |
104 | // Create a reverse vertical quantitybar starting at 50.
105 | rvhealthbar = new uiWidgets.QuantityBar(
106 | this,
107 | {"x": 250, "y": 130},
108 | {"startValue": 30, maxValue: 100},
109 | true,
110 | true,
111 | "vtrack",
112 | "vbar",
113 | {'duration': 400, 'ease': Phaser.Math.Easing.Quadratic.Out}
114 | );
115 |
116 | rvhealthbarText = this.add.text(300, 150, rvhealthbar.valueRange.startValue, textStyle);
117 |
118 | var vlessHealth = new uiWidgets.Button(this, 300, 250, 'subtract', rvdecreaseHealth, this);
119 | var vmoreHealth = new uiWidgets.Button(this, 350, 250, 'add', rvincreaseHealth, this);
120 |
121 | }
122 |
123 | function increaseHealth() {
124 | healthbar.adjustBar(10);
125 | healthbarText.setText(healthbar.valueRange.getCurrentValue());
126 | }
127 |
128 | function decreaseHealth() {
129 | healthbar.adjustBar(-10);
130 | healthbarText.setText(healthbar.valueRange.getCurrentValue());
131 | }
132 |
133 | function vincreaseHealth() {
134 | vhealthbar.adjustBar(1);
135 | vhealthbarText.setText(vhealthbar.valueRange.getCurrentValue());
136 | }
137 |
138 | function vdecreaseHealth() {
139 | vhealthbar.adjustBar(-1);
140 | vhealthbarText.setText(vhealthbar.valueRange.getCurrentValue());
141 | }
142 |
143 | function rincreaseHealth() {
144 | rhealthbar.adjustBar(10);
145 | rhealthbarText.setText(rhealthbar.valueRange.getCurrentValue());
146 | }
147 |
148 | function rdecreaseHealth() {
149 | rhealthbar.adjustBar(-10);
150 | rhealthbarText.setText(rhealthbar.valueRange.getCurrentValue());
151 | }
152 |
153 | function rvincreaseHealth() {
154 | rvhealthbar.adjustBar(10);
155 | rvhealthbarText.setText(rvhealthbar.valueRange.getCurrentValue());
156 | }
157 |
158 | function rvdecreaseHealth() {
159 | rvhealthbar.adjustBar(-10);
160 | rvhealthbarText.setText(rvhealthbar.valueRange.getCurrentValue());
161 | }
162 |
--------------------------------------------------------------------------------
/docs/examples/js/phaser3/textover.js:
--------------------------------------------------------------------------------
1 | var config = {
2 | type: Phaser.AUTO,
3 | width: 400,
4 | height: 300,
5 | scene: {
6 | preload: preload,
7 | create: create,
8 | }
9 | };
10 |
11 | var game = new Phaser.Game(config);
12 |
13 | var textStyle = {'fill': '#FFF', 'font': '16px Courier New'};
14 |
15 | function preload() {
16 | var assetRoot = '../../assets/textover/';
17 | this.load.image("header", assetRoot + "header.png");
18 | this.load.spritesheet('button', assetRoot + 'button.png', { frameWidth: 128, frameHeight: 48 });
19 | }
20 |
21 | function create() {
22 | this.header = new uiWidgets.TextSprite(this, 0, 0, "header").setText('Header', textStyle).setOrigin(0.0, 0.0);
23 |
24 | var buttonOne = new uiWidgets.TextButton(this, 0, 0, "button", newGameCallback, this, 1, 0, 2, 1)
25 | .setText("New Game", textStyle)
26 | .eventTextYAdjustment(3);
27 | var buttonTwo = new uiWidgets.TextButton(this, 0, 0, "button", continueCallback, this, 1, 0, 2, 1)
28 | .setText("Continue", textStyle)
29 | .eventTextYAdjustment(3);
30 | var buttonThree = new uiWidgets.TextButton(this, 0, 0, "button", optionsCallback, this, 1, 0, 2, 1)
31 | .setText("Options", textStyle)
32 | .eventTextYAdjustment(3);
33 |
34 | var column = new uiWidgets.Column(this, 200, 100);
35 | column.addNode(buttonOne, paddingX=0, paddingY=10);
36 | column.addNode(buttonTwo, paddingX=0, paddingY=10);
37 | column.addNode(buttonThree, paddingX=0, paddingY=10);
38 |
39 | }
40 |
41 | function newGameCallback() {
42 | this.header.text.setText('You clicked the New Game button');
43 | }
44 |
45 |
46 | function continueCallback() {
47 | this.header.text.setText('You clicked the Continue button');
48 | }
49 |
50 |
51 | function optionsCallback() {
52 | this.header.text.setText('You clicked the Options button');
53 | }
54 |
--------------------------------------------------------------------------------
/docs/examples/js/phaser3/valuebar.js:
--------------------------------------------------------------------------------
1 | var config = {
2 | type: Phaser.AUTO,
3 | width: 600,
4 | height: 400,
5 | scene: {
6 | preload: preload,
7 | create: create,
8 | }
9 | };
10 |
11 | var game = new Phaser.Game(config);
12 |
13 | function preload() {
14 | var assetRoot = '../../assets/valuebar/';
15 | this.load.image("track", assetRoot + "track.png");
16 | this.load.spritesheet('bar', assetRoot + 'bar.png', { frameWidth: 32, frameHeight: 32 });
17 | this.load.image("vtrack", assetRoot + "vtrack.png");
18 | this.load.image("alphaImage", assetRoot + "alpha.png");
19 | }
20 |
21 | var valuebar0;
22 | var valuebar50;
23 | var vvaluebar50;
24 | var valuebar0_text;
25 | var valuebar50_text;
26 | var vvaluebar50_text;
27 | var alphaImage;
28 |
29 | function create() {
30 |
31 | // Create a valuebar starting at 0.
32 | valuebar0 = new uiWidgets.ValueBar(
33 | this,
34 | {"x": 50, "y": 10},
35 | {"step": 1, "startValue": 0, maxValue: 100},
36 | true,
37 | false,
38 | "track",
39 | "bar",
40 | {'duration': 100, 'ease': Phaser.Math.Easing.Quadratic.Out}
41 | );
42 |
43 | valuebar0_text = this.add.text(50, 50, valuebar0.valueRange.startValue, {
44 | font: "65px Arial",
45 | fill: "#ff0044",
46 | align: "center"
47 | });
48 |
49 | valuebar0.valueDisplay = valuebar0_text;
50 |
51 | // Use the onMovement signal to update the display of text when the bar is moved.
52 | valuebar0.emitter.on('movement', updateValueDisplay);
53 |
54 | // Create a valuebar starting at 50.
55 | valuebar50 = new uiWidgets.ValueBar(
56 | this,
57 | {"x": 50, "y": 150},
58 | {"step": 25, "startValue": 50, maxValue: 100},
59 | true,
60 | false,
61 | "track",
62 | "bar",
63 | {'duration': 100, 'ease': Phaser.Math.Easing.Quadratic.Out}
64 | );
65 |
66 | valuebar50_text = this.add.text(50, 200, valuebar50.valueRange.startValue, {
67 | font: "65px Arial",
68 | fill: "#ff0044",
69 | align: "center"
70 | });
71 |
72 | valuebar50.valueDisplay = valuebar50_text;
73 | valuebar50.emitter.on('movement', updateValueDisplay);
74 |
75 | // Create a vertical valuebar starting at 50.
76 | vvaluebar50 = new uiWidgets.ValueBar(
77 | this,
78 | {"x": 350, "y": 50},
79 | {"step": 25, "startValue": 50, maxValue: 100},
80 | true,
81 | true,
82 | "vtrack",
83 | "bar",
84 | {'duration': 100, 'ease': Phaser.Math.Easing.Quadratic.Out}
85 | );
86 |
87 | vvaluebar50_text = this.add.text(400, 200, valuebar50.valueRange.startValue, {
88 | font: "65px Arial",
89 | fill: "#ff0044",
90 | align: "center"
91 | });
92 |
93 | vvaluebar50.valueDisplay = vvaluebar50_text;
94 | vvaluebar50.emitter.on('movement', updateValueDisplay);
95 |
96 | alphaImage = this.add.sprite(500, 100, "alphaImage");
97 | alphaImage.alpha = vvaluebar50.valueRange.getCurrentValue() / 100;
98 |
99 | // Use the onMovement signal to updte the sprite's opacity whenever the bar is moved.
100 | vvaluebar50.emitter.on('movement', updateAlpha);
101 | }
102 |
103 | function updateValueDisplay(bar) {
104 | bar.valueDisplay.setText(bar.valueRange.getCurrentValue());
105 | }
106 |
107 | function updateAlpha(bar) {
108 | alphaImage.alpha = bar.valueRange.getCurrentValue() / 100;
109 | }
110 |
--------------------------------------------------------------------------------
/docs/examples/js/phaser3/vscrollbar.js:
--------------------------------------------------------------------------------
1 | var config = {
2 | type: Phaser.AUTO,
3 | width: 600,
4 | height: 400,
5 | scene: {
6 | preload: preload,
7 | create: create,
8 | }
9 | };
10 |
11 | var game = new Phaser.Game(config);
12 |
13 | function preload() {
14 | var assetRoot = '../../assets/vertical/';
15 | this.load.image("dummyButton", assetRoot + "sprite.png");
16 | this.load.image("track", assetRoot + "track.png");
17 | this.load.spritesheet('bar', assetRoot + 'bar.png', { frameWidth: 22, frameHeight: 44 });
18 |
19 | }
20 |
21 | function create() {
22 |
23 | // Create a viewport. A "window" with a limited area of view.
24 | var viewport = new uiWidgets.Viewport(this, 75, 75, 600, 260);
25 |
26 | // Create a column. Anything added to a column is placed under the previous thing added.
27 | var column = new uiWidgets.Column(this);
28 |
29 | // Put the column inside the viewport.
30 | viewport.addNode(column);
31 |
32 | // Add things to the column.
33 | var dummy_sprite_a = this.add.image(0, 0, "dummyButton");
34 | var dummy_sprite_b = this.add.image(0, 0, "dummyButton");
35 | var dummy_sprite_c = this.add.image(0, 0, "dummyButton");
36 | var dummy_sprite_d = this.add.image(0, 0, "dummyButton");
37 | var dummy_sprite_e = this.add.image(0, 0, "dummyButton");
38 | var dummy_sprite_f = this.add.image(0, 0, "dummyButton");
39 |
40 | column.addNode(dummy_sprite_a);
41 | column.addNode(dummy_sprite_b);
42 | column.addNode(dummy_sprite_c);
43 | column.addNode(dummy_sprite_d);
44 | column.addNode(dummy_sprite_e);
45 | column.addNode(dummy_sprite_f);
46 |
47 | // Create a scrollbar for the viewport.
48 | var scrollbar = new uiWidgets.Scrollbar(
49 | this,
50 | viewport,
51 | true,
52 | true,
53 | "track",
54 | "bar",
55 | {'duration': 300, 'ease': Phaser.Math.Easing.Quadratic.Out}
56 | );
57 |
58 | // Place scrollbar next to viewport.
59 | Phaser.Display.Align.To.RightCenter(scrollbar, viewport, 300 + 10, 0);
60 |
61 | }
62 |
--------------------------------------------------------------------------------
/docs/examples/js/phaser3/wheel3D.js:
--------------------------------------------------------------------------------
1 | var config = {
2 | type: Phaser.AUTO,
3 | width: 600,
4 | height: 400,
5 | scene: {
6 | preload: preload,
7 | create: create,
8 | }
9 | };
10 |
11 | var game = new Phaser.Game(config);
12 |
13 | function preload() {
14 | var assetRoot = '../../assets/wheel3D/';
15 | this.load.image("icon", assetRoot + "icon.png");
16 | this.load.image("mmUp", assetRoot + "mmUp.png");
17 | this.load.image("mmDown", assetRoot + "mmDown.png");
18 | }
19 |
20 | function create() {
21 |
22 | var centerX = this.cameras.main.centerX;
23 | var centerY = this.cameras.main.centerY;
24 |
25 | // Example 1: Rotate around the Y axis
26 |
27 | // Build a list of sprites for the wheel.
28 | var menuListY = [];
29 |
30 | for (var i = 0; i < 12; i++) {
31 | var icon = this.add.sprite(0, 0, 'icon');
32 | menuListY.push(icon);
33 | }
34 |
35 | // Create the wheel.
36 | menuWheelY = new uiWidgets.Wheel3D(
37 | this,
38 | {"x": centerX - 100, "y": centerY},
39 | menuListY,
40 | 0,
41 | 90,
42 | "y",
43 | {"x":0, "y": -90, "z": 0}
44 | );
45 |
46 | // Tint the active item after each movement.
47 | menuWheelY.emitter.on('complete',
48 | function(wheel) {
49 | for (var i = 0; i < 12; i++) {
50 | wheel.sprites[i].tint = 0xFFFFFF;
51 | }
52 | wheel.active.tint = 0xdc21ff;
53 | }
54 | );
55 |
56 | menuWheelY.activate();
57 |
58 | // Scroll Buttons
59 | var scrollLeftButtonY = new uiWidgets.Button(
60 | this,
61 | 0,
62 | centerY,
63 | "mmUp",
64 | scrollTheWheel,
65 | null,
66 | );
67 | scrollLeftButtonY.wheel = menuWheelY;
68 | scrollLeftButtonY.dir = 0;
69 |
70 | var scrollRightButtonY = new uiWidgets.Button(
71 | this,
72 | 0,
73 | centerY + 48,
74 | "mmDown",
75 | scrollTheWheel,
76 | null,
77 | );
78 | scrollRightButtonY.wheel = menuWheelY;
79 | scrollRightButtonY.dir = 1;
80 |
81 | // Example 2: Rotate around the X axis
82 | var menuListX = [];
83 |
84 | for (var i = 0; i < 12; i++) {
85 | var icon = this.add.sprite(0, 0, 'icon');
86 | menuListX.push(icon);
87 | }
88 |
89 | // Create the wheel.
90 | menuWheelX = new uiWidgets.Wheel3D(
91 | this,
92 | {"x": centerX - 100, "y": 100},
93 | menuListX,
94 | 0,
95 | 90,
96 | "x",
97 | {"x": 90, "y": 0, "z": 0}
98 | );
99 |
100 | // Tint the active item after each movement.
101 | menuWheelX.emitter.on('complete',
102 | function(wheel) {
103 | for (var i = 0; i < 12; i++) {
104 | wheel.sprites[i].tint = 0xFFFFFF;
105 | }
106 | wheel.active.tint = 0xdc21ff;
107 | }
108 | );
109 |
110 | menuWheelX.activate();
111 |
112 | // Scroll Buttons
113 | var scrollLeftButtonX = new uiWidgets.Button(
114 | this,
115 | 0,
116 | 0,
117 | "mmUp",
118 | scrollTheWheel,
119 | null,
120 | );
121 | scrollLeftButtonX.wheel = menuWheelX;
122 | scrollLeftButtonX.dir = 0;
123 |
124 | var scrollRightButtonX = new uiWidgets.Button(
125 | this,
126 | 0,
127 | 48,
128 | "mmDown",
129 | scrollTheWheel,
130 | null,
131 | );
132 | scrollRightButtonX.wheel = menuWheelX;
133 | scrollRightButtonX.dir = 1;
134 |
135 | // Example 3: Rotate around the Z axis
136 | var menuListZ = [];
137 |
138 | for (var i = 0; i < 12; i++) {
139 | var icon = this.add.sprite(0, 0, 'icon');
140 | menuListZ.push(icon);
141 | }
142 |
143 | // Create the wheel.
144 | menuWheelZ = new uiWidgets.Wheel3D(
145 | this,
146 | {"x": centerX, "y": 100},
147 | menuListZ,
148 | 0,
149 | 90,
150 | "z",
151 | {"x": 0, "y": 0, "z": 90}
152 | );
153 |
154 | menuWheelZ.emitter.on('start',
155 | function(wheel) {
156 | for (var i = 0; i < 12; i++) {
157 | wheel.sprites[i].tint = 0xFFFFFF;
158 | }
159 | wheel.active.tint = 0xdc21ff;
160 | }
161 | );
162 |
163 | menuWheelZ.activate();
164 |
165 | // Scroll Buttons
166 | var scrollLeftButtonZ = new uiWidgets.Button(
167 | this,
168 | this.cameras.main.width - 100,
169 | 0,
170 | "mmUp",
171 | scrollTheWheel,
172 | null,
173 | );
174 | scrollLeftButtonZ.wheel = menuWheelZ;
175 | scrollLeftButtonZ.dir = 0;
176 |
177 | var scrollRightButtonZ = new uiWidgets.Button(
178 | this,
179 | this.cameras.main.width - 100,
180 | 48,
181 | "mmDown",
182 | scrollTheWheel,
183 | null,
184 | );
185 | scrollRightButtonZ.wheel = menuWheelZ;
186 | scrollRightButtonZ.dir = 1;
187 | }
188 |
189 | var scrollTheWheel = function () {
190 | if (this.dir === 0){
191 | this.wheel.moveBack();
192 | }
193 | if (this.dir === 1){
194 | this.wheel.moveForward();
195 | }
196 | };
197 |
--------------------------------------------------------------------------------
/docs/examples/js/phaserce/hscrollbar.js:
--------------------------------------------------------------------------------
1 | var game = new Phaser.Game(400, 300, Phaser.AUTO, 'hscrollbar', { preload: preload, create: create });
2 |
3 | function preload() {
4 | var assetRoot = '../../assets/horizontal/';
5 | game.load.image("dummyButton", assetRoot + "sprite.png");
6 | game.load.image("track", assetRoot + "track.png");
7 | game.load.spritesheet('bar', assetRoot + 'bar.png', 44, 22);
8 | }
9 |
10 | function create() {
11 | // Create a viewport. A "window" with a limited area of view.
12 | var viewport = new uiWidgets.Viewport(game, 75, 75, 260, 128);
13 |
14 | // Create a row. Anything added to a row is placed next to the previous thing added.
15 | var row = new uiWidgets.Row(game);
16 |
17 | // Put the row inside the viewport.
18 | viewport.addNode(row);
19 |
20 | // Add things to the row.
21 | var dummy_sprite_a = this.game.add.sprite(0, 0, "dummyButton");
22 | var dummy_sprite_b = this.game.add.sprite(0, 0, "dummyButton");
23 | var dummy_sprite_c = this.game.add.sprite(0, 0, "dummyButton");
24 | var dummy_sprite_d = this.game.add.sprite(0, 0, "dummyButton");
25 | var dummy_sprite_e = this.game.add.sprite(0, 0, "dummyButton");
26 | var dummy_sprite_f = this.game.add.sprite(0, 0, "dummyButton");
27 | var dummy_sprite_g = this.game.add.sprite(0, 0, "dummyButton");
28 | var dummy_sprite_h = this.game.add.sprite(0, 0, "dummyButton");
29 | var dummy_sprite_i = this.game.add.sprite(0, 0, "dummyButton");
30 |
31 | row.addNode(dummy_sprite_a);
32 | row.addNode(dummy_sprite_b);
33 | row.addNode(dummy_sprite_c);
34 | row.addNode(dummy_sprite_d);
35 | row.addNode(dummy_sprite_e);
36 | row.addNode(dummy_sprite_f);
37 | row.addNode(dummy_sprite_g);
38 | row.addNode(dummy_sprite_h);
39 | row.addNode(dummy_sprite_i);
40 |
41 | // Create a scrollbar for the viewport.
42 | var scrollbar = new uiWidgets.Scrollbar(
43 | game,
44 | viewport,
45 | true,
46 | false,
47 | "track",
48 | "bar",
49 | {'duration': 300, 'ease': Phaser.Easing.Quadratic.Out}
50 | );
51 |
52 | // Place scrollbar below viewport.
53 | scrollbar.alignTo(viewport, Phaser.BOTTOM_LEFT, 0, 10);
54 | }
55 |
--------------------------------------------------------------------------------
/docs/examples/js/phaserce/modal.js:
--------------------------------------------------------------------------------
1 | var game = new Phaser.Game(
2 | 400,
3 | 300,
4 | Phaser.AUTO,
5 | 'textover',
6 | {preload: preload, create: create},
7 | false,
8 | false,
9 | );
10 |
11 | var textStyle = {'fill': '#FFF', 'font': '16px Courier New'};
12 |
13 | function preload() {
14 | var assetRoot = '../../assets/modal/';
15 | game.load.image("modalBg", assetRoot + "background.png");
16 | game.load.spritesheet('button', assetRoot + 'button.png', 128, 48);
17 | }
18 |
19 | function create() {
20 | var openButton = new uiWidgets.TextButton(this.game, 100, 50, "button", createModal, this, 1, 0, 2, 1)
21 | .setText("Open", textStyle)
22 | .eventTextYAdjustment(3);
23 |
24 | console.log(openButton.x, openButton.button.x, openButton.text.x)
25 |
26 | }
27 |
28 | function createModal() {
29 | var closeButton = new uiWidgets.TextButton(this.game, 0, 0, "button", closeCallback, this, 1, 0, 2, 1)
30 | .setText("Close", textStyle)
31 | .eventTextYAdjustment(3);
32 |
33 | this.modal = new uiWidgets.Column(this.game, 100, 100, bg="modalBg", modal=true);
34 | this.modal.addNode(closeButton, paddingX=0, paddingY=10);
35 | }
36 |
37 | function closeCallback() {
38 | this.modal.dismiss();
39 | }
40 |
--------------------------------------------------------------------------------
/docs/examples/js/phaserce/quantitybar.js:
--------------------------------------------------------------------------------
1 | var game = new Phaser.Game(600, 400, Phaser.AUTO, 'quantitybar', { preload: preload, create: create});
2 |
3 | function preload() {
4 | var assetRoot = '../../assets/quantitybar/';
5 | game.load.image("track", assetRoot + "horizontal/track.png");
6 | game.load.spritesheet('bar', assetRoot + 'horizontal/bar.png', 260, 32);
7 |
8 | game.load.image("vtrack", assetRoot + "vertical/track.png");
9 | game.load.spritesheet('vbar', assetRoot + 'vertical/bar.png', 32, 260);
10 |
11 | game.load.image("add", assetRoot + "add.png");
12 | game.load.image("subtract", assetRoot + "subtract.png");
13 | }
14 |
15 | var healthbar,
16 | healthbarText,
17 | vhealthbar,
18 | vhealthbarText,
19 | rhealthbar,
20 | rhealthbarText,
21 | rvhealthbar,
22 | rvhealthbarText,
23 | decreaseHealth,
24 | increaseHealth,
25 | vdecreaseHealth,
26 | vincreaseHealth,
27 | rdecreaseHealth,
28 | rincreaseHealth,
29 | rvdecreaseHealth,
30 | rvincreaseHealth,
31 | textStyle;
32 |
33 | function create() {
34 |
35 | textStyle = {
36 | font: "65px Arial",
37 | fill: "#ff0044",
38 | align: "center"
39 | };
40 |
41 | // Create a quantitybar starting at 50.
42 | healthbar = new uiWidgets.QuantityBar(
43 | game,
44 | {"x": 50, "y": 10},
45 | {"startValue": 50, maxValue: 100},
46 | false,
47 | false,
48 | "track",
49 | "bar",
50 | {'duration': 400, 'ease': Phaser.Easing.Quadratic.Out}
51 | );
52 |
53 | // Display the quantitybar's value.
54 | healthbarText = game.add.text(50, 50, healthbar.valueRange.startValue, textStyle);
55 |
56 | // Add buttons to modify the value.
57 | var lessHealth = game.add.button(150, 10, 'subtract', decreaseHealth, this);
58 | var moreHealth = game.add.button(200, 10, 'add', increaseHealth, this);
59 |
60 | // Create a reverse quantitybar starting at 30.
61 | rhealthbar = new uiWidgets.QuantityBar(
62 | game,
63 | {"x": 350, "y": 10},
64 | {"startValue": 30, maxValue: 100},
65 | false,
66 | true,
67 | "track",
68 | "bar",
69 | {'duration': 400, 'ease': Phaser.Easing.Quadratic.Out}
70 | );
71 |
72 | rhealthbarText = game.add.text(350, 50, rhealthbar.valueRange.startValue, textStyle);
73 |
74 | var rlessHealth = game.add.button(450, 10, 'subtract', rdecreaseHealth, this);
75 | var rmoreHealth = game.add.button(500, 10, 'add', rincreaseHealth, this);
76 |
77 | // Create a vertical quantitybar starting at 50.
78 | vhealthbar = new uiWidgets.QuantityBar(
79 | game,
80 | {"x": 30, "y": 130},
81 | {"startValue": 32, maxValue: 100},
82 | true,
83 | false,
84 | "vtrack",
85 | "vbar",
86 | {'duration': 400, 'ease': Phaser.Easing.Quadratic.Out}
87 | );
88 |
89 | vhealthbarText = game.add.text(100, 150, vhealthbar.valueRange.startValue, textStyle);
90 |
91 | var vlessHealth = game.add.button(100, 250, 'subtract', vdecreaseHealth, this);
92 | var vmoreHealth = game.add.button(150, 250, 'add', vincreaseHealth, this);
93 |
94 | // Create a reverse vertical quantitybar starting at 50.
95 | rvhealthbar = new uiWidgets.QuantityBar(
96 | game,
97 | {"x": 250, "y": 130},
98 | {"startValue": 30, maxValue: 100},
99 | true,
100 | true,
101 | "vtrack",
102 | "vbar",
103 | {'duration': 400, 'ease': Phaser.Easing.Quadratic.Out}
104 | );
105 |
106 | rvhealthbarText = game.add.text(300, 150, rvhealthbar.valueRange.startValue, textStyle);
107 |
108 | var vlessHealth = game.add.button(300, 250, 'subtract', rvdecreaseHealth, this);
109 | var vmoreHealth = game.add.button(350, 250, 'add', rvincreaseHealth, this);
110 |
111 | }
112 |
113 | function increaseHealth() {
114 | healthbar.adjustBar(10);
115 | healthbarText.setText(healthbar.valueRange.getCurrentValue());
116 | }
117 |
118 | function decreaseHealth() {
119 | healthbar.adjustBar(-10);
120 | healthbarText.setText(healthbar.valueRange.getCurrentValue());
121 | }
122 |
123 | function vincreaseHealth() {
124 | vhealthbar.adjustBar(1);
125 | vhealthbarText.setText(vhealthbar.valueRange.getCurrentValue());
126 | }
127 |
128 | function vdecreaseHealth() {
129 | vhealthbar.adjustBar(-1);
130 | vhealthbarText.setText(vhealthbar.valueRange.getCurrentValue());
131 | }
132 |
133 | function rincreaseHealth() {
134 | rhealthbar.adjustBar(10);
135 | rhealthbarText.setText(rhealthbar.valueRange.getCurrentValue());
136 | }
137 |
138 | function rdecreaseHealth() {
139 | rhealthbar.adjustBar(-10);
140 | rhealthbarText.setText(rhealthbar.valueRange.getCurrentValue());
141 | }
142 |
143 | function rvincreaseHealth() {
144 | rvhealthbar.adjustBar(10);
145 | rvhealthbarText.setText(rvhealthbar.valueRange.getCurrentValue());
146 | }
147 |
148 | function rvdecreaseHealth() {
149 | rvhealthbar.adjustBar(-10);
150 | rvhealthbarText.setText(rvhealthbar.valueRange.getCurrentValue());
151 | }
152 |
--------------------------------------------------------------------------------
/docs/examples/js/phaserce/quantitybar_column.js:
--------------------------------------------------------------------------------
1 | var game = new Phaser.Game(600, 400, Phaser.AUTO, 'quantitybar_column', { preload: preload, create: create, update: update });
2 |
3 | function preload() {
4 | var assetRoot = '../../assets/quantitybar/horizontal/';
5 | game.load.image("track", assetRoot + "track.png");
6 | game.load.image('bar', assetRoot + 'bar.png');
7 |
8 | game.load.image("add", assetRoot + "add.png");
9 | game.load.image("subtract", assetRoot + "subtract.png");
10 | }
11 |
12 | var healthbar;
13 | var healthbarText;
14 | var lessHealth;
15 | var moreHealth;
16 | var column;
17 |
18 | function create() {
19 |
20 | // Create a quantitybar starting at 50.
21 | healthbar = new uiWidgets.QuantityBar(
22 | game,
23 | {"x": 0, "y": 0},
24 | {"startValue": 50, maxValue: 100},
25 | false,
26 | false,
27 | "track",
28 | "bar",
29 | {'duration': 400, 'ease': Phaser.Easing.Quadratic.Out}
30 | );
31 |
32 | healthbarText = game.add.text(50, 50, healthbar.valueRange.minValue, {
33 | font: "65px Arial",
34 | fill: "#ff0044",
35 | align: "center"
36 | });
37 |
38 | lessHealth = game.add.button(150, 10, 'subtract', decreaseHealth, this);
39 | moreHealth = game.add.button(200, 10, 'add', increaseHealth, this);
40 |
41 | column = new uiWidgets.Column(game, 30, 0);
42 | column.addNode(healthbar);
43 | column.addNode(healthbarText);
44 | console.log("a", healthbar.track.x, healthbar.track.worldPosition.x)
45 | console.log("b", healthbar.bar.x, healthbar.bar.worldPosition.x)
46 | console.log("c", healthbar.x, healthbar.worldPosition.x)
47 | //console.log(healthbar.bar.mask.x, healthbar.bar.mask.worldPosition.x)
48 | //healthbar.bar.mask.x = 30
49 | //healthbar.bar.mask.y = 30
50 |
51 | }
52 |
53 | function update() {
54 | healthbarText.setText(healthbar.valueRange.getCurrentValue());
55 | }
56 |
57 | function increaseHealth() {
58 | healthbar.adjustBar(10);
59 |
60 | }
61 |
62 | function decreaseHealth() {
63 | healthbar.adjustBar(-10);
64 | }
65 |
--------------------------------------------------------------------------------
/docs/examples/js/phaserce/textover.js:
--------------------------------------------------------------------------------
1 | var game = new Phaser.Game(
2 | 400,
3 | 300,
4 | Phaser.AUTO,
5 | 'textover',
6 | {preload: preload, create: create},
7 | false,
8 | false,
9 | );
10 |
11 | var textStyle = {'fill': '#FFF', 'font': '16px Courier New'};
12 |
13 | function preload() {
14 | var assetRoot = '../../assets/textover/';
15 | game.load.image("header", assetRoot + "header.png");
16 | game.load.spritesheet('button', assetRoot + 'button.png', 128, 48);
17 | }
18 |
19 | function create() {
20 | this.header = new uiWidgets.TextSprite(game, 0, 0, "header").setText('Header', textStyle);
21 |
22 | var buttonOne = new uiWidgets.TextButton(game, 0, 0, "button", newGameCallback, this, 1, 0, 2, 1)
23 | .setText("New Game", textStyle)
24 | .eventTextYAdjustment(3);
25 | var buttonTwo = new uiWidgets.TextButton(game, 0, 0, "button", continueCallback, this, 1, 0, 2, 1)
26 | .setText("Continue", textStyle)
27 | .eventTextYAdjustment(3);
28 | var buttonThree = new uiWidgets.TextButton(game, 0, 0, "button", optionsCallback, this, 1, 0, 2, 1)
29 | .setText("Options", textStyle)
30 | .eventTextYAdjustment(3);
31 |
32 | var column = new uiWidgets.Column(this, 200, 100);
33 | column.addNode(buttonOne, paddingX=0, paddingY=10);
34 | column.addNode(buttonTwo, paddingX=0, paddingY=10);
35 | column.addNode(buttonThree, paddingX=0, paddingY=10);
36 |
37 | }
38 |
39 | function newGameCallback() {
40 | this.header.text.setText('You clicked the New Game button');
41 | }
42 |
43 |
44 | function continueCallback() {
45 | this.header.text.setText('You clicked the Continue button');
46 | }
47 |
48 |
49 | function optionsCallback() {
50 | this.header.text.setText('You clicked the Options button');
51 | }
52 |
--------------------------------------------------------------------------------
/docs/examples/js/phaserce/valuebar.js:
--------------------------------------------------------------------------------
1 | var game = new Phaser.Game(600, 400, Phaser.AUTO, 'valuebar', { preload: preload, create: create });
2 |
3 | function preload() {
4 | var assetRoot = '../../assets/valuebar/';
5 | game.load.image("track", assetRoot + "track.png");
6 | game.load.spritesheet('bar', assetRoot + 'bar.png', 32, 32);
7 | game.load.image("vtrack", assetRoot + "vtrack.png");
8 | game.load.image("alphaImage", assetRoot + "alpha.png");
9 | }
10 |
11 | var valuebar0;
12 | var valuebar50;
13 | var vvaluebar50;
14 | var valuebar0_text;
15 | var valuebar50_text;
16 | var vvaluebar50_text;
17 | var alphaImage;
18 |
19 | function create() {
20 |
21 | // Create a valuebar starting at 0.
22 | valuebar0 = new uiWidgets.ValueBar(
23 | game,
24 | {"x": 50, "y": 10},
25 | {"step": 1, "startValue": 0, maxValue: 100},
26 | true,
27 | false,
28 | "track",
29 | "bar",
30 | {'duration': 100, 'ease': Phaser.Easing.Quadratic.Out}
31 | );
32 |
33 | valuebar0_text = game.add.text(50, 50, valuebar0.valueRange.startValue, {
34 | font: "65px Arial",
35 | fill: "#ff0044",
36 | align: "center"
37 | });
38 |
39 | valuebar0.valueDisplay = valuebar0_text;
40 |
41 | // Use the onMovement signal to update the display of text when the bar is moved.
42 | valuebar0.emitter.on('movement', updateValueDisplay);
43 |
44 | // Create a valuebar starting at 50.
45 | valuebar50 = new uiWidgets.ValueBar(
46 | game,
47 | {"x": 50, "y": 150},
48 | {"step": 25, "startValue": 50, maxValue: 100},
49 | true,
50 | false,
51 | "track",
52 | "bar",
53 | {'duration': 100, 'ease': Phaser.Easing.Quadratic.Out}
54 | );
55 |
56 | valuebar50_text = game.add.text(50, 200, valuebar50.valueRange.startValue, {
57 | font: "65px Arial",
58 | fill: "#ff0044",
59 | align: "center"
60 | });
61 |
62 | valuebar50.valueDisplay = valuebar50_text;
63 | valuebar50.emitter.on('movement', updateValueDisplay);
64 |
65 | // Create a vertical valuebar starting at 50.
66 | vvaluebar50 = new uiWidgets.ValueBar(
67 | game,
68 | {"x": 350, "y": 50},
69 | {"step": 25, "startValue": 50, maxValue: 100},
70 | true,
71 | true,
72 | "vtrack",
73 | "bar",
74 | {'duration': 100, 'ease': Phaser.Easing.Quadratic.Out}
75 | );
76 |
77 | vvaluebar50_text = game.add.text(400, 200, valuebar50.valueRange.startValue, {
78 | font: "65px Arial",
79 | fill: "#ff0044",
80 | align: "center"
81 | });
82 |
83 | vvaluebar50.valueDisplay = vvaluebar50_text;
84 | vvaluebar50.emitter.on('movement', updateValueDisplay);
85 |
86 | alphaImage = game.add.sprite(500, 100, "alphaImage");
87 | alphaImage.alpha = vvaluebar50.valueRange.getCurrentValue() / 100;
88 |
89 | // Use the onMovement signal to updte the sprite's opacity whenever the bar is moved.
90 | vvaluebar50.emitter.on('movement', updateAlpha);
91 | }
92 |
93 | function updateValueDisplay(bar) {
94 | bar.valueDisplay.setText(bar.valueRange.getCurrentValue());
95 | }
96 |
97 | function updateAlpha(bar) {
98 | alphaImage.alpha = bar.valueRange.getCurrentValue() / 100;
99 | }
100 |
--------------------------------------------------------------------------------
/docs/examples/js/phaserce/valuebar_column.js:
--------------------------------------------------------------------------------
1 | var game = new Phaser.Game(600, 400, Phaser.AUTO, 'valuebar_column', { preload: preload, create: create });
2 |
3 | function preload() {
4 | var assetRoot = '../../assets/valuebar/';
5 | game.load.image("bg", assetRoot + "background.png");
6 | game.load.image("track", assetRoot + "track.png");
7 | game.load.spritesheet('bar', assetRoot + 'bar.png', 32, 32);
8 | game.load.image("pointer", assetRoot + "pointer.png");
9 |
10 | game.load.image("vtrack", "assets/valuebar/vtrack.png");
11 | }
12 |
13 | var bar;
14 | var bar2;
15 | var barText;
16 | var barText2;
17 | var lessHealth;
18 | var moreHealth;
19 | var column;
20 | var keyboardGroup;
21 | var cursor;
22 |
23 | function create() {
24 | // Create a quantitybar starting at 50.
25 | bar = new uiWidgets.ValueBar(
26 | game,
27 | {"x": 0, "y": 0},
28 | {"step": 25, "startValue": 50, maxValue: 100},
29 | true,
30 | false,
31 | "track",
32 | "bar",
33 | {'duration': 100, 'ease': Phaser.Easing.Quadratic.Out}
34 | );
35 |
36 | barText = game.add.text(50, 50, bar.valueRange.startValue, {
37 | font: "24px Arial",
38 | fill: "#ff0044",
39 | align: "center"
40 | });
41 |
42 | bar.valueDisplay = barText;
43 | bar.emitter.on('movement', updateValueDisplay);
44 |
45 | // Create a quantitybar starting at 50.
46 | bar2 = new uiWidgets.ValueBar(
47 | game,
48 | {"x": 0, "y": 0},
49 | {"step": 25, "startValue": 50, maxValue: 100},
50 | true,
51 | false,
52 | "track",
53 | "bar",
54 | {'duration': 100, 'ease': Phaser.Easing.Quadratic.Out}
55 | );
56 |
57 | barText2 = game.add.text(50, 50, bar2.valueRange.startValue, {
58 | font: "24px Arial",
59 | fill: "#ff0044",
60 | align: "center"
61 | });
62 |
63 | bar2.valueDisplay = barText2;
64 | bar2.emitter.on('movement', updateValueDisplay);
65 |
66 | var prevItemCallback = function (group, context) {
67 | cursor.y = group.selected.worldPosition.y - (group.selected.height/4);
68 | };
69 |
70 | var nextItemCallback = function (group, context) {
71 | cursor.y = group.selected.worldPosition.y - (group.selected.height/4);
72 | };
73 |
74 | column = new uiWidgets.Column(game, 150, 50, "bg");
75 | column.addNode(bar);
76 | column.addNode(barText);
77 | column.addNode(bar2);
78 | column.addNode(barText2);
79 |
80 | // Example of using prev/nextItem callback.
81 | // Create a cursor and move to the selected child's position.
82 | cursor = game.add.sprite(100, 42, 'pointer');
83 |
84 | // Create a KeyboardGroup and add the Bars to it.
85 | keyboardGroup = new uiWidgets.KeyboardGroup(game, true, this);
86 | keyboardGroup.addNode(bar);
87 | keyboardGroup.addNode(bar2);
88 |
89 | keyboardGroup.emitter.on('previous', prevItemCallback);
90 | keyboardGroup.emitter.on('next', nextItemCallback);
91 | }
92 |
93 | function updateValueDisplay(bar) {
94 | bar.valueDisplay.setText(bar.valueRange.getCurrentValue());
95 | }
96 |
--------------------------------------------------------------------------------
/docs/examples/js/phaserce/vscrollbar.js:
--------------------------------------------------------------------------------
1 | var game = new Phaser.Game(600, 400, Phaser.AUTO, 'vscrollbar', { preload: preload, create: create });
2 |
3 | function preload() {
4 | var assetRoot = '../../assets/vertical/';
5 | game.load.image("dummyButton", assetRoot + "sprite.png");
6 | game.load.image("track", assetRoot + "track.png");
7 | game.load.spritesheet('bar', assetRoot + 'bar.png', 22, 44);
8 | }
9 |
10 | function create() {
11 |
12 | // Create a viewport. A "window" with a limited area of view.
13 | var viewport = new uiWidgets.Viewport(game, 75, 75, 600, 260);
14 |
15 | // Create a column. Anything added to a column is placed under the previous thing added.
16 | var column = new uiWidgets.Column(game);
17 |
18 | // Put the column inside the viewport.
19 | viewport.addNode(column);
20 |
21 | // Add things to the column.
22 | var dummy_sprite_a = this.game.add.sprite(0, 0, "dummyButton");
23 | var dummy_sprite_b = this.game.add.sprite(0, 0, "dummyButton");
24 | var dummy_sprite_c = this.game.add.sprite(0, 0, "dummyButton");
25 | var dummy_sprite_d = this.game.add.sprite(0, 0, "dummyButton");
26 | var dummy_sprite_e = this.game.add.sprite(0, 0, "dummyButton");
27 | var dummy_sprite_f = this.game.add.sprite(0, 0, "dummyButton");
28 |
29 | column.addNode(dummy_sprite_a);
30 | column.addNode(dummy_sprite_b);
31 | column.addNode(dummy_sprite_c);
32 | column.addNode(dummy_sprite_d);
33 | column.addNode(dummy_sprite_e);
34 | column.addNode(dummy_sprite_f);
35 |
36 | // Create a scrollbar for the viewport.
37 | var scrollbar = new uiWidgets.Scrollbar(
38 | game,
39 | viewport,
40 | true,
41 | true,
42 | "track",
43 | "bar",
44 | {'duration': 300, 'ease': Phaser.Easing.Quadratic.Out}
45 | );
46 |
47 | // Place scrollbar next to viewport.
48 | scrollbar.alignTo(viewport, Phaser.RIGHT_TOP, 10);
49 | }
50 |
--------------------------------------------------------------------------------
/docs/examples/js/phaserce/wheel3D.js:
--------------------------------------------------------------------------------
1 | var game = new Phaser.Game(
2 | 600,
3 | 400,
4 | Phaser.AUTO,
5 | 'wheel3D',
6 | { preload: preload, create: create },
7 | );
8 |
9 | function preload() {
10 | var assetRoot = '../../assets/wheel3D/';
11 | game.load.image("icon", assetRoot + "icon.png");
12 | game.load.image("mmUp", assetRoot + "mmUp.png");
13 | game.load.image("mmDown", assetRoot + "mmDown.png");
14 | }
15 |
16 | function create() {
17 |
18 | // Example 1: Rotate around the Y axis
19 |
20 | // Build a list of sprites for the wheel.
21 | var menuListY = [];
22 |
23 | for (var i = 0; i < 12; i++) {
24 | var icon = game.add.sprite(0, 0, 'icon');
25 | menuListY.push(icon);
26 | }
27 |
28 | // Create the wheel.
29 | menuWheelY = new uiWidgets.Wheel3D(
30 | game,
31 | {"x": game.world.centerX - 100, "y": game.world.centerY},
32 | menuListY,
33 | 0,
34 | 90,
35 | "y",
36 | {"x":0, "y": -90, "z": 0}
37 | );
38 |
39 | // Tint the active item after each movement.
40 | menuWheelY.emitter.on('complete',
41 | function(wheel) {
42 | for (var i = 0; i < 12; i++) {
43 | wheel.sprites[i].tint = 0xFFFFFF;
44 | }
45 | wheel.active.tint = 0xdc21ff;
46 | }
47 | );
48 |
49 | menuWheelY.activate();
50 |
51 | // Scroll Buttons
52 | var scrollLeftButtonY = this.game.add.button(
53 | 0,
54 | game.world.centerY,
55 | "mmUp",
56 | scrollTheWheel,
57 | this,
58 | );
59 | scrollLeftButtonY.wheel = menuWheelY;
60 | scrollLeftButtonY.dir = 0;
61 |
62 | var scrollRightButtonY = this.game.add.button(
63 | 0,
64 | game.world.centerY + 48,
65 | "mmDown",
66 | scrollTheWheel,
67 | this,
68 | );
69 | scrollRightButtonY.wheel = menuWheelY;
70 | scrollRightButtonY.dir = 1;
71 |
72 | // Example 2: Rotate around the X axis
73 | var menuListX = [];
74 |
75 | for (var i = 0; i < 12; i++) {
76 | var icon = game.add.sprite(0, 0, 'icon');
77 | menuListX.push(icon);
78 | }
79 |
80 | // Create the wheel.
81 | menuWheelX = new uiWidgets.Wheel3D(
82 | game,
83 | {"x": game.world.centerX - 100, "y": 100},
84 | menuListX,
85 | 0,
86 | 90,
87 | "x",
88 | {"x": 90, "y": 0, "z": 0}
89 | );
90 |
91 | // Tint the active item after each movement.
92 | menuWheelX.emitter.on('complete',
93 | function(wheel) {
94 | for (var i = 0; i < 12; i++) {
95 | wheel.sprites[i].tint = 0xFFFFFF;
96 | }
97 | wheel.active.tint = 0xdc21ff;
98 | }
99 | );
100 |
101 | menuWheelX.activate();
102 |
103 | // Scroll Buttons
104 | var scrollLeftButtonX = this.game.add.button(
105 | 0,
106 | 0,
107 | "mmUp",
108 | scrollTheWheel,
109 | this,
110 | );
111 | scrollLeftButtonX.wheel = menuWheelX;
112 | scrollLeftButtonX.dir = 0;
113 |
114 | var scrollRightButtonX = this.game.add.button(
115 | 0,
116 | 48,
117 | "mmDown",
118 | scrollTheWheel,
119 | this,
120 | );
121 | scrollRightButtonX.wheel = menuWheelX;
122 | scrollRightButtonX.dir = 1;
123 |
124 | // Example 3: Rotate around the Z axis
125 | var menuListZ = [];
126 |
127 | for (var i = 0; i < 12; i++) {
128 | var icon = game.add.sprite(0, 0, 'icon');
129 | menuListZ.push(icon);
130 | }
131 |
132 | // Create the wheel.
133 | menuWheelZ = new uiWidgets.Wheel3D(
134 | game,
135 | {"x": game.world.centerX, "y": 100},
136 | menuListZ,
137 | 0,
138 | 90,
139 | "z",
140 | {"x": 0, "y": 0, "z": 90}
141 | );
142 |
143 | menuWheelZ.emitter.on('start',
144 | function(wheel) {
145 | for (var i = 0; i < 12; i++) {
146 | wheel.sprites[i].tint = 0xFFFFFF;
147 | }
148 | wheel.active.tint = 0xdc21ff;
149 | }
150 | );
151 |
152 | menuWheelZ.activate();
153 |
154 | // Scroll Buttons
155 | var scrollLeftButtonZ = this.game.add.button(
156 | game.world.width - 100,
157 | 0,
158 | "mmUp",
159 | scrollTheWheel,
160 | this,
161 | );
162 | scrollLeftButtonZ.wheel = menuWheelZ;
163 | scrollLeftButtonZ.dir = 0;
164 |
165 | var scrollRightButtonZ = this.game.add.button(
166 | game.world.width - 100,
167 | 48,
168 | "mmDown",
169 | scrollTheWheel,
170 | this,
171 | );
172 | scrollRightButtonZ.wheel = menuWheelZ;
173 | scrollRightButtonZ.dir = 1;
174 | }
175 |
176 | var scrollTheWheel = function (item) {
177 | if (item.dir === 0){
178 | item.wheel.moveBack();
179 | }
180 | if (item.dir === 1){
181 | item.wheel.moveForward();
182 | }
183 | };
184 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "phaser-ui-tools",
3 | "version": "2.0.0-beta",
4 | "description": "User Interface objects for the phaser game engine.",
5 | "author": "Joshua Fehler ",
6 | "repository": "github:jsfehler/phaser-ui-tools",
7 | "license": "MIT",
8 | "main": "dist/phaser-ui-tools.js",
9 | "unpkg": "dist/phaser-ui-tools.min.js",
10 | "files": [
11 | "dist/phaser-ui-tools.js",
12 | "dist/phaser-ui-tools.min.js"
13 | ],
14 | "scripts": {
15 | "build:js": "webpack",
16 | "test": "mocha-headless-chrome -f tests/testrunner.html",
17 | "test3": "mocha-headless-chrome -f tests/testrunner_phaser3.html",
18 | "copyToExamplesLib": "copyfiles -f dist/phaser-ui-tools.js docs/examples/lib/",
19 | "build:docs": "documentation build src/** -f html -o docs"
20 | },
21 | "devDependencies": {
22 | "@babel/core": "^7.23.9",
23 | "@babel/preset-env": "^7.23.9",
24 | "babel-loader": "^9.1.3",
25 | "chai": "^4.3.4",
26 | "copyfiles": "^2.4.1",
27 | "documentation": "^14.0.2",
28 | "eslint": "^8.56.0",
29 | "eslint-config-airbnb-base": "^15.0.0",
30 | "eslint-plugin-import": "^2.29.1",
31 | "eslint-webpack-plugin": "^4.0.1",
32 | "mocha": "^10.2.0",
33 | "mocha-headless-chrome": "^4.0.0",
34 | "webpack": "^5.90.0",
35 | "webpack-cli": "^5.1.4"
36 | },
37 | "peerDependencies": {
38 | "phaser": "^3.60.0",
39 | "phaser-ce": "2.20.0"
40 | },
41 | "peerDependenciesMeta": {
42 | "phaser-ce": {
43 | "optional": true
44 | },
45 | "phaser": {
46 | "optional": true
47 | }
48 | },
49 | "dependencies": {
50 | "eventemitter3": "^5.0.1"
51 | }
52 | }
53 |
--------------------------------------------------------------------------------
/src/bars/bar.js:
--------------------------------------------------------------------------------
1 | import * as PhaserObjects from '../phaserObjects';
2 |
3 | /** Base object for all Bar-like Widgets.
4 | * @private
5 | * @extends PhaserObjects.Group
6 | */
7 | export class Bar extends PhaserObjects.Group {
8 | /**
9 | * @param {Object} game - Current game instance.
10 | * @param {number} x - The Bar's x position.
11 | * @param {number} y - The Bar's y position.
12 | * @param {boolean} vertical - Sets the Bar's alignment as vertical.
13 | * @param {string} trackKey - The key to use for the track.
14 | * @param {string} barKey - The key to use for the bar.
15 | */
16 | constructor(game, x = 0, y = 0, vertical = false, trackKey = '', barKey = '') {
17 | super(game);
18 | game.add.existing(this);
19 |
20 | this.game = game;
21 | this.x = x;
22 | this.y = y;
23 |
24 | this.vertical = vertical;
25 |
26 | this.trackKey = trackKey;
27 | this.barKey = barKey;
28 | }
29 |
30 | /**
31 | * @private
32 | * Determine the distance the bar can scroll over.
33 | */
34 | setTrackScrollAreaSize() {
35 | if (this.vertical) {
36 | this.trackScrollAreaSize = this.track.height - this.vslice;
37 | } else {
38 | this.trackScrollAreaSize = this.track.width - this.hslice;
39 | }
40 | }
41 |
42 | /**
43 | * @private
44 | * Sets position for the bar's non-moving axis. Centers it inside the track.
45 | */
46 | centerStaticAxis() {
47 | if (this.vertical) {
48 | this.bar.x = (
49 | this.track.x + (this.track.width / 2)) - (this.bar.displayWidth / 2);
50 | } else {
51 | this.bar.y = (
52 | this.track.y + (this.track.height / 2)) - (this.bar.displayHeight / 2);
53 | }
54 | }
55 | }
56 |
--------------------------------------------------------------------------------
/src/bars/draggable_bar.js:
--------------------------------------------------------------------------------
1 | import { EventEmitter } from 'eventemitter3';
2 | import * as PhaserObjects from '../phaserObjects';
3 | import { Bar } from './bar';
4 |
5 | /**
6 | * Base object for Bars that can be manipulated with a mouse.
7 | * @private
8 | * @extends Bar
9 | */
10 | export class DraggableBar extends Bar {
11 | constructor(game, x = 0, y = 0, vertical = false, trackKey = '', barKey = '') {
12 | super(game, x, y, vertical, trackKey, barKey);
13 | this.emitter = new EventEmitter();
14 | }
15 |
16 | /**
17 | * @private
18 | * If the vertical scrollbar is draggable,
19 | * this function is called when the track is clicked.
20 | */
21 | verticalTrackClick() {
22 | // Don't register mouse clicks on the bar itself.
23 | const mouseY = this.game.input.mousePointer.y;
24 |
25 | let barY = this.bar.y + this.worldPosition.y;
26 |
27 | if (this.parentContainer) {
28 | barY += this.parentContainer.y;
29 | }
30 |
31 | if (mouseY > barY + this.bar.displayHeight) {
32 | this.scrollDown();
33 | } else if (mouseY < barY) {
34 | this.scrollUp();
35 | }
36 | }
37 |
38 | /**
39 | * @private
40 | * If the horizontal scrollbar is draggable,
41 | * this function is called when the track is clicked.
42 | */
43 | horizontalTrackClick() {
44 | // Don't register mouse clicks on the bar itself.
45 | const mouseX = this.game.input.mousePointer.x;
46 |
47 | let barX = this.bar.x + this.worldPosition.x;
48 |
49 | if (this.parentContainer) {
50 | barX += this.parentContainer.x;
51 | }
52 |
53 | if (mouseX > barX + this.bar.displayWidth) {
54 | this.scrollRight();
55 | } else if (mouseX < barX) {
56 | this.scrollLeft();
57 | }
58 | }
59 |
60 | /**
61 | * @private
62 | * Allows the bar to scroll when the track is clicked directly.
63 | */
64 | enableTrackClick() {
65 | let event;
66 |
67 | this.track.setInteractive();
68 |
69 | if (this.vertical) {
70 | event = this.verticalTrackClick;
71 | } else {
72 | event = this.horizontalTrackClick;
73 | }
74 |
75 | this.track.addDownEvent(event, this);
76 | }
77 |
78 | /**
79 | * @private
80 | * When called, ensures the bar can be moved.
81 | * Must be called once the bar has finished scrolling.
82 | */
83 | enableBarInput() {
84 | this.trackClicked = false;
85 | this.barMoving = false;
86 | this.bar.enableDragging(this.vertical);
87 | }
88 |
89 | /**
90 | * @private
91 | * Enables clicking and dragging on the bar.
92 | */
93 | enableBarDrag() {
94 | this.setDraggableArea();
95 |
96 | this.bar.enableDragging(this.vertical);
97 |
98 | let draggableArea;
99 |
100 | if (this.vertical) {
101 | draggableArea = this.verticalDraggableArea;
102 | } else {
103 | draggableArea = this.horizontalDraggableArea;
104 | }
105 |
106 | this.bar.setDragBounds(draggableArea);
107 |
108 | if (this.snapping) {
109 | this.bar.addUpEvent(this.snapToClosestPosition, this);
110 | }
111 | this.bar.addDownEvent(this.saveMousePosition, this);
112 | this.bar.addDragEvent(this.moveContent, this);
113 | }
114 |
115 | saveMousePosition(pointer) {
116 | // When the bar is dragged, record where the mouse clicked down.
117 | this.mousePointer = {
118 | x: pointer.x - (pointer.x - this.bar.x),
119 | y: pointer.y - (pointer.y - this.bar.y),
120 | };
121 | }
122 |
123 | getBarPosition() {
124 | const currentValue = this.valueRange.getCurrentValue();
125 | const windowPositionRatio = currentValue / this.windowScrollAreaSize;
126 | return this.trackScrollAreaSize * windowPositionRatio;
127 | }
128 |
129 | getMouseDelta() {
130 | // Only difference between clicking the track/using the keyboard vs mouse drag.
131 | let newMousePointer;
132 | if (this.trackClicked) {
133 | newMousePointer = { x: this.bar.x, y: this.bar.y };
134 | } else {
135 | const { mousePointer } = this.game.input;
136 | newMousePointer = {
137 | x: mousePointer.x - (mousePointer.x - this.bar.x),
138 | y: mousePointer.y - (mousePointer.y - this.bar.y),
139 | };
140 | }
141 |
142 | let oldMousePosition;
143 | let newMousePosition;
144 | if (this.vertical) {
145 | oldMousePosition = this.mousePointer.y;
146 | newMousePosition = newMousePointer.y;
147 | } else {
148 | oldMousePosition = this.mousePointer.x;
149 | newMousePosition = newMousePointer.x;
150 | }
151 |
152 | this.mousePointer = newMousePointer;
153 |
154 | // Only update when the new position is inside the track.
155 | let mousePositionDelta;
156 | if (newMousePosition < this.maxValue) {
157 | mousePositionDelta = oldMousePosition - newMousePosition;
158 | } else {
159 | mousePositionDelta = 0;
160 | }
161 |
162 | return mousePositionDelta;
163 | }
164 |
165 | /**
166 | * @private
167 | * Creates the tween for moving the bar to a new position.
168 | */
169 | addScrollTween(properties) {
170 | this.mousePointer = { x: this.bar.x, y: this.bar.y };
171 | this.trackClicked = true;
172 |
173 | new PhaserObjects.Tween(this.game).add(
174 | this.bar,
175 | properties,
176 | this.tweenParams.duration,
177 | this.tweenParams.ease,
178 | this.enableBarInput,
179 | this.moveContent,
180 | null,
181 | this,
182 | );
183 | }
184 |
185 | /** For Vertical Scrollbars. Scrolls up by one step. */
186 | scrollUp() {
187 | // Prevents users from moving the bar while it's moving.
188 | if (this.bar.y !== this.track.y && !this.barMoving) {
189 | const testPosition = this.bar.y - this.vslice;
190 | let moveToY = null;
191 | this.barMoving = true;
192 |
193 | // Ensure the bar can't move above the track.
194 | if (testPosition <= this.track.y) {
195 | moveToY = this.minY;
196 | } else {
197 | moveToY = this.bar.y - this.vslice;
198 | }
199 |
200 | this.addScrollTween({ y: moveToY });
201 | }
202 | }
203 |
204 | /** For Vertical Scrollbars. Scrolls down by one step. */
205 | scrollDown() {
206 | if (this.bar.y + this.bar.displayHeight !== this.track.y + this.track.height && !this.barMoving) {
207 | const testPosition = this.bar.y + (this.vslice * 2);
208 | let moveToY = null;
209 | this.barMoving = true;
210 | this.bar.disableInteractive();
211 |
212 | // Ensure the bar can't move below the track.
213 | if (testPosition >= this.track.y + this.track.height) {
214 | moveToY = this.maxY;
215 | } else {
216 | moveToY = this.bar.y + this.vslice;
217 | }
218 |
219 | this.addScrollTween({ y: moveToY });
220 | }
221 | }
222 |
223 | /** For Horizontal Scrollbars. Scrolls left by one step. */
224 | scrollLeft() {
225 | if (this.bar.x !== this.track.x && !this.barMoving) {
226 | const testPosition = this.bar.x - this.hslice;
227 | let moveToX = null;
228 | this.barMoving = true;
229 | this.bar.disableInteractive();
230 |
231 | // Ensure the bar can't move above the track.
232 | if (testPosition <= this.track.x) {
233 | moveToX = this.minX;
234 | } else {
235 | moveToX = this.bar.x - this.hslice;
236 | }
237 |
238 | this.addScrollTween({ x: moveToX });
239 | }
240 | }
241 |
242 | /** For Horizontal Scrollbars. Scrolls right by one step. */
243 | scrollRight() {
244 | if (this.bar.x + this.bar.displayWidth !== this.track.x + this.track.width && !this.barMoving) {
245 | const testPosition = this.bar.x + (this.hslice * 2);
246 | let moveToX = null;
247 | this.barMoving = true;
248 | this.bar.disableInteractive();
249 |
250 | // Ensure the bar can't move below the track.
251 | if (testPosition >= this.track.x + this.track.width) {
252 | moveToX = this.maxX;
253 | } else {
254 | moveToX = this.bar.x + this.hslice;
255 | }
256 |
257 | this.addScrollTween({ x: moveToX });
258 | }
259 | }
260 |
261 | /** Called when the scrollbar needs to move the viewport.
262 | * Causes the content to move relative to the bar's position on the track.
263 | */
264 | moveContent() {
265 | const newGripPositionRatio = this.getGripPositionRatio();
266 |
267 | const newContentPosition = newGripPositionRatio * this.windowScrollAreaSize;
268 |
269 | this.valueRange.adjustValue(newContentPosition);
270 |
271 | this.emitter.emit('movement', this);
272 | }
273 | }
274 |
--------------------------------------------------------------------------------
/src/bars/quantitybar.js:
--------------------------------------------------------------------------------
1 | import { Bar } from './bar';
2 |
3 | import { QuantityRange } from './ranges';
4 |
5 | import * as PhaserObjects from '../phaserObjects';
6 |
7 | /**
8 | * Bar that adjusts the size of a static sprite based on a value.
9 | * This is done by masking the sprite and then resizing the mask.
10 | * @extends Bar
11 | */
12 | export class QuantityBar extends Bar {
13 | /**
14 | * @param {Object} game - Current game instance.
15 | * @param {Object} xy - Dictionary with the values for the bar's x and y position.
16 | * @param {Object} values - The numerical values for the bar.
17 | * @param {boolean} vertical - Determines if the bar should be vertical or horizontal.
18 | * @param {boolean} reverse - Determines the direction the bar moves when adjusted.
19 | * @param {string} trackKey - The key to use for the track.
20 | * @param {string} barKey - The key to use for the bar.
21 | * @param {Object} tweenParams - Object with duration and easing function for the scrolling tween.
22 | */
23 | constructor(
24 | game,
25 | xy,
26 | values,
27 | vertical = false,
28 | reverse = false,
29 | trackKey = '',
30 | barKey = '',
31 | tweenParams = null,
32 | ) {
33 | super(game, xy.x, xy.y, vertical, trackKey, barKey);
34 |
35 | this.uiWidgetsObjectRole = 'quantitybar';
36 |
37 | this.valueRange = new QuantityRange(
38 | this,
39 | values.startValue,
40 | values.maxValue,
41 | );
42 |
43 | this.reverse = reverse || false;
44 |
45 | this.tweenParams = tweenParams || { duration: 300, ease: PhaserObjects.Easing.Quadratic.Out };
46 |
47 | // The track is the static area the bar will move along.
48 | this.track = new PhaserObjects.Sprite(game, 0, 0, this.trackKey);
49 |
50 | // Phaser 3:
51 | // Anchor the track to 0 instead of 0.5
52 | this.track.displayOriginX = 0;
53 | this.track.displayOriginY = 0;
54 |
55 | this.add(this.track);
56 |
57 | this.width = this.track.width;
58 | this.height = this.track.height;
59 |
60 | // The bar is a static image taking up the width of the track.
61 | this.bar = new PhaserObjects.Sprite(
62 | game,
63 | 0,
64 | 0,
65 | this.barKey,
66 | );
67 |
68 | // Phaser 3:
69 | // Anchor the track to 0 instead of 0.5
70 | this.bar.displayOriginX = 0;
71 | this.bar.displayOriginY = 0;
72 |
73 | this.add(this.bar);
74 |
75 | this.create();
76 | }
77 |
78 | /**
79 | * @private
80 | * Sets the bar's mask.
81 | */
82 | setMask() {
83 | if (this.bar.mask !== null) {
84 | this.bar.mask.destroy();
85 | this.bar.mask = null;
86 | }
87 |
88 | const mask = new PhaserObjects.ViewportMask(this.game, this.maskX, this.maskY);
89 | this.bar.mask = mask.create(this.bar.maskX, this.bar.maskY, this.maskW, this.maskH);
90 |
91 | // Phaser CE: Mask must be added to Group
92 | if (this.version === undefined) {
93 | this.add(mask);
94 | }
95 | }
96 |
97 | getMaskXY(x = 0, y = 0) {
98 | // Phaser CE: Mask starts at bar xy, Phaser 3: Mask starts at group xy
99 | let maskX;
100 | let maskY;
101 |
102 | maskX = x;
103 | maskY = y;
104 |
105 | if (this.version === undefined) {
106 | maskX += this.bar.x;
107 | maskY += this.bar.y;
108 | } else {
109 | maskX += this.x;
110 | maskY += this.y;
111 | }
112 |
113 | // Resizes the bar.
114 | if (this.reverse) {
115 | if (this.vertical) {
116 | maskY += this.getBarFraction();
117 | } else {
118 | maskX += this.getBarFraction();
119 | }
120 | }
121 |
122 | return [maskX, maskY];
123 | }
124 |
125 | getBarPosition() {
126 | const windowPositionRatio = this.valueRange.getRatio() / this.windowScrollAreaSize;
127 | return this.trackScrollAreaSize * windowPositionRatio;
128 | }
129 |
130 | create(maskX = 0, maskY = 0) {
131 | this.centerStaticAxis();
132 |
133 | // Values for the bar's mask.
134 | this.maskW = this.bar.width;
135 | this.maskH = this.bar.height;
136 |
137 | const [mx, my] = this.getMaskXY(maskX, maskY);
138 | this.maskX = mx;
139 | this.maskY = my;
140 |
141 | this.setMask();
142 |
143 | const barSize = this.getBarSize();
144 |
145 | if (this.reverse) {
146 | if (this.vertical) {
147 | this.bar.mask.geometryMask.scaleY = barSize;
148 | } else {
149 | this.bar.mask.geometryMask.scaleX = barSize;
150 | }
151 | } else {
152 | if (this.vertical) {
153 | this.bar.mask.geometryMask.scaleY = barSize;
154 | } else {
155 | this.bar.mask.geometryMask.scaleX = barSize;
156 | }
157 | }
158 |
159 | // Determine the distance the window can scroll over
160 | this.windowScrollAreaSize = this.valueRange.maxValue;
161 |
162 | // Represents one fraction of the track.
163 | this.vslice = (this.track.height * this.valueRange.getRatio());
164 | this.hslice = (this.track.width * this.valueRange.getRatio());
165 |
166 | this.setTrackScrollAreaSize();
167 | }
168 |
169 | /**
170 | * @private
171 | * Creates the tween for adjusting the size of the mask.
172 | * @param {Object} properties - Values for the tween's movement.
173 | */
174 | addScrollTweenMask(properties, duration, ease) {
175 | new PhaserObjects.Tween(this.game).add(
176 | this.bar.mask.geometryMask,
177 | properties,
178 | duration,
179 | ease,
180 | );
181 | }
182 |
183 | /**
184 | * Adjusts the bar by a given value.
185 | * @param {number} newValue - The value to adjust the bar by.
186 | */
187 | adjustBar(newValue) {
188 | this.valueRange.currentValue += newValue;
189 |
190 | let tween;
191 | const barSize = this.getBarSize();
192 |
193 | let moveToX;
194 | let moveToY;
195 |
196 | // Phaser 3 requires an offset
197 | if (this.version === undefined) {
198 | moveToX = 0;
199 | moveToY = 0;
200 | } else {
201 | moveToX = this.x;
202 | moveToY = this.y;
203 | }
204 |
205 | if (this.reverse) {
206 | if (this.vertical) {
207 | tween = { scaleY: barSize, y: moveToY + this.getBarFraction() };
208 | } else {
209 | tween = { scaleX: barSize, x: moveToX + this.getBarFraction() };
210 | }
211 | } else {
212 | if (this.vertical) {
213 | tween = { scaleY: barSize };
214 | } else {
215 | tween = { scaleX: barSize };
216 | }
217 | }
218 |
219 | this.addScrollTweenMask(tween, this.tweenParams.duration, this.tweenParams.ease);
220 | }
221 |
222 | getBarFraction() {
223 | let fraction;
224 | if (this.vertical) {
225 | fraction = this.track.height * this.valueRange.getRatio();
226 | } else {
227 | fraction = this.track.width * this.valueRange.getRatio();
228 | }
229 |
230 | return fraction;
231 | }
232 |
233 | /**
234 | * @private
235 | * Given a ratio between total content size and viewport size,
236 | * return an appropriate percentage of the track.
237 | */
238 | getBarSize() {
239 | let barSize;
240 |
241 | if (this.reverse) {
242 | barSize = 1 - this.valueRange.getRatio();
243 | } else {
244 | barSize = this.valueRange.getRatio();
245 | }
246 |
247 | return barSize;
248 | }
249 | }
250 |
--------------------------------------------------------------------------------
/src/bars/ranges.js:
--------------------------------------------------------------------------------
1 | /** Used by a QuantityBar to hold the bar's values. */
2 | export class QuantityRange {
3 | /**
4 | * @param {number} bar - The QuantityBar object that uses the range.
5 | * @param {number} startValue - The initial value for the bar.
6 | * @param {number} maxValue - The maximum value the bar can have.
7 | */
8 | constructor(bar, startValue, maxValue) {
9 | this.bar = bar;
10 | this.startValue = startValue;
11 | this.maxValue = maxValue;
12 |
13 | this.currentValue = startValue;
14 | }
15 |
16 | /** Returns the current ratio for how large the bar is compared to the track. */
17 | getRatio() {
18 | const ratio = this.currentValue / this.maxValue;
19 | return ratio;
20 | }
21 |
22 | /** Returns the bar's current value. */
23 | getCurrentValue() {
24 | return this.currentValue;
25 | }
26 | }
27 |
28 | /** Used by a ValueBar to hold the bar's values. */
29 | export class ValueRange {
30 | /**
31 | * @param {number} step - The amount the bar is changed by.
32 | * @param {number} startValue - The initial value for the bar.
33 | * @param {number} maxValue - The maximum value the bar can have.
34 | */
35 | constructor(step, startValue, maxValue) {
36 | this.step = step;
37 | this.startValue = startValue;
38 | this.maxValue = maxValue + step;
39 |
40 | this.ratio = step / maxValue;
41 |
42 | // The ratio between the step and max can't be greater than 1.
43 | // ie: There can't be more steps than the max value.
44 | if (this.ratio > 1) {
45 | this.ratio = 1;
46 | }
47 |
48 | this.currentValue = startValue;
49 |
50 | // List of every possible step. Used for snapping into position by the ValueBar.
51 | this.steps = [];
52 | for (let i = 0; i < this.maxValue; i += step) {
53 | this.steps.push(i);
54 | }
55 | }
56 |
57 | /** Adjusts the current value for the bar.
58 | * @param {number} newValue - The new current value.
59 | */
60 | adjustValue(newValue) {
61 | this.currentValue = newValue;
62 | }
63 |
64 | /** Returns the bar's current value. */
65 | getCurrentValue() {
66 | return this.currentValue;
67 | }
68 | }
69 |
70 | /** Used by a Scrollbar to hold the values and adjust a viewport's position. */
71 | export class ViewportRange {
72 | /**
73 | * @param {Object} viewport - The viewport to adjust.
74 | * @param {boolean} vertical - If the viewport is vertical or horizontal.
75 | */
76 | constructor(viewport, vertical) {
77 | this.viewport = viewport;
78 | this.vertical = vertical;
79 |
80 | if (vertical) {
81 | this.step = viewport.area.height;
82 | this.maxValue = viewport.realHeight;
83 | } else {
84 | this.step = viewport.area.width;
85 | this.maxValue = viewport.realWidth;
86 | }
87 |
88 | this.ratio = this.step / this.maxValue;
89 |
90 | // The ratio between the step and max can't be greater than 1.
91 | // ie: There can't be more steps than the max value.
92 | if (this.ratio > 1) {
93 | this.ratio = 1;
94 | }
95 | }
96 |
97 | /** Adjusts the viewport's position. */
98 | adjustValue(newValue) {
99 | // Set the content's new position. Uses an offset for where the viewport is on screen.
100 | if (this.vertical) {
101 | this.viewport.y = newValue + this.viewport.area.y;
102 | } else {
103 | this.viewport.x = newValue + this.viewport.area.x;
104 | }
105 |
106 | this.viewport.disableOutOfBounds(
107 | this.viewport.children,
108 | this,
109 | this.vertical,
110 | );
111 | }
112 |
113 | getCurrentValue() {
114 | let currentValue;
115 | if (this.vertical) {
116 | // y - an offset for where the viewport is on screen.
117 | currentValue = this.viewport.y - this.viewport.area.y;
118 | } else {
119 | currentValue = this.viewport.x - this.viewport.area.x;
120 | }
121 |
122 | return currentValue;
123 | }
124 | }
125 |
--------------------------------------------------------------------------------
/src/bars/scrollbar.js:
--------------------------------------------------------------------------------
1 | import { DraggableBar } from './draggable_bar';
2 |
3 | import { ViewportRange } from './ranges';
4 |
5 | import * as PhaserObjects from '../phaserObjects';
6 |
7 | /**
8 | * A bar that moves along a track.
9 | * The bar is resized relative to the size of the track and size of the content to be scrolled.
10 | * @extends DraggableBar
11 | */
12 | export class Scrollbar extends DraggableBar {
13 | /**
14 | * @param {Object} game - Current game instance.
15 | * @param {Object} content - An object to adjust via the scrollbar.
16 | * @param {boolean} draggable - Determines if the scrollbar responds to mouse clicks.
17 | * @param {boolean} vertical - Determines if the scrollbar should be vertical or horizontal.
18 | * @param {string} trackKey - The key to use for the track.
19 | * @param {string} barKey - The key to use for the bar.
20 | * @param {Object} tweenParams - Object with duration and easing function for the scrolling tween.
21 | */
22 | constructor(game, content, draggable, vertical = false, trackKey = '', barKey = '', tweenParams = null) {
23 | super(game, 0, 0, vertical, trackKey, barKey);
24 |
25 | this.game = game;
26 |
27 | this.content = content;
28 |
29 | this.valueRange = new ViewportRange(content, vertical);
30 |
31 | this.draggable = draggable || false;
32 |
33 | // The smallest pixel size allowed for the bar.
34 | this.minBarSize = 1;
35 |
36 | this.tweenParams = tweenParams || { duration: 300, ease: PhaserObjects.Easing.Quadratic.Out };
37 |
38 | // Flag switched on when the track is clicked, switched off after the bar movement is finished.
39 | this.trackClicked = false;
40 | this.barMoving = false;
41 |
42 | // Records mouse pointer when clicking the bar.
43 | this.mousePointer = null;
44 |
45 | // The track is the static area the bar will move along.
46 | this.track = new PhaserObjects.Sprite(
47 | game,
48 | this.x,
49 | this.y,
50 | this.trackKey,
51 | );
52 |
53 | // Phaser 3:
54 | // Anchor the track to 0 instead of 0.5
55 | this.track.displayOriginX = 0;
56 | this.track.displayOriginY = 0;
57 |
58 | this.add(this.track);
59 |
60 | if (this.draggable) {
61 | // If the bar is draggable, clicking the track will move the bar up or down.
62 | this.enableTrackClick();
63 | }
64 |
65 | this.createBar();
66 | this.add(this.bar);
67 |
68 | this.minY = this.track.y;
69 | this.maxY = (this.track.y + this.track.height) - this.bar.displayHeight;
70 | this.minX = this.track.x;
71 | this.maxX = (this.track.x + this.track.width) - this.bar.displayWidth;
72 |
73 | this.create();
74 |
75 | // Maximum value for the mouse position.
76 | if (this.vertical) {
77 | this.maxValue = this.track.displayHeight + this.worldPosition.y;
78 |
79 | if (this.version === undefined) {
80 | this.maxValue += this.bar.displayHeight;
81 | }
82 | } else {
83 | this.maxValue = this.track.displayWidth + this.worldPosition.x;
84 |
85 | if (this.version === undefined) {
86 | this.maxValue += this.bar.displayWidth;
87 | }
88 | }
89 | }
90 |
91 | /**
92 | * @private
93 | * The bar is the part that moves, controlling the value of the scrollbar.
94 | */
95 | createBar() {
96 | const bar = new PhaserObjects.Button(
97 | this.game,
98 | this.x,
99 | this.y,
100 | this.barKey,
101 | this.moveContent,
102 | this,
103 | 1,
104 | 0,
105 | );
106 |
107 | // Phaser 3:
108 | // Anchor the bar to 0 instead of 0.5
109 | bar.displayOriginX = 0;
110 | bar.displayOriginY = 0;
111 |
112 | this.bar = bar;
113 |
114 | this.resizeBar();
115 |
116 | return bar;
117 | }
118 |
119 | /**
120 | * @private
121 | * Given a ratio between total content size and viewport size,
122 | * resize the bar sprite to the appropriate percentage of the track.
123 | */
124 | resizeBar() {
125 | let barSize;
126 | if (this.vertical) {
127 | barSize = this.track.height * this.valueRange.ratio;
128 | } else {
129 | barSize = this.track.width * this.valueRange.ratio;
130 | }
131 |
132 | // Prevents bar from becoming zero pixels.
133 | if (barSize < this.minBarSize) {
134 | barSize = this.minBarSize;
135 | }
136 |
137 | // Resizes the bar.
138 | if (this.vertical) {
139 | this.bar.displayHeight = barSize;
140 | } else {
141 | this.bar.displayWidth = barSize;
142 | }
143 | }
144 |
145 | create() {
146 | this.centerStaticAxis();
147 |
148 | if (this.draggable) {
149 | this.enableBarDrag();
150 | }
151 |
152 | // Determine the distance the window can scroll over
153 | this.windowScrollAreaSize = this.valueRange.maxValue - this.valueRange.step;
154 |
155 | // Represents one fraction of the track.
156 | this.vslice = (this.track.displayHeight * this.valueRange.ratio);
157 | this.hslice = (this.track.displayWidth * this.valueRange.ratio);
158 |
159 | this.setTrackScrollAreaSize();
160 |
161 | // Initial position for the bar.
162 | this.mousePointer = { x: this.bar.x, y: this.bar.y };
163 |
164 | this.setInitialBarPosition();
165 | }
166 |
167 | /**
168 | * @private
169 | * Sets the draggable area of the bar.
170 | */
171 | setDraggableArea() {
172 | let vh = this.track.displayHeight;
173 |
174 | if (this.version === 3) {
175 | vh -= this.bar.displayHeight;
176 | }
177 |
178 | this.verticalDraggableArea = {
179 | x: this.track.x - ((this.bar.displayWidth - this.track.width) / 2),
180 | y: this.track.y,
181 | w: this.bar.displayWidth,
182 | h: vh,
183 | };
184 |
185 | this.horizontalDraggableArea = {
186 | x: this.track.x,
187 | y: this.track.y - ((this.bar.displayHeight - this.track.height) / 2),
188 | w: this.track.width,
189 | h: this.bar.displayHeight,
190 | };
191 | }
192 |
193 | /**
194 | * @private
195 | * Ensure the bar starts off where it should be, according to the bar's logical position.
196 | */
197 | setInitialBarPosition() {
198 | const gripPositionOnTrack = this.getBarPosition();
199 |
200 | // Make sure the bar is physically where it should be.
201 | if (this.vertical) {
202 | this.bar.y = gripPositionOnTrack + this.track.y;
203 | } else {
204 | this.bar.x = gripPositionOnTrack + this.track.x;
205 | }
206 | }
207 |
208 | getGripPositionRatio() {
209 | const gripPositionOnTrack = this.getBarPosition();
210 | const mousePositionDelta = this.getMouseDelta();
211 |
212 | let newGripPosition = gripPositionOnTrack + mousePositionDelta;
213 |
214 | // Don't let the content scroll above or below the track's size
215 | if (newGripPosition > 0) {
216 | newGripPosition = 0;
217 | } else if (newGripPosition <= -this.trackScrollAreaSize) {
218 | newGripPosition = -this.trackScrollAreaSize;
219 | }
220 |
221 | // When the scrollbar is at the top or bottom, prevent a mouse movement that
222 | // doesn't move the scrollbar from moving the content.
223 | if (this.vertical) {
224 | if (this.bar.y <= this.track.y) {
225 | newGripPosition = 0;
226 | } else if (this.bar.y + this.bar.displayHeight >= this.track.y + this.track.displayHeight) {
227 | newGripPosition = -this.trackScrollAreaSize;
228 | }
229 | } else {
230 | if (this.bar.x <= this.track.x) {
231 | newGripPosition = 0;
232 | } else if (this.bar.x + this.bar.displayWidth >= this.track.x + this.track.displayWidth) {
233 | newGripPosition = -this.trackScrollAreaSize;
234 | }
235 | }
236 |
237 | let newGripPositionRatio = newGripPosition / this.trackScrollAreaSize;
238 |
239 | // If the scrollable area is less than the size of the scrollbar,
240 | // the bar and track will be the same size.
241 | // In this scenario, a divide by zero occurs. Capture that and turn it into zero.
242 | if (Number.isNaN(newGripPositionRatio)) {
243 | newGripPositionRatio = 0;
244 | }
245 |
246 | return newGripPositionRatio;
247 | }
248 | }
249 |
--------------------------------------------------------------------------------
/src/bars/valuebar.js:
--------------------------------------------------------------------------------
1 | import { DraggableBar } from './draggable_bar';
2 |
3 | import { ValueRange } from './ranges';
4 |
5 | import * as PhaserObjects from '../phaserObjects';
6 |
7 | /**
8 | * Bar that adjusts a number.
9 | * This is done by masking the sprite and then resizing the mask.
10 | * @extends DraggableBar
11 | */
12 | export class ValueBar extends DraggableBar {
13 | /**
14 | * @param {Object} game - Current game instance.
15 | * @param {Object} xy - Dictionary with the values for the bar's x and y position.
16 | * @param {Object} values - The numerical values for the bar.
17 | * @param {boolean} draggable - Determines if the scrollbar responds to mouse clicks.
18 | * @param {boolean} vertical - Determines if the bar should be vertical or horizontal.
19 | * @param {string} trackKey - The key to use for the track.
20 | * @param {string} barKey - The key to use for the bar.
21 | * @param {Object} tweenParams - Object with duration and easing function for the scrolling tween.
22 | */
23 | constructor(game, xy, values, draggable, vertical = false, trackKey = '', barKey = '', tweenParams = null) {
24 | super(game, xy.x, xy.y, vertical, trackKey, barKey);
25 |
26 | this.valueRange = new ValueRange(values.step, values.startValue, values.maxValue);
27 |
28 | this.draggable = draggable || false;
29 |
30 | this.tweenParams = tweenParams || { duration: 300, ease: PhaserObjects.Easing.Quadratic.Out };
31 |
32 | // Flag flipped when the track is clicked, switched off after the bar movement is finished.
33 | this.trackClicked = false;
34 | this.barMoving = false;
35 |
36 | // Records mouse pointer when clicking the bar.
37 | this.mousePointer = null;
38 |
39 | // The track is the static area the bar will move along.
40 | this.track = new PhaserObjects.Sprite(game, 0, 0, this.trackKey);
41 |
42 | // Phaser 3:
43 | // Anchor the track to 0 instead of 0.5
44 | this.track.displayOriginX = 0;
45 | this.track.displayOriginY = 0;
46 |
47 | this.add(this.track);
48 |
49 | this.width = this.track.width;
50 | this.height = this.track.height;
51 |
52 | // If the bar is draggable, clicking the track will move the bar up or down.
53 | if (this.draggable) {
54 | this.enableTrackClick();
55 | }
56 |
57 | // The bar is the part that moves, controlling the value of the scrollbar.
58 | this.bar = new PhaserObjects.Button(
59 | game,
60 | this.x,
61 | this.y,
62 | this.barKey,
63 | this.moveContent,
64 | this,
65 | 1,
66 | 0,
67 | );
68 |
69 | // Phaser 3:
70 | // Anchor the track to 0 instead of 0.5
71 | this.bar.displayOriginX = 0;
72 | this.bar.displayOriginY = 0;
73 |
74 | // Add an invisible background.
75 | // This ensures the bar can always be entered correctly, no matter where the grip is.
76 | this.bg = new PhaserObjects.Graphics(game, { x: 0, y: 0 });
77 | this.add(this.bg);
78 | this.sendToBack(this.bg);
79 | this.bg.fillStyle(0xff0000, 0);
80 |
81 | if (this.vertical) {
82 | this.bg.fillRect(0, 0 - (this.bar.height / 2), 1, this.track.height + this.bar.height);
83 | } else {
84 | this.bg.fillRect(0 - (this.bar.width / 2), 0, this.track.width + this.bar.width, 1);
85 | }
86 |
87 | this.snapping = true;
88 |
89 | this.add(this.bar);
90 | this.minY = this.track.y - (this.bar.height / 2);
91 | this.maxY = (this.track.y + this.track.height) - (this.bar.height / 2);
92 | this.minX = this.track.x - (this.bar.width / 2);
93 | this.maxX = (this.track.x + this.track.width) - (this.bar.width / 2);
94 |
95 | this.create();
96 |
97 | if (this.vertical) {
98 | this.upEvent = this.scrollUp;
99 | this.downEvent = this.scrollDown;
100 | } else {
101 | this.upEvent = this.scrollLeft;
102 | this.downEvent = this.scrollRight;
103 | }
104 |
105 | // Maximum value for the mouse position.
106 | if (this.vertical) {
107 | this.maxValue = this.track.displayHeight + this.worldPosition.y;
108 |
109 | if (this.version === undefined) {
110 | this.maxValue += this.bar.displayHeight;
111 | }
112 | } else {
113 | this.maxValue = this.track.displayWidth + this.worldPosition.x;
114 |
115 | if (this.version === undefined) {
116 | this.maxValue += this.bar.displayWidth;
117 | }
118 | }
119 | }
120 |
121 | create() {
122 | this.centerStaticAxis();
123 |
124 | if (this.draggable) {
125 | this.enableBarDrag();
126 | }
127 |
128 | // Determine the distance the window can scroll over
129 | this.windowScrollAreaSize = this.valueRange.maxValue - this.valueRange.step;
130 |
131 | // Represents one fraction of the track.
132 | this.vslice = (this.track.displayHeight * this.valueRange.ratio);
133 | this.hslice = (this.track.displayWidth * this.valueRange.ratio);
134 |
135 | this.setTrackScrollAreaSize();
136 |
137 | this.setInitialBarPosition();
138 |
139 | // Initial position for the bar.
140 | this.mousePointer = { x: this.bar.x, y: this.bar.y };
141 | }
142 |
143 | /** Sets the draggable area of the bar. */
144 | setDraggableArea() {
145 | let vh = this.track.displayHeight;
146 |
147 | if (this.version === undefined) {
148 | vh += this.bar.displayHeight;
149 | }
150 |
151 | this.verticalDraggableArea = {
152 | x: this.track.x - ((this.bar.displayWidth - this.track.displayWidth) / 2),
153 | y: this.track.y - (this.bar.displayHeight / 2),
154 | w: this.bar.displayWidth,
155 | h: vh,
156 | };
157 |
158 | this.horizontalDraggableArea = {
159 | x: this.track.x - (this.bar.displayWidth / 2),
160 | y: this.track.y - ((this.bar.displayHeight - this.track.displayHeight) / 2),
161 | w: this.track.displayWidth + this.bar.displayWidth,
162 | h: this.bar.displayHeight,
163 | };
164 | }
165 |
166 | /** Determine the distance the bar can scroll over */
167 | setTrackScrollAreaSize() {
168 | if (this.vertical) {
169 | this.trackScrollAreaSize = this.track.displayHeight;
170 | } else {
171 | this.trackScrollAreaSize = this.track.displayWidth;
172 | }
173 | }
174 |
175 | /** Ensure the bar starts off where it should be, according to the bar's logical position. */
176 | setInitialBarPosition() {
177 | const gripPositionOnTrack = this.getBarPosition();
178 |
179 | // The bar should always be in centered on it's current position.
180 | if (this.vertical) {
181 | this.bar.y = (gripPositionOnTrack + this.track.y) - (this.bar.displayHeight / 2);
182 | } else {
183 | this.bar.x = (gripPositionOnTrack + this.track.x) - (this.bar.displayWidth / 2);
184 | }
185 | }
186 |
187 | /** @returns {number} the closest valid value. */
188 | getClosestPosition() {
189 | const currentValue = this.valueRange.getCurrentValue();
190 |
191 | let diff = Math.abs(currentValue - this.valueRange.steps[0]);
192 | let closestPosition = this.valueRange.steps[0];
193 |
194 | for (let i = 0; i < this.valueRange.steps.length; i++) {
195 | const newdiff = Math.abs(currentValue - this.valueRange.steps[i]);
196 | if (newdiff < diff) {
197 | diff = newdiff;
198 | closestPosition = this.valueRange.steps[i];
199 | }
200 | }
201 |
202 | return closestPosition;
203 | }
204 |
205 | /** On mouse up, forces the value to equal the closest step. */
206 | snapToClosestPosition() {
207 | const closestPosition = this.getClosestPosition();
208 |
209 | this.valueRange.adjustValue(closestPosition);
210 | this.moveContent();
211 | this.setInitialBarPosition();
212 | }
213 |
214 | /**
215 | * @private
216 | * Creates the tween for moving the bar to a new position.
217 | */
218 | addScrollTween(properties) {
219 | this.mousePointer = { x: this.bar.x, y: this.bar.y };
220 | this.trackClicked = true;
221 |
222 | new PhaserObjects.Tween(this.game).add(
223 | this.bar,
224 | properties,
225 | this.tweenParams.duration,
226 | this.tweenParams.ease,
227 | () => { this.moveContent(); this.enableBarInput(); },
228 | null,
229 | null,
230 | this,
231 | null,
232 | null,
233 | );
234 | }
235 |
236 | getGripPositionRatio() {
237 | const gripPositionOnTrack = this.getBarPosition();
238 | const mousePositionDelta = this.getMouseDelta();
239 |
240 | let newGripPosition = gripPositionOnTrack - mousePositionDelta;
241 | // Don't let the content scroll above or below the track's size
242 | if (newGripPosition < 0) {
243 | newGripPosition = 0;
244 | } else if (newGripPosition >= this.trackScrollAreaSize) {
245 | newGripPosition = this.trackScrollAreaSize;
246 | }
247 |
248 | // When the scrollbar is at the top or bottom, prevent a mouse movement that
249 | // doesn't move the scrollbar from moving the content.
250 | if (this.vertical) {
251 | if (this.bar.y <= this.track.y) {
252 | newGripPosition = 0;
253 | } else if (this.bar.y + this.bar.displayHeight >= this.track.y + this.track.displayHeight) {
254 | newGripPosition = this.trackScrollAreaSize;
255 | }
256 | } else {
257 | if (this.bar.x <= this.track.x) {
258 | newGripPosition = 0;
259 | } else if (this.bar.x + this.bar.displayWidth >= this.track.x + this.track.displayWidth) {
260 | newGripPosition = this.trackScrollAreaSize;
261 | }
262 | }
263 |
264 | let newGripPositionRatio = newGripPosition / this.trackScrollAreaSize;
265 |
266 | // If the scrollable area is less than the size of the scrollbar,
267 | // the bar and track will be the same size.
268 | // In this scenario, a divide by zero occurs. Capture that and turn it into zero.
269 | if (Number.isNaN(newGripPositionRatio)) {
270 | newGripPositionRatio = 0;
271 | }
272 |
273 | return newGripPositionRatio;
274 | }
275 | }
276 |
--------------------------------------------------------------------------------
/src/buttons.js:
--------------------------------------------------------------------------------
1 | import * as PhaserObjects from './phaserObjects';
2 |
3 | /** Group with some added functionality for text overlays.
4 | * @private
5 | * @extends Phaser.Group
6 | */
7 | class TextGroup extends PhaserObjects.Group {
8 | /**
9 | * @param {Object} game - Current game instance.
10 | * @param {number} x - The x coordinate on screen where the sprite will be placed.
11 | * @param {number} y - The y coordinate on screen where the sprite will be placed.
12 | */
13 | constructor(game, x, y) {
14 | super(game, x, y);
15 |
16 | this.game = game;
17 | this.x = x;
18 | this.y = y;
19 |
20 | game.add.existing(this);
21 | }
22 |
23 | /**
24 | * @param {string} label - The text to place on top of the sprite.
25 | * @param {Object} style - The style properties to be set on the Text.
26 | */
27 | setText(label, style) {
28 | if (this.text) {
29 | this.text.destroy();
30 | }
31 |
32 | if (this.version === 3) {
33 | this.text = this.game.add.text(0, 0, label, style);
34 | this.text.setOrigin(0.5, 0.5);
35 | } else {
36 | this.text = this.game.add.text(this.width / 2, this.height / 2, label, style);
37 | this.text.anchor.set(0.5, 0.5);
38 | }
39 |
40 | this.add(this.text);
41 |
42 | this.text.wordWrap = true;
43 | this.text.wordWrapWidth = this.width;
44 |
45 | return this;
46 | }
47 |
48 | /** When setOrigin is called, align the text as well. */
49 | setOrigin(x, y) {
50 | if (this.version === 3) {
51 | this.sprite.setOrigin(x, y);
52 | } else {
53 | this.sprite.anchor.set(x, y);
54 | }
55 |
56 | this.text.x = this.sprite.width / 2;
57 | this.text.y = this.sprite.height / 2;
58 | this.currentTextY = this.sprite.height / 2;
59 |
60 | return this;
61 | }
62 | }
63 |
64 | /** Sprite with text added as a child.
65 | * @extends Phaser.Group
66 | */
67 | export class TextSprite extends TextGroup {
68 | /**
69 | * @param {Object} game - Current game instance.
70 | * @param {number} x - The x coordinate on screen where the textSprite will be placed.
71 | * @param {number} y - The y coordinate on screen where the textSprite will be placed.
72 | * @param {string} key - The image to create a sprite with.
73 | */
74 | constructor(game, x, y, key) {
75 | super(game, x, y);
76 |
77 | this.sprite = new PhaserObjects.Sprite(game, 0, 0, key);
78 | this.add(this.sprite);
79 |
80 | this.width = this.sprite.width;
81 | this.height = this.sprite.height;
82 | }
83 | }
84 |
85 | /** Phaser Group containing a button with text anchored in the button's center.
86 | * @extends Phaser.Group
87 | */
88 | export class TextButton extends TextGroup {
89 | /**
90 | * @param {Object} game - Current game instance.
91 | * @param {number} x - The x coordinate on screen where the textSprite will be placed.
92 | * @param {number} y - The y coordinate on screen where the textSprite will be placed.
93 | * @param {string} key - The image to create a sprite with.
94 | * @param {Object} callback - Callback to use when the button is clicked.
95 | * @param {Object} callbackContext - The context the callback is called in.
96 | * @param {number} overKey - The frame to switch to when an over event is triggered.
97 | * @param {number} outKey - The frame to switch to when an out event is triggered.
98 | * @param {number} downKey - The frame to switch to when a down event is triggered.
99 | * @param {number} upKey - The frame to switch to when an up event is triggered.
100 | */
101 | constructor(game, x, y, key, callback, callbackContext, overKey, outKey, downKey, upKey) {
102 | super(game, x, y);
103 |
104 | this.sprite = new PhaserObjects.Button(
105 | game,
106 | 0,
107 | 0,
108 | key,
109 | callback,
110 | callbackContext,
111 | overKey,
112 | outKey,
113 | downKey,
114 | upKey,
115 | );
116 | this.add(this.sprite);
117 |
118 | this.width = this.sprite.width;
119 | this.height = this.sprite.height;
120 |
121 | this.currentTextY = 0;
122 | }
123 |
124 | /** Adds an adjustment to the text on down/up events. */
125 | eventTextYAdjustment(number) {
126 | this.sprite.addDownEvent(() => { this.text.y += number; });
127 | this.sprite.addUpEvent(() => { this.text.y = this.currentTextY; });
128 |
129 | // A pointerout event should reset the text position too.
130 | this.sprite.addOutEvent(() => { this.text.y = this.currentTextY; });
131 |
132 | return this;
133 | }
134 | }
135 |
--------------------------------------------------------------------------------
/src/const.js:
--------------------------------------------------------------------------------
1 | // Matches the alignment consts from Phaser
2 | export const alignments = {
3 | TOP_LEFT: 0,
4 | TOP_CENTER: 1,
5 | TOP_RIGHT: 2,
6 | LEFT_TOP: 3,
7 | LEFT_CENTER: 4,
8 | LEFT_BOTTOM: 5,
9 | CENTER: 6,
10 | RIGHT_TOP: 7,
11 | RIGHT_CENTER: 8,
12 | RIGHT_BOTTOM: 9,
13 | BOTTOM_LEFT: 10,
14 | BOTTOM_CENTER: 11,
15 | BOTTOM_RIGHT: 12,
16 | };
17 |
--------------------------------------------------------------------------------
/src/containers/column.js:
--------------------------------------------------------------------------------
1 | import { alignments } from '../const';
2 | import { Frame } from './frame';
3 |
4 | /** Frame that places new child nodes directly under the previous child.
5 | * @extends Frame
6 | */
7 | export class Column extends Frame {
8 | /**
9 | * @param {Object} game - Current game instance.
10 | * @param {Number} x - The x position of the Column.
11 | * @param {Number} y - The y position of the Column.
12 | * @param {string} bg - The background image to use.
13 | * @param {boolean} modal - If the Column should block external interaction until killed.
14 | */
15 | constructor(game, x = 0, y = 0, bg = null, modal = false) {
16 | super(game, x, y, bg, modal);
17 | this.alignment = alignments.BOTTOM_CENTER;
18 | }
19 | }
20 |
--------------------------------------------------------------------------------
/src/containers/frame.js:
--------------------------------------------------------------------------------
1 | import { alignments } from '../const';
2 | import * as PhaserObjects from '../phaserObjects';
3 |
4 | /** Group with a dedicated background image.
5 | * @extends Phaser.Group
6 | */
7 | export class Frame extends PhaserObjects.Group {
8 | /**
9 | * @param {Object} game - Current game instance.
10 | * @param {number} x - The x position of the Frame.
11 | * @param {number} y - The y position of the Frame.
12 | * @param {string} bg - The background image to use.
13 | * @param {boolean} modal - If the Frame should block external interaction until killed.
14 | */
15 | constructor(game, x = 0, y = 0, bg = null, modal = false) {
16 | super(game);
17 | game.add.existing(this);
18 |
19 | this.x = x;
20 | this.y = y;
21 |
22 | this.uiWidgetsObjectRole = 'layout';
23 |
24 | this.modal = modal;
25 |
26 | this.background = null;
27 | this.back = null;
28 |
29 | if (modal) {
30 | // Create a sprite to use as the modal back.
31 | const texture = new PhaserObjects.Graphics(game, { x: 0, y: 0 });
32 | texture.fillStyle(0xffffff, 1);
33 | texture.fillRect(0, 0, game.scale.width, game.scale.height);
34 |
35 | let modalBack;
36 | if (this.version === 3) {
37 | texture.generateTexture('modalBack');
38 | modalBack = 'modalBack';
39 | } else {
40 | modalBack = texture.generateTexture();
41 | }
42 |
43 | this.back = new PhaserObjects.Sprite(game, 0, 0, modalBack);
44 | this.back.setInteractive();
45 | this.back.setOrigin(0, 0);
46 | // CE requirement
47 | this.back.moveDown();
48 |
49 | // Mask the back, causing it to be invisible.
50 | const backMask = new PhaserObjects.ViewportMask(game, 0, 0);
51 | this.back.mask = backMask.create(0, 0, 0, 0);
52 |
53 | game.add.existing(this.back);
54 |
55 | // Ensure the modal doesn't hide the Frame's content.
56 | this.depth = 9999;
57 | this.back.depth = 9998;
58 | }
59 |
60 | if (bg !== null) {
61 | this.background = new PhaserObjects.Sprite(game, 0, 0, bg);
62 | game.add.existing(this.background);
63 | this.background.sendToBack();
64 | this.background.alignIn(this, alignments.TOP_LEFT);
65 | }
66 | }
67 |
68 | /** Destroy the Frame, along with any background or modal effect. */
69 | dismiss() {
70 | if (this.modal) {
71 | this.back.destroy();
72 | }
73 |
74 | if (this.background !== null) {
75 | this.background.destroy();
76 | }
77 |
78 | this.destroy();
79 | }
80 |
81 | /** Adds a new object into the Frame, then aligns it with the previous object.
82 | * @param {Object} node - The object to add to the Frame.
83 | * @param {number} paddingX - The amount of horizontal space between objects.
84 | * @param {number} paddingY - The amount of vertical space between objects.
85 | * @param {number} alignment - The alignment relative to the previous child.
86 | */
87 | addNode(node, paddingX = 0, paddingY = 0, alignment = null) {
88 | const align = alignment || this.alignment;
89 |
90 | if (!(node instanceof PhaserObjects.Group)) {
91 | node.displayOriginX = 0; // eslint-disable-line
92 | node.displayOriginY = 0; // eslint-disable-line
93 | } else if (node instanceof PhaserObjects.Group) {
94 | node.setOrigin(0, 0);
95 | }
96 |
97 | this.add(node);
98 | this.alignNodeToPrevious(node, align, paddingX, paddingY);
99 | node.parentContainer = this; // eslint-disable-line
100 |
101 | // Phaser 3: QuantityBar's mask needs a personal touch.
102 | if (node.uiWidgetsObjectRole === 'quantitybar') {
103 | if (node.version === 3) {
104 | node.create(this.x, this.y);
105 | }
106 | }
107 |
108 | // Reset the positions for the bar's draggable area.
109 | if ('enableBarDrag' in node) {
110 | node.enableBarDrag();
111 | }
112 | }
113 | }
114 |
--------------------------------------------------------------------------------
/src/containers/row.js:
--------------------------------------------------------------------------------
1 | import { alignments } from '../const';
2 | import { Frame } from './frame';
3 |
4 | /** Frame that places new child nodes directly next to the previous child.
5 | * @extends Frame
6 | */
7 | export class Row extends Frame {
8 | /**
9 | * @param {Object} game - Current game instance.
10 | * @param {Number} x - The x position of the Row.
11 | * @param {Number} y - The y position of the Row.
12 | * @param {string} bg - The background image to use.
13 | * @param {boolean} modal - If the Row should block external interaction until killed.
14 | */
15 | constructor(game, x = 0, y = 0, bg = null, modal = false) {
16 | super(game, x, y, bg, modal);
17 | this.alignment = alignments.RIGHT_CENTER;
18 | }
19 | }
20 |
--------------------------------------------------------------------------------
/src/containers/viewport.js:
--------------------------------------------------------------------------------
1 | import * as PhaserObjects from '../phaserObjects';
2 |
3 | /**
4 | * A container with a limited viewable area. Uses a mask to hide children outside of the specified x/y/width/height area.
5 | * Content outside the viewport has their input disabled.
6 | * @extends PhaserObjects.Group
7 | */
8 | export class Viewport extends PhaserObjects.Group {
9 | /**
10 | * @param {Object} game - Current game instance.
11 | * @param {number} x - The x coordinate on screen where the viewport will be placed.
12 | * @param {number} y - The y coordinate on screen where the viewport will be placed.
13 | * @param {number} width - The width of the viewport.
14 | * @param {number} height - The height of the viewport.
15 | */
16 | constructor(game, x, y, width, height) {
17 | super(game);
18 | game.add.existing(this);
19 |
20 | this.x = x;
21 | this.y = y;
22 |
23 | this.uiWidgetsObjectRole = 'layout';
24 |
25 | // Viewport size and position, distinct from the total window size.
26 | this.area = {
27 | x,
28 | y,
29 | width,
30 | height,
31 | };
32 |
33 | // Adding the mask attribute to a group hides objects outside the mask.
34 | const mask = new PhaserObjects.ViewportMask(game, x, y);
35 |
36 | if (this.version === undefined) {
37 | this.mask = mask.create(x, y, width, height);
38 | } else {
39 | this.mask = mask.create(0, 0, width, height);
40 | }
41 | }
42 |
43 | /** Adds a new object into the Viewport.
44 | * @param {Object} node - The object to add to the Viewport.
45 | */
46 | addNode(node) {
47 | this.add(node);
48 | node.parentContainer = this; // eslint-disable-line
49 | }
50 |
51 | /** Disable input for all objets outside the viewport's visible area.
52 | * Recursively checks all the object's children.
53 | * @param {Object} children - The objects to disable, if they're outside the viewport.
54 | * @param {Object} context - The context the function is run in.
55 | * @param {boolean} vertical - If the bounds should be checked horizontally or vertically.
56 | */
57 | disableOutOfBounds(children, context, vertical) {
58 | let child;
59 | let location;
60 | let contentLocation;
61 | let trueCoords;
62 |
63 | // Makes sure the recursive function stops when there's no children.
64 | if (children !== undefined) {
65 | for (let i = 0; i < children.length; i++) {
66 | child = children[i];
67 | child.inputEnabled = true;
68 |
69 | // An object's x/y is relative to it's parent.
70 | // The world gives an x/y relative to the whole game.
71 | trueCoords = child.world || child;
72 |
73 | if (vertical) {
74 | location = trueCoords.y;
75 | contentLocation = context.viewport.area.y;
76 | } else {
77 | location = trueCoords.x;
78 | contentLocation = context.viewport.area.x;
79 | }
80 |
81 | if (location < contentLocation) {
82 | child.inputEnabled = false;
83 | }
84 |
85 | this.disableOutOfBounds(child.children, context, vertical);
86 | }
87 | }
88 | }
89 | }
90 |
--------------------------------------------------------------------------------
/src/entry.js:
--------------------------------------------------------------------------------
1 | import { TextSprite, TextButton } from './buttons';
2 | import { KeyboardGroup } from './keyboard_group';
3 | import { Wheel3D } from './wheel3D';
4 |
5 | import { QuantityBar } from './bars/quantitybar';
6 | import { Scrollbar } from './bars/scrollbar';
7 | import { ValueBar } from './bars/valuebar';
8 |
9 | import { Frame } from './containers/frame';
10 | import { Column } from './containers/column';
11 | import { Row } from './containers/row';
12 | import { Viewport } from './containers/viewport';
13 |
14 | import { Button } from './phaserObjects/button';
15 |
16 | export {
17 | TextSprite,
18 | TextButton,
19 | KeyboardGroup,
20 | Wheel3D,
21 | QuantityBar,
22 | Scrollbar,
23 | ValueBar,
24 | Frame,
25 | Column,
26 | Row,
27 | Viewport,
28 | Button,
29 | };
30 |
--------------------------------------------------------------------------------
/src/keyboard_group.js:
--------------------------------------------------------------------------------
1 | import { EventEmitter } from 'eventemitter3';
2 | import * as PhaserObjects from './phaserObjects';
3 |
4 | import { utils } from './utils';
5 |
6 | /** Collection of sprites that can be selected with the keyboard.
7 | * When the select key is hit, the sprite that was selected is now connected to the keyboard.
8 | */
9 | export class KeyboardGroup {
10 | /**
11 | * @param {Object} game - Current game instance.
12 | * @param {Boolean} vertical - If the selection should be controlled with up/down or left/right arrow keys.
13 | * @param {Object} callbackContext - The context for the previous and next Events.
14 | */
15 | constructor(game, vertical, callbackContext) {
16 | this.game = game;
17 | this.vertical = vertical || false;
18 | this.callbackContext = callbackContext;
19 |
20 | this.children = [];
21 |
22 | this.selected = null;
23 | this.idx = 0;
24 |
25 | this.upKey = this.game.input.keyboard.addKey(PhaserObjects.KeyCodes.UP);
26 | this.downKey = this.game.input.keyboard.addKey(PhaserObjects.KeyCodes.DOWN);
27 | this.leftKey = this.game.input.keyboard.addKey(PhaserObjects.KeyCodes.LEFT);
28 | this.rightKey = this.game.input.keyboard.addKey(PhaserObjects.KeyCodes.RIGHT);
29 |
30 | this.upEvent = this.prevItem;
31 | this.downEvent = this.nextItem;
32 |
33 | this.emitter = new EventEmitter();
34 |
35 | this.activateGroup();
36 | }
37 |
38 | /** Add a new child to the group
39 | * @param {Object} newNode - The sprite to add to the group.
40 | */
41 | addNode(newNode) {
42 | this.children.push(newNode);
43 |
44 | // Ensure the first child is already selected when the game loads.
45 | this.selected = this.children[0]; // eslint-disable-line prefer-destructuring
46 | this.useBar();
47 | }
48 |
49 | /** Selects the previous child. */
50 | prevItem() {
51 | this.idx = utils.modulo(this.idx - 1, this.children.length);
52 |
53 | this.selected = this.children[this.idx];
54 |
55 | this.useBar();
56 |
57 | this.emitter.emit('previous', this, this.callbackContext);
58 | }
59 |
60 | /** Selects the next child. */
61 | nextItem() {
62 | this.idx = utils.modulo(this.idx + 1, this.children.length);
63 |
64 | this.selected = this.children[this.idx];
65 |
66 | this.useBar();
67 |
68 | this.emitter.emit('next', this, this.callbackContext);
69 | }
70 |
71 | /**
72 | * Enables keyboard input for the group.
73 | * @private
74 | */
75 | activateGroup() {
76 | const keyboard = new PhaserObjects.Keyboard();
77 |
78 | if (this.vertical) {
79 | keyboard.addDownEvent(this.upKey, this.upEvent, this);
80 | keyboard.addDownEvent(this.downKey, this.downEvent, this);
81 | } else {
82 | keyboard.addDownEvent(this.leftKey, this.upEvent, this);
83 | keyboard.addDownEvent(this.rightKey, this.downEvent, this);
84 | }
85 | }
86 |
87 | /**
88 | * Enables keyboard input on a child.
89 | * @private
90 | */
91 | useBar() {
92 | const keyboard = new PhaserObjects.Keyboard();
93 |
94 | if (this.vertical) {
95 | keyboard.removeDownEvent(this.leftKey);
96 | keyboard.removeDownEvent(this.rightKey);
97 |
98 | keyboard.addDownEvent(this.leftKey, this.selected.upEvent, this.selected);
99 | keyboard.addDownEvent(this.rightKey, this.selected.downEvent, this.selected);
100 | } else {
101 | keyboard.removeDownEvent(this.upKey);
102 | keyboard.removeDownEvent(this.downKey);
103 |
104 | keyboard.addDownEvent(this.upKey, this.selected.upEvent, this.selected);
105 | keyboard.addDownEvent(this.downKey, this.selected.downEvent, this.selected);
106 | }
107 | }
108 | }
109 |
--------------------------------------------------------------------------------
/src/phaserObjects/button.js:
--------------------------------------------------------------------------------
1 | import { Sprite } from './sprite';
2 |
3 | let exportObject;
4 |
5 | if (Phaser.Button === undefined) {
6 | /** Build a button object that immitates Phaser CE's Button
7 | * @private
8 | */
9 | class Phaser3Button extends Sprite {
10 | constructor(game, x, y, key, callback, callbackContext, overKey, outKey, downKey, upKey) {
11 | super(game, x, y, key);
12 |
13 | game.add.existing(this);
14 |
15 | this.game = game;
16 |
17 | this.overKey = overKey;
18 | this.outKey = outKey;
19 | this.downKey = downKey;
20 | this.upKey = upKey;
21 |
22 | this.setInteractive({ useHandCursor: true });
23 |
24 | let cbContext = callbackContext;
25 |
26 | if (cbContext === null || (cbContext === undefined)) {
27 | cbContext = this;
28 | }
29 |
30 | this.on('pointerdown', (pointer) => {
31 | callback.call(cbContext, pointer);
32 | });
33 | this.on('pointerover', this.onOver, this);
34 | this.on('pointerout', this.onOut, this);
35 | this.on('pointerdown', this.onDown, this);
36 | this.on('pointerup', this.onUp, this);
37 | }
38 |
39 | onOver() {
40 | this.setFrame(this.overKey);
41 | }
42 |
43 | onOut() {
44 | this.setFrame(this.outKey);
45 | }
46 |
47 | onDown() {
48 | this.setFrame(this.downKey);
49 | }
50 |
51 | onUp() {
52 | this.setFrame(this.upKey);
53 | }
54 |
55 | updateDrag(pointer, gameObject, x, y) { //eslint-disable-line
56 | if (gameObject.vertical) {
57 | if ((gameObject.draggableArea.y + gameObject.draggableArea.h) >= y) {
58 | if (gameObject.draggableArea.y <= y) {
59 | gameObject.y = y; //eslint-disable-line
60 | }
61 | }
62 | } else {
63 | const fx = (x + gameObject.displayWidth);
64 | if ((gameObject.draggableArea.x + gameObject.draggableArea.w) >= fx) {
65 | if (gameObject.draggableArea.x <= x) {
66 | gameObject.x = x; //eslint-disable-line
67 | }
68 | }
69 | }
70 | }
71 |
72 | setDragBounds(draggableArea) {
73 | this.draggableArea = draggableArea;
74 | }
75 |
76 | enableDragging(vertical = false) {
77 | this.setInteractive({ useHandCursor: true });
78 | this.vertical = vertical;
79 | this.game.input.setDraggable(this);
80 | this.game.input.on('drag', this.updateDrag);
81 | }
82 | }
83 |
84 | exportObject = Phaser3Button;
85 | } else {
86 | class PhaserCEButton extends Phaser.Button {
87 | constructor(game, x, y, key, callback, callbackContext, overKey, outKey, downKey, upKey) {
88 | super(game, x, y, key, callback, callbackContext, overKey, outKey, downKey, upKey);
89 |
90 | game.add.existing(this);
91 | }
92 |
93 | setInteractive() {
94 | this.inputEnabled = true;
95 | this.input.useHandCursor = true;
96 | }
97 |
98 | disableInteractive() {
99 | this.inputEnabled = false;
100 | }
101 |
102 | setDragBounds(draggableArea) {
103 | this.input.boundsRect = new Phaser.Rectangle(
104 | draggableArea.x,
105 | draggableArea.y,
106 | draggableArea.w,
107 | draggableArea.h,
108 | );
109 | }
110 |
111 | enableDragging(vertical = false) {
112 | this.inputEnabled = true;
113 | this.input.enableDrag();
114 |
115 | if (vertical) {
116 | this.input.allowHorizontalDrag = false;
117 | } else {
118 | this.input.allowVerticalDrag = false;
119 | }
120 | }
121 |
122 | /**
123 | * @private
124 | * Add a callback that is triggered when the pointer moves out of the object.
125 | */
126 | addOutEvent(callback, callbackContext) {
127 | this.events.onInputOut.add(callback, callbackContext);
128 | }
129 |
130 | /**
131 | * @private
132 | * Add a callback that is triggered when the object is unclicked.
133 | */
134 | addUpEvent(callback, callbackContext) {
135 | this.events.onInputUp.add(callback, callbackContext);
136 | }
137 |
138 | /**
139 | * @private
140 | * Add a callback that is triggered when the object is clicked.
141 | */
142 | addDownEvent(callback, callbackContext) {
143 | this.events.onInputDown.add(callback, callbackContext);
144 | }
145 |
146 | /**
147 | * @private
148 | * Add a callback that is triggered when the object is dragged.
149 | */
150 | addDragEvent(callback, callbackContext) {
151 | this.events.onDragUpdate.add(callback, callbackContext);
152 | }
153 |
154 | /** Immitate the API of Phaser3.
155 | * The current displayed height of the Object.
156 | * @type {number}
157 | * @private
158 | */
159 | get displayHeight() {
160 | return this.height;
161 | }
162 |
163 | set displayHeight(newValue) {
164 | this.height = newValue;
165 | }
166 |
167 | /** Immitate the API of Phaser3.
168 | * The current displayed width of the Object.
169 | * @type {number}
170 | * @private
171 | */
172 | get displayWidth() {
173 | return this.width;
174 | }
175 |
176 | set displayWidth(newValue) {
177 | this.width = newValue;
178 | }
179 | }
180 |
181 | exportObject = PhaserCEButton;
182 | }
183 |
184 | export const Button = exportObject;
185 |
--------------------------------------------------------------------------------
/src/phaserObjects/easing.js:
--------------------------------------------------------------------------------
1 | let exportObject;
2 |
3 | if (Phaser.Easing === undefined) {
4 | exportObject = Phaser.Math.Easing;
5 | } else {
6 | exportObject = Phaser.Easing;
7 | }
8 |
9 | export const Easing = exportObject;
10 |
--------------------------------------------------------------------------------
/src/phaserObjects/graphics.js:
--------------------------------------------------------------------------------
1 | let exportObject;
2 |
3 | if (Phaser.Graphics === undefined) {
4 | exportObject = Phaser.GameObjects.Graphics;
5 | } else {
6 | class PhaserCEGraphics extends Phaser.Graphics {
7 | constructor(game, options) {
8 | super(game, options.x, options.y);
9 | }
10 |
11 | fillStyle(colour, alpha) {
12 | this.beginFill(colour, alpha);
13 | }
14 |
15 | fillRect(x, y, width, height) {
16 | this.drawRect(x, y, width, height);
17 | this.endFill();
18 | }
19 | }
20 |
21 | exportObject = PhaserCEGraphics;
22 | }
23 |
24 | export const Graphics = exportObject;
25 |
--------------------------------------------------------------------------------
/src/phaserObjects/group.js:
--------------------------------------------------------------------------------
1 | let exportObject;
2 |
3 | if (Phaser.Group === undefined) {
4 | // Emulate Phaser CE's worldPosition object
5 | class WorldPosition {
6 | constructor(parent) {
7 | this.parent = parent;
8 | }
9 |
10 | get x() {
11 | return this.parent.x;
12 | }
13 |
14 | get y() {
15 | return this.parent.y;
16 | }
17 | }
18 |
19 | class Phaser3Group extends Phaser.GameObjects.Container {
20 | constructor(game, x = 0, y = 0) {
21 | super(game, x, y);
22 |
23 | game.add.existing(this);
24 |
25 | this.parentContainer = null;
26 |
27 | this.SORT_ASCENDING = -1;
28 | this.SORT_DESCENDING = 1;
29 |
30 | this.version = 3;
31 |
32 | // Emulate Phaser CE's GameObject.alignTo
33 | this.alignToMapping = {
34 | 0: Phaser.Display.Align.To.TopLeft,
35 | 1: Phaser.Display.Align.To.TopCenter,
36 | 8: Phaser.Display.Align.To.RightCenter,
37 | 11: Phaser.Display.Align.To.BottomCenter,
38 | };
39 |
40 | this.worldPosition = new WorldPosition(this);
41 | }
42 |
43 | /** @private Alias to match Phaser CE */
44 | get children() {
45 | return this.list;
46 | }
47 |
48 | get realHeight() {
49 | return this.getBounds().height - this.y;
50 | }
51 |
52 | get realWidth() {
53 | return this.getBounds().width - this.x;
54 | }
55 |
56 | /** Set the origin points of every member of the group. */
57 | setOrigin(x = 0, y = 0) {
58 | let node;
59 |
60 | for (let i = 0; i < this.children.length; i++) {
61 | node = this.children[i];
62 |
63 | if (!(node instanceof Phaser3Group)) {
64 | node.displayOriginX = x; // eslint-disable-line
65 | node.displayOriginY = y; // eslint-disable-line
66 | } else if (node instanceof Phaser3Group) {
67 | node.setOrigin(x, y);
68 | }
69 | }
70 | }
71 |
72 | getNodes() {
73 | return this.getAll();
74 | }
75 |
76 | // Can't align to Containers automatically
77 | alignToContainerBottom(previousNode, child) {
78 | // realWidth changes inside a Container
79 |
80 | let previousNodeWidth;
81 | let previousNodeHeight;
82 | if (child.uiWidgetsObjectRole === 'layout') {
83 | const bounds = previousNode.getBounds();
84 | previousNodeWidth = bounds.width - this.worldPosition.x;
85 | previousNodeHeight = bounds.height - this.worldPosition.y;
86 | } else {
87 | previousNodeWidth = previousNode.width;
88 | previousNodeHeight = previousNode.height;
89 | }
90 |
91 | const centerX = (previousNodeWidth * 0.5);
92 | const bottomY = previousNodeHeight;
93 |
94 | let w = child.width;
95 | if (child.uiWidgetsObjectRole === 'layout') {
96 | w = child.getBounds().width;
97 | }
98 |
99 | child.x = centerX - (w * 0.5); // eslint-disable-line
100 | child.y = previousNode.getBounds().y + bottomY; // eslint-disable-line
101 | }
102 |
103 | /** Containers can't be aligned automatically */
104 | alignContainerToBottom(previousNode, child) { // eslint-disable-line
105 | const bounds = previousNode.getBounds();
106 | const centerX = previousNode.x + (bounds.width * 0.5);
107 |
108 | child.x = centerX - (child.width * 0.5); // eslint-disable-line
109 | child.y = bounds.y; // eslint-disable-line
110 | }
111 |
112 | alignContainerToRight(previousNode, child) { // eslint-disable-line
113 | const rightX = previousNode.x + previousNode.width;
114 |
115 | child.x = rightX; // eslint-disable-line
116 | }
117 |
118 | /** Aligns child to the last object in the group.
119 | * @private
120 | */
121 | alignNodeToPrevious(child, align, paddingX, paddingY) {
122 | const nodes = this.getNodes();
123 | const previousNode = nodes[nodes.length - 2];
124 |
125 | const toGroupAlignFuncs = {
126 | 11: this.alignContainerToBottom, // Column
127 | 8: this.alignContainerToRight, // Row
128 | };
129 |
130 | const toGroupAlignFunc = toGroupAlignFuncs[align];
131 |
132 | if (previousNode instanceof Phaser3Group) {
133 | this.alignToContainerBottom(previousNode, child);
134 | } else if (child instanceof Phaser3Group && (previousNode !== undefined)) {
135 | toGroupAlignFunc(previousNode, child);
136 | } else if (previousNode !== undefined) {
137 | this.alignToMapping[align](child, previousNode, paddingX, paddingY);
138 | }
139 | }
140 | }
141 |
142 | exportObject = Phaser3Group;
143 | } else {
144 | class PhaserCEGroup extends Phaser.Group {
145 | constructor(game) {
146 | super(game);
147 |
148 | this.parentContainer = null;
149 |
150 | this.SORT_ASCENDING = -1;
151 | this.SORT_DESCENDING = 1;
152 | }
153 |
154 | get realHeight() {
155 | return this.height;
156 | }
157 |
158 | get realWidth() {
159 | return this.width;
160 | }
161 |
162 | getNodes() {
163 | return this.children;
164 | }
165 |
166 | /** Aligns child to the last object in the group.
167 | * @private
168 | */
169 | alignNodeToPrevious(child, align, paddingX, paddingY) {
170 | const nodes = this.getNodes();
171 | const previousNode = nodes[nodes.length - 2];
172 |
173 | if (previousNode !== undefined) {
174 | child.alignTo(previousNode, align, paddingX, paddingY);
175 | }
176 | }
177 | }
178 |
179 | exportObject = PhaserCEGroup;
180 | }
181 |
182 | export const Group = exportObject;
183 |
--------------------------------------------------------------------------------
/src/phaserObjects/index.js:
--------------------------------------------------------------------------------
1 | /** Collection of shims that provide different objects based on the Phaser
2 | * version used.
3 | * @module phaserObjects
4 | * @private
5 | */
6 | import { Button } from './button';
7 | import { Easing } from './easing';
8 | import { Graphics } from './graphics';
9 | import { Group } from './group';
10 | import { Keyboard } from './keyboard';
11 | import { KeyCodes } from './keycodes';
12 | import { Rectangle } from './rectangle';
13 | import { Sprite } from './sprite';
14 | import { Tween } from './tween';
15 | import { ViewportMask } from './viewport_mask';
16 |
17 | export {
18 | Button,
19 | Easing,
20 | Graphics,
21 | Group,
22 | Keyboard,
23 | KeyCodes,
24 | Rectangle,
25 | Sprite,
26 | Tween,
27 | ViewportMask,
28 | };
29 |
--------------------------------------------------------------------------------
/src/phaserObjects/keyboard.js:
--------------------------------------------------------------------------------
1 | let exportObject;
2 |
3 | if (Phaser.Keyboard === undefined) {
4 | class Phaser3Keyboard {
5 | addDownEvent(key, event, context) { // eslint-disable-line
6 | key.on('down', event, context);
7 | }
8 |
9 | removeDownEvent(key) { // eslint-disable-line
10 | key.removeListener('down');
11 | }
12 | }
13 |
14 | exportObject = Phaser3Keyboard;
15 | } else {
16 | class PhaserCEKeyboard {
17 | addDownEvent(key, event, context) { // eslint-disable-line
18 | key.onDown.add(event, context);
19 | }
20 |
21 | removeDownEvent(key) { // eslint-disable-line
22 | key.onDown.removeAll();
23 | }
24 | }
25 |
26 | exportObject = PhaserCEKeyboard;
27 | }
28 |
29 | export const Keyboard = exportObject;
30 |
--------------------------------------------------------------------------------
/src/phaserObjects/keycodes.js:
--------------------------------------------------------------------------------
1 | let exportKeyCodes;
2 |
3 | if (Phaser.Keyboard === undefined) {
4 | exportKeyCodes = Phaser.Input.Keyboard.KeyCodes;
5 | } else {
6 | exportKeyCodes = Phaser.Keyboard;
7 | }
8 |
9 | export const KeyCodes = exportKeyCodes;
10 |
--------------------------------------------------------------------------------
/src/phaserObjects/rectangle.js:
--------------------------------------------------------------------------------
1 | let exportObject;
2 |
3 | if (Phaser.Rectangle === undefined) {
4 | exportObject = Phaser.Geom.Rectangle;
5 | } else {
6 | exportObject = Phaser.Rectangle;
7 | }
8 |
9 | export const Rectangle = exportObject;
10 |
--------------------------------------------------------------------------------
/src/phaserObjects/sprite.js:
--------------------------------------------------------------------------------
1 | let exportObject;
2 |
3 | if (Phaser.Sprite === undefined) {
4 | class Phaser3Sprite extends Phaser.GameObjects.Sprite {
5 | constructor(game, x, y, image) {
6 | super(game, x, y, image);
7 |
8 | game.add.existing(this);
9 |
10 | // Map alignment constants to Phaser 3 Align functions
11 | const alignIn = Phaser.Display.Align.In;
12 | this.alignInMapping = {
13 | 0: alignIn.TopLeft,
14 | 1: alignIn.TopCenter,
15 | 2: alignIn.TopRight,
16 | 3: alignIn.LeftTop,
17 | 4: alignIn.LeftCenter,
18 | 5: alignIn.LeftBottom,
19 | 6: alignIn.Center,
20 | 7: alignIn.RightTop,
21 | 8: alignIn.RightCenter,
22 | 9: alignIn.RightBottom,
23 | 10: alignIn.BottomLeft,
24 | 11: alignIn.BottomCenter,
25 | 12: alignIn.BottomRight,
26 | };
27 | }
28 |
29 | get maskX() {
30 | return this.x;
31 | }
32 |
33 | get maskY() {
34 | return this.y;
35 | }
36 |
37 | /** Immitate the API of PhaserCE.
38 | * Move this object behind all others.
39 | * @private
40 | */
41 | sendToBack() {
42 | this.depth = -9999;
43 | }
44 |
45 | /** Immitate the API of PhaserCE.
46 | * Move this object down the depth order by 1.
47 | * @private
48 | */
49 | moveDown() {
50 | this.depth -= 1;
51 | }
52 |
53 | /** Immitate the API of PhaserCE.
54 | * Align this object in another.
55 | * @private
56 | */
57 | alignIn(other, align) {
58 | this.alignInMapping[align](this, other, other.width);
59 | }
60 |
61 | /**
62 | * @private
63 | * Add a callback that is triggered when the pointer moves out of the object.
64 | */
65 | addOutEvent(callback, callbackContext) {
66 | this.on('pointerout', (pointer) => {
67 | callback.call(callbackContext, pointer);
68 | });
69 | }
70 |
71 | /**
72 | * @private
73 | * Add a callback that is triggered when the object is unclicked.
74 | */
75 | addUpEvent(callback, callbackContext) {
76 | this.on('pointerup', (pointer) => {
77 | callback.call(callbackContext, pointer);
78 | });
79 | }
80 |
81 | /**
82 | * @private
83 | * Add a callback that is triggered when the object is clicked.
84 | */
85 | addDownEvent(callback, callbackContext) {
86 | this.on('pointerdown', (pointer) => {
87 | callback.call(callbackContext, pointer);
88 | });
89 | }
90 |
91 | /**
92 | * @private
93 | * Add a callback that is triggered when the object is dragged.
94 | */
95 | addDragEvent(callback, callbackContext) {
96 | this.on('drag', () => { callback.call(callbackContext); });
97 | }
98 | }
99 |
100 | exportObject = Phaser3Sprite;
101 | } else {
102 | class PhaserCESprite extends Phaser.Sprite {
103 | constructor(game, x, y, key) {
104 | super(game, x, y, key);
105 |
106 | game.add.existing(this);
107 | }
108 |
109 | setOrigin(x, y) {
110 | this.anchor = new Phaser.Point(x, y);
111 | }
112 |
113 | get maskX() { //eslint-disable-line
114 | return 0;
115 | }
116 |
117 | get maskY() { //eslint-disable-line
118 | return 0;
119 | }
120 |
121 | setInteractive() {
122 | this.inputEnabled = true;
123 | }
124 |
125 | disableInteractive() {
126 | this.inputEnabled = false;
127 | }
128 |
129 | /**
130 | * @private
131 | * Add a callback that is triggered when the pointer moves out of the object.
132 | */
133 | addOutEvent(callback, callbackContext) {
134 | this.events.onInputOut.add(callback, callbackContext);
135 | }
136 |
137 | /**
138 | * @private
139 | * Add a callback that is triggered when the object is unclicked.
140 | */
141 | addUpEvent(callback, callbackContext) {
142 | this.events.onInputUp.add(callback, callbackContext);
143 | }
144 |
145 | /**
146 | * @private
147 | * Add a callback that is triggered when the object is clicked.
148 | */
149 | addDownEvent(callback, callbackContext) {
150 | this.events.onInputDown.add(callback, callbackContext);
151 | }
152 |
153 | /** Immitate the API of Phaser3.
154 | * The current displayed height of the Object.
155 | * @type {number}
156 | * @private
157 | */
158 | get displayHeight() {
159 | return this.height;
160 | }
161 |
162 | set displayHeight(newValue) {
163 | this.height = newValue;
164 | }
165 |
166 | /** Immitate the API of Phaser3.
167 | * The current displayed width of the Object.
168 | * @type {number}
169 | * @private
170 | */
171 | get displayWidth() {
172 | return this.width;
173 | }
174 |
175 | set displayWidth(newValue) {
176 | this.width = newValue;
177 | }
178 | }
179 |
180 | exportObject = PhaserCESprite;
181 | }
182 |
183 | export const Sprite = exportObject;
184 |
--------------------------------------------------------------------------------
/src/phaserObjects/tween.js:
--------------------------------------------------------------------------------
1 | let exportObject;
2 |
3 | if (Phaser.Tween === undefined) {
4 | class Phaser3Tween {
5 | constructor(game) {
6 | this.game = game;
7 | }
8 |
9 | add(target, properties, duration, ease, onComplete, onUpdate, onStart, callbackScope) {
10 | const config = {
11 | targets: target,
12 | duration,
13 | ease,
14 | onComplete,
15 | onUpdate,
16 | onStart,
17 | callbackScope,
18 | };
19 | const params = Object.assign(config, properties);
20 | this.game.tweens.add(params);
21 | }
22 | }
23 |
24 | exportObject = Phaser3Tween;
25 | } else {
26 | class PhaserCETween {
27 | constructor(game) {
28 | this.game = game;
29 | }
30 |
31 | add(target, properties, duration, ease, onComplete, onUpdate, onStart, onCompleteScope, onUpdateScope, onStartScope) {
32 | const tween = this.game.add.tween(target).to(
33 | properties,
34 | duration,
35 | ease,
36 | true,
37 | );
38 |
39 | // Add the necessary events to the tween.
40 | if (onUpdate) {
41 | tween.onUpdateCallback(onUpdate, onUpdateScope);
42 | }
43 |
44 | if (onComplete) {
45 | tween.onComplete.add(onComplete, onCompleteScope);
46 | }
47 |
48 | if (onStart) {
49 | tween.onStart.add(onStart, onStartScope);
50 | }
51 | }
52 | }
53 |
54 | exportObject = PhaserCETween;
55 | }
56 |
57 | export const Tween = exportObject;
58 |
--------------------------------------------------------------------------------
/src/phaserObjects/viewport_mask.js:
--------------------------------------------------------------------------------
1 | let exportObject;
2 |
3 | if (Phaser.Graphics === undefined) {
4 | class ViewportMask3 extends Phaser.GameObjects.Graphics {
5 | constructor(game, x, y) {
6 | super(game, { x, y });
7 |
8 | this.game = game;
9 | this.x = x;
10 | this.y = y;
11 | }
12 |
13 | create(x, y, width, height) {
14 | this.fillStyle(0xffffff);
15 | this.beginPath();
16 | this.fillRect(x, y, width, height);
17 |
18 | const m = this.createGeometryMask();
19 | m.x = m.geometryMask.x;
20 | m.y = m.geometryMask.y;
21 | m.width = width;
22 | m.height = height;
23 |
24 | return m;
25 | }
26 | }
27 | exportObject = ViewportMask3;
28 | } else {
29 | class ViewportMaskCE extends Phaser.Graphics {
30 | get scaleX() {
31 | return this.scale.x;
32 | }
33 |
34 | set scaleX(value) {
35 | this.scale.x = value;
36 | }
37 |
38 | get scaleY() {
39 | return this.scale.y;
40 | }
41 |
42 | set scaleY(value) {
43 | this.scale.y = value;
44 | }
45 |
46 | create(x, y, width, height) {
47 | this.beginFill(0x0000ff);
48 | this.drawRect(x, y, width, height);
49 | this.endFill();
50 |
51 | // Match the internals of Phaser3
52 | this.geometryMask = this;
53 |
54 | return this;
55 | }
56 | }
57 |
58 | exportObject = ViewportMaskCE;
59 | }
60 |
61 | export const ViewportMask = exportObject;
62 |
--------------------------------------------------------------------------------
/src/utils.js:
--------------------------------------------------------------------------------
1 | const utils = {};
2 |
3 | /**
4 | * @private
5 | * A modulo operator that doesn't allow negative numbers.
6 | * @param {number} dividend
7 | * @param {number} divisor
8 | * @returns {number}
9 | */
10 | utils.modulo = function modulo(dividend, divisor) {
11 | return ((((dividend) % divisor) + divisor) % divisor);
12 | };
13 |
14 | /**
15 | * @private
16 | * Select an operator action using a string value.
17 | */
18 | utils.operators = {
19 | '+': function add(a, b) { return a + b; },
20 | '-': function sub(a, b) { return a - b; },
21 | };
22 |
23 | export { utils };
24 |
--------------------------------------------------------------------------------
/src/wheel3D.js:
--------------------------------------------------------------------------------
1 | import { EventEmitter } from 'eventemitter3';
2 |
3 | import { utils } from './utils';
4 |
5 | import * as PhaserObjects from './phaserObjects';
6 |
7 | /**
8 | * Represents a single point in a Wheel3D.
9 | * @ignore
10 | */
11 | class VectorPoint {
12 | /**
13 | * @ignore
14 | * @param {number} x - The point's virtual x location.
15 | * @param {number} y - The point's virtual y location.
16 | * @param {number} z - The point's virtual z location.
17 | * @param {Object} sprite - The sprite associated with this point.
18 | * @param {number} position - The point's position on the Wheel3D.
19 | */
20 | constructor(x = 0, y = 0, z = 0, sprite = null, position = 0) {
21 | this.x = x;
22 | this.y = y;
23 | this.z = z;
24 | this.sprite = sprite;
25 | this.position = position;
26 | }
27 |
28 | getSinCosOfAngle(angle) { // eslint-disable-line class-methods-use-this
29 | const rad = (angle * Math.PI) / 180;
30 | const cosAngle = Math.cos(rad);
31 | const sinAngle = Math.sin(rad);
32 |
33 | return { cosine: cosAngle, sine: sinAngle };
34 | }
35 |
36 | rotateY(angle) {
37 | const angles = this.getSinCosOfAngle(angle);
38 | const z = (this.z * angles.cosine) - (this.x * angles.sine);
39 | const x = (this.z * angles.sine) + (this.x * angles.cosine);
40 |
41 | return new VectorPoint(x, this.y, z);
42 | }
43 |
44 | rotateX(angle) {
45 | const angles = this.getSinCosOfAngle(angle);
46 | const y = (this.y * angles.cosine) - (this.z * angles.sine);
47 | const z = (this.y * angles.sine) + (this.z * angles.cosine);
48 |
49 | return new VectorPoint(this.x, y, z);
50 | }
51 |
52 | rotateZ(angle) {
53 | const angles = this.getSinCosOfAngle(angle);
54 | const x = (this.x * angles.cosine) - (this.y * angles.sine);
55 | const y = (this.x * angles.sine) + (this.y * angles.cosine);
56 |
57 | return new VectorPoint(x, y, this.z);
58 | }
59 |
60 | /** Rotate the point along the given axis by the given angle.
61 | * @param {string} axis - The axis to rotate.
62 | * @param {number} angle - The angle to rotate by.
63 | */
64 | rotate(axis, angle) {
65 | let newVector = null;
66 |
67 | if (axis === 'x') {
68 | newVector = this.rotateX(angle);
69 | } else if (axis === 'y') {
70 | newVector = this.rotateY(angle);
71 | } else if (axis === 'z') {
72 | newVector = this.rotateZ(angle);
73 | }
74 |
75 | return newVector;
76 | }
77 |
78 | /** Project the point to the correct physical location on screen.
79 | * z axis is not projected, because screens are 2D.
80 | */
81 | project(width, height, factor) {
82 | const x = (this.x * factor) + width;
83 | const y = (-this.y * factor) + height;
84 |
85 | return new VectorPoint(x, y, this.z);
86 | }
87 | }
88 |
89 | /**
90 | * A Wheel of sprites where each item's position is projected in 3D space.
91 | * The number of points is automatically determined by the number of items.
92 | */
93 | export class Wheel3D {
94 | /**
95 | * @param {Object} game - Reference to current game instance.
96 | * @param {array} sprites - List of sprites to use in the wheel.
97 | * @param {Number} firstPlace - Determines which position on the wheel is the active one.
98 | * @param {Number} zoom - Determines how far to project the points.
99 | * @param {String} axis - The axis the wheel3D places items around.
100 | * @param {Object} rotations - Axis and angle to rotate the entire wheel after the initial projection.
101 | * @param {Object} visibleRange - Determines which items on the wheel should be visible. If none provided, assumes all items should be visible.
102 | * @param {Object} tweenParams - Array with the duration and easing function for the movement tween.
103 | */
104 | constructor(game, xy, sprites, firstPlace, zoom, axis, rotations, visibleRange, tweenParams) {
105 | this.game = game;
106 | this.xy = xy;
107 | this.sprites = sprites;
108 | this.firstPlace = firstPlace;
109 | this.zoom = zoom;
110 | this.axis = axis;
111 | this.rotationAxis = rotations;
112 | this.visibleRange = visibleRange || null;
113 | this.tweenParams = tweenParams || { duration: 300, ease: PhaserObjects.Easing.Quadratic.Out };
114 |
115 | this.emitter = new EventEmitter();
116 |
117 | // Group to store wheel sprites in, used for zindex sorting.
118 | this.group = new PhaserObjects.Group(game);
119 | }
120 |
121 | activate() {
122 | let angle;
123 | let radCos;
124 | let radSin;
125 | let nx;
126 | let ny;
127 | let nz;
128 |
129 | this.pointsAmount = this.sprites.length;
130 | this.totalPositions = this.pointsAmount - 1;
131 | this.rotationAmount = 360 / this.pointsAmount;
132 |
133 | // Determines which items in the wheel should be visible.
134 | if (this.visibleRange !== null) {
135 | const allPositions = [];
136 | for (let i = 0; i <= this.totalPositions; i++) {
137 | allPositions.push(i);
138 | }
139 |
140 | const maxV = allPositions.slice(0, this.visibleRange.max);
141 | const minV = allPositions.slice(this.visibleRange.min);
142 |
143 | this.visiblePositions = maxV.concat(minV);
144 | }
145 |
146 | // Prevents slamming down the move keys.
147 | this.moving = false;
148 |
149 | // Stores the direction the wheel is moving in.
150 | this.direction = null;
151 |
152 | // Set point positions on logical circle.
153 | this.wheelItems = [];
154 | const radius = 1;
155 | const slice = (2 * Math.PI) / this.pointsAmount;
156 |
157 | // For a vertical wheel, the X axis is -1. Points are laid out on the Y axis (height) and Z axis (depth). The wheel rotates around the X axis.
158 | // For a horizontal wheel, the Y Axis is -1. Points are laid out on the X axis (width) and Z axis (depth). The wheel rotates around the Y axis.
159 | // For flat circular wheel, the Z Axis is -1. Points are laid out on the X axis (width) and Y axis (height). The wheel rotates around the Z axis.
160 | for (let i = 0; i < this.pointsAmount; i++) {
161 | // Add sprite to group.
162 | this.sprites[i].wheelPosition = i;
163 | this.group.add(this.sprites[i]);
164 |
165 | angle = slice * i;
166 | radCos = radius * Math.cos(angle);
167 | radSin = radius * Math.sin(angle);
168 |
169 | if (this.axis === 'x') {
170 | nx = -1;
171 | ny = radCos;
172 | nz = radSin;
173 | } else if (this.axis === 'y') {
174 | nx = radCos;
175 | ny = -1;
176 | nz = radSin;
177 | } else if (this.axis === 'z') {
178 | nx = radCos;
179 | ny = radSin;
180 | nz = -1;
181 | }
182 |
183 | this.wheelItems.push(new VectorPoint(nx, ny, nz, this.sprites[i], i));
184 | }
185 |
186 | // Active Point
187 | this.active = this.wheelItems[this.firstPlace].sprite;
188 |
189 | this.project();
190 | }
191 |
192 | /** Move the wheel backwards. */
193 | moveBack() {
194 | if (this.moving === false) {
195 | this.moving = true;
196 | this.direction = 0;
197 |
198 | if (this.axis === 'x' || this.axis === 'z') {
199 | this.rotationAxis[this.axis] += this.rotationAmount;
200 | } else { // y axis needs to go in the opposite direction.
201 | this.rotationAxis[this.axis] -= this.rotationAmount;
202 | }
203 |
204 | this.updatePosition('+');
205 |
206 | this.project();
207 | this.resetAngle();
208 | }
209 | }
210 |
211 | /** Move the wheel forward. */
212 | moveForward() {
213 | if (this.moving === false) {
214 | this.moving = true;
215 | this.direction = 1;
216 |
217 | if (this.axis === 'x' || this.axis === 'z') {
218 | this.rotationAxis[this.axis] -= this.rotationAmount;
219 | } else { // y axis needs to go in the opposite direction.
220 | this.rotationAxis[this.axis] += this.rotationAmount;
221 | }
222 |
223 | this.updatePosition('-');
224 |
225 | this.project();
226 | this.resetAngle();
227 | }
228 | }
229 |
230 | /** Project every item in the wheel to it's physical location. */
231 | project() {
232 | // Create a list with the axes, then remove the projected axis.
233 | const arr = ['x', 'y', 'z'];
234 | const idx = arr.indexOf(this.axis);
235 | arr.splice(idx, 1);
236 |
237 | // We only need to call all this when moving. It doesn't need to be done
238 | // every update
239 | for (let i = 0; i < this.wheelItems.length; i++) {
240 | // Rotate along the projected axis
241 | const rotationOne = this.wheelItems[i].rotate(
242 | this.axis,
243 | this.rotationAxis[this.axis],
244 | );
245 |
246 | // Rotate the other 2 axes
247 | const rotationTwo = rotationOne.rotate(arr[0], this.rotationAxis[arr[0]]);
248 | const rotationThree = rotationTwo.rotate(arr[1], this.rotationAxis[arr[1]]);
249 |
250 | const p = rotationThree.project(
251 | this.xy.x,
252 | this.xy.y,
253 | this.zoom,
254 | );
255 |
256 | const transformed = this.wheelItems[i].sprite;
257 | transformed.lz = p.z;
258 |
259 | // Ensure active sprite has no scale/alpha changes.
260 | if (this.wheelItems[i].position === this.firstPlace) {
261 | transformed.alpha = 1.0;
262 | this.active = this.wheelItems[i].sprite;
263 | } else {
264 | if (this.visibleRange !== null) {
265 | const includes = this.visiblePositions.includes(this.wheelItems[i].position);
266 | // const includes = _.includes(
267 | // this.visiblePositions,
268 | // this.wheelItems[i].position,
269 | // );
270 |
271 | if (includes) {
272 | transformed.alpha = 1.0;
273 | } else {
274 | transformed.alpha = 0.0;
275 | }
276 | }
277 | }
278 |
279 | const newTween = new PhaserObjects.Tween(this.game);
280 |
281 | if (i !== this.wheelItems.length - 1) {
282 | newTween.add(
283 | transformed,
284 | { x: p.x, y: p.y },
285 | this.tweenParams.duration,
286 | this.tweenParams.ease,
287 | );
288 | } else {
289 | // Wheel's signals are dispatched by the last tween.
290 | newTween.add(
291 | transformed,
292 | { x: p.x, y: p.y },
293 | this.tweenParams.duration,
294 | this.tweenParams.ease,
295 | () => { this.enableMoving(); this.dispatchOnComplete(); },
296 | null,
297 | () => { this.dispatchOnStart(); },
298 | this,
299 | );
300 | }
301 | }
302 |
303 | // Sort wheelItems by the projection's z axis for correct z-order when drawing.
304 | this.group.sort('lz', PhaserObjects.Group.SORT_ASCENDING);
305 | }
306 |
307 | /** Called after movement starts. */
308 | dispatchOnStart() {
309 | this.emitter.emit('start', this);
310 | }
311 |
312 | /** Called after movement is finished. */
313 | dispatchOnComplete() {
314 | if (this.direction === 0) {
315 | this.emitter.emit('backComplete', this);
316 | } else if (this.direction === 1) {
317 | this.emitter.emit('forwardComplete', this);
318 | }
319 |
320 | this.emitter.emit('complete', this);
321 | }
322 |
323 | /** Once the buttons have finished their move animation, allow them to move again. */
324 | enableMoving() {
325 | this.moving = false;
326 | }
327 |
328 | /** Move all the WheelItem's position by 1. */
329 | updatePosition(operator) {
330 | for (let i = 0; i < this.wheelItems.length; i++) {
331 | const { position } = this.wheelItems[i];
332 | this.wheelItems[i].position = utils.operators[operator](position, 1);
333 |
334 | const m = utils.modulo(this.wheelItems[i].position, this.pointsAmount);
335 | this.wheelItems[i].position = m;
336 | this.wheelItems[i].sprite.wheelPosition = m;
337 | }
338 | }
339 |
340 | /** Make sure rotation can't go past 360 in either direction. */
341 | resetAngle() {
342 | const angle = this.rotationAxis[this.axis];
343 |
344 | if (angle === 360 || angle === -360) {
345 | this.rotationAxis[this.axis] = 0;
346 | }
347 | }
348 | }
349 |
--------------------------------------------------------------------------------
/tests/column.spec.js:
--------------------------------------------------------------------------------
1 | describe("Column", function(){
2 | var column;
3 |
4 | describe("When I add a second node to a Column", function(){
5 | it("Then should be placed under the first node", function(){
6 | var dummySprite = game.add.sprite(0, 0, "dummySprite");
7 | var dummySprite2 = game.add.sprite(0, 0, "dummySprite");
8 | column = new uiWidgets.Column(game);
9 | column.addNode(dummySprite);
10 | column.addNode(dummySprite2);
11 |
12 | chai.expect(column.children[1].y).to.equal(78);
13 | });
14 | });
15 |
16 | describe("When I add a second node to a Column with padding", function(){
17 | it("Then should be placed under to the first node, offset by the padding", function(){
18 | var dummySprite = game.add.sprite(0, 0, "dummySprite");
19 | var dummySprite2 = game.add.sprite(0, 0, "dummySprite");
20 | column = new uiWidgets.Column(game);
21 | column.addNode(dummySprite);
22 | column.addNode(dummySprite2, 0, 23);
23 |
24 | chai.expect(column.children[1].y).to.equal(101);
25 | });
26 | });
27 |
28 | afterEach(function() {
29 | column.destroy();
30 | });
31 | });
32 |
--------------------------------------------------------------------------------
/tests/frame.spec.js:
--------------------------------------------------------------------------------
1 | describe("Frame", function(){
2 | describe("When I create a new Frame", function(){
3 | it("Then the x, y attributes should be set correctly", function(){
4 | var frame = new uiWidgets.Frame(game, 12, 37);
5 |
6 | chai.expect(frame.x).to.equal(12);
7 | chai.expect(frame.y).to.equal(37);
8 |
9 | });
10 | });
11 | describe("When I create a new Frame with no x, y attributes", function(){
12 | it("Then the x, y attributes should be 0", function(){
13 | var frame = new uiWidgets.Frame(game);
14 |
15 | chai.expect(frame.x).to.equal(0);
16 | chai.expect(frame.y).to.equal(0);
17 |
18 | });
19 | });
20 | describe("When I add a new node to a Frame", function(){
21 | it("Then it should be added as a child of the Frame", function(){
22 | var frame = new uiWidgets.Frame(game);
23 | var dummySprite = game.add.sprite(0, 0);
24 | frame.addNode(dummySprite);
25 |
26 | chai.expect(frame.children[0]).to.equal(dummySprite);
27 | });
28 | });
29 | });
30 |
--------------------------------------------------------------------------------
/tests/row.spec.js:
--------------------------------------------------------------------------------
1 | describe("Row", function(){
2 | var row;
3 |
4 | describe("When I add a second node to a Row", function(){
5 | it("Then should be placed next to the first node", function(){
6 | var dummySprite = game.add.sprite(0, 0, "dummySprite");
7 | var dummySprite2 = game.add.sprite(0, 0, "dummySprite");
8 | row = new uiWidgets.Row(game);
9 | row.addNode(dummySprite);
10 | row.addNode(dummySprite2);
11 |
12 | chai.expect(row.children[1].x).to.equal(300);
13 | });
14 | });
15 |
16 | describe("When I add a second node to a Row with padding", function(){
17 | it("Then should be placed next to the first node, offset by the padding", function(){
18 | var dummySprite = game.add.sprite(0, 0, "dummySprite");
19 | var dummySprite2 = game.add.sprite(0, 0, "dummySprite");
20 | row = new uiWidgets.Row(game);
21 | row.addNode(dummySprite);
22 | row.addNode(dummySprite2, 17, 0);
23 |
24 | chai.expect(row.children[1].x).to.equal(317);
25 | });
26 | });
27 |
28 | afterEach(function() {
29 | row.destroy();
30 | });
31 | });
32 |
--------------------------------------------------------------------------------
/tests/testrunner.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | Mocha Tests
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
15 |
16 |
17 |
18 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 |
47 |
--------------------------------------------------------------------------------
/tests/testrunner_phaser3.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | Mocha Tests
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
15 |
16 |
17 |
18 |
46 |
47 |
48 |
49 |
50 |
51 |
52 |
53 |
54 |
55 |
56 |
57 |
58 |
59 |
60 |
--------------------------------------------------------------------------------
/tests/viewport.spec.js:
--------------------------------------------------------------------------------
1 | describe("Viewport", function(){
2 | var viewport;
3 |
4 | describe("When I create a new Viewport", function(){
5 | it("Then the x, y attributes should be set correctly", function(){
6 | viewport = new uiWidgets.Viewport(game, 17, 102, 103, 104);
7 | chai.expect(viewport.x).to.equal(17);
8 | chai.expect(viewport.y).to.equal(102);
9 | });
10 |
11 | it("Then the area attribute should be set correctly", function(){
12 | viewport = new uiWidgets.Viewport(game, 17, 102, 103, 104);
13 | chai.expect(viewport.area.x).to.equal(17);
14 | chai.expect(viewport.area.y).to.equal(102);
15 | chai.expect(viewport.area.width).to.equal(103);
16 | chai.expect(viewport.area.height).to.equal(104);
17 | });
18 |
19 | it("Then the viewport mask attributes should be set correctly", function(){
20 | viewport = new uiWidgets.Viewport(game, 17, 102, 103, 104);
21 | chai.expect(viewport.mask.x).to.equal(17);
22 | chai.expect(viewport.mask.y).to.equal(102);
23 | });
24 | });
25 |
26 | afterEach(function() {
27 | viewport.destroy();
28 | });
29 |
30 | });
31 |
--------------------------------------------------------------------------------
/tests/wheel3D.spec.js:
--------------------------------------------------------------------------------
1 | describe("Wheel3D", function(){
2 | describe("When I create a new Wheel3D", function(){
3 | describe("And I move the wheel backwards", function(){
4 | it("Then the active sprite should be the previous one", function(done){
5 | var menuList = [];
6 | for (var i = 0; i < 4; i++) {
7 | var dummySprite = game.add.sprite(0, 0, "dummySprite");
8 | menuList.push(dummySprite);
9 | }
10 |
11 | var wheel = new uiWidgets.Wheel3D(
12 | game,
13 | {"x": 0, "y": 0},
14 | menuList,
15 | 0,
16 | 90,
17 | "x",
18 | {"x":0, "y": -90, "z": 0}
19 | );
20 |
21 | var isDone = false;
22 |
23 | wheel.emitter.on('complete',
24 | function (wheel) {
25 | isDone = true;
26 | });
27 | wheel.activate();
28 | wheel.moveBack();
29 |
30 | setTimeout( function () {
31 | try {
32 | chai.expect(isDone).to.equal(true);
33 | done();
34 | } catch(err) {
35 | done(err);
36 | }
37 | }, 500 );
38 | });
39 | });
40 | describe("And I move the wheel forward", function(){
41 | it("Then the active sprite should be the next one", function(done){
42 | var menuList = [];
43 | for (var i = 0; i < 4; i++) {
44 | var dummySprite = game.add.sprite(0, 0, "dummySprite");
45 | menuList.push(dummySprite);
46 | }
47 |
48 | var wheel = new uiWidgets.Wheel3D(
49 | game,
50 | {"x": 0, "y": 0},
51 | menuList,
52 | 0,
53 | 90,
54 | "x",
55 | {"x":0, "y": -90, "z": 0}
56 | );
57 |
58 | var isDone = false;
59 |
60 | wheel.emitter.on('complete',
61 | function (wheel) {
62 | isDone = true;
63 | });
64 | wheel.activate();
65 | wheel.moveForward();
66 |
67 | setTimeout( function () {
68 | try {
69 | chai.expect(isDone).to.equal(true);
70 | done();
71 | } catch(err) {
72 | done(err);
73 | }
74 | }, 500 );
75 |
76 | });
77 | });
78 | });
79 | });
80 |
--------------------------------------------------------------------------------
/webpack.config.js:
--------------------------------------------------------------------------------
1 | const path = require('path');
2 |
3 | const ESLintPlugin = require('eslint-webpack-plugin');
4 |
5 | module.exports = {
6 | mode: 'production',
7 | plugins: [new ESLintPlugin()],
8 | module: {
9 | rules: [
10 | {
11 | test: /\.(js)$/,
12 | exclude: /node_modules/,
13 | use: {
14 | loader: 'babel-loader',
15 | options: {
16 | presets: ['@babel/preset-env'],
17 | },
18 | },
19 | },
20 | ],
21 | },
22 | entry: './src/entry.js',
23 | output: {
24 | library: 'uiWidgets',
25 | libraryTarget: 'umd',
26 | umdNamedDefine: true,
27 | filename: 'phaser-ui-tools.js',
28 | path: path.resolve(__dirname, 'dist'),
29 | },
30 | };
31 |
--------------------------------------------------------------------------------