muigui ⍺
What / Why?
Ranges:
Like <input type="range">
, manipulates a number with optional conversions.
const s = {
value1: 40,
value2: 50,
value3: 80,
value4: 60,
value5: 65,
angleRad: Math.PI * 0.25,
};
const gui = new GUI();
gui.add(s, 'value1', 0, 100); // 0 to 100
gui.add(s, 'value2', 0, 100, 1); // 0 to 100 by in steps of 1
gui.add(s, 'value3', 0, 100, 5); // 0 to 100 by steps of 5
GUI.makeMinMaxPair(gui, s, 'value4', 'value5', {
min: 0, max: 100, minRange: 5,
});
const degToRad = d => d * Math.PI / 180;
const radToDeg = r => r * 180 / Math.PI;
gui.add(s, 'angleRad', {
min: -360, max: 360, step: 1,
converters: {
to: radToDeg,
from: v => [true, degToRad(v)]},
});
Dropdowns:
Like <select>
, lets you pick one of many options.
const s = {
byIndex: 2,
byString: 'dog',
byKeyValue: 0.5,
byKVArray: 2,
};
const gui = new GUI();
gui.add(s, 'byIndex', ['apple', 'banana', 'cumquat']);
gui.add(s, 'byString', ['bird', 'cat', 'dog']);
gui.add(s, 'byKeyValue', [['Slow', 0.1], ['Medium', 0.5], ['Fast', 1.0]]);
gui.add(s, 'byKVArray', {keyValues: {'Loafers': 0, 'Sandals': 1, 'Sneakers': 2}});
Checkbox:
Sets a boolean
const s = {
on: true,
};
const gui = new GUI();
gui.add(s, 'on');
Colors:
Select a color in various formats
const s = {
byHashHex6: '#569AEF',
byHex6: 'EF569A',
byHashHex3: '#F88',
byHex3: '8F8',
byCSSRgb: 'rgb(170,68,240)',
byCSSHsl: 'hsl(170,100%,50%)',
byUint24: 0xFEA956,
byUint8: [255, 192, 255],
byUint8Array: new Uint8Array([75, 150, 225]),
byFloat32Array: new Float32Array([0.9, 0.7, 0.5]),
byNumberArray: [0.2, 0.9, 0.5],
byRGBObject: {r: 0, g: 0, b: 1},
byHashHex8: '#5438a180',
byUint32: 0xEF569A80,
byCSSRgba: 'rgba(64, 128, 255, 0.25)',
byCSSHslAlpha: 'hsl(180 100% 50% / 0.75)',
};
const gui = new GUI();
gui.addColor(s, 'byHashHex6');
gui.addColor(s, 'byHex6');
gui.addColor(s, 'byHashHex3');
gui.addColor(s, 'byHex3');
gui.addColor(s, 'byCSSRgb');
gui.addColor(s, 'byCSSHsl');
gui.addColor(s, 'byUint24', {format: 'uint32-rgb'});
gui.addColor(s, 'byUint8', {format: 'uint8-rgb'});
gui.addColor(s, 'byUint8Array');
gui.addColor(s, 'byFloat32Array');
gui.addColor(s, 'byNumberArray');
gui.addColor(s, 'byRGBObject');
gui.addLabel('with alpha');
gui.addColor(s, 'byHashHex8');
gui.addColor(s, 'byUint32', {format: 'uint32-rgba'});
gui.addColor(s, 'byCSSRgba');
gui.addColor(s, 'byCSSHslAlpha');
Canvas:
Lets you draw something
const waveData = new Array(100).fill(0);
const waveData2 = new Array(100).fill(0);
const gui = new GUI();
helpers.graph(
gui.addCanvas('wave').canvas,
waveData,
{
min: -2,
max: +2,
interval: 200,
});
helpers.graph(
gui.addCanvas('wave 2').canvas,
waveData2,
{
min: -2,
max: +2,
interval: 10,
});
Returning just a canvas means you can use whatever API you want (2d, WebGL, WebGPU)
const hsl = (h, s, l) =>
`hsl(${h * 360 | 0}, ${s * 100}%, ${l * 100 | 0}%)`;
const gui = new GUI();
const canvas = gui.addCanvas('funky').canvas;
const ctx = canvas.getContext('2d');
function render(t) {
t *= 0.0001;
canvas.width = canvas.width;
ctx.translate(canvas.width / 2, canvas.height / 2);
const size = Math.max(canvas.width, canvas.height);
ctx.rotate(t);
for (let i = 0; i < 16; ++i) {
ctx.fillStyle = hsl(i / 48 + t * 0.2, 1, i % 2 ? 0.5 : 0.25);
ctx.fillRect(-size, -size, size * 2, size * 2);
ctx.scale(0.8, 0.8);
ctx.rotate(Math.sin(t * 3 + i * 0.1) * 0.2);
}
requestAnimationFrame(render);
}
requestAnimationFrame(render);
And you can use whatever Library (three.js, babylon.js, pixi.js) you want.
// three.js example
import {model} from '/examples/js/model.js';
const gui = new GUI();
const s = model(gui.addCanvas('model').canvas);
gui.addColor(s.material, 'color');
gui.add(s.material, 'shininess', 0, 300);
gui.addColor(s.light, 'color').name('light');
gui.add(s.light, 'intensity', 0, 5, 0.01);
gui.add(s.camera, 'fov', 1, 179);
Button:
Like <button type="button">
, calls a function
function doIt() { log('doIt'); }
function doThat() { log('doThat'); }
const s = {
clickMe() { log("clicked"); },
clickMeToo: () => { log("clicked too"); },
doIt,
doSomething: doThat,
};
const gui = new GUI();
gui.add(s, 'clickMe');
gui.add(s, 'clickMeToo');
gui.add(s, 'doIt');
gui.add(s, 'doSomething');
gui.addButton('easy', _ => log('e-z'));
Label:
Displays text. Can be used to label sections or to display info
const gui = new GUI();
gui.addLabel('Seconds:');
const output = gui.addLabel('');
setInterval(() => {
const secs = new Date().getSeconds();
const s = secs.toString().padStart(2, '0');
output.text(`---\n${s}\n---`);
}, 1000);
A small utility to monitor a value
const s = {
frameCount: 0,
stats: {
mode: 'searching',
speed: 12,
},
};
const gui = new GUI();
helpers.monitor(gui.addLabel('Seconds:'), s, 'frameCount');
helpers.monitor(gui.addLabel('Status:'), s, 'stats');
Text:
Like <input type="text">
, manipulates a string.
const s = {
name: "Sandy",
};
const gui = new GUI();
gui.add(s, 'name');
TextNumbers:
Like <input type="number">
, manipulates a number via text editing with optional conversions.
const s = {
value: 123,
angleDeg: 45,
angleRad: Math.PI,
};
const degToRad = d => d * Math.PI / 180;
const radToDeg = r => r * 180 / Math.PI;
const gui = new GUI();
gui.add(new TextNumber(s, 'value'));
gui.add(new TextNumber(s, 'angleRad', {
converters: {
to: radToDeg,
from: v => [true, degToRad(v)]},
step: 1,
}));
gui.add(new TextNumber(s, 'angleDeg', {
converters: {
to: degToRad,
from: v => [true, radToDeg(v)]},
step: 0.001,
}));
Respond to changes:
Get called when the user changes a value.
const s = {
count: 123,
name: 'Buffy',
speed: 45,
};
const gui = new GUI();
gui.add(s, 'count').onChange(logCount);
gui.add(s, 'name').onChange(logName);
gui.add(s, 'speed', 0, 100).onChange(logSpeed);
gui.onChange(logAll); // or folder
function logCount(v) { log('count:', v); }
function logName(v) { log('name:', v); }
function logSpeed(v) { log('speed:', v); }
function logAll() { log(JSON.stringify(s)); }
Listen:
Listens for changes. (auto update)
const s = {
amount: 5,
name: 'Jim',
pet: 'cat',
randomize,
};
function randomize() {
s.amount = Math.random() * 50;
s.name = randElem(['Jane', 'Lee', 'Anne']);
s.pet = randElem(pets);
}
const pets = ['cat', 'dog', 'bird'];
const gui = new GUI();
gui.add(s, 'amount', 0, 50).listen();
gui.add(s, 'name').listen();
gui.add(s, 'pet', pets).listen();
gui.add(s, 'randomize');
updateDisplay:
Updates the UI from current value(s).
const s = {
amount: 5,
name: 'Jim',
pet: 'cat',
};
function randomize() {
s.amount = Math.random() * 50;
s.name = randElem(['Jane', 'Lee', 'Anne']);
s.pet = randElem(pets);
}
const pets = ['cat', 'dog', 'bird'];
const gui = new GUI();
const a = gui.add(s, 'amount', 0, 50);
const n = gui.add(s, 'name');
const p = gui.add(s, 'pet', pets);
gui.addButton('randomize', randomize);
gui.addButton('update amount', _ => a.updateDisplay());
gui.addButton('update name', _ => n.updateDisplay());
gui.addButton('update pet', _ => p.updateDisplay());
gui.addButton('update all', _ => gui.updateDisplay());
Folder:
A way to make collapsable sections.
const position = {
x: 5,
y: 20,
z: -15,
};
const rotation = {
x: 45,
y: 90,
z: -15,
};
const gui = new GUI();
const p = gui.addFolder('position');
p.add(position, 'x', -100, 100);
p.add(position, 'y', -100, 100);
p.add(position, 'z', -100, 100);
const r = gui.addFolder('rotation');
r.add(rotation, 'x', -100, 100);
r.add(rotation, 'y', -100, 100);
r.add(rotation, 'z', -100, 100);
Float Theme:
Useful when you want the UI over something
const gui = new GUI();
gui.setTheme('float');
gui.add(settings, 'period1', 0.1, 20);
gui.add(settings, 'period2', 0.1, 20);
gui.add(settings, 'p1', 0.1, 20);
gui.add(settings, 'p2', 0.1, 20);
gui.add(settings, 'heightMult', 1, 10);
gui.addColor(settings, 'baseColor');
Form Theme:
const s = {
name: "Jane Cheng",
address1: "B 1, No. 5, Xuzhou R",
address2: "Taipei 100218",
email: "jane_c@notreally.notcom",
receipt: true,
currency: '$',
};
const gui = new GUI();
gui.setTheme('form');
gui.add(s, 'name');
gui.add(s, 'address1');
gui.add(s, 'address2');
gui.add(s, 'email');
gui.add(s, 'receipt');
gui.add(
s, 'currency',
['$', '¥', '€', '£', '₣']);
gui.addButton('submit', submit);
//