-
Notifications
You must be signed in to change notification settings - Fork 2
/
tiny-graphics-widgets.js
431 lines (373 loc) · 25.2 KB
/
tiny-graphics-widgets.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
// This file defines a lot of panels that can be placed on websites to create interactive graphics programs that use tiny-graphics.js.
import {tiny} from './tiny-graphics.js';
const { color, Scene } = tiny; // Pull these names into this module's scope for convenience.
export const widgets = {};
const Canvas_Widget = widgets.Canvas_Widget =
class Canvas_Widget
{ // **Canvas_Widget** embeds a WebGL demo onto a website in place of the given placeholder document
// element. It creates a WebGL canvas and loads onto it any initial Scene objects in the
// arguments. Optionally spawns a Text_Widget and Controls_Widget for showing more information
// or interactive UI buttons, divided into one panel per each loaded Scene. You can use up to
// 16 Canvas_Widgets; browsers support up to 16 WebGL contexts per page.
constructor( element, initial_scenes, options = {} )
{ this.element = element;
const defaults = { show_canvas: true, make_controls: true, show_explanation: true,
make_editor: false, make_code_nav: true };
if( initial_scenes && initial_scenes[0] )
Object.assign( options, initial_scenes[0].widget_options );
Object.assign( this, defaults, options )
const rules = [ ".canvas-widget { width: 1080px; background: DimGray; margin:auto }",
".canvas-widget canvas { width: 1080px; height: 600px; margin-bottom:-3px }" ];
if( document.styleSheets.length == 0 ) document.head.appendChild( document.createElement( "style" ) );
for( const r of rules ) document.styleSheets[document.styleSheets.length - 1].insertRule( r, 0 )
// Fill in the document elements:
if( this.show_explanation )
{ this.embedded_explanation_area = this.element.appendChild( document.createElement( "div" ) );
this.embedded_explanation_area.className = "text-widget";
}
const canvas = this.element.appendChild( document.createElement( "canvas" ) );
if( this.make_controls )
{ this.embedded_controls_area = this.element.appendChild( document.createElement( "div" ) );
this.embedded_controls_area.className = "controls-widget";
}
if( this.make_code_nav )
{ this.embedded_code_nav_area = this.element.appendChild( document.createElement( "div" ) );
this.embedded_code_nav_area.className = "code-widget";
}
if( this.make_editor )
{ this.embedded_editor_area = this.element.appendChild( document.createElement( "div" ) );
this.embedded_editor_area.className = "editor-widget";
}
if( !this.show_canvas )
canvas.style.display = "none";
this.webgl_manager = new tiny.Webgl_Manager( canvas, color( 0,0,0,1 ) ); // Second parameter sets background color.
// Add scenes and child widgets
if( initial_scenes )
this.webgl_manager.scenes.push( ...initial_scenes );
const primary_scene = initial_scenes ? initial_scenes[0] : undefined;
const additional_scenes = initial_scenes ? initial_scenes.slice(1) : [];
const primary_scene_definiton = primary_scene ? primary_scene.constructor : undefined;
if( this.show_explanation )
this.embedded_explanation = new Text_Widget( this.embedded_explanation_area, this.webgl_manager.scenes, this.webgl_manager );
if( this.make_controls )
this.embedded_controls = new Controls_Widget( this.embedded_controls_area, this.webgl_manager.scenes );
if( this.make_editor )
this.embedded_editor = new Editor_Widget( this.embedded_editor_area, primary_scene_definiton, this );
if( this.make_code_nav )
this.embedded_code_nav = new Code_Widget( this.embedded_code_nav_area, primary_scene_definiton,
additional_scenes, { associated_editor: this.embedded_editor } );
// Start WebGL initialization. Note that render() will re-queue itself for continuous calls.
this.webgl_manager.render();
}
}
const Controls_Widget = widgets.Controls_Widget =
class Controls_Widget
{ // **Controls_Widget** adds an array of panels to the document, one per loaded
// Scene object, each providing interactive elements such as buttons with key
// bindings, live readouts of Scene data members, etc.
constructor( element, scenes )
{ const rules = [ ".controls-widget * { font-family: monospace }",
".controls-widget div { background: white }",
".controls-widget table { border-collapse: collapse; display:block; overflow-x: auto; }",
".controls-widget table.control-box { width: 1080px; border:0; margin:0; max-height:380px; transition:.5s; overflow-y:scroll; background:DimGray }",
".controls-widget table.control-box:hover { max-height:500px }",
".controls-widget table.control-box td { overflow:hidden; border:0; background:DimGray; border-radius:30px }",
".controls-widget table.control-box td .control-div { background: #EEEEEE; height:338px; padding: 5px 5px 5px 30px; box-shadow: 25px 0px 60px -15px inset }",
".controls-widget table.control-box td * { background:transparent }",
".controls-widget table.control-box .control-div td { border-radius:unset }",
".controls-widget table.control-box .control-title { padding:7px 40px; color:white; background:DarkSlateGray; box-shadow: 25px 0px 70px -15px inset black }",
".controls-widget *.live_string { display:inline-block; background:unset }",
".dropdown { display:inline-block }",
".dropdown-content { display:inline-block; transition:.2s; transform: scaleY(0); overflow:hidden; position: absolute; \
z-index: 1; background:#E8F6FF; padding: 16px; margin-left:30px; min-width: 100px; \
box-shadow: 5px 10px 16px 0px rgba(0,0,0,0.2) inset; border-radius:10px }",
".dropdown-content a { color: black; padding: 4px 4px; display: block }",
".dropdown a:hover { background: #f1f1f1 }",
".controls-widget button { background: #4C9F50; color: white; padding: 6px; border-radius:9px; \
box-shadow: 4px 6px 16px 0px rgba(0,0,0,0.3); transition: background .3s, transform .3s }",
".controls-widget button:hover, button:focus { transform: scale(1.3); color:gold }",
".link { text-decoration:underline; cursor: pointer }",
".show { transform: scaleY(1); height:200px; overflow:auto }",
".hide { transform: scaleY(0); height:0px; overflow:hidden }" ];
const style = document.head.appendChild( document.createElement( "style" ) );
for( const r of rules ) document.styleSheets[document.styleSheets.length - 1].insertRule( r, 0 )
const table = element.appendChild( document.createElement( "table" ) );
table.className = "control-box";
this.row = table.insertRow( 0 );
this.panels = [];
this.scenes = scenes;
this.render();
}
make_panels( time )
{ this.timestamp = time;
this.row.innerHTML = "";
// Traverse all scenes and their children, recursively:
const open_list = [ ...this.scenes ];
while( open_list.length )
{ open_list.push( ...open_list[0].children );
const scene = open_list.shift();
const control_box = this.row.insertCell();
this.panels.push( control_box );
// Draw top label bar:
control_box.appendChild( Object.assign( document.createElement("div"), {
textContent: scene.constructor.name, className: "control-title" } ) )
const control_panel = control_box.appendChild( document.createElement( "div" ) );
control_panel.className = "control-div";
scene.control_panel = control_panel;
scene.timestamp = time;
// Draw each registered animation:
scene.make_control_panel();
}
}
render( time = 0 )
{ // Check to see if we need to re-create the panels due to any scene being new.
// Traverse all scenes and their children, recursively:
const open_list = [ ...this.scenes ];
while( open_list.length )
{ open_list.push( ...open_list[0].children );
const scene = open_list.shift();
if( !scene.timestamp || scene.timestamp > this.timestamp )
{ this.make_panels( time );
break;
}
// TODO: Check for updates to each scene's desired_controls_position, including if the
// scene just appeared in the tree, in which case call make_control_panel().
}
for( let panel of this.panels )
for( let live_string of panel.querySelectorAll(".live_string") ) live_string.onload( live_string );
// TODO: Cap this so that it can't be called faster than a human can read?
this.event = window.requestAnimFrame( this.render.bind( this ) );
}
}
const Code_Manager = widgets.Code_Manager =
class Code_Manager
{ // **Code_Manager** breaks up a string containing code (any ES6 JavaScript). The RegEx being used
// to parse is from https://github.com/lydell/js-tokens which states the following limitation:
// "If the end of a statement looks like a regex literal (even if it isn’t), it will be treated
// as one." (This can miscolor lines of code containing divisions and comments).
constructor( code )
{ const es6_tokens_parser = RegExp( [
/((['"])(?:(?!\2|\\).|\\(?:\r\n|[\s\S]))*(\2)?|`(?:[^`\\$]|\\[\s\S]|\$(?!\{)|\$\{(?:[^{}]|\{[^}]*\}?)*\}?)*(`)?)/, // Any string.
/(\/\/.*)|(\/\*(?:[^*]|\*(?!\/))*(\*\/)?)/, // Any comment (2 forms). And next, any regex:
/(\/(?!\*)(?:\[(?:(?![\]\\]).|\\.)*\]|(?![\/\]\\]).|\\.)+\/(?:(?!\s*(?:\b|[\u0080-\uFFFF$\\'"~({]|[+\-!](?!=)|\.?\d))|[gmiyu]{1,5}\b(?![\u0080-\uFFFF$\\]|\s*(?:[+\-*%&|^<>!=?({]|\/(?![\/*])))))/,
/(0[xX][\da-fA-F]+|0[oO][0-7]+|0[bB][01]+|(?:\d*\.\d+|\d+\.?)(?:[eE][+-]?\d+)?)/, // Any number.
/((?!\d)(?:(?!\s)[$\w\u0080-\uFFFF]|\\u[\da-fA-F]{4}|\\u\{[\da-fA-F]+\})+)/, // Any name.
/(--|\+\+|&&|\|\||=>|\.{3}|(?:[+\-\/%&|^]|\*{1,2}|<{1,2}|>{1,3}|!=?|={1,2})=?|[?~.,:;[\](){}])/, // Any punctuator.
/(\s+)|(^$|[\s\S])/ // Any whitespace. Lastly, blank/invalid.
].map( r => r.source ).join('|'), 'g' );
this.tokens = []; this.no_comments = []; let single_token = null;
while( ( single_token = es6_tokens_parser.exec( code ) ) !== null )
{ let token = { type: "invalid", value: single_token[0] }
if ( single_token[ 1 ] ) token.type = "string" , token.closed = !!( single_token[3] || single_token[4] )
else if ( single_token[ 5 ] ) token.type = "comment"
else if ( single_token[ 6 ] ) token.type = "comment", token.closed = !!single_token[7]
else if ( single_token[ 8 ] ) token.type = "regex"
else if ( single_token[ 9 ] ) token.type = "number"
else if ( single_token[ 10 ] ) token.type = "name"
else if ( single_token[ 11 ] ) token.type = "punctuator"
else if ( single_token[ 12 ] ) token.type = "whitespace"
this.tokens.push( token )
if( token.type != "whitespace" && token.type != "comment" ) this.no_comments.push( token.value );
}
}
}
const Code_Widget = widgets.Code_Widget =
class Code_Widget
{ // **Code_Widget** draws a code navigator panel with inline links to the entire program source code.
constructor( element, main_scene, additional_scenes, options = {} )
{ const rules = [ ".code-widget .code-panel { margin:auto; background:white; overflow:auto; font-family:monospace; width:1060px; padding:10px; padding-bottom:40px; max-height: 500px; \
border-radius:12px; box-shadow: 20px 20px 90px 0px powderblue inset, 5px 5px 30px 0px blue inset }",
".code-widget .code-display { min-width:1200px; padding:10px; white-space:pre-wrap; background:transparent }",
".code-widget table { display:block; margin:auto; overflow-x:auto; width:1080px; border-radius:25px; border-collapse:collapse; border: 2px solid black }",
".code-widget table.class-list td { border-width:thin; background: #EEEEEE; padding:12px; font-family:monospace; border: 1px solid black }"
];
if( document.styleSheets.length == 0 ) document.head.appendChild( document.createElement( "style" ) );
for( const r of rules ) document.styleSheets[document.styleSheets.length - 1].insertRule( r, 0 )
this.associated_editor_widget = options.associated_editor;
if( !main_scene )
return;
import( './main-scene.js' )
.then( module => {
this.build_reader( element, main_scene, additional_scenes, module.defs );
if( !options.hide_navigator )
this.build_navigator( element, main_scene, additional_scenes, module.defs );
} )
}
build_reader( element, main_scene, additional_scenes, definitions )
{ // (Internal helper function)
this.definitions = definitions;
const code_panel = element.appendChild( document.createElement( "div" ) );
code_panel.className = "code-panel";
// const text = code_panel.appendChild( document.createElement( "p" ) );
// text.textContent = "Code for the above scene:";
this.code_display = code_panel.appendChild( document.createElement( "div" ) );
this.code_display.className = "code-display";
// Default textbox contents:
this.display_code( main_scene );
}
build_navigator( element, main_scene, additional_scenes, definitions )
{ // (Internal helper function)
const class_list = element.appendChild( document.createElement( "table" ) );
class_list.className = "class-list";
const top_cell = class_list.insertRow( -1 ).insertCell( -1 );
top_cell.colSpan = 2;
top_cell.appendChild( document.createTextNode("Click below to navigate through all classes that are defined.") );
const content = top_cell.appendChild( document.createElement( "p" ) );
content.style = "text-align:center; margin:0; font-weight:bold";
content.innerHTML = "main-scene.js<br>Main Scene: ";
const main_scene_link = content.appendChild( document.createElement( "a" ) );
main_scene_link.href = "javascript:void(0);"
main_scene_link.addEventListener( 'click', () => this.display_code( main_scene ) );
main_scene_link.textContent = main_scene.name;
const second_cell = class_list.insertRow( -1 ).insertCell( -1 );
second_cell.colSpan = 2;
second_cell.style = "text-align:center; font-weight:bold";
const index_src_link = second_cell.appendChild( document.createElement( "a" ) );
index_src_link.href = "javascript:void(0);"
index_src_link.addEventListener( 'click', () => this.display_code() );
index_src_link.textContent = "This page's complete HTML source";
const third_row = class_list.insertRow( -1 );
third_row.style = "text-align:center";
third_row.innerHTML = "<td><b>tiny-graphics.js</b><br>(Always the same)</td> \
<td><b>All other class definitions from dependencies:</td>";
const fourth_row = class_list.insertRow( -1 );
// Generate the navigator table of links:
for( let list of [ tiny, definitions ] )
{ const cell = fourth_row.appendChild( document.createElement( "td" ) );
// List all class names except the main one, which we'll display separately:
const class_names = Object.keys( list ).filter( x => x != main_scene.name );
cell.style = "white-space:normal"
for( let name of class_names )
{ const class_link = cell.appendChild( document.createElement( "a" ) );
class_link.style["margin-right"] = "80px"
class_link.href = "javascript:void(0);"
class_link.addEventListener( 'click', () => this.display_code( tiny[name] || definitions[name] ) );
class_link.textContent = name;
cell.appendChild( document.createTextNode(" ") );
}
}
}
display_code( class_to_display )
{ // display_code(): Populate the code textbox.
// Pass undefined to choose index.html source.
if( this.associated_editor_widget )
this.associated_editor_widget.select_class( class_to_display );
if( class_to_display ) this.format_code( class_to_display.toString() );
else fetch( document.location.href )
.then( response => response.text() )
.then( pageSource => this.format_code( pageSource ) );
}
format_code( code_string )
{ // (Internal helper function)
this.code_display.innerHTML = "";
const color_map = { string: "chocolate", comment: "green", regex: "blue", number: "magenta",
name: "black", punctuator: "red", whitespace: "black" };
for( let t of new Code_Manager( code_string ).tokens )
if( t.type == "name" && [ ...Object.keys( tiny ), ...Object.keys( this.definitions ) ].includes( t.value ) )
{ const link = this.code_display.appendChild( document.createElement( 'a' ) );
link.href = "javascript:void(0);"
link.addEventListener( 'click', () => this.display_code( tiny[t.value] || this.definitions[t.value] ) );
link.textContent = t.value;
}
else
{ const span = this.code_display.appendChild( document.createElement( 'span' ) );
span.style.color = color_map[t.type];
span.textContent = t.value;
}
}
}
const Editor_Widget = widgets.Editor_Widget =
class Editor_Widget
{ constructor( element, initially_selected_class, canvas_widget, options = {} )
{ let rules = [ ".editor-widget { margin:auto; background:white; overflow:auto; font-family:monospace; width:1060px; padding:10px; \
border-radius:12px; box-shadow: 20px 20px 90px 0px powderblue inset, 5px 5px 30px 0px blue inset }",
".editor-widget button { background: #4C9F50; color: white; padding: 6px; border-radius:9px; margin-right:5px; \
box-shadow: 4px 6px 16px 0px rgba(0,0,0,0.3); transition: background .3s, transform .3s }",
".editor-widget input { margin-right:5px }",
".editor-widget textarea { white-space:pre; width:1040px; margin-bottom:30px }",
".editor-widget button:hover, button:focus { transform: scale(1.3); color:gold }"
];
for( const r of rules ) document.styleSheets[0].insertRule( r, 1 );
this.associated_canvas = canvas_widget;
this.options = options;
const form = this.form = element.appendChild( document.createElement( "form" ) );
// Don't refresh the page on submit:
form.addEventListener( 'submit', event =>
{ event.preventDefault(); this.submit_demo() }, false );
const explanation = form.appendChild( document.createElement( "p" ) );
explanation.innerHTML = `<i><b>What can I put here?</b></i> A JavaScript class, with any valid JavaScript inside. Your code can use classes from this demo,
<br>or from ANY demo on Demopedia -- the dependencies will automatically be pulled in to run your demo!<br>`;
const run_button = this.run_button = form.appendChild( document.createElement( "button" ) );
run_button.type = "button";
run_button.style = "background:maroon";
run_button.textContent = "Run with Changes";
const submit = this.submit = form.appendChild( document.createElement( "button" ) );
submit.type = "submit";
submit.textContent = "Save as New Webpage";
const author_box = this.author_box = form.appendChild( document.createElement( "input" ) );
author_box.name = "author";
author_box.type = "text";
author_box.placeholder = "Author name";
const password_box = this.password_box = form.appendChild( document.createElement( "input" ) );
password_box.name = "password";
password_box.type = "text";
password_box.placeholder = "Password";
password_box.style = "display:none";
const overwrite_panel = this.overwrite_panel = form.appendChild( document.createElement( "span" ) );
overwrite_panel.style = "display:none";
overwrite_panel.innerHTML = "<label>Overwrite?<input type='checkbox' name='overwrite' autocomplete='off'></label>";
const submit_result = this.submit_result = form.appendChild( document.createElement( "div" ) );
submit_result.style = "margin: 10px 0";
const new_demo_code = this.new_demo_code = form.appendChild( document.createElement( "textarea" ) );
new_demo_code.name = "new_demo_code";
new_demo_code.rows = this.options.rows || 25;
new_demo_code.cols = 140;
if( initially_selected_class )
this.select_class( initially_selected_class );
}
select_class( class_definition )
{ this.new_demo_code.value = class_definition.toString(); }
fetch_handler( url, body ) // A general utility function for sending / receiving JSON, with error handling.
{ return fetch( url,
{ body: body, method: body === undefined ? 'GET' : 'POST',
headers: { 'content-type': 'application/json' }
}).then( response =>
{ if ( response.ok ) return Promise.resolve( response.json() )
else return Promise.reject ( response.status )
})
}
submit_demo()
{ const form_fields = Array.from( this.form.elements ).reduce( ( accum, elem ) =>
{ if( elem.value && !( ['checkbox', 'radio'].includes( elem.type ) && !elem.checked ) )
accum[ elem.name ] = elem.value;
return accum;
}, {} );
this.submit_result.innerHTML = "";
return this.fetch_handler( "/submit-demo?Unapproved", JSON.stringify( form_fields ) )
.then ( response => { if( response.show_password ) this.password_box.style.display = "inline";
if( response.show_overwrite ) this.overwrite_panel.style.display = "inline";
this.submit_result.innerHTML += response.message + "<br>"; } )
.catch( error => { this.submit_result.innerHTML += "Error " + error + " when trying to upload.<br>" } )
}
}
const Text_Widget = widgets.Text_Widget =
class Text_Widget
{ // **Text_Widget** generates HTML documentation and fills a panel with it. This
// documentation is extracted from whichever Scene object gets loaded first.
constructor( element, scenes, webgl_manager )
{ const rules = [ ".text-widget { background: white; width:1060px;\
padding:0 10px; overflow:auto; transition:1s; overflow-y:scroll; box-shadow: 10px 10px 90px 0 inset LightGray}",
".text-widget div { transition:none } "
];
if( document.styleSheets.length == 0 ) document.head.appendChild( document.createElement( "style" ) );
for( const r of rules ) document.styleSheets[document.styleSheets.length - 1].insertRule( r, 0 )
Object.assign( this, { element, scenes, webgl_manager } );
this.render();
}
render( time = 0 )
{ if( this.scenes[0] )
this.scenes[0].show_explanation( this.element, this.webgl_manager )
else
this.event = window.requestAnimFrame( this.render.bind( this ) )
}
}