├── .gitignore ├── LICENSE ├── README.md ├── dist ├── perfect-gui.mjs └── perfect-gui.umd.js ├── package-lock.json ├── package.json ├── src ├── addons │ └── three.js ├── components │ └── Slider.js ├── index.js └── styles │ ├── _button.css.js │ ├── _color.css.js │ ├── _folder.css.js │ ├── _image.css.js │ ├── _list.css.js │ ├── _slider.css.js │ ├── _switch.css.js │ ├── _vector2.css.js │ └── styles.js └── vite.config.js /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | .DS_Store 3 | Thumbs.db 4 | build -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2019 Thibaut Foussard 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 | # Perfect GUI 2 | A nice, simple and (probably not) perfect GUI for JavaScript. 3 | 4 | ## API 5 | [Documentation](https://thibka.github.io/perfect-gui/public/) 6 | 7 | 8 | ## Features 9 | - image buttons 10 | - multiple panels 11 | - easy positioning 12 | - draggable panels 13 | - two-dimensional vector visualization 14 | 15 | 16 | 17 | ## Install 18 | 19 | ### With NPM 20 | 21 | ```bash 22 | npm i perfect-gui 23 | ``` 24 | 25 | ### Import from a CDN 26 | 27 | ```javascript 28 | 35 | ``` 36 | 37 | ## Hello world 38 | 39 | ```javascript 40 | import GUI from 'perfect-gui'; 41 | 42 | const gui = new GUI(); 43 | 44 | gui.button('Click me', callback); 45 | ``` 46 | 47 | ## Options 48 | ```javascript 49 | const gui = new GUI({ 50 | name: 'My GUI', 51 | // Name of the panel. 52 | // Default is null. 53 | 54 | container: '#container', 55 | // Element containing the GUI 56 | // Default is document.body 57 | 58 | width: 250, 59 | // Width of the panel (in pixels). 60 | // Default is 290. 61 | 62 | maxHeight: 500, 63 | // Maximum height beyond which scrolling is necessary. 64 | // Default is the smallest value between the height of the window and the height of the container. 65 | 66 | closed: false, 67 | // Defines whether the panel should be closed by default. 68 | // Default is false. 69 | 70 | position: 'bottom right', 71 | // Defines where to place the panel on screen. 72 | // Accepted values are 'top', 'bottom', 'left' and 'right' in no particular order ('bottom right' = 'right bottom'). 73 | // If multiple instances have the same position, they will be stacked horizontally. 74 | // Default is 'top right'. 75 | 76 | draggable: false, 77 | // Defines if the panel can be manually moved across the screen. 78 | // Default is false. 79 | 80 | autoRepositioning: true, 81 | // If set to true, the panel position will be reset when the screen is resized. 82 | // If a panel has been dragged, it won't be be affected. 83 | // Default is true. 84 | 85 | color: '#bada55', 86 | // Default is #2e2e2e 87 | 88 | onUpdate: () => { 89 | // Callback function triggered each time this GUI instance is updated. 90 | } 91 | }); 92 | ``` 93 | 94 | ## [Methods](https://thibka.github.io/perfect-gui/public/) 95 | 96 | * [button](https://thibka.github.io/perfect-gui/public/#button) 97 | * [slider](https://thibka.github.io/perfect-gui/public/#slider) 98 | * [toggle](https://thibka.github.io/perfect-gui/public/#toggle) 99 | * [list](https://thibka.github.io/perfect-gui/public/#list) 100 | * [image](https://thibka.github.io/perfect-gui/public/#image) 101 | * [color](https://thibka.github.io/perfect-gui/public/#color) 102 | * [vector2](https://thibka.github.io/perfect-gui/public/#vector2) 103 | * [folder](https://thibka.github.io/perfect-gui/public/#folder) 104 | * [toggleClose](https://thibka.github.io/perfect-gui/public) -------------------------------------------------------------------------------- /dist/perfect-gui.mjs: -------------------------------------------------------------------------------- 1 | class R { 2 | constructor(e, t = {}, i) { 3 | if (this.parent = e, this.propReferences = [], typeof t != "object") 4 | throw Error(`[GUI] slider() first parameter must be an object. Received: ${typeof t}.`); 5 | let a = typeof t.name == "string" && t.name || " "; 6 | this.isObject = !1; 7 | let s = null; 8 | this.obj = t.obj, this.prop = t.prop; 9 | let o = typeof t.value == "number" ? t.value : null; 10 | if (this.min = t.min ?? 0, this.max = t.max ?? 1, this.step = t.step || (this.max - this.min) / 100, this.decimals = this.parent._countDecimals(this.step), this.callback = typeof i == "function" ? i : null, o !== null) 11 | (this.prop != null || this.obj != null) && console.warn('[GUI] slider() "obj" and "prop" parameters are ignored when a "value" parameter is used.'); 12 | else if (this.prop != null && this.obj != null) { 13 | if (typeof this.prop != "string") 14 | throw Error(`[GUI] slider() "prop" parameter must be an string. Received: ${typeof this.prop}.`); 15 | if (typeof this.obj != "object") 16 | throw Error(`[GUI] slider() "obj" parameter must be an object. Received: ${typeof this.obj}.`); 17 | a == " " && (a = this.prop), s = this.propReferences.push(this.obj[this.prop]) - 1, this.isObject = !0; 18 | } else 19 | (this.prop != null && this.obj == null || this.prop == null && this.obj != null) && console.warn('[GUI] slider() "obj" and "prop" parameters must be used together.'), o = (this.max - this.min) / 2; 20 | const r = typeof t.tooltip == "string" ? t.tooltip : t.tooltip === !0 ? a : null; 21 | this.imageContainer = null; 22 | const n = document.createElement("div"); 23 | n.className = "p-gui__slider", r && n.setAttribute("title", r); 24 | const c = document.createElement("div"); 25 | c.className = "p-gui__slider-name", c.textContent = a, n.append(c), this.ctrlDiv = document.createElement("div"), this.ctrlDiv.className = "p-gui__slider-ctrl", this.ctrlDiv.setAttribute("type", "range"), this.ctrlDiv.setAttribute("min", this.min), this.ctrlDiv.setAttribute("max", this.max), n.append(this.ctrlDiv); 26 | const l = document.createElement("div"); 27 | return l.className = "p-gui__slider-bar", this.ctrlDiv.append(l), this.handle = document.createElement("div"), this.handle.className = "p-gui__slider-handle", this.ctrlDiv.append(this.handle), this.filling = document.createElement("div"), this.filling.className = "p-gui__slider-filling", l.append(this.filling), this.valueInput = document.createElement("input"), this.valueInput.className = "p-gui__slider-value", this.valueInput.value = this.isObject ? this.obj[this.prop] : o, n.append(this.valueInput), setTimeout(() => { 28 | const d = this.handle.offsetWidth; 29 | this.handle.position = this._mapLinear(this.valueInput.value, this.min, this.max, d / 2, 88 - d / 2), this.handle.position = Math.min(this.handle.position, 88 - d / 2), this.handle.position = Math.max(this.handle.position, d / 2), this.handle.style.transform = `translate(-50%, -50%) translateX(${this.handle.position}px)`, this.filling.style.width = `${this.handle.position}px`; 30 | }, 0), this.valueInput.addEventListener("change", () => { 31 | this._updateHandlePositionFromValue(), this._triggerCallbacks(); 32 | }), this.ctrlDiv.addEventListener("pointerdown", (d) => { 33 | this.ctrlDiv.pointerDown = !0, this.ctrlDiv.prevPosition = d.clientX, this._updateHandlePositionFromPointer(d, !0); 34 | }), window.addEventListener("pointerup", (d) => { 35 | this.ctrlDiv.pointerDown = !1; 36 | }), window.addEventListener("pointermove", (d) => { 37 | this.ctrlDiv.pointerDown && (this.ctrlDiv.pointerDelta = d.clientX - this.ctrlDiv.prevPosition, this._updateHandlePositionFromPointer(d)); 38 | }), this.isObject && Object.defineProperty(this.obj, this.prop, { 39 | set: (d) => { 40 | this.propReferences[s] = d, this.valueInput.value = d, this._updateHandlePositionFromValue(), this.callback && this.callback(parseFloat(this.valueInput.value)); 41 | }, 42 | get: () => this.propReferences[s] 43 | }), n; 44 | } 45 | _updateHandlePositionFromPointer(e, t = !1) { 46 | const i = this.ctrlDiv.offsetWidth, a = this.handle.offsetWidth, s = e.clientX - this.ctrlDiv.prevPosition, o = parseFloat(this.valueInput.value); 47 | let r; 48 | t ? r = e.offsetX : r = this.handle.position + s, r = Math.max(a / 2, Math.min(r, i - a / 2)); 49 | let n = this.min + (this.max - this.min) * (r - a / 2) / (i - a); 50 | n > o ? n = this._quantizeFloor(n, this.step) : n = this._quantizeCeil(n, this.step), n = parseFloat(n.toFixed(9)); 51 | const c = parseFloat((o + this.step).toFixed(9)), l = parseFloat((o - this.step).toFixed(9)); 52 | (n >= c || n <= l) && (n = n.toFixed(this.decimals), this.valueInput.value = n, this.ctrlDiv.prevPosition = e.clientX, this.handle.style.transform = `translate(-50%, -50%) translateX(${r}px)`, this.handle.position = r, this.filling.style.width = this.handle.position + "px", this._triggerCallbacks()); 53 | } 54 | _updateHandlePositionFromValue() { 55 | const e = this.ctrlDiv.offsetWidth, t = this.handle.offsetWidth; 56 | let i = this._mapLinear(this.valueInput.value, this.min, this.max, t / 2, e - t / 2); 57 | i = Math.max(t / 2, Math.min(i, e - t / 2)), this.handle.style.transform = `translate(-50%, -50%) translateX(${i}px)`, this.handle.position = i, this.filling.style.width = this.handle.position + "px"; 58 | } 59 | _triggerCallbacks() { 60 | this.isObject ? this.obj[this.prop] = parseFloat(this.valueInput.value) : this.callback && this.callback(parseFloat(this.valueInput.value)), this.parent.onUpdate ? this.parent.onUpdate() : this.parent.isFolder && this.parent.firstParent.onUpdate && this.parent.firstParent.onUpdate(); 61 | } 62 | _mapLinear(e, t, i, a, s) { 63 | return a + (e - t) * (s - a) / (i - t); 64 | } 65 | _quantize(e, t) { 66 | return t * Math.round(e / t); 67 | } 68 | _quantizeCeil(e, t) { 69 | return t * Math.ceil(e / t); 70 | } 71 | _quantizeFloor(e, t) { 72 | return t * Math.floor(e / t); 73 | } 74 | } 75 | const P = ( 76 | /* css */ 77 | ` 78 | .p-gui__button { 79 | background: var(--color-accent); 80 | text-align: center; 81 | color: white; 82 | border: none; 83 | border: 1px solid transparent; 84 | box-sizing: border-box; 85 | transition: var(--transition) background, var(--transition) border-color; 86 | } 87 | 88 | .p-gui__button:hover { 89 | background: var(--color-accent-hover); 90 | color: var(--color-text-light); 91 | border-color: rgba(255, 255, 255, 0.2); 92 | } 93 | 94 | .p-gui__folder .p-gui__button { 95 | margin-inline: 0; 96 | } 97 | ` 98 | ), O = ( 99 | /* css */ 100 | ` 101 | .p-gui__slider { 102 | position: relative; 103 | min-height: 14px; 104 | display: flex; 105 | align-items: center; 106 | justify-content: space-between; 107 | gap: 10px; 108 | color: var(--color-text-dark); 109 | transition: color var(--transition); 110 | } 111 | 112 | .p-gui__slider:hover { 113 | color: var(--color-text-light); 114 | } 115 | 116 | .p-gui__slider-name { 117 | width: 50%; 118 | text-overflow: ellipsis; 119 | overflow: hidden; 120 | } 121 | 122 | .p-gui__slider-ctrl { 123 | -webkit-appearance: none; 124 | padding: 0; 125 | font: inherit; 126 | outline: none; 127 | box-sizing: border-box; 128 | cursor: pointer; 129 | position: relative; 130 | right: 0; 131 | height: 14px; 132 | margin: 0 0 0 auto; 133 | width: 37%; 134 | } 135 | 136 | .p-gui__slider-bar { 137 | position: absolute; 138 | top: 50%; 139 | left: 0; 140 | height: 2px; 141 | background: rgba(255, 255, 255, .2); 142 | width: 100%; 143 | transform: translateY(-50%); 144 | } 145 | 146 | .p-gui__slider-filling { 147 | position: absolute; 148 | top: -25%; 149 | left: 0; 150 | height: 150%; 151 | background: var(--color-accent); 152 | width: 0; 153 | } 154 | 155 | .p-gui__slider:hover .p-gui__slider-filling { 156 | background: var(--color-accent-hover); 157 | } 158 | 159 | .p-gui__slider-handle { 160 | width: 15px; 161 | height: 8px; 162 | position: absolute; 163 | top: 50%; 164 | left: 0; 165 | border-radius: 2px; 166 | transform: translate(-50%, -50%); 167 | pointer-events: none; 168 | background: var(--color-text-dark); 169 | box-shadow: 0 0 2px rgba(0, 0, 0, .5); 170 | } 171 | 172 | .p-gui__slider:hover .p-gui__slider-handle { 173 | background: var(--color-text-light); 174 | } 175 | 176 | .p-gui__slider-value { 177 | display: inline-block; 178 | right: 7px; 179 | width: 13%; 180 | border: none; 181 | color: white; 182 | border-radius: 2px; 183 | background: rgba(255, 255, 255, 0.1); 184 | padding: 2px 4px; 185 | color: inherit; 186 | } 187 | 188 | .p-gui__slider-value:focus { 189 | outline: none; 190 | } 191 | ` 192 | ), $ = ( 193 | /* css */ 194 | ` 195 | .p-gui__list { 196 | cursor: default; 197 | color: var(--color-text-dark); 198 | transition: var(--transition) color; 199 | } 200 | 201 | .p-gui__list:hover { 202 | color: var(--color-text-light); 203 | } 204 | 205 | .p-gui__list-dropdown { 206 | background: rgba(255, 255, 255,.05); 207 | color: white; 208 | padding: 0 12px 0 5px; 209 | top: 0px; 210 | } 211 | 212 | .p-gui__list-dropdown { 213 | position: absolute; 214 | right: 5px; 215 | top: 0; 216 | bottom: 0; 217 | margin: auto; 218 | height: 21px; 219 | cursor: pointer; 220 | border-radius: 3px; 221 | border: 1px solid var(--color-border-2); 222 | outline: none; 223 | } 224 | 225 | .p-gui__list-dropdown:hover { 226 | background: rgba(255, 255, 255, .1); 227 | } 228 | ` 229 | ), D = ( 230 | /* css */ 231 | ` 232 | .p-gui__switch { 233 | background: rgba(255, 255, 255, .05); 234 | color: var(--color-text-dark); 235 | transition: var(--transition) background, var(--transition) color; 236 | } 237 | 238 | .p-gui__switch:hover { 239 | background: rgba(255, 255, 255, .1); 240 | color: var(--color-text-light); 241 | } 242 | 243 | .p-gui__folder .p-gui__switch { 244 | margin-inline: 0; 245 | } 246 | 247 | .p-gui__switch-checkbox { 248 | width: 5px; 249 | height: 5px; 250 | background-color: rgba(0, 0, 0, .5); 251 | border: 1px solid grey; 252 | position: absolute; 253 | top: 0; 254 | right: 10px; 255 | bottom: 0; 256 | margin: auto; 257 | border-radius: 50%; 258 | pointer-events: none; 259 | } 260 | 261 | .p-gui__switch-checkbox--active { 262 | background-color: #00ff89; 263 | box-shadow: 0 0 7px #00ff89; 264 | } 265 | ` 266 | ), F = ( 267 | /* css */ 268 | ` 269 | .p-gui__color { 270 | cursor: default; 271 | color: var(--color-text-dark); 272 | transition: var(--transition) color; 273 | } 274 | 275 | .p-gui__color:hover { 276 | color: var(--color-text-light); 277 | } 278 | 279 | .p-gui__color-picker { 280 | position: absolute; 281 | right: 5px; 282 | top: 0; 283 | bottom: 0; 284 | margin: auto; 285 | height: 21px; 286 | cursor: pointer; 287 | border-radius: 3px; 288 | border: 1px solid var(--color-border-2); 289 | outline: none; 290 | -webkit-appearance: none; 291 | padding: 0; 292 | background-color: transparent; 293 | border: 1px solid #222222; 294 | overflow: hidden; 295 | } 296 | 297 | .p-gui__color-picker::-webkit-color-swatch-wrapper { 298 | padding: 0; 299 | } 300 | .p-gui__color-picker::-webkit-color-swatch { 301 | border: none; 302 | } 303 | ` 304 | ), L = ( 305 | /* css */ 306 | ` 307 | .p-gui__vector2 { 308 | background: transparent; 309 | aspect-ratio: 1; 310 | padding-bottom: 0; 311 | color: var(--color-text-dark); 312 | } 313 | 314 | .p-gui__vector2:hover { 315 | color: var(--color-text-light); 316 | } 317 | 318 | .p-gui__vector2-area { 319 | position: relative; 320 | background: rgba(0, 0, 0, .3); 321 | margin-top: 8px; 322 | width: 100%; 323 | height: calc(100% - 28px); 324 | touch-action: none; 325 | } 326 | 327 | .p-gui__vector2-line { 328 | position: absolute; 329 | background: white; 330 | opacity: .3; 331 | pointer-events: none; 332 | } 333 | 334 | .p-gui__vector2-line-x { 335 | width: 100%; 336 | height: 1px; 337 | left: 0; 338 | top: 50%; 339 | transform: translateY(-50%); 340 | } 341 | 342 | .p-gui__vector2-line-y { 343 | width: 1px; 344 | height: 100%; 345 | top: 0; 346 | left: 50%; 347 | transform: translateX(-50%); 348 | } 349 | 350 | .p-gui__vector2-dot { 351 | position: absolute; 352 | top: 0; 353 | left: 0; 354 | width: 8px; 355 | height: 8px; 356 | border-radius: 50%; 357 | background: #d5d5d5; 358 | border: 2px solid #ff9999; 359 | transform: translate(-50%, -50%); 360 | pointer-events: none; 361 | } 362 | 363 | .p-gui__vector-value { 364 | display: inline-block; 365 | right: 7px; 366 | position: absolute; 367 | } 368 | ` 369 | ), H = ( 370 | /* css */ 371 | ` 372 | .p-gui__image-container { 373 | width: 100%; 374 | padding: 3px; 375 | display: flex; 376 | justify-content: flex-start; 377 | flex-wrap: wrap; 378 | box-sizing: border-box; 379 | } 380 | 381 | .p-gui__image { 382 | background-size: cover; 383 | cursor: pointer; 384 | position: relative; 385 | margin: 1px 2.5px 19px 2.5px; 386 | border-radius: var(--main-border-radius); 387 | flex: 0 0 calc(33.333% - 5px); 388 | height: 90px; 389 | background-position: center; 390 | color: var(--color-text-dark); 391 | transition: var(--transition) color; 392 | } 393 | 394 | .p-gui__image:hover { 395 | color: var(--color-text-light); 396 | } 397 | 398 | .p-gui__image::after { 399 | position: absolute; 400 | top: 0; 401 | left: 0; 402 | width: 100%; 403 | height: 100%; 404 | content: ''; 405 | border: 1px solid transparent; 406 | box-sizing: border-box; 407 | border-radius: var(--main-border-radius); 408 | transition: var(--transition) border-color; 409 | } 410 | .p-gui__image--selected::after { 411 | border-color: #06FF89; 412 | } 413 | 414 | .p-gui__image-text { 415 | position: absolute; 416 | bottom: -15px; 417 | text-shadow: 0 -1px 0 #111; 418 | white-space: nowrap; 419 | width: 100%; 420 | overflow: hidden; 421 | text-overflow: ellipsis; 422 | 423 | } 424 | ` 425 | ), N = ( 426 | /* css */ 427 | ` 428 | .p-gui__folder { 429 | width: 100%; 430 | position: relative; 431 | background: #434343; 432 | overflow: auto; 433 | margin-bottom: 2px; 434 | display: flex; 435 | flex-wrap: wrap; 436 | border: 1px solid grey; 437 | padding: 0 3px 0 3px; 438 | border-radius: var(--main-border-radius); 439 | box-sizing: border-box; 440 | } 441 | 442 | .p-gui__folder--first { 443 | margin-top: 0; 444 | } 445 | 446 | .p-gui__folder--closed { 447 | height: 32px; 448 | overflow: hidden; 449 | } 450 | 451 | .p-gui__folder-header { 452 | padding: 10px 5px; 453 | background-color: rgba(0, 0, 0, .5); 454 | color: white; 455 | cursor: pointer; 456 | width: 100%; 457 | margin: 0 -2px 2px -3px; 458 | } 459 | 460 | .p-gui__folder-header:hover { 461 | background-color: rgba(0, 0, 0, .75); 462 | } 463 | 464 | .p-gui__folder-arrow { 465 | width: 8px; 466 | height: 8px; 467 | display: inline-block; 468 | background-image: url(); 469 | background-size: contain; 470 | margin-right: 5px; 471 | transform: rotate(90deg) 472 | } 473 | 474 | .p-gui__folder--closed .p-gui__folder-arrow { 475 | transform: rotate(0deg); 476 | } 477 | ` 478 | ); 479 | function M(U) { 480 | return ( 481 | /* css */ 482 | ` 483 | .p-gui { 484 | --main-border-radius: 5px; 485 | --color-bg: #121212; 486 | --color-border: #484848; 487 | --color-border-2: rgba(255,255,255,.1); 488 | --color-text-light: #ffffff; 489 | --color-text-dark: #bbbbbb; 490 | --color-accent: #1681ca; 491 | --color-accent-hover: #218fda; 492 | --transition: .1s linear; 493 | 494 | position: ${U}; 495 | top: 0; 496 | left: 0; 497 | transform: translate3d(0,0,0); 498 | padding-top: 21px; 499 | padding-inline: 3px; 500 | background: var(--color-bg); 501 | display: block; 502 | justify-content: center; 503 | flex-wrap: wrap; 504 | font-family: "Arial Rounded MT Bold", Arial, sans-serif; 505 | width: 290px; 506 | overflow: auto; 507 | box-shadow: 0 0 2px black; 508 | box-sizing: border-box; 509 | z-index: 99999; 510 | user-select: none; 511 | border-bottom-right-radius: 3px; 512 | border-bottom-left-radius: 3px; 513 | cursor: auto; 514 | border-radius: var(--main-border-radius); 515 | border: 1px solid var(--color-border); 516 | line-height: normal; 517 | transition: var(--transition) opacity; 518 | } 519 | 520 | .p-gui:hover { 521 | opacity: 1!important; 522 | } 523 | 524 | .p-gui * { 525 | font-size: 11px; 526 | } 527 | 528 | .p-gui::-webkit-scrollbar, 529 | .p-gui *::-webkit-scrollbar { 530 | width: 10px; 531 | } 532 | 533 | .p-gui::-webkit-scrollbar-track, 534 | .p-gui *::-webkit-scrollbar-track { 535 | background: #2f2f2f; 536 | border-radius: 3px; 537 | } 538 | 539 | .p-gui::-webkit-scrollbar-thumb, 540 | .p-gui *::-webkit-scrollbar-thumb { 541 | background: #757576; 542 | border-radius: 10px; 543 | box-sizing: border-box; 544 | border: 1px solid #2f2f2f; 545 | } 546 | 547 | .p-gui--collapsed { 548 | height: 0; 549 | padding: 21px 10px 0 10px; 550 | overflow: hidden; 551 | } 552 | 553 | .p-gui__header { 554 | position: absolute; 555 | top: 0; 556 | left: 0; 557 | width: 100%; 558 | height: 20px; 559 | background-color: rgba(0, 0, 0, .8); 560 | cursor: grab; 561 | color: grey; 562 | font-size: 10px; 563 | line-height: 20px; 564 | padding-left: 12px; 565 | box-sizing: border-box; 566 | touch-action: none; 567 | } 568 | 569 | .p-gui__header-close { 570 | width: 20px; 571 | height: 20px; 572 | position: absolute; 573 | top: 0; 574 | right: 5px; 575 | cursor: pointer; 576 | background-image: url(); 577 | background-size: 50% 50%; 578 | background-position: center; 579 | background-repeat: no-repeat; 580 | } 581 | 582 | .p-gui--collapsed .p-gui__header-close { 583 | background-image: url(); 584 | } 585 | 586 | .p-gui__slider, 587 | .p-gui__button, 588 | .p-gui__switch, 589 | .p-gui__list, 590 | .p-gui__vector2, 591 | .p-gui__color { 592 | width: 100%; 593 | padding: 7px; 594 | cursor: pointer; 595 | position: relative; 596 | box-sizing: border-box; 597 | margin-block: 3px; 598 | border: 1px solid var(--color-border-2); 599 | border-radius: var(--main-border-radius); 600 | transition: var(--transition) border-color; 601 | } 602 | 603 | .p-gui__slider:hover, 604 | .p-gui__button:hover, 605 | .p-gui__switch:hover, 606 | .p-gui__list:hover, 607 | .p-gui__vector2:hover, 608 | .p-gui__color:hover { 609 | border-color: rgba(255,255,255,.2); 610 | } 611 | 612 | ${P} 613 | 614 | ${H} 615 | 616 | ${$} 617 | 618 | ${D} 619 | 620 | ${O} 621 | 622 | ${F} 623 | 624 | ${L} 625 | 626 | ${N} 627 | ` 628 | ); 629 | } 630 | class C { 631 | constructor(e = {}) { 632 | if (this.firstParent = this, e.container ? (this.container = typeof e.container == "string" ? document.querySelector(e.container) : e.container, this.position_type = "absolute") : (this.container = document.body, this.position_type = "fixed"), this.propReferences = [], this.folders = [], e.isFolder) { 633 | this._folderConstructor(e.folderOptions); 634 | return; 635 | } 636 | typeof e.onUpdate == "function" && (this.onUpdate = e.onUpdate), this.name = e != null && typeof e.name == "string" ? e.name : "", this.backgroundColor = e.color || null, this.opacity = e.opacity || 1, this.container == document.body ? this.maxHeight = window.innerHeight : this.maxHeight = Math.min(this.container.clientHeight, window.innerHeight), e.maxHeight && (this.initMaxHeight = e.maxHeight, this.maxHeight = Math.min(this.initMaxHeight, this.maxHeight)), this.screenCorner = this._parseScreenCorner(e.position), window.perfectGUI || (window.perfectGUI = {}), window.perfectGUI.instanceCounter == null ? window.perfectGUI.instanceCounter = 0 : window.perfectGUI.instanceCounter++, this.instanceId = window.perfectGUI.instanceCounter, this.wrapperWidth = e.width || 290, this.stylesheet = document.createElement("style"), this.stylesheet.setAttribute("type", "text/css"), this.stylesheet.setAttribute("id", "lm-gui-stylesheet"), document.head.append(this.stylesheet), this.instanceId == 0 && this._addStyles(`${M(this.position_type)}`), this._styleInstance(), this._addWrapper(), this.wrapper.setAttribute("data-corner-x", this.screenCorner.x), this.wrapper.setAttribute("data-corner-y", this.screenCorner.y), e.autoRepositioning != !1 && window.addEventListener("resize", this._handleResize.bind(this)), this._handleResize(), this.hasBeenDragged = !1, e.draggable == !0 && this._makeDraggable(), this.closed = !1, e.closed && this.toggleClose(); 637 | } 638 | _styleInstance() { 639 | let e = this._getScrollbarWidth(this.container); 640 | if (this.screenCorner.x == "left" ? this.xOffset = 0 : this.xOffset = this.container.clientWidth - this.wrapperWidth - e, this.instanceId > 0) { 641 | let t = this.container.querySelectorAll(".p-gui"); 642 | for (let i = 0; i < t.length; i++) 643 | this.screenCorner.y == t[i].dataset.cornerY && (this.screenCorner.x == "left" && t[i].dataset.cornerX == "left" ? this.xOffset += t[i].offsetWidth : this.screenCorner.x == "right" && t[i].dataset.cornerX == "right" && (this.xOffset -= t[i].offsetWidth)); 644 | } 645 | this.yOffset = 0, this.position = { 646 | prevX: this.xOffset, 647 | prevY: this.yOffset, 648 | x: this.xOffset, 649 | y: this.yOffset 650 | }, this._addStyles(`#p-gui-${this.instanceId} { 651 | width: ${this.wrapperWidth}px; 652 | max-height: ${this.maxHeight}px; 653 | transform: translate3d(${this.xOffset}px,${this.yOffset}px,0); 654 | ${this.screenCorner.y == "top" ? "" : "top: auto; bottom: 0;"} 655 | ${this.backgroundColor ? "background: " + this.backgroundColor + ";" : ""} 656 | opacity: ${this.opacity}; 657 | }`); 658 | } 659 | _folderConstructor(e) { 660 | this.wrapper = e.wrapper, this.isFolder = !0, this.parent = e.parent, this.firstParent = e.firstParent; 661 | } 662 | _parseScreenCorner(e) { 663 | let t = { x: "right", y: "top" }; 664 | return e == null || (typeof e != "string" && console.error("[perfect-gui] Position must be a string."), e.includes("left") && (t.x = "left"), e.includes("bottom") && (t.y = "bottom")), t; 665 | } 666 | _getScrollbarWidth(e) { 667 | return e === document.body ? window.innerWidth - document.documentElement.clientWidth : e.offsetWidth - e.clientWidth; 668 | } 669 | _handleResize() { 670 | if (this.container == document.body ? this.maxHeight = window.innerHeight : this.maxHeight = Math.min(this.container.clientHeight, window.innerHeight), this.initMaxHeight && (this.maxHeight = Math.min(this.initMaxHeight, this.maxHeight)), this.wrapper.style.maxHeight = this.maxHeight + "px", this.hasBeenDragged) 671 | return; 672 | let e = this._getScrollbarWidth(this.container); 673 | if (this.xOffset = this.screenCorner.x == "left" ? 0 : this.container.clientWidth - this.wrapperWidth - e, this.instanceId > 0) { 674 | let t = this.container.querySelectorAll(`.p-gui:not(#${this.wrapper.id}):not([data-dragged])`); 675 | for (let i = 0; i < t.length && !(parseInt(t[i].id.replace("p-gui-", "")) > this.instanceId); i++) 676 | this.screenCorner.y == t[i].dataset.cornerY && (this.screenCorner.x == "left" && t[i].dataset.cornerX == "left" ? this.xOffset += t[i].offsetWidth : this.screenCorner.x == "right" && t[i].dataset.cornerX == "right" && (this.xOffset -= t[i].offsetWidth)); 677 | } 678 | this.position = { prevX: this.xOffset, prevY: this.yOffset, x: this.xOffset, y: this.yOffset }, this.wrapper.style.transform = `translate3d(${this.position.x}px, ${this.position.y}px, 0)`; 679 | } 680 | _addStyles(e) { 681 | this.stylesheet.innerHTML += e; 682 | } 683 | _addWrapper() { 684 | this.wrapper = document.createElement("div"), this.wrapper.id = "p-gui-" + this.instanceId, this.wrapper.className = "p-gui", this.container.append(this.wrapper), this.header = document.createElement("div"), this.header.className = "p-gui__header", this.header.textContent = this.name, this.header.style = `${this.backgroundColor ? "border-color: " + this.backgroundColor + ";" : ""}`, this.wrapper.append(this.header); 685 | const e = document.createElement("div"); 686 | e.className = "p-gui__header-close", e.addEventListener("click", this.toggleClose.bind(this)), this.header.append(e); 687 | } 688 | button(e, t) { 689 | let i = ""; 690 | typeof e != "string" ? typeof e == "object" && (e != null && e.hasOwnProperty("name")) ? i = e.name == "" ? " " : e.name : i = " " : i = e == "" ? " " : e; 691 | const a = typeof e.tooltip == "string" ? e.tooltip : e.tooltip === !0 ? i : null; 692 | this.imageContainer = null; 693 | const s = document.createElement("div"); 694 | s.className = "p-gui__button", s.textContent = i, a && s.setAttribute("title", a), s.addEventListener("click", () => { 695 | t && t(), this.onUpdate ? this.onUpdate() : this.isFolder && this.firstParent.onUpdate && this.firstParent.onUpdate(); 696 | }), this.wrapper.append(s), typeof e.color == "string" && (s.style.setProperty("--color-accent", e.color), s.style.setProperty("--color-accent-hover", e.hoverColor || e.color)); 697 | } 698 | image(e = {}, t) { 699 | if (typeof e != "object") 700 | throw Error(`[GUI] image() first parameter must be an object. Received: ${typeof e}.`); 701 | let i; 702 | if (typeof e.path == "string") 703 | i = e.path; 704 | else 705 | throw typeof e.path == null ? Error("[GUI] image() path must be provided.") : Error("[GUI] image() path must be a string."); 706 | let a = i.replace(/^.*[\\\/]/, ""), s; 707 | e.name == null ? s = a : s = typeof e.name == "string" && e.name || " "; 708 | const o = typeof e.tooltip == "string" ? e.tooltip : e.tooltip === !0 ? s : null, r = e.selected === !0, n = e.selectionBorder !== !1; 709 | let c = ""; 710 | e.width && (typeof e.width == "number" && (e.width += "px"), c += `flex: 0 0 calc(${e.width} - 5px); `), e.height && (typeof e.height == "number" && (e.height += "px"), c += `height: ${e.height}; `), this.imageContainer || (this.imageContainer = document.createElement("div"), this.imageContainer.className = "p-gui__image-container", this.wrapper.append(this.imageContainer)); 711 | const l = document.createElement("div"); 712 | l.className = "p-gui__image", l.style = "background-image: url(" + i + "); " + c, o && l.setAttribute("title", o), this.imageContainer.append(l), r && n && l.classList.add("p-gui__image--selected"); 713 | const d = document.createElement("div"); 714 | return d.className = "p-gui__image-text", d.textContent = s, l.append(d), l.addEventListener("click", () => { 715 | let p = l.parentElement.querySelectorAll(".p-gui__image--selected"); 716 | for (let h = 0; h < p.length; h++) 717 | p[h].classList.remove("p-gui__image--selected"); 718 | n && l.classList.add("p-gui__image--selected"), typeof t == "function" && t({ path: i, text: s }), this.onUpdate ? this.onUpdate() : this.isFolder && this.firstParent.onUpdate && this.firstParent.onUpdate(); 719 | }), l; 720 | } 721 | slider(e = {}, t) { 722 | const i = new R(this, e, t); 723 | this.wrapper.append(i); 724 | } 725 | toggle(e = {}, t) { 726 | if (typeof e != "object") 727 | throw Error(`[GUI] toggle() first parameter must be an object. Received: ${typeof e}.`); 728 | let i = typeof e.name == "string" && e.name || " ", a = !1, s = null, o = e.obj, r = e.prop, n = typeof e.value == "boolean" ? e.value : null; 729 | if (n !== null) 730 | (r != null || o != null) && console.warn('[GUI] toggle() "obj" and "prop" parameters are ignored when a "value" parameter is used.'); 731 | else if (r != null && o != null) { 732 | if (typeof r != "string") 733 | throw Error(`[GUI] toggle() "prop" parameter must be an string. Received: ${typeof r}.`); 734 | if (typeof o != "object") 735 | throw Error(`[GUI] toggle() "obj" parameter must be an object. Received: ${typeof o}.`); 736 | i == " " && (i = r), s = this.propReferences.push(o[r]) - 1, a = !0; 737 | } else 738 | (r != null && o == null || r == null && o == null) && console.warn('[GUI] toggle() "obj" and "prop" parameters must be used together.'); 739 | const c = typeof e.tooltip == "string" ? e.tooltip : e.tooltip === !0 ? i : null; 740 | this.imageContainer = null; 741 | const l = document.createElement("div"); 742 | l.textContent = i, l.className = "p-gui__switch", c && l.setAttribute("title", c), this.wrapper.append(l), l.addEventListener("click", (h) => { 743 | const u = h.target.childNodes[1]; 744 | let f = !0; 745 | u.classList.contains("p-gui__switch-checkbox--active") && (f = !1), u.classList.toggle("p-gui__switch-checkbox--active"), a ? o[r] = f : typeof t == "function" && t(f), this.onUpdate ? this.onUpdate() : this.isFolder && this.firstParent.onUpdate && this.firstParent.onUpdate(); 746 | }); 747 | let d = (() => a ? o[r] ? " p-gui__switch-checkbox--active" : "" : n ? " p-gui__switch-checkbox--active" : "")(); 748 | const p = document.createElement("div"); 749 | p.className = "p-gui__switch-checkbox" + d, l.append(p), a && Object.defineProperty(o, r, { 750 | set: (h) => { 751 | this.propReferences[s] = h, h ? p.classList.add("p-gui__switch-checkbox--active") : p.classList.remove("p-gui__switch-checkbox--active"), typeof t == "function" && t(h); 752 | }, 753 | get: () => this.propReferences[s] 754 | }); 755 | } 756 | list(e = {}, t) { 757 | if (typeof e != "object") 758 | throw Error(`[GUI] list() first parameter must be an object. Received: ${typeof e}.`); 759 | let i = typeof e.name == "string" ? e.name : " ", a = !1, s = null, o = e.obj, r = e.prop, n = Array.isArray(e.values) ? e.values : null, c, l = typeof n[0] != "string"; 760 | const d = typeof e.tooltip == "string" ? e.tooltip : e.tooltip === !0 ? i : null; 761 | if (t = typeof t == "function" ? t : null, e.value !== void 0 || e.value === void 0 && o === void 0 && r === void 0) 762 | (r != null || o != null) && console.warn('[GUI] list() "obj" and "prop" parameters are ignored when a "value" parameter is used.'), c = (() => { 763 | if (!n) 764 | return null; 765 | if (typeof e.value == "string") 766 | return n.indexOf(e.value); 767 | if (typeof e.value == "number") 768 | return e.value; 769 | })(); 770 | else if (r != null && o != null) { 771 | if (typeof r != "string") 772 | throw Error(`[GUI] list() "prop" parameter must be an string. Received: ${typeof r}.`); 773 | if (typeof o != "object") 774 | throw Error(`[GUI] list() "obj" parameter must be an object. Received: ${typeof o}.`); 775 | c = (() => { 776 | if (!n) 777 | return null; 778 | if (typeof o[r] == "string") 779 | return l ? n.find((u) => u.value === o[r]).value : n.indexOf(o[r]); 780 | if (typeof o[r] == "number") 781 | return l ? n.find((u) => u.value === o[r]).value : o[r]; 782 | })(), s = this.propReferences.push(o[r]) - 1, a = !0; 783 | } else 784 | (r != null && o == null || r == null && o == null) && console.warn('[GUI] list() "obj" and "prop" parameters must be used together.'); 785 | this.imageContainer = null; 786 | let p = document.createElement("div"); 787 | p.className = "p-gui__list", p.textContent = i, d && p.setAttribute("title", d), this.wrapper.append(p); 788 | let h = document.createElement("select"); 789 | p.append(h), h.className = "p-gui__list-dropdown", h.addEventListener("change", (u) => { 790 | a ? o[r] = u.target.value : t && t(u.target.value), this.onUpdate ? this.onUpdate() : this.isFolder && this.firstParent.onUpdate && this.firstParent.onUpdate(); 791 | }), n && n.forEach((u, f) => { 792 | const g = l ? u.name : u, m = l ? u.value : u; 793 | let x = document.createElement("option"); 794 | x.setAttribute("value", m), x.textContent = g, h.append(x), (!l && c == f || l && c == m) && x.setAttribute("selected", ""); 795 | }), a && Object.defineProperty(o, r, { 796 | set: (u) => { 797 | let f, g, m; 798 | l ? (m = n.find((v) => v.value == u), g = (m == null ? void 0 : m.value) || n[0].value, f = n.indexOf(m)) : (typeof u == "string" && (f = n.indexOf(u), g = u), typeof u == "number" && (f = u, g = n[u])), this.propReferences[s] = l ? g : u; 799 | const x = h.querySelector("[selected]"); 800 | x && x.removeAttribute("selected"), h.querySelectorAll("option")[f].setAttribute("selected", ""), typeof t == "function" && t(l ? m : g, f); 801 | }, 802 | get: () => this.propReferences[s] 803 | }); 804 | } 805 | color(e = {}, t) { 806 | if (typeof e != "object") 807 | throw Error(`[GUI] color() first parameter must be an object. Received: ${typeof e}.`); 808 | let i = typeof e.name == "string" && e.name || " ", a = !1, s = null, o = e.obj, r = e.prop, n; 809 | const c = typeof e.tooltip == "string" ? e.tooltip : e.tooltip === !0 ? i : null; 810 | if (typeof e.value == "string" && (e.value.length != 7 || e.value[0] != "#" ? console.error(`[GUI] color() 'value' parameter must be an hexadecimal string in the format "#ffffff". Received: "${e.value}".`) : n = e.value), n || (n = "#000000"), e.value !== void 0) 811 | (r != null || o != null) && console.warn('[GUI] color() "obj" and "prop" parameters are ignored when a "value" parameter is used.'); 812 | else if (r != null && o != null) { 813 | if (typeof r != "string") 814 | throw Error(`[GUI] color() "prop" parameter must be an string. Received: ${typeof r}.`); 815 | if (typeof o != "object") 816 | throw Error(`[GUI] color() "obj" parameter must be an object. Received: ${typeof o}.`); 817 | i == " " && (i = r), s = this.propReferences.push(o[r]) - 1, a = !0; 818 | } else 819 | (r != null && o == null || r == null && o == null) && console.warn('[GUI] color() "obj" and "prop" parameters must be used together.'); 820 | this.imageContainer = null; 821 | const l = document.createElement("div"); 822 | l.className = "p-gui__color", l.textContent = i, c && l.setAttribute("title", c), this.wrapper.append(l); 823 | const d = document.createElement("input"); 824 | d.className = "p-gui__color-picker", d.setAttribute("type", "color"), d.value = n, l.append(d), typeof t == "function" && d.addEventListener("input", () => { 825 | a ? o[r] = d.value : typeof t == "function" && t(d.value), this.onUpdate ? this.onUpdate() : this.isFolder && this.firstParent.onUpdate && this.firstParent.onUpdate(); 826 | }), a && Object.defineProperty(o, r, { 827 | set: (p) => { 828 | this.propReferences[s] = p, d.value = p, typeof t == "function" && t(p); 829 | }, 830 | get: () => this.propReferences[s] 831 | }); 832 | } 833 | vector2(e = {}, t) { 834 | if (typeof e != "object") 835 | throw Error(`[GUI] vector2() first parameter must be an object. Received: ${typeof e}.`); 836 | let i = typeof e.name == "string" && e.name || " "; 837 | const a = e.x.min ?? 0, s = e.x.max ?? 1, o = e.y.min ?? 0, r = e.y.max ?? 1, n = e.x.step || (s - a) / 100, c = e.y.step || (r - o) / 100, l = this._countDecimals(n), d = this._countDecimals(c), p = e.x.obj, h = e.x.prop, u = this.propReferences.push(p[h]) - 1, f = e.y.obj, g = e.y.prop, m = this.propReferences.push(f[g]) - 1, x = typeof e.tooltip == "string" ? e.tooltip : e.tooltip === !0 ? i : null; 838 | t = typeof t == "function" ? t : null, this.imageContainer = null; 839 | const v = document.createElement("div"); 840 | v.className = "p-gui__vector2", v.textContent = i, x && v.setAttribute("title", x), this.wrapper.append(v); 841 | const y = document.createElement("div"); 842 | y.className = "p-gui__vector-value", y.textContent = p[h] + ", " + f[g], v.append(y); 843 | const b = document.createElement("div"); 844 | b.className = "p-gui__vector2-area", v.append(b), b.addEventListener("click", (_) => { 845 | const k = parseFloat(this._mapLinear(_.offsetX, 0, b.clientWidth, a, s)), E = parseFloat(this._mapLinear(_.offsetY, 0, b.clientHeight, r, o)); 846 | p[h] = k.toFixed(l), f[g] = E.toFixed(d), t && t(p[h], p[g]), this.onUpdate ? this.onUpdate() : this.isFolder && this.firstParent.onUpdate && this.firstParent.onUpdate(); 847 | }); 848 | let A = !1; 849 | b.addEventListener("pointerdown", (_) => { 850 | A = !0; 851 | }), b.addEventListener("pointerup", () => { 852 | A = !1; 853 | }), b.addEventListener("pointermove", (_) => { 854 | if (A) { 855 | const k = parseFloat(this._mapLinear(_.offsetX, 0, b.clientWidth, a, s)), E = parseFloat(this._mapLinear(_.offsetY, 0, b.clientHeight, r, o)); 856 | p[h] = k.toFixed(l), f[g] = E.toFixed(d), t && t(p[h], p[g]), this.onUpdate ? this.onUpdate() : this.isFolder && this.firstParent.onUpdate && this.firstParent.onUpdate(); 857 | } 858 | }); 859 | const j = document.createElement("div"); 860 | j.className = "p-gui__vector2-line p-gui__vector2-line-x", b.append(j); 861 | const I = document.createElement("div"); 862 | I.className = "p-gui__vector2-line p-gui__vector2-line-y", b.append(I); 863 | const w = document.createElement("div"); 864 | w.className = "p-gui__vector2-dot", b.append(w), w.style.left = this._mapLinear(p[h], a, s, 0, b.clientWidth) + "px", w.style.top = this._mapLinear(f[g], o, r, b.clientHeight, 0) + "px", Object.defineProperty(p, h, { 865 | set: (_) => { 866 | this.propReferences[u] = _, w.style.left = this._mapLinear(_, a, s, 0, b.clientWidth) + "px", y.textContent = String(_) + ", " + f[g]; 867 | }, 868 | get: () => this.propReferences[u] 869 | }), Object.defineProperty(f, g, { 870 | set: (_) => { 871 | this.propReferences[m] = _, w.style.top = this._mapLinear(_, o, r, b.clientHeight, 0) + "px", y.textContent = p[h] + ", " + String(_); 872 | }, 873 | get: () => this.propReferences[m] 874 | }); 875 | } 876 | folder(e = {}) { 877 | let t = typeof e.closed == "boolean" ? e.closed : !1, i = e.name || "", a = e.color || null, s = e.maxHeight || null; 878 | this.imageContainer = null; 879 | let o = "p-gui__folder"; 880 | this.folders.length == 0 && (o += " p-gui__folder--first"), t && (o += " p-gui__folder--closed"); 881 | let r = a ? `background-color: ${a};` : ""; 882 | r += s ? `max-height: ${s}px;` : ""; 883 | const n = document.createElement("div"); 884 | n.className = o, n.style = r, this.wrapper.append(n); 885 | const c = document.createElement("div"); 886 | c.innerHTML = `${i}`, c.className = "p-gui__folder-header", n.append(c), c.addEventListener("click", () => { 887 | n.classList.toggle("p-gui__folder--closed"); 888 | }); 889 | let l = new C({ isFolder: !0, folderOptions: { 890 | wrapper: n, 891 | parent: this, 892 | firstParent: this.firstParent 893 | } }); 894 | return this.folders.push(l), l; 895 | } 896 | _makeDraggable() { 897 | var e = this; 898 | this.header.addEventListener("pointerdown", t), this.header.addEventListener("pointerup", a); 899 | function t(s) { 900 | s.preventDefault(), e.position.initX = e.position.x, e.position.initY = e.position.y, e.position.prevX = s.clientX, e.position.prevY = s.clientY, document.addEventListener("pointermove", i); 901 | } 902 | function i(s) { 903 | s.preventDefault(), e.hasBeenDragged || (e.hasBeenDragged = !0, e.wrapper.setAttribute("data-dragged", "true")), e.position.x = e.position.initX + s.clientX - e.position.prevX, e.position.y = e.position.initY + s.clientY - e.position.prevY, e.wrapper.style.transform = "translate3d(" + e.position.x + "px," + e.position.y + "px,0)"; 904 | } 905 | function a(s) { 906 | document.removeEventListener("pointermove", i); 907 | } 908 | } 909 | toggleClose() { 910 | this.closed = !this.closed, this.closed ? (this.previousInnerScroll = this.wrapper.scrollTop, this.wrapper.scrollTo(0, 0)) : this.wrapper.scrollTo(0, this.previousInnerScroll), this.wrapper.classList.toggle("p-gui--collapsed"); 911 | } 912 | kill() { 913 | this.wrapper.remove(); 914 | } 915 | _mapLinear(e, t, i, a, s) { 916 | return a + (e - t) * (s - a) / (i - t); 917 | } 918 | _countDecimals(e) { 919 | const t = e.toString(), i = t.indexOf("."); 920 | return i === -1 ? 0 : t.length - i - 1; 921 | } 922 | } 923 | export { 924 | C as default 925 | }; 926 | -------------------------------------------------------------------------------- /dist/perfect-gui.umd.js: -------------------------------------------------------------------------------- 1 | (function(y,A){typeof exports=="object"&&typeof module<"u"?module.exports=A():typeof define=="function"&&define.amd?define(A):(y=typeof globalThis<"u"?globalThis:y||self,y["Perfect GUI"]=A())})(this,function(){"use strict";class y{constructor(e,t={},i){if(this.parent=e,this.propReferences=[],typeof t!="object")throw Error(`[GUI] slider() first parameter must be an object. Received: ${typeof t}.`);let a=typeof t.name=="string"&&t.name||" ";this.isObject=!1;let s=null;this.obj=t.obj,this.prop=t.prop;let o=typeof t.value=="number"?t.value:null;if(this.min=t.min??0,this.max=t.max??1,this.step=t.step||(this.max-this.min)/100,this.decimals=this.parent._countDecimals(this.step),this.callback=typeof i=="function"?i:null,o!==null)(this.prop!=null||this.obj!=null)&&console.warn('[GUI] slider() "obj" and "prop" parameters are ignored when a "value" parameter is used.');else if(this.prop!=null&&this.obj!=null){if(typeof this.prop!="string")throw Error(`[GUI] slider() "prop" parameter must be an string. Received: ${typeof this.prop}.`);if(typeof this.obj!="object")throw Error(`[GUI] slider() "obj" parameter must be an object. Received: ${typeof this.obj}.`);a==" "&&(a=this.prop),s=this.propReferences.push(this.obj[this.prop])-1,this.isObject=!0}else(this.prop!=null&&this.obj==null||this.prop==null&&this.obj!=null)&&console.warn('[GUI] slider() "obj" and "prop" parameters must be used together.'),o=(this.max-this.min)/2;const r=typeof t.tooltip=="string"?t.tooltip:t.tooltip===!0?a:null;this.imageContainer=null;const n=document.createElement("div");n.className="p-gui__slider",r&&n.setAttribute("title",r);const c=document.createElement("div");c.className="p-gui__slider-name",c.textContent=a,n.append(c),this.ctrlDiv=document.createElement("div"),this.ctrlDiv.className="p-gui__slider-ctrl",this.ctrlDiv.setAttribute("type","range"),this.ctrlDiv.setAttribute("min",this.min),this.ctrlDiv.setAttribute("max",this.max),n.append(this.ctrlDiv);const l=document.createElement("div");return l.className="p-gui__slider-bar",this.ctrlDiv.append(l),this.handle=document.createElement("div"),this.handle.className="p-gui__slider-handle",this.ctrlDiv.append(this.handle),this.filling=document.createElement("div"),this.filling.className="p-gui__slider-filling",l.append(this.filling),this.valueInput=document.createElement("input"),this.valueInput.className="p-gui__slider-value",this.valueInput.value=this.isObject?this.obj[this.prop]:o,n.append(this.valueInput),setTimeout(()=>{const d=this.handle.offsetWidth;this.handle.position=this._mapLinear(this.valueInput.value,this.min,this.max,d/2,88-d/2),this.handle.position=Math.min(this.handle.position,88-d/2),this.handle.position=Math.max(this.handle.position,d/2),this.handle.style.transform=`translate(-50%, -50%) translateX(${this.handle.position}px)`,this.filling.style.width=`${this.handle.position}px`},0),this.valueInput.addEventListener("change",()=>{this._updateHandlePositionFromValue(),this._triggerCallbacks()}),this.ctrlDiv.addEventListener("pointerdown",d=>{this.ctrlDiv.pointerDown=!0,this.ctrlDiv.prevPosition=d.clientX,this._updateHandlePositionFromPointer(d,!0)}),window.addEventListener("pointerup",d=>{this.ctrlDiv.pointerDown=!1}),window.addEventListener("pointermove",d=>{this.ctrlDiv.pointerDown&&(this.ctrlDiv.pointerDelta=d.clientX-this.ctrlDiv.prevPosition,this._updateHandlePositionFromPointer(d))}),this.isObject&&Object.defineProperty(this.obj,this.prop,{set:d=>{this.propReferences[s]=d,this.valueInput.value=d,this._updateHandlePositionFromValue(),this.callback&&this.callback(parseFloat(this.valueInput.value))},get:()=>this.propReferences[s]}),n}_updateHandlePositionFromPointer(e,t=!1){const i=this.ctrlDiv.offsetWidth,a=this.handle.offsetWidth,s=e.clientX-this.ctrlDiv.prevPosition,o=parseFloat(this.valueInput.value);let r;t?r=e.offsetX:r=this.handle.position+s,r=Math.max(a/2,Math.min(r,i-a/2));let n=this.min+(this.max-this.min)*(r-a/2)/(i-a);n>o?n=this._quantizeFloor(n,this.step):n=this._quantizeCeil(n,this.step),n=parseFloat(n.toFixed(9));const c=parseFloat((o+this.step).toFixed(9)),l=parseFloat((o-this.step).toFixed(9));(n>=c||n<=l)&&(n=n.toFixed(this.decimals),this.valueInput.value=n,this.ctrlDiv.prevPosition=e.clientX,this.handle.style.transform=`translate(-50%, -50%) translateX(${r}px)`,this.handle.position=r,this.filling.style.width=this.handle.position+"px",this._triggerCallbacks())}_updateHandlePositionFromValue(){const e=this.ctrlDiv.offsetWidth,t=this.handle.offsetWidth;let i=this._mapLinear(this.valueInput.value,this.min,this.max,t/2,e-t/2);i=Math.max(t/2,Math.min(i,e-t/2)),this.handle.style.transform=`translate(-50%, -50%) translateX(${i}px)`,this.handle.position=i,this.filling.style.width=this.handle.position+"px"}_triggerCallbacks(){this.isObject?this.obj[this.prop]=parseFloat(this.valueInput.value):this.callback&&this.callback(parseFloat(this.valueInput.value)),this.parent.onUpdate?this.parent.onUpdate():this.parent.isFolder&&this.parent.firstParent.onUpdate&&this.parent.firstParent.onUpdate()}_mapLinear(e,t,i,a,s){return a+(e-t)*(s-a)/(i-t)}_quantize(e,t){return t*Math.round(e/t)}_quantizeCeil(e,t){return t*Math.ceil(e/t)}_quantizeFloor(e,t){return t*Math.floor(e/t)}}const A=` 2 | .p-gui__button { 3 | background: var(--color-accent); 4 | text-align: center; 5 | color: white; 6 | border: none; 7 | border: 1px solid transparent; 8 | box-sizing: border-box; 9 | transition: var(--transition) background, var(--transition) border-color; 10 | } 11 | 12 | .p-gui__button:hover { 13 | background: var(--color-accent-hover); 14 | color: var(--color-text-light); 15 | border-color: rgba(255, 255, 255, 0.2); 16 | } 17 | 18 | .p-gui__folder .p-gui__button { 19 | margin-inline: 0; 20 | } 21 | `,O=` 22 | .p-gui__slider { 23 | position: relative; 24 | min-height: 14px; 25 | display: flex; 26 | align-items: center; 27 | justify-content: space-between; 28 | gap: 10px; 29 | color: var(--color-text-dark); 30 | transition: color var(--transition); 31 | } 32 | 33 | .p-gui__slider:hover { 34 | color: var(--color-text-light); 35 | } 36 | 37 | .p-gui__slider-name { 38 | width: 50%; 39 | text-overflow: ellipsis; 40 | overflow: hidden; 41 | } 42 | 43 | .p-gui__slider-ctrl { 44 | -webkit-appearance: none; 45 | padding: 0; 46 | font: inherit; 47 | outline: none; 48 | box-sizing: border-box; 49 | cursor: pointer; 50 | position: relative; 51 | right: 0; 52 | height: 14px; 53 | margin: 0 0 0 auto; 54 | width: 37%; 55 | } 56 | 57 | .p-gui__slider-bar { 58 | position: absolute; 59 | top: 50%; 60 | left: 0; 61 | height: 2px; 62 | background: rgba(255, 255, 255, .2); 63 | width: 100%; 64 | transform: translateY(-50%); 65 | } 66 | 67 | .p-gui__slider-filling { 68 | position: absolute; 69 | top: -25%; 70 | left: 0; 71 | height: 150%; 72 | background: var(--color-accent); 73 | width: 0; 74 | } 75 | 76 | .p-gui__slider:hover .p-gui__slider-filling { 77 | background: var(--color-accent-hover); 78 | } 79 | 80 | .p-gui__slider-handle { 81 | width: 15px; 82 | height: 8px; 83 | position: absolute; 84 | top: 50%; 85 | left: 0; 86 | border-radius: 2px; 87 | transform: translate(-50%, -50%); 88 | pointer-events: none; 89 | background: var(--color-text-dark); 90 | box-shadow: 0 0 2px rgba(0, 0, 0, .5); 91 | } 92 | 93 | .p-gui__slider:hover .p-gui__slider-handle { 94 | background: var(--color-text-light); 95 | } 96 | 97 | .p-gui__slider-value { 98 | display: inline-block; 99 | right: 7px; 100 | width: 13%; 101 | border: none; 102 | color: white; 103 | border-radius: 2px; 104 | background: rgba(255, 255, 255, 0.1); 105 | padding: 2px 4px; 106 | color: inherit; 107 | } 108 | 109 | .p-gui__slider-value:focus { 110 | outline: none; 111 | } 112 | `,$=` 113 | .p-gui__list { 114 | cursor: default; 115 | color: var(--color-text-dark); 116 | transition: var(--transition) color; 117 | } 118 | 119 | .p-gui__list:hover { 120 | color: var(--color-text-light); 121 | } 122 | 123 | .p-gui__list-dropdown { 124 | background: rgba(255, 255, 255,.05); 125 | color: white; 126 | padding: 0 12px 0 5px; 127 | top: 0px; 128 | } 129 | 130 | .p-gui__list-dropdown { 131 | position: absolute; 132 | right: 5px; 133 | top: 0; 134 | bottom: 0; 135 | margin: auto; 136 | height: 21px; 137 | cursor: pointer; 138 | border-radius: 3px; 139 | border: 1px solid var(--color-border-2); 140 | outline: none; 141 | } 142 | 143 | .p-gui__list-dropdown:hover { 144 | background: rgba(255, 255, 255, .1); 145 | } 146 | `,D=` 147 | .p-gui__switch { 148 | background: rgba(255, 255, 255, .05); 149 | color: var(--color-text-dark); 150 | transition: var(--transition) background, var(--transition) color; 151 | } 152 | 153 | .p-gui__switch:hover { 154 | background: rgba(255, 255, 255, .1); 155 | color: var(--color-text-light); 156 | } 157 | 158 | .p-gui__folder .p-gui__switch { 159 | margin-inline: 0; 160 | } 161 | 162 | .p-gui__switch-checkbox { 163 | width: 5px; 164 | height: 5px; 165 | background-color: rgba(0, 0, 0, .5); 166 | border: 1px solid grey; 167 | position: absolute; 168 | top: 0; 169 | right: 10px; 170 | bottom: 0; 171 | margin: auto; 172 | border-radius: 50%; 173 | pointer-events: none; 174 | } 175 | 176 | .p-gui__switch-checkbox--active { 177 | background-color: #00ff89; 178 | box-shadow: 0 0 7px #00ff89; 179 | } 180 | `,F=` 181 | .p-gui__color { 182 | cursor: default; 183 | color: var(--color-text-dark); 184 | transition: var(--transition) color; 185 | } 186 | 187 | .p-gui__color:hover { 188 | color: var(--color-text-light); 189 | } 190 | 191 | .p-gui__color-picker { 192 | position: absolute; 193 | right: 5px; 194 | top: 0; 195 | bottom: 0; 196 | margin: auto; 197 | height: 21px; 198 | cursor: pointer; 199 | border-radius: 3px; 200 | border: 1px solid var(--color-border-2); 201 | outline: none; 202 | -webkit-appearance: none; 203 | padding: 0; 204 | background-color: transparent; 205 | border: 1px solid #222222; 206 | overflow: hidden; 207 | } 208 | 209 | .p-gui__color-picker::-webkit-color-swatch-wrapper { 210 | padding: 0; 211 | } 212 | .p-gui__color-picker::-webkit-color-swatch { 213 | border: none; 214 | } 215 | `,L=` 216 | .p-gui__vector2 { 217 | background: transparent; 218 | aspect-ratio: 1; 219 | padding-bottom: 0; 220 | color: var(--color-text-dark); 221 | } 222 | 223 | .p-gui__vector2:hover { 224 | color: var(--color-text-light); 225 | } 226 | 227 | .p-gui__vector2-area { 228 | position: relative; 229 | background: rgba(0, 0, 0, .3); 230 | margin-top: 8px; 231 | width: 100%; 232 | height: calc(100% - 28px); 233 | touch-action: none; 234 | } 235 | 236 | .p-gui__vector2-line { 237 | position: absolute; 238 | background: white; 239 | opacity: .3; 240 | pointer-events: none; 241 | } 242 | 243 | .p-gui__vector2-line-x { 244 | width: 100%; 245 | height: 1px; 246 | left: 0; 247 | top: 50%; 248 | transform: translateY(-50%); 249 | } 250 | 251 | .p-gui__vector2-line-y { 252 | width: 1px; 253 | height: 100%; 254 | top: 0; 255 | left: 50%; 256 | transform: translateX(-50%); 257 | } 258 | 259 | .p-gui__vector2-dot { 260 | position: absolute; 261 | top: 0; 262 | left: 0; 263 | width: 8px; 264 | height: 8px; 265 | border-radius: 50%; 266 | background: #d5d5d5; 267 | border: 2px solid #ff9999; 268 | transform: translate(-50%, -50%); 269 | pointer-events: none; 270 | } 271 | 272 | .p-gui__vector-value { 273 | display: inline-block; 274 | right: 7px; 275 | position: absolute; 276 | } 277 | `,H=` 278 | .p-gui__image-container { 279 | width: 100%; 280 | padding: 3px; 281 | display: flex; 282 | justify-content: flex-start; 283 | flex-wrap: wrap; 284 | box-sizing: border-box; 285 | } 286 | 287 | .p-gui__image { 288 | background-size: cover; 289 | cursor: pointer; 290 | position: relative; 291 | margin: 1px 2.5px 19px 2.5px; 292 | border-radius: var(--main-border-radius); 293 | flex: 0 0 calc(33.333% - 5px); 294 | height: 90px; 295 | background-position: center; 296 | color: var(--color-text-dark); 297 | transition: var(--transition) color; 298 | } 299 | 300 | .p-gui__image:hover { 301 | color: var(--color-text-light); 302 | } 303 | 304 | .p-gui__image::after { 305 | position: absolute; 306 | top: 0; 307 | left: 0; 308 | width: 100%; 309 | height: 100%; 310 | content: ''; 311 | border: 1px solid transparent; 312 | box-sizing: border-box; 313 | border-radius: var(--main-border-radius); 314 | transition: var(--transition) border-color; 315 | } 316 | .p-gui__image--selected::after { 317 | border-color: #06FF89; 318 | } 319 | 320 | .p-gui__image-text { 321 | position: absolute; 322 | bottom: -15px; 323 | text-shadow: 0 -1px 0 #111; 324 | white-space: nowrap; 325 | width: 100%; 326 | overflow: hidden; 327 | text-overflow: ellipsis; 328 | 329 | } 330 | `,N=` 331 | .p-gui__folder { 332 | width: 100%; 333 | position: relative; 334 | background: #434343; 335 | overflow: auto; 336 | margin-bottom: 2px; 337 | display: flex; 338 | flex-wrap: wrap; 339 | border: 1px solid grey; 340 | padding: 0 3px 0 3px; 341 | border-radius: var(--main-border-radius); 342 | box-sizing: border-box; 343 | } 344 | 345 | .p-gui__folder--first { 346 | margin-top: 0; 347 | } 348 | 349 | .p-gui__folder--closed { 350 | height: 32px; 351 | overflow: hidden; 352 | } 353 | 354 | .p-gui__folder-header { 355 | padding: 10px 5px; 356 | background-color: rgba(0, 0, 0, .5); 357 | color: white; 358 | cursor: pointer; 359 | width: 100%; 360 | margin: 0 -2px 2px -3px; 361 | } 362 | 363 | .p-gui__folder-header:hover { 364 | background-color: rgba(0, 0, 0, .75); 365 | } 366 | 367 | .p-gui__folder-arrow { 368 | width: 8px; 369 | height: 8px; 370 | display: inline-block; 371 | background-image: url(); 372 | background-size: contain; 373 | margin-right: 5px; 374 | transform: rotate(90deg) 375 | } 376 | 377 | .p-gui__folder--closed .p-gui__folder-arrow { 378 | transform: rotate(0deg); 379 | } 380 | `;function M(C){return` 381 | .p-gui { 382 | --main-border-radius: 5px; 383 | --color-bg: #121212; 384 | --color-border: #484848; 385 | --color-border-2: rgba(255,255,255,.1); 386 | --color-text-light: #ffffff; 387 | --color-text-dark: #bbbbbb; 388 | --color-accent: #1681ca; 389 | --color-accent-hover: #218fda; 390 | --transition: .1s linear; 391 | 392 | position: ${C}; 393 | top: 0; 394 | left: 0; 395 | transform: translate3d(0,0,0); 396 | padding-top: 21px; 397 | padding-inline: 3px; 398 | background: var(--color-bg); 399 | display: block; 400 | justify-content: center; 401 | flex-wrap: wrap; 402 | font-family: "Arial Rounded MT Bold", Arial, sans-serif; 403 | width: 290px; 404 | overflow: auto; 405 | box-shadow: 0 0 2px black; 406 | box-sizing: border-box; 407 | z-index: 99999; 408 | user-select: none; 409 | border-bottom-right-radius: 3px; 410 | border-bottom-left-radius: 3px; 411 | cursor: auto; 412 | border-radius: var(--main-border-radius); 413 | border: 1px solid var(--color-border); 414 | line-height: normal; 415 | transition: var(--transition) opacity; 416 | } 417 | 418 | .p-gui:hover { 419 | opacity: 1!important; 420 | } 421 | 422 | .p-gui * { 423 | font-size: 11px; 424 | } 425 | 426 | .p-gui::-webkit-scrollbar, 427 | .p-gui *::-webkit-scrollbar { 428 | width: 10px; 429 | } 430 | 431 | .p-gui::-webkit-scrollbar-track, 432 | .p-gui *::-webkit-scrollbar-track { 433 | background: #2f2f2f; 434 | border-radius: 3px; 435 | } 436 | 437 | .p-gui::-webkit-scrollbar-thumb, 438 | .p-gui *::-webkit-scrollbar-thumb { 439 | background: #757576; 440 | border-radius: 10px; 441 | box-sizing: border-box; 442 | border: 1px solid #2f2f2f; 443 | } 444 | 445 | .p-gui--collapsed { 446 | height: 0; 447 | padding: 21px 10px 0 10px; 448 | overflow: hidden; 449 | } 450 | 451 | .p-gui__header { 452 | position: absolute; 453 | top: 0; 454 | left: 0; 455 | width: 100%; 456 | height: 20px; 457 | background-color: rgba(0, 0, 0, .8); 458 | cursor: grab; 459 | color: grey; 460 | font-size: 10px; 461 | line-height: 20px; 462 | padding-left: 12px; 463 | box-sizing: border-box; 464 | touch-action: none; 465 | } 466 | 467 | .p-gui__header-close { 468 | width: 20px; 469 | height: 20px; 470 | position: absolute; 471 | top: 0; 472 | right: 5px; 473 | cursor: pointer; 474 | background-image: url(); 475 | background-size: 50% 50%; 476 | background-position: center; 477 | background-repeat: no-repeat; 478 | } 479 | 480 | .p-gui--collapsed .p-gui__header-close { 481 | background-image: url(); 482 | } 483 | 484 | .p-gui__slider, 485 | .p-gui__button, 486 | .p-gui__switch, 487 | .p-gui__list, 488 | .p-gui__vector2, 489 | .p-gui__color { 490 | width: 100%; 491 | padding: 7px; 492 | cursor: pointer; 493 | position: relative; 494 | box-sizing: border-box; 495 | margin-block: 3px; 496 | border: 1px solid var(--color-border-2); 497 | border-radius: var(--main-border-radius); 498 | transition: var(--transition) border-color; 499 | } 500 | 501 | .p-gui__slider:hover, 502 | .p-gui__button:hover, 503 | .p-gui__switch:hover, 504 | .p-gui__list:hover, 505 | .p-gui__vector2:hover, 506 | .p-gui__color:hover { 507 | border-color: rgba(255,255,255,.2); 508 | } 509 | 510 | ${A} 511 | 512 | ${H} 513 | 514 | ${$} 515 | 516 | ${D} 517 | 518 | ${O} 519 | 520 | ${F} 521 | 522 | ${L} 523 | 524 | ${N} 525 | `}class E{constructor(e={}){if(this.firstParent=this,e.container?(this.container=typeof e.container=="string"?document.querySelector(e.container):e.container,this.position_type="absolute"):(this.container=document.body,this.position_type="fixed"),this.propReferences=[],this.folders=[],e.isFolder){this._folderConstructor(e.folderOptions);return}typeof e.onUpdate=="function"&&(this.onUpdate=e.onUpdate),this.name=e!=null&&typeof e.name=="string"?e.name:"",this.backgroundColor=e.color||null,this.opacity=e.opacity||1,this.container==document.body?this.maxHeight=window.innerHeight:this.maxHeight=Math.min(this.container.clientHeight,window.innerHeight),e.maxHeight&&(this.initMaxHeight=e.maxHeight,this.maxHeight=Math.min(this.initMaxHeight,this.maxHeight)),this.screenCorner=this._parseScreenCorner(e.position),window.perfectGUI||(window.perfectGUI={}),window.perfectGUI.instanceCounter==null?window.perfectGUI.instanceCounter=0:window.perfectGUI.instanceCounter++,this.instanceId=window.perfectGUI.instanceCounter,this.wrapperWidth=e.width||290,this.stylesheet=document.createElement("style"),this.stylesheet.setAttribute("type","text/css"),this.stylesheet.setAttribute("id","lm-gui-stylesheet"),document.head.append(this.stylesheet),this.instanceId==0&&this._addStyles(`${M(this.position_type)}`),this._styleInstance(),this._addWrapper(),this.wrapper.setAttribute("data-corner-x",this.screenCorner.x),this.wrapper.setAttribute("data-corner-y",this.screenCorner.y),e.autoRepositioning!=!1&&window.addEventListener("resize",this._handleResize.bind(this)),this._handleResize(),this.hasBeenDragged=!1,e.draggable==!0&&this._makeDraggable(),this.closed=!1,e.closed&&this.toggleClose()}_styleInstance(){let e=this._getScrollbarWidth(this.container);if(this.screenCorner.x=="left"?this.xOffset=0:this.xOffset=this.container.clientWidth-this.wrapperWidth-e,this.instanceId>0){let t=this.container.querySelectorAll(".p-gui");for(let i=0;i0){let t=this.container.querySelectorAll(`.p-gui:not(#${this.wrapper.id}):not([data-dragged])`);for(let i=0;ithis.instanceId);i++)this.screenCorner.y==t[i].dataset.cornerY&&(this.screenCorner.x=="left"&&t[i].dataset.cornerX=="left"?this.xOffset+=t[i].offsetWidth:this.screenCorner.x=="right"&&t[i].dataset.cornerX=="right"&&(this.xOffset-=t[i].offsetWidth))}this.position={prevX:this.xOffset,prevY:this.yOffset,x:this.xOffset,y:this.yOffset},this.wrapper.style.transform=`translate3d(${this.position.x}px, ${this.position.y}px, 0)`}_addStyles(e){this.stylesheet.innerHTML+=e}_addWrapper(){this.wrapper=document.createElement("div"),this.wrapper.id="p-gui-"+this.instanceId,this.wrapper.className="p-gui",this.container.append(this.wrapper),this.header=document.createElement("div"),this.header.className="p-gui__header",this.header.textContent=this.name,this.header.style=`${this.backgroundColor?"border-color: "+this.backgroundColor+";":""}`,this.wrapper.append(this.header);const e=document.createElement("div");e.className="p-gui__header-close",e.addEventListener("click",this.toggleClose.bind(this)),this.header.append(e)}button(e,t){let i="";typeof e!="string"?typeof e=="object"&&(e!=null&&e.hasOwnProperty("name"))?i=e.name==""?" ":e.name:i=" ":i=e==""?" ":e;const a=typeof e.tooltip=="string"?e.tooltip:e.tooltip===!0?i:null;this.imageContainer=null;const s=document.createElement("div");s.className="p-gui__button",s.textContent=i,a&&s.setAttribute("title",a),s.addEventListener("click",()=>{t&&t(),this.onUpdate?this.onUpdate():this.isFolder&&this.firstParent.onUpdate&&this.firstParent.onUpdate()}),this.wrapper.append(s),typeof e.color=="string"&&(s.style.setProperty("--color-accent",e.color),s.style.setProperty("--color-accent-hover",e.hoverColor||e.color))}image(e={},t){if(typeof e!="object")throw Error(`[GUI] image() first parameter must be an object. Received: ${typeof e}.`);let i;if(typeof e.path=="string")i=e.path;else throw typeof e.path==null?Error("[GUI] image() path must be provided."):Error("[GUI] image() path must be a string.");let a=i.replace(/^.*[\\\/]/,""),s;e.name==null?s=a:s=typeof e.name=="string"&&e.name||" ";const o=typeof e.tooltip=="string"?e.tooltip:e.tooltip===!0?s:null,r=e.selected===!0,n=e.selectionBorder!==!1;let c="";e.width&&(typeof e.width=="number"&&(e.width+="px"),c+=`flex: 0 0 calc(${e.width} - 5px); `),e.height&&(typeof e.height=="number"&&(e.height+="px"),c+=`height: ${e.height}; `),this.imageContainer||(this.imageContainer=document.createElement("div"),this.imageContainer.className="p-gui__image-container",this.wrapper.append(this.imageContainer));const l=document.createElement("div");l.className="p-gui__image",l.style="background-image: url("+i+"); "+c,o&&l.setAttribute("title",o),this.imageContainer.append(l),r&&n&&l.classList.add("p-gui__image--selected");const d=document.createElement("div");return d.className="p-gui__image-text",d.textContent=s,l.append(d),l.addEventListener("click",()=>{let p=l.parentElement.querySelectorAll(".p-gui__image--selected");for(let h=0;h{const u=h.target.childNodes[1];let f=!0;u.classList.contains("p-gui__switch-checkbox--active")&&(f=!1),u.classList.toggle("p-gui__switch-checkbox--active"),a?o[r]=f:typeof t=="function"&&t(f),this.onUpdate?this.onUpdate():this.isFolder&&this.firstParent.onUpdate&&this.firstParent.onUpdate()});let d=(()=>a?o[r]?" p-gui__switch-checkbox--active":"":n?" p-gui__switch-checkbox--active":"")();const p=document.createElement("div");p.className="p-gui__switch-checkbox"+d,l.append(p),a&&Object.defineProperty(o,r,{set:h=>{this.propReferences[s]=h,h?p.classList.add("p-gui__switch-checkbox--active"):p.classList.remove("p-gui__switch-checkbox--active"),typeof t=="function"&&t(h)},get:()=>this.propReferences[s]})}list(e={},t){if(typeof e!="object")throw Error(`[GUI] list() first parameter must be an object. Received: ${typeof e}.`);let i=typeof e.name=="string"?e.name:" ",a=!1,s=null,o=e.obj,r=e.prop,n=Array.isArray(e.values)?e.values:null,c,l=typeof n[0]!="string";const d=typeof e.tooltip=="string"?e.tooltip:e.tooltip===!0?i:null;if(t=typeof t=="function"?t:null,e.value!==void 0||e.value===void 0&&o===void 0&&r===void 0)(r!=null||o!=null)&&console.warn('[GUI] list() "obj" and "prop" parameters are ignored when a "value" parameter is used.'),c=(()=>{if(!n)return null;if(typeof e.value=="string")return n.indexOf(e.value);if(typeof e.value=="number")return e.value})();else if(r!=null&&o!=null){if(typeof r!="string")throw Error(`[GUI] list() "prop" parameter must be an string. Received: ${typeof r}.`);if(typeof o!="object")throw Error(`[GUI] list() "obj" parameter must be an object. Received: ${typeof o}.`);c=(()=>{if(!n)return null;if(typeof o[r]=="string")return l?n.find(u=>u.value===o[r]).value:n.indexOf(o[r]);if(typeof o[r]=="number")return l?n.find(u=>u.value===o[r]).value:o[r]})(),s=this.propReferences.push(o[r])-1,a=!0}else(r!=null&&o==null||r==null&&o==null)&&console.warn('[GUI] list() "obj" and "prop" parameters must be used together.');this.imageContainer=null;let p=document.createElement("div");p.className="p-gui__list",p.textContent=i,d&&p.setAttribute("title",d),this.wrapper.append(p);let h=document.createElement("select");p.append(h),h.className="p-gui__list-dropdown",h.addEventListener("change",u=>{a?o[r]=u.target.value:t&&t(u.target.value),this.onUpdate?this.onUpdate():this.isFolder&&this.firstParent.onUpdate&&this.firstParent.onUpdate()}),n&&n.forEach((u,f)=>{const g=l?u.name:u,m=l?u.value:u;let x=document.createElement("option");x.setAttribute("value",m),x.textContent=g,h.append(x),(!l&&c==f||l&&c==m)&&x.setAttribute("selected","")}),a&&Object.defineProperty(o,r,{set:u=>{let f,g,m;l?(m=n.find(v=>v.value==u),g=(m==null?void 0:m.value)||n[0].value,f=n.indexOf(m)):(typeof u=="string"&&(f=n.indexOf(u),g=u),typeof u=="number"&&(f=u,g=n[u])),this.propReferences[s]=l?g:u;const x=h.querySelector("[selected]");x&&x.removeAttribute("selected"),h.querySelectorAll("option")[f].setAttribute("selected",""),typeof t=="function"&&t(l?m:g,f)},get:()=>this.propReferences[s]})}color(e={},t){if(typeof e!="object")throw Error(`[GUI] color() first parameter must be an object. Received: ${typeof e}.`);let i=typeof e.name=="string"&&e.name||" ",a=!1,s=null,o=e.obj,r=e.prop,n;const c=typeof e.tooltip=="string"?e.tooltip:e.tooltip===!0?i:null;if(typeof e.value=="string"&&(e.value.length!=7||e.value[0]!="#"?console.error(`[GUI] color() 'value' parameter must be an hexadecimal string in the format "#ffffff". Received: "${e.value}".`):n=e.value),n||(n="#000000"),e.value!==void 0)(r!=null||o!=null)&&console.warn('[GUI] color() "obj" and "prop" parameters are ignored when a "value" parameter is used.');else if(r!=null&&o!=null){if(typeof r!="string")throw Error(`[GUI] color() "prop" parameter must be an string. Received: ${typeof r}.`);if(typeof o!="object")throw Error(`[GUI] color() "obj" parameter must be an object. Received: ${typeof o}.`);i==" "&&(i=r),s=this.propReferences.push(o[r])-1,a=!0}else(r!=null&&o==null||r==null&&o==null)&&console.warn('[GUI] color() "obj" and "prop" parameters must be used together.');this.imageContainer=null;const l=document.createElement("div");l.className="p-gui__color",l.textContent=i,c&&l.setAttribute("title",c),this.wrapper.append(l);const d=document.createElement("input");d.className="p-gui__color-picker",d.setAttribute("type","color"),d.value=n,l.append(d),typeof t=="function"&&d.addEventListener("input",()=>{a?o[r]=d.value:typeof t=="function"&&t(d.value),this.onUpdate?this.onUpdate():this.isFolder&&this.firstParent.onUpdate&&this.firstParent.onUpdate()}),a&&Object.defineProperty(o,r,{set:p=>{this.propReferences[s]=p,d.value=p,typeof t=="function"&&t(p)},get:()=>this.propReferences[s]})}vector2(e={},t){if(typeof e!="object")throw Error(`[GUI] vector2() first parameter must be an object. Received: ${typeof e}.`);let i=typeof e.name=="string"&&e.name||" ";const a=e.x.min??0,s=e.x.max??1,o=e.y.min??0,r=e.y.max??1,n=e.x.step||(s-a)/100,c=e.y.step||(r-o)/100,l=this._countDecimals(n),d=this._countDecimals(c),p=e.x.obj,h=e.x.prop,u=this.propReferences.push(p[h])-1,f=e.y.obj,g=e.y.prop,m=this.propReferences.push(f[g])-1,x=typeof e.tooltip=="string"?e.tooltip:e.tooltip===!0?i:null;t=typeof t=="function"?t:null,this.imageContainer=null;const v=document.createElement("div");v.className="p-gui__vector2",v.textContent=i,x&&v.setAttribute("title",x),this.wrapper.append(v);const k=document.createElement("div");k.className="p-gui__vector-value",k.textContent=p[h]+", "+f[g],v.append(k);const b=document.createElement("div");b.className="p-gui__vector2-area",v.append(b),b.addEventListener("click",_=>{const j=parseFloat(this._mapLinear(_.offsetX,0,b.clientWidth,a,s)),I=parseFloat(this._mapLinear(_.offsetY,0,b.clientHeight,r,o));p[h]=j.toFixed(l),f[g]=I.toFixed(d),t&&t(p[h],p[g]),this.onUpdate?this.onUpdate():this.isFolder&&this.firstParent.onUpdate&&this.firstParent.onUpdate()});let U=!1;b.addEventListener("pointerdown",_=>{U=!0}),b.addEventListener("pointerup",()=>{U=!1}),b.addEventListener("pointermove",_=>{if(U){const j=parseFloat(this._mapLinear(_.offsetX,0,b.clientWidth,a,s)),I=parseFloat(this._mapLinear(_.offsetY,0,b.clientHeight,r,o));p[h]=j.toFixed(l),f[g]=I.toFixed(d),t&&t(p[h],p[g]),this.onUpdate?this.onUpdate():this.isFolder&&this.firstParent.onUpdate&&this.firstParent.onUpdate()}});const R=document.createElement("div");R.className="p-gui__vector2-line p-gui__vector2-line-x",b.append(R);const P=document.createElement("div");P.className="p-gui__vector2-line p-gui__vector2-line-y",b.append(P);const w=document.createElement("div");w.className="p-gui__vector2-dot",b.append(w),w.style.left=this._mapLinear(p[h],a,s,0,b.clientWidth)+"px",w.style.top=this._mapLinear(f[g],o,r,b.clientHeight,0)+"px",Object.defineProperty(p,h,{set:_=>{this.propReferences[u]=_,w.style.left=this._mapLinear(_,a,s,0,b.clientWidth)+"px",k.textContent=String(_)+", "+f[g]},get:()=>this.propReferences[u]}),Object.defineProperty(f,g,{set:_=>{this.propReferences[m]=_,w.style.top=this._mapLinear(_,o,r,b.clientHeight,0)+"px",k.textContent=p[h]+", "+String(_)},get:()=>this.propReferences[m]})}folder(e={}){let t=typeof e.closed=="boolean"?e.closed:!1,i=e.name||"",a=e.color||null,s=e.maxHeight||null;this.imageContainer=null;let o="p-gui__folder";this.folders.length==0&&(o+=" p-gui__folder--first"),t&&(o+=" p-gui__folder--closed");let r=a?`background-color: ${a};`:"";r+=s?`max-height: ${s}px;`:"";const n=document.createElement("div");n.className=o,n.style=r,this.wrapper.append(n);const c=document.createElement("div");c.innerHTML=`${i}`,c.className="p-gui__folder-header",n.append(c),c.addEventListener("click",()=>{n.classList.toggle("p-gui__folder--closed")});let l=new E({isFolder:!0,folderOptions:{wrapper:n,parent:this,firstParent:this.firstParent}});return this.folders.push(l),l}_makeDraggable(){var e=this;this.header.addEventListener("pointerdown",t),this.header.addEventListener("pointerup",a);function t(s){s.preventDefault(),e.position.initX=e.position.x,e.position.initY=e.position.y,e.position.prevX=s.clientX,e.position.prevY=s.clientY,document.addEventListener("pointermove",i)}function i(s){s.preventDefault(),e.hasBeenDragged||(e.hasBeenDragged=!0,e.wrapper.setAttribute("data-dragged","true")),e.position.x=e.position.initX+s.clientX-e.position.prevX,e.position.y=e.position.initY+s.clientY-e.position.prevY,e.wrapper.style.transform="translate3d("+e.position.x+"px,"+e.position.y+"px,0)"}function a(s){document.removeEventListener("pointermove",i)}}toggleClose(){this.closed=!this.closed,this.closed?(this.previousInnerScroll=this.wrapper.scrollTop,this.wrapper.scrollTo(0,0)):this.wrapper.scrollTo(0,this.previousInnerScroll),this.wrapper.classList.toggle("p-gui--collapsed")}kill(){this.wrapper.remove()}_mapLinear(e,t,i,a,s){return a+(e-t)*(s-a)/(i-t)}_countDecimals(e){const t=e.toString(),i=t.indexOf(".");return i===-1?0:t.length-i-1}}return E}); 533 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "perfect-gui", 3 | "version": "4.9.3", 4 | "lockfileVersion": 3, 5 | "requires": true, 6 | "packages": { 7 | "": { 8 | "name": "perfect-gui", 9 | "version": "4.9.3", 10 | "license": "MIT", 11 | "devDependencies": { 12 | "vite": "^4.5.0" 13 | } 14 | }, 15 | "node_modules/@esbuild/android-arm": { 16 | "version": "0.18.20", 17 | "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.18.20.tgz", 18 | "integrity": "sha512-fyi7TDI/ijKKNZTUJAQqiG5T7YjJXgnzkURqmGj13C6dCqckZBLdl4h7bkhHt/t0WP+zO9/zwroDvANaOqO5Sw==", 19 | "cpu": [ 20 | "arm" 21 | ], 22 | "dev": true, 23 | "optional": true, 24 | "os": [ 25 | "android" 26 | ], 27 | "engines": { 28 | "node": ">=12" 29 | } 30 | }, 31 | "node_modules/@esbuild/android-arm64": { 32 | "version": "0.18.20", 33 | "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.18.20.tgz", 34 | "integrity": "sha512-Nz4rJcchGDtENV0eMKUNa6L12zz2zBDXuhj/Vjh18zGqB44Bi7MBMSXjgunJgjRhCmKOjnPuZp4Mb6OKqtMHLQ==", 35 | "cpu": [ 36 | "arm64" 37 | ], 38 | "dev": true, 39 | "optional": true, 40 | "os": [ 41 | "android" 42 | ], 43 | "engines": { 44 | "node": ">=12" 45 | } 46 | }, 47 | "node_modules/@esbuild/android-x64": { 48 | "version": "0.18.20", 49 | "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.18.20.tgz", 50 | "integrity": "sha512-8GDdlePJA8D6zlZYJV/jnrRAi6rOiNaCC/JclcXpB+KIuvfBN4owLtgzY2bsxnx666XjJx2kDPUmnTtR8qKQUg==", 51 | "cpu": [ 52 | "x64" 53 | ], 54 | "dev": true, 55 | "optional": true, 56 | "os": [ 57 | "android" 58 | ], 59 | "engines": { 60 | "node": ">=12" 61 | } 62 | }, 63 | "node_modules/@esbuild/darwin-arm64": { 64 | "version": "0.18.20", 65 | "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.18.20.tgz", 66 | "integrity": "sha512-bxRHW5kHU38zS2lPTPOyuyTm+S+eobPUnTNkdJEfAddYgEcll4xkT8DB9d2008DtTbl7uJag2HuE5NZAZgnNEA==", 67 | "cpu": [ 68 | "arm64" 69 | ], 70 | "dev": true, 71 | "optional": true, 72 | "os": [ 73 | "darwin" 74 | ], 75 | "engines": { 76 | "node": ">=12" 77 | } 78 | }, 79 | "node_modules/@esbuild/darwin-x64": { 80 | "version": "0.18.20", 81 | "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.18.20.tgz", 82 | "integrity": "sha512-pc5gxlMDxzm513qPGbCbDukOdsGtKhfxD1zJKXjCCcU7ju50O7MeAZ8c4krSJcOIJGFR+qx21yMMVYwiQvyTyQ==", 83 | "cpu": [ 84 | "x64" 85 | ], 86 | "dev": true, 87 | "optional": true, 88 | "os": [ 89 | "darwin" 90 | ], 91 | "engines": { 92 | "node": ">=12" 93 | } 94 | }, 95 | "node_modules/@esbuild/freebsd-arm64": { 96 | "version": "0.18.20", 97 | "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.18.20.tgz", 98 | "integrity": "sha512-yqDQHy4QHevpMAaxhhIwYPMv1NECwOvIpGCZkECn8w2WFHXjEwrBn3CeNIYsibZ/iZEUemj++M26W3cNR5h+Tw==", 99 | "cpu": [ 100 | "arm64" 101 | ], 102 | "dev": true, 103 | "optional": true, 104 | "os": [ 105 | "freebsd" 106 | ], 107 | "engines": { 108 | "node": ">=12" 109 | } 110 | }, 111 | "node_modules/@esbuild/freebsd-x64": { 112 | "version": "0.18.20", 113 | "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.18.20.tgz", 114 | "integrity": "sha512-tgWRPPuQsd3RmBZwarGVHZQvtzfEBOreNuxEMKFcd5DaDn2PbBxfwLcj4+aenoh7ctXcbXmOQIn8HI6mCSw5MQ==", 115 | "cpu": [ 116 | "x64" 117 | ], 118 | "dev": true, 119 | "optional": true, 120 | "os": [ 121 | "freebsd" 122 | ], 123 | "engines": { 124 | "node": ">=12" 125 | } 126 | }, 127 | "node_modules/@esbuild/linux-arm": { 128 | "version": "0.18.20", 129 | "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.18.20.tgz", 130 | "integrity": "sha512-/5bHkMWnq1EgKr1V+Ybz3s1hWXok7mDFUMQ4cG10AfW3wL02PSZi5kFpYKrptDsgb2WAJIvRcDm+qIvXf/apvg==", 131 | "cpu": [ 132 | "arm" 133 | ], 134 | "dev": true, 135 | "optional": true, 136 | "os": [ 137 | "linux" 138 | ], 139 | "engines": { 140 | "node": ">=12" 141 | } 142 | }, 143 | "node_modules/@esbuild/linux-arm64": { 144 | "version": "0.18.20", 145 | "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.18.20.tgz", 146 | "integrity": "sha512-2YbscF+UL7SQAVIpnWvYwM+3LskyDmPhe31pE7/aoTMFKKzIc9lLbyGUpmmb8a8AixOL61sQ/mFh3jEjHYFvdA==", 147 | "cpu": [ 148 | "arm64" 149 | ], 150 | "dev": true, 151 | "optional": true, 152 | "os": [ 153 | "linux" 154 | ], 155 | "engines": { 156 | "node": ">=12" 157 | } 158 | }, 159 | "node_modules/@esbuild/linux-ia32": { 160 | "version": "0.18.20", 161 | "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.18.20.tgz", 162 | "integrity": "sha512-P4etWwq6IsReT0E1KHU40bOnzMHoH73aXp96Fs8TIT6z9Hu8G6+0SHSw9i2isWrD2nbx2qo5yUqACgdfVGx7TA==", 163 | "cpu": [ 164 | "ia32" 165 | ], 166 | "dev": true, 167 | "optional": true, 168 | "os": [ 169 | "linux" 170 | ], 171 | "engines": { 172 | "node": ">=12" 173 | } 174 | }, 175 | "node_modules/@esbuild/linux-loong64": { 176 | "version": "0.18.20", 177 | "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.18.20.tgz", 178 | "integrity": "sha512-nXW8nqBTrOpDLPgPY9uV+/1DjxoQ7DoB2N8eocyq8I9XuqJ7BiAMDMf9n1xZM9TgW0J8zrquIb/A7s3BJv7rjg==", 179 | "cpu": [ 180 | "loong64" 181 | ], 182 | "dev": true, 183 | "optional": true, 184 | "os": [ 185 | "linux" 186 | ], 187 | "engines": { 188 | "node": ">=12" 189 | } 190 | }, 191 | "node_modules/@esbuild/linux-mips64el": { 192 | "version": "0.18.20", 193 | "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.18.20.tgz", 194 | "integrity": "sha512-d5NeaXZcHp8PzYy5VnXV3VSd2D328Zb+9dEq5HE6bw6+N86JVPExrA6O68OPwobntbNJ0pzCpUFZTo3w0GyetQ==", 195 | "cpu": [ 196 | "mips64el" 197 | ], 198 | "dev": true, 199 | "optional": true, 200 | "os": [ 201 | "linux" 202 | ], 203 | "engines": { 204 | "node": ">=12" 205 | } 206 | }, 207 | "node_modules/@esbuild/linux-ppc64": { 208 | "version": "0.18.20", 209 | "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.18.20.tgz", 210 | "integrity": "sha512-WHPyeScRNcmANnLQkq6AfyXRFr5D6N2sKgkFo2FqguP44Nw2eyDlbTdZwd9GYk98DZG9QItIiTlFLHJHjxP3FA==", 211 | "cpu": [ 212 | "ppc64" 213 | ], 214 | "dev": true, 215 | "optional": true, 216 | "os": [ 217 | "linux" 218 | ], 219 | "engines": { 220 | "node": ">=12" 221 | } 222 | }, 223 | "node_modules/@esbuild/linux-riscv64": { 224 | "version": "0.18.20", 225 | "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.18.20.tgz", 226 | "integrity": "sha512-WSxo6h5ecI5XH34KC7w5veNnKkju3zBRLEQNY7mv5mtBmrP/MjNBCAlsM2u5hDBlS3NGcTQpoBvRzqBcRtpq1A==", 227 | "cpu": [ 228 | "riscv64" 229 | ], 230 | "dev": true, 231 | "optional": true, 232 | "os": [ 233 | "linux" 234 | ], 235 | "engines": { 236 | "node": ">=12" 237 | } 238 | }, 239 | "node_modules/@esbuild/linux-s390x": { 240 | "version": "0.18.20", 241 | "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.18.20.tgz", 242 | "integrity": "sha512-+8231GMs3mAEth6Ja1iK0a1sQ3ohfcpzpRLH8uuc5/KVDFneH6jtAJLFGafpzpMRO6DzJ6AvXKze9LfFMrIHVQ==", 243 | "cpu": [ 244 | "s390x" 245 | ], 246 | "dev": true, 247 | "optional": true, 248 | "os": [ 249 | "linux" 250 | ], 251 | "engines": { 252 | "node": ">=12" 253 | } 254 | }, 255 | "node_modules/@esbuild/linux-x64": { 256 | "version": "0.18.20", 257 | "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.18.20.tgz", 258 | "integrity": "sha512-UYqiqemphJcNsFEskc73jQ7B9jgwjWrSayxawS6UVFZGWrAAtkzjxSqnoclCXxWtfwLdzU+vTpcNYhpn43uP1w==", 259 | "cpu": [ 260 | "x64" 261 | ], 262 | "dev": true, 263 | "optional": true, 264 | "os": [ 265 | "linux" 266 | ], 267 | "engines": { 268 | "node": ">=12" 269 | } 270 | }, 271 | "node_modules/@esbuild/netbsd-x64": { 272 | "version": "0.18.20", 273 | "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.18.20.tgz", 274 | "integrity": "sha512-iO1c++VP6xUBUmltHZoMtCUdPlnPGdBom6IrO4gyKPFFVBKioIImVooR5I83nTew5UOYrk3gIJhbZh8X44y06A==", 275 | "cpu": [ 276 | "x64" 277 | ], 278 | "dev": true, 279 | "optional": true, 280 | "os": [ 281 | "netbsd" 282 | ], 283 | "engines": { 284 | "node": ">=12" 285 | } 286 | }, 287 | "node_modules/@esbuild/openbsd-x64": { 288 | "version": "0.18.20", 289 | "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.18.20.tgz", 290 | "integrity": "sha512-e5e4YSsuQfX4cxcygw/UCPIEP6wbIL+se3sxPdCiMbFLBWu0eiZOJ7WoD+ptCLrmjZBK1Wk7I6D/I3NglUGOxg==", 291 | "cpu": [ 292 | "x64" 293 | ], 294 | "dev": true, 295 | "optional": true, 296 | "os": [ 297 | "openbsd" 298 | ], 299 | "engines": { 300 | "node": ">=12" 301 | } 302 | }, 303 | "node_modules/@esbuild/sunos-x64": { 304 | "version": "0.18.20", 305 | "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.18.20.tgz", 306 | "integrity": "sha512-kDbFRFp0YpTQVVrqUd5FTYmWo45zGaXe0X8E1G/LKFC0v8x0vWrhOWSLITcCn63lmZIxfOMXtCfti/RxN/0wnQ==", 307 | "cpu": [ 308 | "x64" 309 | ], 310 | "dev": true, 311 | "optional": true, 312 | "os": [ 313 | "sunos" 314 | ], 315 | "engines": { 316 | "node": ">=12" 317 | } 318 | }, 319 | "node_modules/@esbuild/win32-arm64": { 320 | "version": "0.18.20", 321 | "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.18.20.tgz", 322 | "integrity": "sha512-ddYFR6ItYgoaq4v4JmQQaAI5s7npztfV4Ag6NrhiaW0RrnOXqBkgwZLofVTlq1daVTQNhtI5oieTvkRPfZrePg==", 323 | "cpu": [ 324 | "arm64" 325 | ], 326 | "dev": true, 327 | "optional": true, 328 | "os": [ 329 | "win32" 330 | ], 331 | "engines": { 332 | "node": ">=12" 333 | } 334 | }, 335 | "node_modules/@esbuild/win32-ia32": { 336 | "version": "0.18.20", 337 | "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.18.20.tgz", 338 | "integrity": "sha512-Wv7QBi3ID/rROT08SABTS7eV4hX26sVduqDOTe1MvGMjNd3EjOz4b7zeexIR62GTIEKrfJXKL9LFxTYgkyeu7g==", 339 | "cpu": [ 340 | "ia32" 341 | ], 342 | "dev": true, 343 | "optional": true, 344 | "os": [ 345 | "win32" 346 | ], 347 | "engines": { 348 | "node": ">=12" 349 | } 350 | }, 351 | "node_modules/@esbuild/win32-x64": { 352 | "version": "0.18.20", 353 | "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.18.20.tgz", 354 | "integrity": "sha512-kTdfRcSiDfQca/y9QIkng02avJ+NCaQvrMejlsB3RRv5sE9rRoeBPISaZpKxHELzRxZyLvNts1P27W3wV+8geQ==", 355 | "cpu": [ 356 | "x64" 357 | ], 358 | "dev": true, 359 | "optional": true, 360 | "os": [ 361 | "win32" 362 | ], 363 | "engines": { 364 | "node": ">=12" 365 | } 366 | }, 367 | "node_modules/esbuild": { 368 | "version": "0.18.20", 369 | "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.18.20.tgz", 370 | "integrity": "sha512-ceqxoedUrcayh7Y7ZX6NdbbDzGROiyVBgC4PriJThBKSVPWnnFHZAkfI1lJT8QFkOwH4qOS2SJkS4wvpGl8BpA==", 371 | "dev": true, 372 | "hasInstallScript": true, 373 | "bin": { 374 | "esbuild": "bin/esbuild" 375 | }, 376 | "engines": { 377 | "node": ">=12" 378 | }, 379 | "optionalDependencies": { 380 | "@esbuild/android-arm": "0.18.20", 381 | "@esbuild/android-arm64": "0.18.20", 382 | "@esbuild/android-x64": "0.18.20", 383 | "@esbuild/darwin-arm64": "0.18.20", 384 | "@esbuild/darwin-x64": "0.18.20", 385 | "@esbuild/freebsd-arm64": "0.18.20", 386 | "@esbuild/freebsd-x64": "0.18.20", 387 | "@esbuild/linux-arm": "0.18.20", 388 | "@esbuild/linux-arm64": "0.18.20", 389 | "@esbuild/linux-ia32": "0.18.20", 390 | "@esbuild/linux-loong64": "0.18.20", 391 | "@esbuild/linux-mips64el": "0.18.20", 392 | "@esbuild/linux-ppc64": "0.18.20", 393 | "@esbuild/linux-riscv64": "0.18.20", 394 | "@esbuild/linux-s390x": "0.18.20", 395 | "@esbuild/linux-x64": "0.18.20", 396 | "@esbuild/netbsd-x64": "0.18.20", 397 | "@esbuild/openbsd-x64": "0.18.20", 398 | "@esbuild/sunos-x64": "0.18.20", 399 | "@esbuild/win32-arm64": "0.18.20", 400 | "@esbuild/win32-ia32": "0.18.20", 401 | "@esbuild/win32-x64": "0.18.20" 402 | } 403 | }, 404 | "node_modules/fsevents": { 405 | "version": "2.3.3", 406 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", 407 | "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", 408 | "dev": true, 409 | "hasInstallScript": true, 410 | "optional": true, 411 | "os": [ 412 | "darwin" 413 | ], 414 | "engines": { 415 | "node": "^8.16.0 || ^10.6.0 || >=11.0.0" 416 | } 417 | }, 418 | "node_modules/nanoid": { 419 | "version": "3.3.6", 420 | "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.6.tgz", 421 | "integrity": "sha512-BGcqMMJuToF7i1rt+2PWSNVnWIkGCU78jBG3RxO/bZlnZPK2Cmi2QaffxGO/2RvWi9sL+FAiRiXMgsyxQ1DIDA==", 422 | "dev": true, 423 | "funding": [ 424 | { 425 | "type": "github", 426 | "url": "https://github.com/sponsors/ai" 427 | } 428 | ], 429 | "bin": { 430 | "nanoid": "bin/nanoid.cjs" 431 | }, 432 | "engines": { 433 | "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" 434 | } 435 | }, 436 | "node_modules/picocolors": { 437 | "version": "1.0.0", 438 | "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.0.tgz", 439 | "integrity": "sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==", 440 | "dev": true 441 | }, 442 | "node_modules/postcss": { 443 | "version": "8.4.31", 444 | "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.31.tgz", 445 | "integrity": "sha512-PS08Iboia9mts/2ygV3eLpY5ghnUcfLV/EXTOW1E2qYxJKGGBUtNjN76FYHnMs36RmARn41bC0AZmn+rR0OVpQ==", 446 | "dev": true, 447 | "funding": [ 448 | { 449 | "type": "opencollective", 450 | "url": "https://opencollective.com/postcss/" 451 | }, 452 | { 453 | "type": "tidelift", 454 | "url": "https://tidelift.com/funding/github/npm/postcss" 455 | }, 456 | { 457 | "type": "github", 458 | "url": "https://github.com/sponsors/ai" 459 | } 460 | ], 461 | "dependencies": { 462 | "nanoid": "^3.3.6", 463 | "picocolors": "^1.0.0", 464 | "source-map-js": "^1.0.2" 465 | }, 466 | "engines": { 467 | "node": "^10 || ^12 || >=14" 468 | } 469 | }, 470 | "node_modules/rollup": { 471 | "version": "3.29.4", 472 | "resolved": "https://registry.npmjs.org/rollup/-/rollup-3.29.4.tgz", 473 | "integrity": "sha512-oWzmBZwvYrU0iJHtDmhsm662rC15FRXmcjCk1xD771dFDx5jJ02ufAQQTn0etB2emNk4J9EZg/yWKpsn9BWGRw==", 474 | "dev": true, 475 | "bin": { 476 | "rollup": "dist/bin/rollup" 477 | }, 478 | "engines": { 479 | "node": ">=14.18.0", 480 | "npm": ">=8.0.0" 481 | }, 482 | "optionalDependencies": { 483 | "fsevents": "~2.3.2" 484 | } 485 | }, 486 | "node_modules/source-map-js": { 487 | "version": "1.0.2", 488 | "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.0.2.tgz", 489 | "integrity": "sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw==", 490 | "dev": true, 491 | "engines": { 492 | "node": ">=0.10.0" 493 | } 494 | }, 495 | "node_modules/vite": { 496 | "version": "4.5.0", 497 | "resolved": "https://registry.npmjs.org/vite/-/vite-4.5.0.tgz", 498 | "integrity": "sha512-ulr8rNLA6rkyFAlVWw2q5YJ91v098AFQ2R0PRFwPzREXOUJQPtFUG0t+/ZikhaOCDqFoDhN6/v8Sq0o4araFAw==", 499 | "dev": true, 500 | "dependencies": { 501 | "esbuild": "^0.18.10", 502 | "postcss": "^8.4.27", 503 | "rollup": "^3.27.1" 504 | }, 505 | "bin": { 506 | "vite": "bin/vite.js" 507 | }, 508 | "engines": { 509 | "node": "^14.18.0 || >=16.0.0" 510 | }, 511 | "funding": { 512 | "url": "https://github.com/vitejs/vite?sponsor=1" 513 | }, 514 | "optionalDependencies": { 515 | "fsevents": "~2.3.2" 516 | }, 517 | "peerDependencies": { 518 | "@types/node": ">= 14", 519 | "less": "*", 520 | "lightningcss": "^1.21.0", 521 | "sass": "*", 522 | "stylus": "*", 523 | "sugarss": "*", 524 | "terser": "^5.4.0" 525 | }, 526 | "peerDependenciesMeta": { 527 | "@types/node": { 528 | "optional": true 529 | }, 530 | "less": { 531 | "optional": true 532 | }, 533 | "lightningcss": { 534 | "optional": true 535 | }, 536 | "sass": { 537 | "optional": true 538 | }, 539 | "stylus": { 540 | "optional": true 541 | }, 542 | "sugarss": { 543 | "optional": true 544 | }, 545 | "terser": { 546 | "optional": true 547 | } 548 | } 549 | } 550 | } 551 | } 552 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "perfect-gui", 3 | "version": "4.11.9", 4 | "description": "GUI for JavaScript", 5 | "main": "src/index.js", 6 | "scripts": { 7 | "build": "vite build" 8 | }, 9 | "repository": { 10 | "type": "git", 11 | "url": "git+https://github.com/thibka/perfect-gui.git" 12 | }, 13 | "keywords": [ 14 | "gui" 15 | ], 16 | "author": "thibka", 17 | "license": "MIT", 18 | "bugs": { 19 | "url": "https://github.com/thibka/perfect-gui/issues" 20 | }, 21 | "homepage": "https://thibka.github.io/perfect-gui/public/", 22 | "devDependencies": { 23 | "vite": "^4.5.0" 24 | } 25 | } 26 | -------------------------------------------------------------------------------- /src/addons/three.js: -------------------------------------------------------------------------------- 1 | import GUI from '../index.js'; 2 | 3 | /* 4 | GUI.prototype.materialFolder = materialFolder; 5 | gui.materialFolder({name: 'material', obj: material}) 6 | */ 7 | export function materialFolder(params = {}) { 8 | const material = params.obj; 9 | 10 | if (material.isMeshStandardMaterial) { 11 | console.error('[GUI materialFolder]', 'materialFolder() only works with meshStandardMaterial'); 12 | } 13 | 14 | const folder = this.folder({ name: params.name }); 15 | 16 | folder.slider({ obj: material, prop: 'roughness' }); 17 | folder.slider({ obj: material, prop: 'metalness' }); 18 | folder.color({ name: 'color', obj: material, value: '#'+material.color.getHexString() }, val => { 19 | material.color.set(val) 20 | }); 21 | folder.slider({ obj: material, prop: 'envMapIntensity' }); 22 | return folder; 23 | } 24 | 25 | export function threejsColor() { 26 | 27 | } -------------------------------------------------------------------------------- /src/components/Slider.js: -------------------------------------------------------------------------------- 1 | export default class Slider { 2 | constructor(parent, params = {}, callback) { 3 | this.parent = parent; 4 | this.propReferences = []; 5 | 6 | if (typeof params != 'object') { 7 | throw Error(`[GUI] slider() first parameter must be an object. Received: ${typeof params}.`); 8 | } 9 | 10 | let name = typeof params.name == 'string' ? params.name || ' ' : ' '; 11 | this.isObject = false; 12 | let propReferenceIndex = null; 13 | this.obj = params.obj; 14 | this.prop = params.prop; 15 | let value = typeof params.value == 'number' ? params.value : null; 16 | this.min = params.min ?? 0; 17 | this.max = params.max ?? 1; 18 | this.step = params.step || (this.max - this.min) / 100; 19 | this.decimals = this.parent._countDecimals(this.step); 20 | this.callback = typeof callback == 'function' ? callback : null; 21 | 22 | // callback mode 23 | if ( value !== null ) { 24 | if (this.prop != undefined || this.obj != undefined) { 25 | console.warn(`[GUI] slider() "obj" and "prop" parameters are ignored when a "value" parameter is used.`); 26 | } 27 | } 28 | // object-binding 29 | else if (this.prop != undefined && this.obj != undefined) { 30 | if (typeof this.prop != 'string') { 31 | throw Error(`[GUI] slider() "prop" parameter must be an string. Received: ${typeof this.prop}.`); 32 | } 33 | if (typeof this.obj != 'object') { 34 | throw Error(`[GUI] slider() "obj" parameter must be an object. Received: ${typeof this.obj}.`); 35 | } 36 | 37 | if (name == ' ') { 38 | name = this.prop; 39 | } 40 | 41 | propReferenceIndex = this.propReferences.push(this.obj[this.prop]) - 1; 42 | this.isObject = true; 43 | } 44 | else { 45 | if ((this.prop != undefined && this.obj == undefined) || (this.prop == undefined && this.obj != undefined)) { 46 | console.warn(`[GUI] slider() "obj" and "prop" parameters must be used together.`); 47 | } 48 | 49 | value = (this.max - this.min) / 2; 50 | } 51 | const tooltip = (typeof params.tooltip === 'string') ? params.tooltip : (params.tooltip === true ? name : null); 52 | 53 | this.imageContainer = null; 54 | 55 | const container = document.createElement('div'); 56 | container.className = 'p-gui__slider'; 57 | 58 | if (tooltip) { 59 | container.setAttribute('title', tooltip); 60 | } 61 | 62 | const slider_name = document.createElement('div'); 63 | slider_name.className = 'p-gui__slider-name'; 64 | slider_name.textContent = name; 65 | container.append(slider_name); 66 | 67 | this.ctrlDiv = document.createElement('div'); 68 | this.ctrlDiv.className = 'p-gui__slider-ctrl'; 69 | this.ctrlDiv.setAttribute('type', 'range'); 70 | this.ctrlDiv.setAttribute('min', this.min); 71 | this.ctrlDiv.setAttribute('max', this.max); 72 | container.append(this.ctrlDiv); 73 | 74 | const slider_bar = document.createElement('div'); 75 | slider_bar.className = 'p-gui__slider-bar'; 76 | this.ctrlDiv.append(slider_bar); 77 | 78 | this.handle = document.createElement('div'); 79 | this.handle.className = 'p-gui__slider-handle'; 80 | this.ctrlDiv.append(this.handle); 81 | 82 | this.filling = document.createElement('div'); 83 | this.filling.className = 'p-gui__slider-filling'; 84 | slider_bar.append(this.filling); 85 | 86 | this.valueInput = document.createElement('input'); 87 | this.valueInput.className = 'p-gui__slider-value'; 88 | this.valueInput.value = this.isObject ? this.obj[this.prop] : value; 89 | container.append(this.valueInput); 90 | 91 | // init position 92 | setTimeout(() => { 93 | const handleWidth = this.handle.offsetWidth; 94 | this.handle.position = this._mapLinear(this.valueInput.value, this.min, this.max, handleWidth / 2, 88 - handleWidth / 2); 95 | this.handle.position = Math.min(this.handle.position, 88 - handleWidth / 2); 96 | this.handle.position = Math.max(this.handle.position, handleWidth / 2); 97 | this.handle.style.transform = `translate(-50%, -50%) translateX(${this.handle.position}px)`; 98 | this.filling.style.width = `${this.handle.position}px`; 99 | }, 0); // wait for render 100 | 101 | this.valueInput.addEventListener('change', () => { 102 | this._updateHandlePositionFromValue(); 103 | this._triggerCallbacks(); 104 | }) 105 | 106 | this.ctrlDiv.addEventListener('pointerdown', (evt) => { 107 | this.ctrlDiv.pointerDown = true; 108 | this.ctrlDiv.prevPosition = evt.clientX; 109 | this._updateHandlePositionFromPointer(evt, true); 110 | }); 111 | 112 | window.addEventListener('pointerup', (evt) => { 113 | this.ctrlDiv.pointerDown = false; 114 | }); 115 | 116 | window.addEventListener('pointermove', (evt) => { 117 | if (this.ctrlDiv.pointerDown) { 118 | this.ctrlDiv.pointerDelta = evt.clientX - this.ctrlDiv.prevPosition; 119 | this._updateHandlePositionFromPointer(evt); 120 | } 121 | }); 122 | 123 | if ( this.isObject ) { 124 | Object.defineProperty( this.obj, this.prop, { 125 | set: val => { 126 | this.propReferences[propReferenceIndex] = val; 127 | this.valueInput.value = val; 128 | 129 | this._updateHandlePositionFromValue(); 130 | 131 | if (this.callback) { 132 | this.callback(parseFloat(this.valueInput.value)); 133 | } 134 | }, 135 | get: () => { 136 | return this.propReferences[propReferenceIndex]; 137 | } 138 | }); 139 | } 140 | 141 | return container; 142 | } 143 | 144 | _updateHandlePositionFromPointer(evt, firstDown = false) { 145 | const sliderWidth = this.ctrlDiv.offsetWidth; 146 | const handleWidth = this.handle.offsetWidth; 147 | const pointerDelta = evt.clientX - this.ctrlDiv.prevPosition; 148 | const currentValue = parseFloat(this.valueInput.value); 149 | let handlePosition; 150 | 151 | if (firstDown) { 152 | handlePosition = evt.offsetX; 153 | } else { 154 | handlePosition = this.handle.position + pointerDelta; 155 | } 156 | 157 | handlePosition = Math.max(handleWidth / 2, Math.min(handlePosition, sliderWidth - handleWidth / 2)); 158 | 159 | let newValue = this.min + (this.max - this.min) * (handlePosition - handleWidth / 2) / (sliderWidth - handleWidth); 160 | if ( newValue > currentValue ) { 161 | newValue = this._quantizeFloor(newValue, this.step); 162 | } else { 163 | newValue = this._quantizeCeil(newValue, this.step); 164 | } 165 | 166 | // toFixed(9) avoids weird javascript infinite decimals 167 | newValue = parseFloat(newValue.toFixed(9)); 168 | const nextValue = parseFloat((currentValue + this.step).toFixed(9)); 169 | const prevValue = parseFloat((currentValue - this.step).toFixed(9)); 170 | 171 | if (newValue >= nextValue || newValue <= prevValue) { 172 | newValue = newValue.toFixed(this.decimals); 173 | 174 | this.valueInput.value = newValue; 175 | 176 | this.ctrlDiv.prevPosition = evt.clientX; 177 | 178 | this.handle.style.transform = `translate(-50%, -50%) translateX(${handlePosition}px)`; 179 | this.handle.position = handlePosition; 180 | 181 | this.filling.style.width = this.handle.position + 'px'; 182 | 183 | this._triggerCallbacks(); 184 | } 185 | } 186 | 187 | _updateHandlePositionFromValue() { 188 | const sliderWidth = this.ctrlDiv.offsetWidth; 189 | const handleWidth = this.handle.offsetWidth; 190 | let handlePosition = this._mapLinear(this.valueInput.value, this.min, this.max, handleWidth / 2, sliderWidth - handleWidth / 2); 191 | 192 | handlePosition = Math.max(handleWidth / 2, Math.min(handlePosition, sliderWidth - handleWidth / 2)); 193 | 194 | this.handle.style.transform = `translate(-50%, -50%) translateX(${handlePosition}px)`; 195 | this.handle.position = handlePosition; 196 | 197 | this.filling.style.width = this.handle.position + 'px'; 198 | } 199 | 200 | _triggerCallbacks() { 201 | if ( this.isObject ) { 202 | this.obj[this.prop] = parseFloat(this.valueInput.value); 203 | } 204 | else { 205 | if (this.callback) { 206 | this.callback(parseFloat(this.valueInput.value)); 207 | } 208 | } 209 | 210 | if (this.parent.onUpdate) { 211 | this.parent.onUpdate(); 212 | } else if (this.parent.isFolder && this.parent.firstParent.onUpdate) { 213 | this.parent.firstParent.onUpdate(); 214 | } 215 | } 216 | 217 | _mapLinear( x, a1, a2, b1, b2 ) { 218 | return b1 + ( x - a1 ) * ( b2 - b1 ) / ( a2 - a1 ); 219 | } 220 | 221 | _quantize(x, step) { 222 | return step * Math.round(x / step); 223 | } 224 | 225 | _quantizeCeil(x, step) { 226 | return step * Math.ceil(x / step); 227 | } 228 | 229 | _quantizeFloor(x, step) { 230 | return step * Math.floor(x / step); 231 | } 232 | } -------------------------------------------------------------------------------- /src/index.js: -------------------------------------------------------------------------------- 1 | import Slider from './components/Slider'; 2 | import styles from './styles/styles'; 3 | 4 | export default class GUI { 5 | constructor(options = {}) { 6 | this.firstParent = this; 7 | 8 | if ( options.container ) { 9 | this.container = typeof options.container == "string" ? document.querySelector(options.container) : options.container; 10 | this.position_type = 'absolute'; 11 | } else { 12 | this.container = document.body; 13 | this.position_type = 'fixed'; 14 | } 15 | 16 | this.propReferences = []; 17 | this.folders = []; 18 | 19 | if ( options.isFolder ) { 20 | this._folderConstructor(options.folderOptions); 21 | return; 22 | } 23 | 24 | if ( typeof options.onUpdate == 'function' ) { 25 | this.onUpdate = options.onUpdate; 26 | } 27 | 28 | this.name = (options != undefined && typeof options.name == "string") ? options.name : ''; 29 | 30 | this.backgroundColor = options.color || null; 31 | this.opacity = options.opacity || 1; 32 | 33 | if (this.container == document.body) { 34 | this.maxHeight = window.innerHeight; 35 | } else { 36 | this.maxHeight = Math.min(this.container.clientHeight, window.innerHeight) 37 | } 38 | if ( options.maxHeight ) { 39 | this.initMaxHeight = options.maxHeight; 40 | this.maxHeight = Math.min(this.initMaxHeight, this.maxHeight); 41 | } 42 | 43 | this.screenCorner = this._parseScreenCorner(options.position); 44 | 45 | if (!window.perfectGUI) { 46 | window.perfectGUI = {}; 47 | } 48 | if ( window.perfectGUI.instanceCounter == undefined ) { 49 | window.perfectGUI.instanceCounter = 0; 50 | } else { 51 | window.perfectGUI.instanceCounter++; 52 | } 53 | this.instanceId = window.perfectGUI.instanceCounter; 54 | 55 | this.wrapperWidth = options.width || 290; 56 | this.stylesheet = document.createElement('style'); 57 | this.stylesheet.setAttribute('type', 'text/css'); 58 | this.stylesheet.setAttribute('id', 'lm-gui-stylesheet'); 59 | document.head.append(this.stylesheet); 60 | 61 | // Common styles 62 | if (this.instanceId == 0) { 63 | this._addStyles(`${styles(this.position_type)}`); 64 | } 65 | 66 | // Instance specific styles 67 | this._styleInstance(); 68 | 69 | this._addWrapper(); 70 | this.wrapper.setAttribute('data-corner-x', this.screenCorner.x); 71 | this.wrapper.setAttribute('data-corner-y', this.screenCorner.y); 72 | 73 | if (options.autoRepositioning != false) { 74 | window.addEventListener('resize', this._handleResize.bind(this)); 75 | } 76 | this._handleResize(); 77 | 78 | this.hasBeenDragged = false; 79 | if (options.draggable == true) this._makeDraggable(); 80 | 81 | this.closed = false; 82 | if (options.closed) this.toggleClose(); 83 | } 84 | 85 | _styleInstance() { 86 | let scrollbar_width = this._getScrollbarWidth(this.container); 87 | if (this.screenCorner.x == 'left') { 88 | this.xOffset = 0; 89 | } else { 90 | this.xOffset = this.container.clientWidth - this.wrapperWidth - scrollbar_width; 91 | } 92 | 93 | if (this.instanceId > 0) { 94 | let existingDomInstances = this.container.querySelectorAll('.p-gui'); 95 | for (let i = 0; i < existingDomInstances.length; i++) { 96 | if (this.screenCorner.y == existingDomInstances[i].dataset.cornerY) { 97 | if (this.screenCorner.x == 'left' && existingDomInstances[i].dataset.cornerX == 'left') { 98 | this.xOffset += existingDomInstances[i].offsetWidth; 99 | } 100 | else if (this.screenCorner.x == 'right' && existingDomInstances[i].dataset.cornerX == 'right') { 101 | this.xOffset -= existingDomInstances[i].offsetWidth; 102 | } 103 | } 104 | } 105 | } 106 | this.yOffset = 0; 107 | this.position = { 108 | prevX: this.xOffset, 109 | prevY: this.yOffset, 110 | x: this.xOffset, 111 | y: this.yOffset 112 | }; 113 | 114 | this._addStyles(`#p-gui-${this.instanceId} { 115 | width: ${this.wrapperWidth}px; 116 | max-height: ${this.maxHeight}px; 117 | transform: translate3d(${this.xOffset}px,${this.yOffset}px,0); 118 | ${ this.screenCorner.y == 'top' ? '' : 'top: auto; bottom: 0;' } 119 | ${ this.backgroundColor ? 'background: ' + this.backgroundColor + ';' : '' } 120 | opacity: ${this.opacity}; 121 | }`); 122 | } 123 | 124 | _folderConstructor(folderOptions) { 125 | this.wrapper = folderOptions.wrapper; 126 | this.isFolder = true; 127 | this.parent = folderOptions.parent; 128 | this.firstParent = folderOptions.firstParent; 129 | } 130 | 131 | _parseScreenCorner(position) { 132 | let parsedPosition = {x: 'right', y: 'top'}; 133 | 134 | if (position == undefined) return parsedPosition; 135 | else if (typeof position != 'string') console.error('[perfect-gui] Position must be a string.'); 136 | 137 | if (position.includes('left')) parsedPosition.x = 'left'; 138 | if (position.includes('bottom')) parsedPosition.y = 'bottom'; 139 | 140 | return parsedPosition; 141 | } 142 | 143 | _getScrollbarWidth(element) { 144 | if (element === document.body) { 145 | return window.innerWidth - document.documentElement.clientWidth; 146 | } else { 147 | return element.offsetWidth - element.clientWidth; 148 | } 149 | } 150 | 151 | _handleResize() { 152 | if (this.container == document.body) { 153 | this.maxHeight = window.innerHeight; 154 | } else { 155 | this.maxHeight = Math.min(this.container.clientHeight, window.innerHeight) 156 | } 157 | if (this.initMaxHeight) { 158 | this.maxHeight = Math.min(this.initMaxHeight, this.maxHeight); 159 | } 160 | this.wrapper.style.maxHeight = this.maxHeight + 'px'; 161 | 162 | if (this.hasBeenDragged) { 163 | return; 164 | } 165 | 166 | let scrollbar_width = this._getScrollbarWidth(this.container); 167 | this.xOffset = this.screenCorner.x == 'left' ? 0 : this.container.clientWidth - this.wrapperWidth - scrollbar_width; 168 | if (this.instanceId > 0) { 169 | let existingDomInstances = this.container.querySelectorAll(`.p-gui:not(#${this.wrapper.id}):not([data-dragged])`); 170 | for (let i = 0; i < existingDomInstances.length; i++) { 171 | let instanceId = parseInt(existingDomInstances[i].id.replace('p-gui-', '')); 172 | if (instanceId > this.instanceId) break; 173 | if (this.screenCorner.y == existingDomInstances[i].dataset.cornerY) { 174 | if (this.screenCorner.x == 'left' && existingDomInstances[i].dataset.cornerX == 'left') { 175 | this.xOffset += existingDomInstances[i].offsetWidth; 176 | } 177 | else if (this.screenCorner.x == 'right' && existingDomInstances[i].dataset.cornerX == 'right') { 178 | this.xOffset -= existingDomInstances[i].offsetWidth; 179 | } 180 | } 181 | } 182 | } 183 | this.position = {prevX:this.xOffset, prevY:this.yOffset, x:this.xOffset, y:this.yOffset}; 184 | this.wrapper.style.transform = `translate3d(${this.position.x}px, ${this.position.y}px, 0)`; 185 | } 186 | 187 | _addStyles(styles) { 188 | this.stylesheet.innerHTML += styles; 189 | } 190 | 191 | _addWrapper() { 192 | this.wrapper = document.createElement('div'); 193 | this.wrapper.id = 'p-gui-'+this.instanceId; 194 | this.wrapper.className = 'p-gui'; 195 | this.wrapper.setAttribute('data-lenis-prevent', ''); 196 | this.container.append(this.wrapper); 197 | 198 | this.header = document.createElement('div'); 199 | this.header.className = 'p-gui__header'; 200 | this.header.textContent = this.name; 201 | this.header.style = `${ this.backgroundColor ? 'border-color: ' + this.backgroundColor + ';' : ''}`; 202 | this.wrapper.append(this.header); 203 | 204 | const close_btn = document.createElement('div'); 205 | close_btn.className = 'p-gui__header-close'; 206 | close_btn.addEventListener('click', this.toggleClose.bind(this)); 207 | this.header.append(close_btn); 208 | } 209 | 210 | button(options, callback) { 211 | let name = ''; 212 | if (typeof options != 'string') { 213 | if (typeof options == 'object' && options?.hasOwnProperty('name')) { 214 | name = options.name == '' ? ' ' : options.name; 215 | } else { 216 | name = ' '; 217 | } 218 | } else { 219 | name = options == '' ? ' ' : options; 220 | } 221 | 222 | const tooltip = (typeof options.tooltip === 'string') ? options.tooltip : (options.tooltip === true ? name : null); 223 | 224 | this.imageContainer = null; 225 | 226 | const el = document.createElement('div'); 227 | el.className = 'p-gui__button'; 228 | el.textContent = name; 229 | if ( tooltip ) { 230 | el.setAttribute('title', tooltip); 231 | } 232 | el.addEventListener('click', () => { 233 | if (callback) { 234 | callback(); 235 | } 236 | 237 | if (this.onUpdate) { 238 | this.onUpdate(); 239 | } else if (this.isFolder && this.firstParent.onUpdate) { 240 | this.firstParent.onUpdate(); 241 | } 242 | }); 243 | this.wrapper.append(el); 244 | 245 | if (typeof options.color == 'string') { 246 | el.style.setProperty('--color-accent', options.color); 247 | el.style.setProperty('--color-accent-hover', options.hoverColor || options.color); 248 | } 249 | } 250 | 251 | image(params = {}, callback) { 252 | if (typeof params != 'object') { 253 | throw Error(`[GUI] image() first parameter must be an object. Received: ${typeof params}.`); 254 | } 255 | 256 | let path; 257 | if (typeof params.path == 'string') { 258 | path = params.path; 259 | } else { 260 | if (typeof params.path == undefined) { 261 | throw Error(`[GUI] image() path must be provided.`); 262 | } else { 263 | throw Error(`[GUI] image() path must be a string.`); 264 | } 265 | } 266 | let filename = path.replace(/^.*[\\\/]/, ''); 267 | let name; 268 | if (params.name == undefined) { 269 | name = filename; 270 | } else { 271 | name = typeof params.name == 'string' ? params.name || ' ' : ' '; 272 | } 273 | 274 | const tooltip = (typeof params.tooltip === 'string') ? params.tooltip : (params.tooltip === true ? name : null); 275 | 276 | const selected = params.selected === true; 277 | const selectionBorder = params.selectionBorder !== false; 278 | 279 | // width & height options 280 | let inline_styles = ''; 281 | if (params.width) { 282 | if (typeof params.width == 'number') { 283 | params.width += 'px'; 284 | } 285 | inline_styles += `flex: 0 0 calc(${params.width} - 5px); `; 286 | } 287 | 288 | if (params.height) { 289 | if (typeof params.height == 'number') { 290 | params.height += 'px'; 291 | } 292 | inline_styles += `height: ${params.height}; `; 293 | } 294 | 295 | if (!this.imageContainer) { 296 | this.imageContainer = document.createElement('div'); 297 | this.imageContainer.className = 'p-gui__image-container'; 298 | this.wrapper.append(this.imageContainer); 299 | } 300 | 301 | // Image button 302 | const image = document.createElement('div'); 303 | image.className = 'p-gui__image'; 304 | image.style = 'background-image: url(' + path + '); ' + inline_styles; 305 | if ( tooltip ) { 306 | image.setAttribute('title', tooltip); 307 | } 308 | this.imageContainer.append(image); 309 | 310 | if (selected && selectionBorder) { 311 | image.classList.add('p-gui__image--selected'); 312 | } 313 | 314 | // Text inside image 315 | const text = document.createElement('div'); 316 | text.className = 'p-gui__image-text'; 317 | text.textContent = name; 318 | image.append(text); 319 | 320 | image.addEventListener('click', () => { 321 | let selected_items = image.parentElement.querySelectorAll('.p-gui__image--selected'); 322 | for (let i = 0; i < selected_items.length; i++) { 323 | selected_items[i].classList.remove('p-gui__image--selected'); 324 | } 325 | if (selectionBorder) { 326 | image.classList.add('p-gui__image--selected'); 327 | } 328 | if (typeof callback == 'function') { 329 | callback({ path, text: name }); 330 | } 331 | if (this.onUpdate) { 332 | this.onUpdate(); 333 | } else if (this.isFolder && this.firstParent.onUpdate) { 334 | this.firstParent.onUpdate(); 335 | } 336 | }); 337 | 338 | return image; 339 | } 340 | 341 | slider (params = {}, callback) { 342 | const el = new Slider(this, params, callback); 343 | this.wrapper.append(el); 344 | } 345 | 346 | toggle(params = {}, callback) { 347 | if (typeof params != 'object') { 348 | throw Error(`[GUI] toggle() first parameter must be an object. Received: ${typeof params}.`); 349 | } 350 | 351 | let name = typeof params.name == 'string' ? params.name || ' ' : ' '; 352 | let isObject = false; 353 | let propReferenceIndex = null; 354 | let obj = params.obj; 355 | let prop = params.prop; 356 | let value = typeof params.value === 'boolean' ? params.value : null; 357 | 358 | // callback mode 359 | if ( value !== null ) { 360 | if (prop != undefined || obj != undefined) { 361 | console.warn(`[GUI] toggle() "obj" and "prop" parameters are ignored when a "value" parameter is used.`); 362 | } 363 | } 364 | 365 | // object-binding 366 | else if (prop != undefined && obj != undefined) { 367 | if (typeof prop != 'string') { 368 | throw Error(`[GUI] toggle() "prop" parameter must be an string. Received: ${typeof prop}.`); 369 | } 370 | if (typeof obj != 'object') { 371 | throw Error(`[GUI] toggle() "obj" parameter must be an object. Received: ${typeof obj}.`); 372 | } 373 | 374 | if (name == ' ') { 375 | name = prop; 376 | } 377 | 378 | propReferenceIndex = this.propReferences.push(obj[prop]) - 1; 379 | isObject = true; 380 | } 381 | else { 382 | if ((prop != undefined && obj == undefined) || (prop == undefined && obj == undefined)) { 383 | console.warn(`[GUI] toggle() "obj" and "prop" parameters must be used together.`); 384 | } 385 | } 386 | 387 | const tooltip = (typeof params.tooltip === 'string') ? params.tooltip : (params.tooltip === true ? name : null); 388 | 389 | this.imageContainer = null; 390 | 391 | const container = document.createElement('div'); 392 | container.textContent = name; 393 | container.className = 'p-gui__switch'; 394 | if ( tooltip ) { 395 | container.setAttribute('title', tooltip); 396 | } 397 | this.wrapper.append(container); 398 | 399 | container.addEventListener('click', (ev) => { 400 | const checkbox = ev.target.childNodes[1]; 401 | 402 | let value = true; 403 | 404 | if (checkbox.classList.contains('p-gui__switch-checkbox--active')) { 405 | value = false; 406 | } 407 | 408 | checkbox.classList.toggle('p-gui__switch-checkbox--active'); 409 | 410 | if ( isObject ) { 411 | obj[prop] = value; 412 | } 413 | 414 | else { 415 | if (typeof callback == 'function') { 416 | callback(value); 417 | } 418 | } 419 | 420 | if (this.onUpdate) { 421 | this.onUpdate(); 422 | } else if (this.isFolder && this.firstParent.onUpdate) { 423 | this.firstParent.onUpdate(); 424 | } 425 | }); 426 | 427 | let activeClass = (() => { 428 | if (!isObject) { 429 | return value ? ' p-gui__switch-checkbox--active' : ''; 430 | } else { 431 | return obj[prop] ? ' p-gui__switch-checkbox--active' : ''; 432 | } 433 | })(); 434 | 435 | const checkbox = document.createElement('div'); 436 | checkbox.className = 'p-gui__switch-checkbox' + activeClass; 437 | container.append(checkbox); 438 | 439 | if ( isObject ) { 440 | Object.defineProperty( obj, prop, { 441 | set: val => { 442 | this.propReferences[propReferenceIndex] = val; 443 | 444 | if (val) { 445 | checkbox.classList.add('p-gui__switch-checkbox--active'); 446 | } else { 447 | checkbox.classList.remove('p-gui__switch-checkbox--active'); 448 | } 449 | 450 | if (typeof callback == 'function') { 451 | callback(val); 452 | } 453 | }, 454 | get: () => { 455 | return this.propReferences[propReferenceIndex]; 456 | } 457 | }); 458 | } 459 | } 460 | 461 | list(params = {}, callback) { 462 | if (typeof params != 'object') { 463 | throw Error(`[GUI] list() first parameter must be an object. Received: ${typeof params}.`); 464 | } 465 | 466 | let name = typeof params.name == 'string' ? params.name : ' '; 467 | let isObject = false; 468 | let propReferenceIndex = null; 469 | let obj = params.obj; 470 | let prop = params.prop; 471 | let values = Array.isArray(params.values) ? params.values : null; 472 | let value; 473 | let objectValues = typeof values[0] == 'string' ? false : true; 474 | const tooltip = (typeof params.tooltip === 'string') ? params.tooltip : (params.tooltip === true ? name : null); 475 | 476 | callback = typeof callback == 'function' ? callback : null; 477 | 478 | // callback mode 479 | if ( params.value !== undefined || 480 | (params.value === undefined && obj === undefined && prop === undefined)) { 481 | if (prop != undefined || obj != undefined) { 482 | console.warn(`[GUI] list() "obj" and "prop" parameters are ignored when a "value" parameter is used.`); 483 | } 484 | 485 | value = (() => { 486 | if (!values) { 487 | return null; 488 | } 489 | if (typeof params.value == 'string') { 490 | return values.indexOf(params.value); 491 | } 492 | if (typeof params.value == 'number') { 493 | return params.value; 494 | } 495 | })(); 496 | } 497 | 498 | // object-binding mode 499 | else if (prop != undefined && obj != undefined) { 500 | if (typeof prop != 'string') { 501 | throw Error(`[GUI] list() "prop" parameter must be an string. Received: ${typeof prop}.`); 502 | } 503 | if (typeof obj != 'object') { 504 | throw Error(`[GUI] list() "obj" parameter must be an object. Received: ${typeof obj}.`); 505 | } 506 | 507 | value = (() => { 508 | if (!values) { 509 | return null; 510 | } 511 | if (typeof obj[prop] == 'string') { 512 | if ( !objectValues ) { // values is an array of strings 513 | return values.indexOf(obj[prop]); 514 | } 515 | else { // values is an array of objects 516 | return values.find(item => item.value === obj[prop]).value; 517 | } 518 | } 519 | if (typeof obj[prop] == 'number') { 520 | if ( !objectValues ) { // values is an array of strings 521 | return obj[prop]; 522 | } 523 | else { // values is an array of objects 524 | return values.find(item => item.value === obj[prop]).value; 525 | } 526 | } 527 | })(); 528 | 529 | propReferenceIndex = this.propReferences.push(obj[prop]) - 1; 530 | isObject = true; 531 | } 532 | 533 | else { 534 | if ((prop != undefined && obj == undefined) || (prop == undefined && obj == undefined)) { 535 | console.warn(`[GUI] list() "obj" and "prop" parameters must be used together.`); 536 | } 537 | } 538 | 539 | this.imageContainer = null; 540 | 541 | let container = document.createElement('div'); 542 | container.className = 'p-gui__list'; 543 | container.textContent = name; 544 | if (tooltip) { 545 | container.setAttribute('title', tooltip); 546 | } 547 | this.wrapper.append(container); 548 | 549 | let select = document.createElement('select'); 550 | container.append(select); 551 | select.className = 'p-gui__list-dropdown'; 552 | select.addEventListener('change', (ev) => { 553 | if ( isObject ) { 554 | obj[prop] = ev.target.value; 555 | } 556 | 557 | else if (callback) { 558 | callback(ev.target.value); 559 | } 560 | 561 | if (this.onUpdate) { 562 | this.onUpdate(); 563 | } else if (this.isFolder && this.firstParent.onUpdate) { 564 | this.firstParent.onUpdate(); 565 | } 566 | }); 567 | 568 | if (values) 569 | { 570 | values.forEach((item, index) => 571 | { 572 | const optionName = objectValues ? item.name : item; 573 | const optionValue = objectValues ? item.value : item; 574 | let option = document.createElement('option'); 575 | option.setAttribute('value', optionValue); 576 | option.textContent = optionName; 577 | select.append(option); 578 | 579 | if (!objectValues && value == index || objectValues && value == optionValue) { 580 | option.setAttribute('selected', ''); 581 | } 582 | }); 583 | } 584 | 585 | if ( isObject ) { 586 | Object.defineProperty( obj, prop, { 587 | set: val => { 588 | let newIndex, newValue, newObj; 589 | if (objectValues) { 590 | newObj = values.find(item => { 591 | return item.value == val; 592 | }); 593 | newValue = newObj?.value || values[0].value; 594 | newIndex = values.indexOf(newObj); 595 | } else { 596 | if (typeof val == 'string') { 597 | newIndex = values.indexOf(val); 598 | newValue = val; 599 | } 600 | if (typeof val == 'number') { 601 | newIndex = val; 602 | newValue = values[val]; 603 | } 604 | } 605 | 606 | this.propReferences[propReferenceIndex] = objectValues ? newValue : val; 607 | 608 | const previousSelection = select.querySelector('[selected]'); 609 | if ( previousSelection ) { 610 | previousSelection.removeAttribute('selected') 611 | } 612 | select.querySelectorAll('option')[newIndex].setAttribute('selected', ''); 613 | 614 | if (typeof callback == 'function') { 615 | if (objectValues) { 616 | callback(newObj, newIndex); 617 | } else { 618 | callback(newValue, newIndex); 619 | } 620 | } 621 | }, 622 | get: () => { 623 | return this.propReferences[propReferenceIndex]; 624 | } 625 | }); 626 | } 627 | } 628 | 629 | color(params = {}, callback) { 630 | if (typeof params != 'object') { 631 | throw Error(`[GUI] color() first parameter must be an object. Received: ${typeof params}.`); 632 | } 633 | 634 | let name = typeof params.name == 'string' ? params.name || ' ' : ' '; 635 | 636 | let isObject = false; 637 | let propReferenceIndex = null; 638 | let obj = params.obj; 639 | let prop = params.prop; 640 | let value; 641 | const tooltip = (typeof params.tooltip === 'string') ? params.tooltip : (params.tooltip === true ? name : null); 642 | 643 | if (typeof params.value == 'string') { 644 | if (params.value.length != 7 || params.value[0] != '#') { 645 | console.error(`[GUI] color() 'value' parameter must be an hexadecimal string in the format "#ffffff". Received: "${params.value}".`) 646 | } 647 | else { 648 | value = params.value; 649 | } 650 | } 651 | if (!value) value = '#000000'; 652 | 653 | // callback mode 654 | if ( params.value !== undefined ) { 655 | if (prop != undefined || obj != undefined) { 656 | console.warn(`[GUI] color() "obj" and "prop" parameters are ignored when a "value" parameter is used.`); 657 | } 658 | } 659 | 660 | // object-binding 661 | else if (prop != undefined && obj != undefined) { 662 | if (typeof prop != 'string') { 663 | throw Error(`[GUI] color() "prop" parameter must be an string. Received: ${typeof prop}.`); 664 | } 665 | if (typeof obj != 'object') { 666 | throw Error(`[GUI] color() "obj" parameter must be an object. Received: ${typeof obj}.`); 667 | } 668 | 669 | if (name == ' ') { 670 | name = prop; 671 | } 672 | 673 | propReferenceIndex = this.propReferences.push(obj[prop]) - 1; 674 | isObject = true; 675 | } 676 | else { 677 | if ((prop != undefined && obj == undefined) || (prop == undefined && obj == undefined)) { 678 | console.warn(`[GUI] color() "obj" and "prop" parameters must be used together.`); 679 | } 680 | } 681 | 682 | this.imageContainer = null; 683 | 684 | const container = document.createElement('div'); 685 | container.className = 'p-gui__color'; 686 | container.textContent = name; 687 | if ( tooltip ) { 688 | container.setAttribute('title', tooltip); 689 | } 690 | this.wrapper.append(container); 691 | 692 | const colorpicker = document.createElement('input'); 693 | colorpicker.className = 'p-gui__color-picker'; 694 | colorpicker.setAttribute('type', 'color'); 695 | colorpicker.value = value; 696 | container.append(colorpicker); 697 | 698 | if (typeof callback == 'function') { 699 | colorpicker.addEventListener('input', () => { 700 | if ( isObject ) { 701 | obj[prop] = colorpicker.value; 702 | } 703 | 704 | else if (typeof callback == 'function') { 705 | callback(colorpicker.value); 706 | } 707 | 708 | if (this.onUpdate) { 709 | this.onUpdate(); 710 | } else if (this.isFolder && this.firstParent.onUpdate) { 711 | this.firstParent.onUpdate(); 712 | } 713 | }); 714 | } 715 | 716 | if ( isObject ) { 717 | Object.defineProperty( obj, prop, { 718 | set: val => { 719 | this.propReferences[propReferenceIndex] = val; 720 | 721 | colorpicker.value = val; 722 | 723 | if (typeof callback == 'function') { 724 | callback(val); 725 | } 726 | }, 727 | get: () => { 728 | return this.propReferences[propReferenceIndex]; 729 | } 730 | }); 731 | } 732 | } 733 | 734 | vector2( params = {}, callback) { 735 | if (typeof params != 'object') { 736 | throw Error(`[GUI] vector2() first parameter must be an object. Received: ${typeof params}.`); 737 | } 738 | 739 | let name = typeof params.name == 'string' ? params.name || ' ' : ' '; 740 | 741 | const minX = params.x.min ?? 0; 742 | const maxX = params.x.max ?? 1; 743 | const minY = params.y.min ?? 0; 744 | const maxY = params.y.max ?? 1; 745 | const stepX = params.x.step || (maxX - minX) / 100; 746 | const stepY = params.y.step || (maxY - minY) / 100; 747 | const decimalsX = this._countDecimals(stepX); 748 | const decimalsY = this._countDecimals(stepY); 749 | 750 | const objectX = params.x.obj; 751 | const propX = params.x.prop; 752 | const propXReferenceIndex = this.propReferences.push(objectX[propX]) - 1; 753 | 754 | const objectY = params.y.obj; 755 | const propY = params.y.prop; 756 | const propYReferenceIndex = this.propReferences.push(objectY[propY]) - 1; 757 | 758 | const tooltip = (typeof params.tooltip === 'string') ? params.tooltip : (params.tooltip === true ? name : null); 759 | 760 | callback = typeof callback == 'function' ? callback : null; 761 | 762 | this.imageContainer = null; 763 | 764 | const container = document.createElement('div'); 765 | container.className = 'p-gui__vector2'; 766 | container.textContent = name; 767 | if ( tooltip ) { 768 | container.setAttribute('title', tooltip); 769 | } 770 | this.wrapper.append(container); 771 | 772 | const vector_value = document.createElement('div'); 773 | vector_value.className = 'p-gui__vector-value'; 774 | vector_value.textContent = objectX[propX] + ', ' + objectY[propY]; 775 | container.append(vector_value); 776 | 777 | const area = document.createElement('div'); 778 | area.className = 'p-gui__vector2-area'; 779 | container.append(area); 780 | area.addEventListener('click', evt => { 781 | const newX = parseFloat(this._mapLinear(evt.offsetX, 0, area.clientWidth, minX, maxX)); 782 | const newY = parseFloat(this._mapLinear(evt.offsetY, 0, area.clientHeight, maxY, minY)); 783 | objectX[propX] = newX.toFixed(decimalsX); 784 | objectY[propY] = newY.toFixed(decimalsY); 785 | 786 | if (callback) { 787 | callback(objectX[propX], objectX[propY]); 788 | } 789 | 790 | if (this.onUpdate) { 791 | this.onUpdate(); 792 | } else if (this.isFolder && this.firstParent.onUpdate) { 793 | this.firstParent.onUpdate(); 794 | } 795 | }); 796 | 797 | let pointer_is_down = false; 798 | area.addEventListener('pointerdown', (evt) => { 799 | pointer_is_down = true; 800 | }); 801 | area.addEventListener('pointerup', () => { 802 | pointer_is_down = false; 803 | }); 804 | area.addEventListener('pointermove', (evt) => { 805 | if (pointer_is_down) { 806 | const newX = parseFloat(this._mapLinear(evt.offsetX, 0, area.clientWidth, minX, maxX)); 807 | const newY = parseFloat(this._mapLinear(evt.offsetY, 0, area.clientHeight, maxY, minY)); 808 | objectX[propX] = newX.toFixed(decimalsX); 809 | objectY[propY] = newY.toFixed(decimalsY); 810 | 811 | if (callback) { 812 | callback(objectX[propX], objectX[propY]); 813 | } 814 | 815 | if (this.onUpdate) { 816 | this.onUpdate(); 817 | } else if (this.isFolder && this.firstParent.onUpdate) { 818 | this.firstParent.onUpdate(); 819 | } 820 | } 821 | }); 822 | 823 | const line_x = document.createElement('div'); 824 | line_x.className = 'p-gui__vector2-line p-gui__vector2-line-x'; 825 | area.append(line_x); 826 | 827 | const line_y = document.createElement('div'); 828 | line_y.className = 'p-gui__vector2-line p-gui__vector2-line-y'; 829 | area.append(line_y); 830 | 831 | const dot = document.createElement('div'); 832 | dot.className = 'p-gui__vector2-dot'; 833 | area.append(dot); 834 | 835 | dot.style.left = this._mapLinear(objectX[propX], minX, maxX, 0, area.clientWidth) + 'px'; 836 | dot.style.top = this._mapLinear(objectY[propY], minY, maxY, area.clientHeight, 0) + 'px'; 837 | 838 | Object.defineProperty( objectX, propX, { 839 | set: val => { 840 | this.propReferences[propXReferenceIndex] = val; 841 | dot.style.left = this._mapLinear(val, minX, maxX, 0, area.clientWidth) + 'px'; 842 | vector_value.textContent = String( val ) + ', ' + objectY[propY]; 843 | }, 844 | get: () => { 845 | return this.propReferences[propXReferenceIndex]; 846 | } 847 | }); 848 | 849 | Object.defineProperty( objectY, propY, { 850 | set: val => { 851 | this.propReferences[propYReferenceIndex] = val; 852 | dot.style.top = this._mapLinear(val, minY, maxY, area.clientHeight, 0) + 'px'; 853 | vector_value.textContent = objectX[propX] + ', ' + String( val ); 854 | }, 855 | get: () => { 856 | return this.propReferences[propYReferenceIndex]; 857 | } 858 | }); 859 | } 860 | 861 | folder(options = {}) { 862 | let closed = typeof options.closed == 'boolean' ? options.closed : false; 863 | let name = options.name || ''; 864 | let color = options.color || null; 865 | let maxHeight = options.maxHeight || null; 866 | 867 | this.imageContainer = null; 868 | 869 | let className = 'p-gui__folder'; 870 | 871 | if (this.folders.length == 0) { 872 | className += ' p-gui__folder--first'; 873 | } 874 | 875 | if (closed) { 876 | className += ' p-gui__folder--closed'; 877 | } 878 | 879 | let container_style = color ? `background-color: ${color};` : ''; 880 | container_style += maxHeight ? `max-height: ${maxHeight}px;` : ''; 881 | 882 | const container = document.createElement('div'); 883 | container.className = className; 884 | container.style = container_style; 885 | this.wrapper.append(container); 886 | 887 | const folderHeader = document.createElement('div'); 888 | folderHeader.innerHTML = `${name}`; 889 | folderHeader.className = 'p-gui__folder-header'; 890 | container.append(folderHeader); 891 | folderHeader.addEventListener('click', () => { 892 | container.classList.toggle('p-gui__folder--closed'); 893 | }); 894 | 895 | let folder = new GUI({isFolder: true, folderOptions: { 896 | wrapper: container, 897 | parent: this, 898 | firstParent: this.firstParent 899 | }}); 900 | this.folders.push(folder); 901 | return folder; 902 | } 903 | 904 | _makeDraggable() { 905 | var that = this; 906 | this.header.addEventListener('pointerdown', dragMouseDown); 907 | this.header.addEventListener('pointerup', dragMouseUp); 908 | 909 | function dragMouseDown(ev) { 910 | ev.preventDefault(); 911 | 912 | that.position.initX = that.position.x; 913 | that.position.initY = that.position.y; 914 | 915 | that.position.prevX = ev.clientX; 916 | that.position.prevY = ev.clientY; 917 | 918 | document.addEventListener('pointermove', dragElement); 919 | } 920 | 921 | function dragElement(ev) { 922 | ev.preventDefault(); 923 | if (!that.hasBeenDragged) { 924 | that.hasBeenDragged = true; 925 | that.wrapper.setAttribute('data-dragged', 'true') 926 | } 927 | 928 | that.position.x = that.position.initX + ev.clientX - that.position.prevX; 929 | that.position.y = that.position.initY + ev.clientY - that.position.prevY; 930 | 931 | that.wrapper.style.transform = "translate3d("+that.position.x + "px,"+that.position.y + "px,0)"; 932 | } 933 | 934 | function dragMouseUp(ev) { 935 | document.removeEventListener('pointermove', dragElement); 936 | } 937 | } 938 | 939 | toggleClose() { 940 | this.closed = !this.closed; 941 | 942 | if (this.closed) { 943 | this.previousInnerScroll = this.wrapper.scrollTop; 944 | this.wrapper.scrollTo(0,0); 945 | } else { 946 | this.wrapper.scrollTo(0,this.previousInnerScroll); 947 | } 948 | 949 | this.wrapper.classList.toggle('p-gui--collapsed'); 950 | } 951 | 952 | kill() { 953 | this.wrapper.remove(); 954 | } 955 | 956 | _mapLinear( x, a1, a2, b1, b2 ) { 957 | return b1 + ( x - a1 ) * ( b2 - b1 ) / ( a2 - a1 ); 958 | } 959 | 960 | _countDecimals(num) { 961 | // Convert the number to a string 962 | const numStr = num.toString(); 963 | 964 | // Find the position of the decimal point 965 | const decimalIndex = numStr.indexOf('.'); 966 | 967 | // If there is no decimal point, return 0 968 | if (decimalIndex === -1) { 969 | return 0; 970 | } 971 | 972 | // Calculate the number of digits after the decimal point 973 | const decimalPlaces = numStr.length - decimalIndex - 1; 974 | 975 | return decimalPlaces; 976 | } 977 | } -------------------------------------------------------------------------------- /src/styles/_button.css.js: -------------------------------------------------------------------------------- 1 | export default /* css */ ` 2 | .p-gui__button { 3 | background: var(--color-accent); 4 | text-align: center; 5 | color: white; 6 | border: none; 7 | border: 1px solid transparent; 8 | box-sizing: border-box; 9 | transition: var(--transition) background, var(--transition) border-color; 10 | } 11 | 12 | .p-gui__button:hover { 13 | background: var(--color-accent-hover); 14 | color: var(--color-text-light); 15 | border-color: rgba(255, 255, 255, 0.2); 16 | } 17 | 18 | .p-gui__folder .p-gui__button { 19 | margin-inline: 0; 20 | } 21 | `; -------------------------------------------------------------------------------- /src/styles/_color.css.js: -------------------------------------------------------------------------------- 1 | export default /* css */ ` 2 | .p-gui__color { 3 | cursor: default; 4 | color: var(--color-text-dark); 5 | transition: var(--transition) color; 6 | } 7 | 8 | .p-gui__color:hover { 9 | color: var(--color-text-light); 10 | } 11 | 12 | .p-gui__color-picker { 13 | position: absolute; 14 | right: 5px; 15 | top: 0; 16 | bottom: 0; 17 | margin: auto; 18 | height: 21px; 19 | cursor: pointer; 20 | border-radius: 3px; 21 | border: 1px solid var(--color-border-2); 22 | outline: none; 23 | -webkit-appearance: none; 24 | padding: 0; 25 | background-color: transparent; 26 | border: 1px solid #222222; 27 | overflow: hidden; 28 | } 29 | 30 | .p-gui__color-picker::-webkit-color-swatch-wrapper { 31 | padding: 0; 32 | } 33 | .p-gui__color-picker::-webkit-color-swatch { 34 | border: none; 35 | } 36 | `; -------------------------------------------------------------------------------- /src/styles/_folder.css.js: -------------------------------------------------------------------------------- 1 | export default /* css */ ` 2 | .p-gui__folder { 3 | width: 100%; 4 | position: relative; 5 | background: #434343; 6 | overflow: auto; 7 | margin-bottom: 2px; 8 | display: flex; 9 | flex-wrap: wrap; 10 | border: 1px solid grey; 11 | padding: 0 3px 0 3px; 12 | border-radius: var(--main-border-radius); 13 | box-sizing: border-box; 14 | } 15 | 16 | .p-gui__folder--first { 17 | margin-top: 0; 18 | } 19 | 20 | .p-gui__folder--closed { 21 | height: 32px; 22 | overflow: hidden; 23 | } 24 | 25 | .p-gui__folder-header { 26 | padding: 10px 5px; 27 | background-color: rgba(0, 0, 0, .5); 28 | color: white; 29 | cursor: pointer; 30 | width: 100%; 31 | margin: 0 -2px 2px -3px; 32 | } 33 | 34 | .p-gui__folder-header:hover { 35 | background-color: rgba(0, 0, 0, .75); 36 | } 37 | 38 | .p-gui__folder-arrow { 39 | width: 8px; 40 | height: 8px; 41 | display: inline-block; 42 | background-image: url(); 43 | background-size: contain; 44 | margin-right: 5px; 45 | transform: rotate(90deg) 46 | } 47 | 48 | .p-gui__folder--closed .p-gui__folder-arrow { 49 | transform: rotate(0deg); 50 | } 51 | `; -------------------------------------------------------------------------------- /src/styles/_image.css.js: -------------------------------------------------------------------------------- 1 | export default /* css */ ` 2 | .p-gui__image-container { 3 | width: 100%; 4 | padding: 3px; 5 | display: flex; 6 | justify-content: flex-start; 7 | flex-wrap: wrap; 8 | box-sizing: border-box; 9 | } 10 | 11 | .p-gui__image { 12 | background-size: cover; 13 | cursor: pointer; 14 | position: relative; 15 | margin: 1px 2.5px 19px 2.5px; 16 | border-radius: var(--main-border-radius); 17 | flex: 0 0 calc(33.333% - 5px); 18 | height: 90px; 19 | background-position: center; 20 | color: var(--color-text-dark); 21 | transition: var(--transition) color; 22 | } 23 | 24 | .p-gui__image:hover { 25 | color: var(--color-text-light); 26 | } 27 | 28 | .p-gui__image::after { 29 | position: absolute; 30 | top: 0; 31 | left: 0; 32 | width: 100%; 33 | height: 100%; 34 | content: ''; 35 | border: 1px solid transparent; 36 | box-sizing: border-box; 37 | border-radius: var(--main-border-radius); 38 | transition: var(--transition) border-color; 39 | } 40 | .p-gui__image--selected::after { 41 | border-color: #06FF89; 42 | } 43 | 44 | .p-gui__image-text { 45 | position: absolute; 46 | bottom: -15px; 47 | text-shadow: 0 -1px 0 #111; 48 | white-space: nowrap; 49 | width: 100%; 50 | overflow: hidden; 51 | text-overflow: ellipsis; 52 | 53 | } 54 | `; -------------------------------------------------------------------------------- /src/styles/_list.css.js: -------------------------------------------------------------------------------- 1 | export default /* css */ ` 2 | .p-gui__list { 3 | cursor: default; 4 | color: var(--color-text-dark); 5 | transition: var(--transition) color; 6 | } 7 | 8 | .p-gui__list:hover { 9 | color: var(--color-text-light); 10 | } 11 | 12 | .p-gui__list-dropdown { 13 | background: rgba(255, 255, 255,.05); 14 | color: white; 15 | padding: 0 12px 0 5px; 16 | top: 0px; 17 | } 18 | 19 | .p-gui__list-dropdown { 20 | position: absolute; 21 | right: 5px; 22 | top: 0; 23 | bottom: 0; 24 | margin: auto; 25 | height: 21px; 26 | cursor: pointer; 27 | border-radius: 3px; 28 | border: 1px solid var(--color-border-2); 29 | outline: none; 30 | } 31 | 32 | .p-gui__list-dropdown:hover { 33 | background: rgba(255, 255, 255, .1); 34 | } 35 | `; -------------------------------------------------------------------------------- /src/styles/_slider.css.js: -------------------------------------------------------------------------------- 1 | export default /* css */ ` 2 | .p-gui__slider { 3 | position: relative; 4 | min-height: 14px; 5 | display: flex; 6 | align-items: center; 7 | justify-content: space-between; 8 | gap: 10px; 9 | color: var(--color-text-dark); 10 | transition: color var(--transition); 11 | } 12 | 13 | .p-gui__slider:hover { 14 | color: var(--color-text-light); 15 | } 16 | 17 | .p-gui__slider-name { 18 | width: 50%; 19 | text-overflow: ellipsis; 20 | overflow: hidden; 21 | } 22 | 23 | .p-gui__slider-ctrl { 24 | -webkit-appearance: none; 25 | padding: 0; 26 | font: inherit; 27 | outline: none; 28 | box-sizing: border-box; 29 | cursor: pointer; 30 | position: relative; 31 | right: 0; 32 | height: 14px; 33 | margin: 0 0 0 auto; 34 | width: 37%; 35 | } 36 | 37 | .p-gui__slider-bar { 38 | position: absolute; 39 | top: 50%; 40 | left: 0; 41 | height: 2px; 42 | background: rgba(255, 255, 255, .2); 43 | width: 100%; 44 | transform: translateY(-50%); 45 | } 46 | 47 | .p-gui__slider-filling { 48 | position: absolute; 49 | top: -25%; 50 | left: 0; 51 | height: 150%; 52 | background: var(--color-accent); 53 | width: 0; 54 | } 55 | 56 | .p-gui__slider:hover .p-gui__slider-filling { 57 | background: var(--color-accent-hover); 58 | } 59 | 60 | .p-gui__slider-handle { 61 | width: 15px; 62 | height: 8px; 63 | position: absolute; 64 | top: 50%; 65 | left: 0; 66 | border-radius: 2px; 67 | transform: translate(-50%, -50%); 68 | pointer-events: none; 69 | background: var(--color-text-dark); 70 | box-shadow: 0 0 2px rgba(0, 0, 0, .5); 71 | } 72 | 73 | .p-gui__slider:hover .p-gui__slider-handle { 74 | background: var(--color-text-light); 75 | } 76 | 77 | .p-gui__slider-value { 78 | display: inline-block; 79 | right: 7px; 80 | width: 13%; 81 | border: none; 82 | color: white; 83 | border-radius: 2px; 84 | background: rgba(255, 255, 255, 0.1); 85 | padding: 2px 4px; 86 | color: inherit; 87 | } 88 | 89 | .p-gui__slider-value:focus { 90 | outline: none; 91 | } 92 | `; -------------------------------------------------------------------------------- /src/styles/_switch.css.js: -------------------------------------------------------------------------------- 1 | export default /* css */ ` 2 | .p-gui__switch { 3 | background: rgba(255, 255, 255, .05); 4 | color: var(--color-text-dark); 5 | transition: var(--transition) background, var(--transition) color; 6 | } 7 | 8 | .p-gui__switch:hover { 9 | background: rgba(255, 255, 255, .1); 10 | color: var(--color-text-light); 11 | } 12 | 13 | .p-gui__folder .p-gui__switch { 14 | margin-inline: 0; 15 | } 16 | 17 | .p-gui__switch-checkbox { 18 | width: 5px; 19 | height: 5px; 20 | background-color: rgba(0, 0, 0, .5); 21 | border: 1px solid grey; 22 | position: absolute; 23 | top: 0; 24 | right: 10px; 25 | bottom: 0; 26 | margin: auto; 27 | border-radius: 50%; 28 | pointer-events: none; 29 | } 30 | 31 | .p-gui__switch-checkbox--active { 32 | background-color: #00ff89; 33 | box-shadow: 0 0 7px #00ff89; 34 | } 35 | `; -------------------------------------------------------------------------------- /src/styles/_vector2.css.js: -------------------------------------------------------------------------------- 1 | export default /* css */ ` 2 | .p-gui__vector2 { 3 | background: transparent; 4 | aspect-ratio: 1; 5 | padding-bottom: 0; 6 | color: var(--color-text-dark); 7 | } 8 | 9 | .p-gui__vector2:hover { 10 | color: var(--color-text-light); 11 | } 12 | 13 | .p-gui__vector2-area { 14 | position: relative; 15 | background: rgba(0, 0, 0, .3); 16 | margin-top: 8px; 17 | width: 100%; 18 | height: calc(100% - 28px); 19 | touch-action: none; 20 | } 21 | 22 | .p-gui__vector2-line { 23 | position: absolute; 24 | background: white; 25 | opacity: .3; 26 | pointer-events: none; 27 | } 28 | 29 | .p-gui__vector2-line-x { 30 | width: 100%; 31 | height: 1px; 32 | left: 0; 33 | top: 50%; 34 | transform: translateY(-50%); 35 | } 36 | 37 | .p-gui__vector2-line-y { 38 | width: 1px; 39 | height: 100%; 40 | top: 0; 41 | left: 50%; 42 | transform: translateX(-50%); 43 | } 44 | 45 | .p-gui__vector2-dot { 46 | position: absolute; 47 | top: 0; 48 | left: 0; 49 | width: 8px; 50 | height: 8px; 51 | border-radius: 50%; 52 | background: #d5d5d5; 53 | border: 2px solid #ff9999; 54 | transform: translate(-50%, -50%); 55 | pointer-events: none; 56 | } 57 | 58 | .p-gui__vector-value { 59 | display: inline-block; 60 | right: 7px; 61 | position: absolute; 62 | } 63 | `; -------------------------------------------------------------------------------- /src/styles/styles.js: -------------------------------------------------------------------------------- 1 | import _button from "./_button.css.js" 2 | import _slider from "./_slider.css.js" 3 | import _list from "./_list.css.js" 4 | import _switch from "./_switch.css.js" 5 | import _color from "./_color.css.js" 6 | import _vector2 from "./_vector2.css.js" 7 | import _image from "./_image.css.js" 8 | import _folder from "./_folder.css.js" 9 | 10 | /** 11 | * JS instead of CSS to avoid 12 | * depending on a css loader 13 | */ 14 | 15 | export default function( position_type ) { 16 | return /* css */` 17 | .p-gui { 18 | --main-border-radius: 5px; 19 | --color-bg: #121212; 20 | --color-border: #484848; 21 | --color-border-2: rgba(255,255,255,.1); 22 | --color-text-light: #ffffff; 23 | --color-text-dark: #bbbbbb; 24 | --color-accent: #1681ca; 25 | --color-accent-hover: #218fda; 26 | --transition: .1s linear; 27 | 28 | position: ${ position_type }; 29 | top: 0; 30 | left: 0; 31 | transform: translate3d(0,0,0); 32 | padding-top: 21px; 33 | padding-inline: 3px; 34 | background: var(--color-bg); 35 | display: block; 36 | justify-content: center; 37 | flex-wrap: wrap; 38 | font-family: "Arial Rounded MT Bold", Arial, sans-serif; 39 | width: 290px; 40 | overflow: auto; 41 | box-shadow: 0 0 2px black; 42 | box-sizing: border-box; 43 | z-index: 99999; 44 | user-select: none; 45 | border-bottom-right-radius: 3px; 46 | border-bottom-left-radius: 3px; 47 | cursor: auto; 48 | border-radius: var(--main-border-radius); 49 | border: 1px solid var(--color-border); 50 | line-height: normal; 51 | transition: var(--transition) opacity; 52 | } 53 | 54 | .p-gui:hover { 55 | opacity: 1!important; 56 | } 57 | 58 | .p-gui * { 59 | font-size: 11px; 60 | } 61 | 62 | .p-gui::-webkit-scrollbar, 63 | .p-gui *::-webkit-scrollbar { 64 | width: 10px; 65 | } 66 | 67 | .p-gui::-webkit-scrollbar-track, 68 | .p-gui *::-webkit-scrollbar-track { 69 | background: #2f2f2f; 70 | border-radius: 3px; 71 | } 72 | 73 | .p-gui::-webkit-scrollbar-thumb, 74 | .p-gui *::-webkit-scrollbar-thumb { 75 | background: #757576; 76 | border-radius: 10px; 77 | box-sizing: border-box; 78 | border: 1px solid #2f2f2f; 79 | } 80 | 81 | .p-gui--collapsed { 82 | height: 0; 83 | padding: 21px 10px 0 10px; 84 | overflow: hidden; 85 | } 86 | 87 | .p-gui__header { 88 | position: absolute; 89 | top: 0; 90 | left: 0; 91 | width: 100%; 92 | height: 20px; 93 | background-color: rgba(0, 0, 0, .8); 94 | cursor: grab; 95 | color: grey; 96 | font-size: 10px; 97 | line-height: 20px; 98 | padding-left: 12px; 99 | box-sizing: border-box; 100 | touch-action: none; 101 | } 102 | 103 | .p-gui__header-close { 104 | width: 20px; 105 | height: 20px; 106 | position: absolute; 107 | top: 0; 108 | right: 5px; 109 | cursor: pointer; 110 | background-image: url(); 111 | background-size: 50% 50%; 112 | background-position: center; 113 | background-repeat: no-repeat; 114 | } 115 | 116 | .p-gui--collapsed .p-gui__header-close { 117 | background-image: url(); 118 | } 119 | 120 | .p-gui__slider, 121 | .p-gui__button, 122 | .p-gui__switch, 123 | .p-gui__list, 124 | .p-gui__vector2, 125 | .p-gui__color { 126 | width: 100%; 127 | padding: 7px; 128 | cursor: pointer; 129 | position: relative; 130 | box-sizing: border-box; 131 | margin-block: 3px; 132 | border: 1px solid var(--color-border-2); 133 | border-radius: var(--main-border-radius); 134 | transition: var(--transition) border-color; 135 | } 136 | 137 | .p-gui__slider:hover, 138 | .p-gui__button:hover, 139 | .p-gui__switch:hover, 140 | .p-gui__list:hover, 141 | .p-gui__vector2:hover, 142 | .p-gui__color:hover { 143 | border-color: rgba(255,255,255,.2); 144 | } 145 | 146 | ${ _button } 147 | 148 | ${ _image } 149 | 150 | ${ _list } 151 | 152 | ${ _switch } 153 | 154 | ${ _slider } 155 | 156 | ${ _color } 157 | 158 | ${ _vector2 } 159 | 160 | ${ _folder } 161 | `}; -------------------------------------------------------------------------------- /vite.config.js: -------------------------------------------------------------------------------- 1 | import { resolve } from "path"; 2 | import { defineConfig } from "vite"; 3 | 4 | export default defineConfig({ 5 | build: { 6 | lib: { 7 | entry: resolve(__dirname, "src/index.js"), 8 | name: "Perfect GUI" 9 | }, 10 | minify: true 11 | }, 12 | }); 13 | --------------------------------------------------------------------------------