diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..aae39e8 --- /dev/null +++ b/.gitignore @@ -0,0 +1,3 @@ +.vscode +node_modules +docs/pages.json \ No newline at end of file diff --git a/.vscode/settings.json b/.vscode/settings.json deleted file mode 100644 index 66dc905..0000000 --- a/.vscode/settings.json +++ /dev/null @@ -1 +0,0 @@ -undefined \ No newline at end of file diff --git a/README.md b/README.md index 3cd8d5b..76ac24a 100644 --- a/README.md +++ b/README.md @@ -12,7 +12,7 @@ So the API is minimal and easy to understand. The library is tiny, fast and rend ## Main features * Fast and lightweight graphics command queue builder. -* Adressing HTML canvas 2D context as the default renderer. +* Addressing HTML canvas 2D context as the default renderer. * Generating SVG output using an [addon library](https://github.com/goessner/g2-svg). * Method chaining. * Support of cartesian coordinates. @@ -55,22 +55,27 @@ See the [API Reference for g2](docs/api/g2.core.md) for details. Also see the [API Reference for g2.ext](docs/api/g2.ext.md) and the [API Reference for g2.mec](docs/api/g2.mec.md). +Also available under + - [https://goessner.github.io/g2/api/g2.core](https://goessner.github.io/g2/api/g2.core) + - [https://goessner.github.io/g2/api/g2.ext](https://goessner.github.io/g2/api/g2.ext) + - [https://goessner.github.io/g2/api/g2.mec](https://goessner.github.io/g2/api/g2.mec) + + +## Cheat Sheet +Check out the single page [Cheat Sheet](docs/api/sheet.pdf). + +Also available under [https://goessner.github.io/g2/api/sheet.pdf](https://goessner.github.io/g2/api/sheet.pdf) ## GitCDN -Use the link [https://cdn.jsdelivr.net/gh/goessner/g2@v2.5/src/g2.js](https://cdn.jsdelivr.net/gh/goessner/g2@v2.5/src/g2.js) -for getting the current stable commit as a raw file. +Use the link [https://cdn.jsdelivr.net/gh/goessner/g2/dist/g2.js](https://cdn.jsdelivr.net/gh/goessner/g2/dist/g2.js) for getting the latest commit as a raw file. In HTML use ... ```html - + ``` ### Tests Tests are found on the [Github Page](https://goessner.github.io/g2/tests.html) -## Cheat Sheet -Check out the single page [Cheat Sheet](docs/api/cheatsheet.pdf). - - # License g2 is licensed under the terms of the MIT License. diff --git a/bin/canvasInteractor.js b/bin/canvasInteractor.js new file mode 100644 index 0000000..2c2787e --- /dev/null +++ b/bin/canvasInteractor.js @@ -0,0 +1,214 @@ +/** + * canvasInteractor.js (c) 2018 Stefan Goessner + * @file interaction manager for html `canvas`. + * @author Stefan Goessner + * @license MIT License + */ +/* jshint -W014 */ +// Managing multiple canvases per static interactor as singleton ... +// .. using a single requestAnimationFrame loop ! +const canvasInteractor = { + create() { + const o = Object.create(this.prototype); + o.constructor.apply(o,arguments); + return o; + }, + // global static tickTimer properties + fps: '?', + fpsOrigin: 0, + frames: 0, + rafid: 0, + instances: [], + // global static timer methods + tick(time) { + canvasInteractor.fpsCount(time); + for (const instance of canvasInteractor.instances) { + instance.notify('tick',{t:time,dt:(time-instance.t)/1000,dirty:instance.dirty}); // notify listeners .. + instance.t = time; + instance.dirty = false; + } + canvasInteractor.rafid = requestAnimationFrame(canvasInteractor.tick); // request next animation frame ... + }, + add(instance) { + canvasInteractor.instances.push(instance); + if (canvasInteractor.instances.length === 1) // first instance added ... + canvasInteractor.tick(canvasInteractor.fpsOrigin = performance.now()); + }, + remove(instance) { + canvasInteractor.instances.splice(canvasInteractor.instances.indexOf(instance),1); + if (canvasInteractor.instances.length === 0) // last instance removed ... + cancelAnimationFrame(canvasInteractor.rafid); + }, + fpsCount(time) { + if (time - canvasInteractor.fpsOrigin > 1000) { // one second interval reached ... + const fps = ~~(canvasInteractor.frames*1000/(time - canvasInteractor.fpsOrigin) + 0.5); // ~~ as Math.floor() + if (fps !== canvasInteractor.fps) + for (const instance of canvasInteractor.instances) + instance.notify('fps',canvasInteractor.fps=fps); + canvasInteractor.fpsOrigin = time; + canvasInteractor.frames = 0; + } + canvasInteractor.frames++; + }, + + prototype: { + constructor(ctx, {x,y,scl,cartesian}) { + // canvas interaction properties + this.ctx = ctx; + this.view = {x:x||0,y:y||0,scl:scl||1,cartesian:cartesian||false}; + this.evt = { + type: false, + basetype: false, + x: -2, y:-2, + xi: 0, yi:0, + dx: 0, dy: 0, + btn: 0, + xbtn: 0, ybtn: 0, + xusr: -2, yusr: -2, + dxusr: 0, dyusr: 0, + delta: 0, + inside: false, + hit: false, // something hit by pointer ... + dscl: 1, // for zooming ... + eps: 5 // some pixel tolerance ... + }; + this.dirty = true; + // event handler registration + const canvas = ctx.canvas; + canvas.addEventListener("pointermove", this, false); + canvas.addEventListener("pointerdown", this, false); + canvas.addEventListener("pointerup", this, false); + canvas.addEventListener("pointerenter", this, false); + canvas.addEventListener("pointerleave", this, false); + canvas.addEventListener("wheel", this, false); + canvas.addEventListener("pointercancel", this, false); + }, + deinit() { + const canvas = this.ctx.canvas; + + canvas.removeEventListener("pointermove", this, false); + canvas.removeEventListener("pointerdown", this, false); + canvas.removeEventListener("pointerup", this, false); + canvas.removeEventListener("pointerenter", this, false); + canvas.removeEventListener("pointerleave", this, false); + canvas.removeEventListener("wheel", this, false); + canvas.removeEventListener("pointercancel", this, false); + + this.endTimer(); + + delete this.signals; + delete this.evt; + delete this.ctx; + + return this; + }, + // canvas interaction interface + handleEvent(e) { + if (e.type in this && (e.isPrimary || e.type === 'wheel')) { // can I handle events of type e.type .. ? + const bbox = e.target.getBoundingClientRect && e.target.getBoundingClientRect() || {left:0, top:0}, + x = e.clientX - Math.floor(bbox.left), + y = e.clientY - Math.floor(bbox.top), + btn = e.buttons !== undefined ? e.buttons : e.button || e.which; + + this.evt.type = e.type; + this.evt.basetype = e.type; // obsolete now ... ? + this.evt.xi = this.evt.x; // interim coordinates ... + this.evt.yi = this.evt.y; // ... of previous event. + this.evt.dx = this.evt.dy = 0; + this.evt.x = x; + this.evt.y = this.view.cartesian ? this.ctx.canvas.height - y : y; + this.evt.xusr = (this.evt.x - this.view.x)/this.view.scl; + this.evt.yusr = (this.evt.y - this.view.y)/this.view.scl; + this.evt.dxusr = this.evt.dyusr = 0; + this.evt.dbtn = btn - this.evt.btn; + this.evt.btn = btn; + this.evt.delta = Math.max(-1,Math.min(1,e.deltaY||e.wheelDelta)) || 0; + + if (this.isDefaultPreventer(e.type)) + e.preventDefault(); + this[e.type](); // handle specific event .. ! + this.notify(this.evt.type,this.evt); // .. tell the world .. ! + } + else + console.log(e) + }, + pointermove() { + this.evt.dx = this.evt.x - this.evt.xi; + this.evt.dy = this.evt.y - this.evt.yi; + if (this.evt.btn === 1) { // pointerdown state ... + this.evt.dxusr = this.evt.dx/this.view.scl; // correct usr coordinates ... + this.evt.dyusr = this.evt.dy/this.view.scl; + this.evt.xusr -= this.evt.dxusr; // correct usr coordinates ... + this.evt.yusr -= this.evt.dyusr; + if (!this.evt.hit) { // let outer app perform panning ... + this.evt.type = 'pan'; + } + else + this.evt.type = 'drag'; + } + // view, geometry or graphics might be modified ... + this.dirty = true; + }, + pointerdown() { + this.evt.xbtn = this.evt.x; + this.evt.ybtn = this.evt.y; + }, + pointerup() { + this.evt.type = this.evt.x===this.evt.xbtn && this.evt.y===this.evt.ybtn ? 'click' : 'pointerup'; + this.evt.xbtn = this.evt.x; + this.evt.ybtn = this.evt.y; + }, + pointerleave() { + this.evt.inside = false; + }, + pointerenter() { + this.evt.inside = true; + }, + wheel() { + this.evt.dscl = this.evt.delta>0?8/10:10/8; + this.evt.eps /= this.evt.dscl; + this.dirty = true; + }, + isDefaultPreventer(type) { + return ['pointermove','pointerdown','pointerup','wheel'].includes(type); + }, + pntToUsr: function(p) { + let vw = this.view; + p.x = (p.x - vw.x)/vw.scl; + p.y = (p.y - vw.y)/vw.scl; + return p; + }, + // tickTimer interface + startTimer() { // shouldn't there be a global startTimer method ? + canvasInteractor.add(this); + this.notify('timerStart',this); // notify potential listeners .. + return this; + }, + endTimer() { + this.notify('timerEnd',this.t/1000); // notify potential listeners .. + canvasInteractor.remove(this); + return this; + }, + // observable interface + notify(key,val) { + if (this.signals && this.signals[key]) + for (let hdl of this.signals[key]) + hdl(val); + return this; + }, + on(key,handler) { // support array of keys as first argument. + if (Array.isArray(key)) + for (let k of key) + this.on(k,handler); + else + ((this.signals || (this.signals = {})) && this.signals[key] || (this.signals[key]=[])).push(handler); + + return this; + }, + remove(key,handler) { + const idx = (this.signals && this.signals[key]) ? this.signals[key].indexOf(handler) : -1; + if (idx >= 0) + this.signals[key].splice(idx,1); + } + } +}; diff --git a/changelog.md b/changelog.md new file mode 100644 index 0000000..46e9696 --- /dev/null +++ b/changelog.md @@ -0,0 +1,35 @@ +# CHANGELOG +## [2.6.0]() on May 2020 + +### `g2.core.js` +* additional support of `{p}` property as alternative to `{x,y}` properties for referencing shared points/vectors. +* unknown symbol name with `use` results in drawing unknown symbol instead of runtime error. +* `{ldoff}` as linedash offset style property for line dash animation added. ` +* Implicite element selecting and dragging capability removed. +* Boolean `{draggable}` property added, for explicitly indicating draggable elements. +### `g2.ext.js` +* code cleanup. +* `label` element deprecated. +* `mark` element deprecated. +* `label` property for most element types introduced. +* Handle element `hdl` with default `{draggable:true}` property added. +* `nod, nodfix, nodflt, pol, gnd, vec, dim, adim, origin` elements moved/ported from `g2.mec.js`. +### `g2.io.js` +* works again. +### `g2.selector.js` / `canvasinteractor.js` +* `interactor.on('pan')`, `interactor.on('drag')` and `interactor.on('wheel')` handling is moved out of the library and is now in the responsibility of the application (s. `g2.drag.html`). +* `hdl` elements can be used elegantly to interactively modify geometry. + +## [3.0.0]() on December 2020 + +### `g2.core.js` +* `g2.mixin` is replaced with `g2.mix`. + +### `g2.ext.js` +* Symbols like `nod`, `origin` etc. are moved from `g2.mec` to `g2.ext`. +* Commands `vec`, `avec`, `dim`, `adim` are also moved. +* `label` is no command anymore, but a property on respective `g2` commands. +* `mark` is no command anymore, but a property on respective `g2` commands. + +### `g2.chart.html.js` +* A new custom HTML element used for easy rendering of `g2.chart` commands. diff --git a/dist/g2.chart.html.js b/dist/g2.chart.html.js new file mode 100644 index 0000000..944ac59 --- /dev/null +++ b/dist/g2.chart.html.js @@ -0,0 +1,3016 @@ + +"use strict" + +/** + * g2.core (c) 2013-20 Stefan Goessner + * @author Stefan Goessner + * @license MIT License + * @link https://github.com/goessner/g2 + * @typedef {g2} + * @param {object} [opts] Custom options object. + * @description Create a 2D graphics command queue object. Call without using 'new'. + * @returns {g2} + * @example + * const ctx = document.getElementById("c").getContext("2d"); + * g2() // Create 'g2' instance. + * .lin({x1:50,y1:50,x2:100,y2:100}) // Append ... + * .lin({x1:100,y1:100,x2:200,y2:50}) // ... commands. + * .exe(ctx); // Execute commands addressing canvas context. + */ + +function g2(opts) { + let o = Object.create(g2.prototype); + o.commands = []; + if (opts) Object.assign(o, opts); + return o; +} + +g2.prototype = { + /** + * Clear viewport region.
+ * @method + * @returns {object} g2 + */ + clr() { return this.addCommand({ c: 'clr' }); }, + + /** + * Set the view by placing origin coordinates and scaling factor in device units + * and make viewport cartesian. + * @method + * @returns {object} g2 + * @param {object} - view arguments object. + * @property {number} [scl=1] - absolute scaling factor. + * @property {number} [x=0] - x-origin in device units. + * @property {number} [y=0] - y-origin in device units. + * @property {boolean} [cartesian=false] - set cartesian flag. + */ + view({ scl, x, y, cartesian }) { return this.addCommand({ c: 'view', a: arguments[0] }); }, + + /** + * Draw grid. + * @method + * @returns {object} g2 + * @param {object} - grid arguments object. + * @property {string} [color=#ccc] - change color. + * @property {number} [size=20] - change space between lines. + */ + grid({ color, size } = {}) { return this.addCommand({ c: 'grid', a: arguments[0] }); }, + + /** + * Draw circle by center and radius. + * @method + * @returns {object} g2 + * @param {object} - circle arguments object. + * @property {number} x - x-value center. + * @property {number} y - y-value center. + * @property {number} r - radius. + * @property {number} w - angle. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {array} [sh=[0,0,0,'transparent']] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @example + * g2().cir({x:100,y:80,r:20}) // Draw circle. + */ + cir({ x, y, r, w }) { return this.addCommand({ c: 'cir', a: arguments[0] }); }, + + /** + * Draw ellipse by center and radius for x and y. + * @method + * @returns {object} g2 + * @param {object} - ellispe argument object. + * @property {number} x - x-value center. + * @property {number} y - y-value center. + * @property {number} rx - radius x-axys. + * @property {number} ry - radius y-axys. + * @property {number} w - start angle. + * @property {number} dw - angular range. + * @property {number} rot - rotation. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @example + * g2().ell({x:100,y:80,rx:20,ry:30,w:0,dw:2*Math.PI/4,rot:1}) // Draw circle. + */ + ell({ x, y, rx, ry, w, dw, rot }) { return this.addCommand({ c: 'ell', a: arguments[0] }); }, + + /** + * Draw arc by center point, radius, start angle and angular range. + * @method + * @returns {object} g2 + * @param {object} - arc arguments object. + * @property {number} x - x-value center. + * @property {number} y - y-value center. + * @property {number} r - radius. + * @property {number} [w=0] - start angle (in radian). + * @property {number} [dw=2*pi] - angular range in Radians. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {string} [lc=butt] - line cap [`butt`, `round`, `square`]. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @example + * g2().arc({x:300,y:400,r:390,w:-Math.PI/4,dw:-Math.PI/2}) + * .exe(ctx); + */ + arc({ x, y, r, w, dw }) { return this.addCommand({ c: 'arc', a: arguments[0] }); }, + + /** + * Draw rectangle by anchor point and dimensions. + * @method + * @returns {object} g2 + * @param {object} - rectangle arguments object. + * @property {number} x - x-value upper left corner. + * @property {number} y - y-value upper left corner. + * @property {number} b - width. + * @property {number} h - height. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {string} [lj='miter'] - line join [`round`, `bevel`, `miter`]. + * @property {number} [ml=10] - miter limit. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @example + * g2().rec({x:100,y:80,b:40,h:30}) // Draw rectangle. + */ + rec({ x, y, b, h }) { return this.addCommand({ c: 'rec', a: arguments[0] }); }, + + /** + * Draw line by start point and end point. + * @method + * @returns {object} g2 + * @param {object} - line arguments object. + * @property {number} x1 - start x coordinate. + * @property {number} y1 - start y coordinate. + * @property {number} x2 - end x coordinate. + * @property {number} y2 - end y coordinate. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {string} [lc=butt] - line cap [`butt`, `round`, `square`]. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @example + * g2().lin({x1:10,x2:10,y1:190,y2:10}) // Draw line. + */ + lin({ x1, y1, x2, y2 }) { return this.addCommand({ c: 'lin', a: arguments[0] }); }, + + /** + * Draw polygon by points. + * Using iterator function for getting points from array by index. + * It must return current point object {x,y} or object {done:true}. + * Default iterator expects sequence of x/y-coordinates as a flat array [x,y,...], + * array of [[x,y],...] arrays or array of [{x,y},...] objects. + * @method + * @returns {object} g2 + * @param {object} - polygon arguments object. + * @property {array} pts - array of points. + * @property {string} [format] - format string of points array structure. Useful for handing over initial empty points array. One of `['x,y','[x,y]','{x,y}']`. Has precedence over `pts` content. + * @property {boolean} [closed = false] + * @property {number} x - start x coordinate. + * @property {number} y - start y coordinate. + * @property {number} w - angle. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {string} [lc=butt] - line cap [`butt`, `round`, `square`]. + * @property {string} [lj='miter'] - line join [`round`, `bevel`, `miter`]. + * @property {number} [ml=10] - miter limit. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @example + * g2().ply({pts:[100,50,120,60,80,70]}), + * .ply({pts:[150,60],[170,70],[130,80]],closed:true}), + * .ply({pts:[{x:160,y:70},{x:180,y:80},{x:140,y:90}]}), + * .exe(ctx); + */ + ply({ pts, format, closed, x, y, w }) { + arguments[0]._itr = format && g2.pntIterator[format](pts) || g2.pntItrOf(pts); + return this.addCommand({ c: 'ply', a: arguments[0] }); + }, + + /** + * Draw text string at anchor point. + * @method + * @returns {object} g2 + * @param {object} - text arguments object. + * @property {string} str - text string. + * @property {number} [x=0] - x coordinate of text anchor position. + * @property {number} [y=0] - y coordinate of text anchor position. + * @property {number} [w=0] - w Rotation angle about anchor point with respect to positive x-axis. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @property {string} [thal='start'] + * - Text horizontal alignment [`'start'`,`'end'`,`'left'`,`'right'`,`'center'`] + * @property {string} [tval='alphabetic'] + * - Text vertival alignment [`'top'`,`'hanging'`,`'middle'`,`'alphabetic'`,`'ideographic'`,`'bottom'`] + * @property {string} [font='normal 14px serif'] - + * [Font]{@link https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/font} + * [styling]{@link https://html.spec.whatwg.org/multipage/canvas.html#dom-context-2d-font} + */ + txt({ str, x, y, w }) { return this.addCommand({ c: 'txt', a: arguments[0] }); }, + + /** + * Reference g2 graphics commands from another g2 object or a predefined g2.symbol. + * With this command you can reuse instances of grouped graphics commands + * while applying a similarity transformation and style properties on them. + * In fact you might want to build custom graphics libraries on top of that feature. + * @method + * @returns {object} g2 + * @param {object} - use arguments object. + * @see {@link https://github.com/goessner/g2/blob/master/docs/api/g2.ext.md#g2symbol--object predefined symbols in g2.ext} + * @property {object | string} grp - g2 source object or symbol name found in 'g2.symbol' namespace. + * @property {number} [x=0] - translation value x. + * @property {number} [y=0] - translation value y. + * @property {number} [w=0] - rotation angle (in radians). + * @property {number} [scl=1] - scale factor. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {string} [lc=butt] - line cap [`butt`, `round`, `square`]. + * @property {string} [lj='miter'] - line join [`round`, `bevel`, `miter`]. + * @property {number} [ml=10] - miter limit. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @property {string} [thal='start'] + * - Text horizontal alignment [`'start'`,`'end'`,`'left'`,`'right'`,`'center'`] + * @property {string} [tval='alphabetic'] + * - Text vertival alignment [`'top'`,`'hanging'`,`'middle'`,`'alphabetic'`,`'ideographic'`,`'bottom'`] + * @property {string} [font='normal 14px serif'] - + * [Font]{@link https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/font} + * [styling]{@link https://html.spec.whatwg.org/multipage/canvas.html#dom-context-2d-font} + * @example + * g2.symbol.cross = g2().lin({x1:5,y1:5,x2:-5,y2:-5}).lin({x1:5,y1:-5,x2:-5,y2:5}); // Define symbol. + * g2().use({grp:"cross",x:100,y:100}) // Draw cross at position 100,100. + */ + use({ grp, x, y, w, scl }) { + if (grp && grp !== this) { // avoid self reference .. + if (typeof grp === "string") // must be a member name of the 'g2.symbol' namespace + arguments[0].grp = g2.symbol[(grp in g2.symbol) ? grp : 'unknown']; + this.addCommand({ c: 'use', a: arguments[0] }); + } + return this; + }, + + /** + * Draw image. + * This also applies to images of reused g2 objects. If an image can not be loaded, it will be replaced by a broken-image symbol. + * @method + * @returns {object} g2 + * @param {object} - image arguments object. + * @property {string} uri - image uri or data:url. + * @property {number} [x = 0] - x-coordinate of image (upper left). + * @property {number} [y = 0] - y-coordinate of image (upper left). + * @property {number} [b = image.width] - width. + * @property {number} [h = image.height] - height. + * @property {number} [sx = 0] - source x-offset. + * @property {number} [sy = 0] - source y-offset. + * @property {number} [sb = image.width] - source width. + * @property {number} [sh = image.height] - source height. + * @property {number} [xoff = 0] - x-offset. + * @property {number} [yoff = 0] - y-offset. + * @property {number} [w = 0] - rotation angle (about upper left, in radians). + * @property {number} [scl = 1] - image scaling. + */ + img({ uri, x, y, b, h, sx, sy, sb, sh, xoff, yoff, w, scl }) { return this.addCommand({ c: 'img', a: arguments[0] }); }, + + /** + * Begin subcommands. Current state is saved. + * Optionally apply transformation or style properties. + * @method + * @returns {object} g2 + * @param {object} - beg arguments object. + * @property {number} [x = 0] - translation value x. + * @property {number} [y = 0] - translation value y. + * @property {number} [w = 0] - rotation angle (in radians). + * @property {number} [scl = 1] - scale factor. + * @property {array} [matrix] - matrix instead of single transform arguments (SVG-structure [a,b,c,d,x,y]). + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {string} [lc=butt] - line cap [`butt`, `round`, `square`]. + * @property {string} [lj='miter'] - line join [`round`, `bevel`, `miter`]. + * @property {number} [ml=10] - miter limit. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @property {string} [thal='start'] + * - text horizontal alignment [`'start'`,`'end'`,`'left'`,`'right'`,`'center'`] + * @property {string} [tval='alphabetic'] + * - text vertival alignment [`'top'`,`'hanging'`,`'middle'`,`'alphabetic'`,`'ideographic'`,`'bottom'`] + * @property {string} [font='normal 14px serif'] - + * [Font]{@link https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/font} + * [styling]{@link https://html.spec.whatwg.org/multipage/canvas.html#dom-context-2d-font} + */ + beg({ x, y, w, scl, matrix } = {}) { return this.addCommand({ c: 'beg', a: arguments[0] }); }, + + /** + * End subcommands. Previous state is restored. + * @method + * @returns {object} g2 + * @param {object} - end arguments object. + */ + end() { // ignore 'end' commands without matching 'beg' + let myBeg = 1, + findMyBeg = (cmd) => { // care about nested beg...end blocks ... + if (cmd.c === 'beg') myBeg--; + else if (cmd.c === 'end') myBeg++; + return myBeg === 0; + } + return g2.cmdIdxBy(this.commands, findMyBeg) !== false ? this.addCommand({ c: 'end' }) : this; + }, + + /** + * Begin new path. + * @method + * @returns {object} g2 + */ + p() { return this.addCommand({ c: 'p' }); }, + + /** + * Close current path by straight line. + * @method + * @returns {object} g2 + */ + z() { return this.addCommand({ c: 'z' }); }, + + /** + * Move to point. + * @method + * @returns {object} g2 + * @param {object} - move arguments object. + * @property {number} x - move to x coordinate + * @property {number} y - move to y coordinate + */ + m({ x, y }) { return this.addCommand({ c: 'm', a: arguments[0] }); }, + + /** + * Create line segment to point. + * @method + * @returns {object} g2 + * @param {object} - line segment argument object. + * @property {number} x - x coordinate of target point. + * @property {number} y - y coordinate of target point. + * @example + * g2().p() // Begin path. + * .m({x:0,y:50}) // Move to point. + * .l({x:300,y:0}) // Line segment to point. + * .l({x:400,y:100}) // ... + * .stroke() // Stroke path. + */ + l({ x, y }) { return this.addCommand({ c: 'l', a: arguments[0] }); }, + + /** + * Create quadratic bezier curve segment to point. + * @method + * @returns {object} g2 + * @param {object} - quadratic curve arguments object. + * @property {number} x1 - x coordinate of control point. + * @property {number} y1 - y coordinate of control point. + * @property {number} x - x coordinate of target point. + * @property {number} y - y coordinate of target point. + * @example + * g2().p() // Begin path. + * .m({x:0,y:0}) // Move to point. + * .q({x1:200,y1:200,x:400,y:0}) // Quadratic bezier curve segment. + * .stroke() // Stroke path. + */ + q({ x1, y1, x, y }) { return this.addCommand({ c: 'q', a: arguments[0] }); }, + + /** + * Create cubic bezier curve to point. + * @method + * @returns {object} g2 + * @param {object} - cubic curve arguments object. + * @property {number} x1 - x coordinate of first control point. + * @property {number} y1 - y coordinate of first control point. + * @property {number} x2 - x coordinate of second control point. + * @property {number} y2 - y coordinate of second control point. + * @property {number} x - x coordinate of target point. + * @property {number} y - y coordinate of target point. + * @example + * g2().p() // Begin path. + * .m({x:0,y:100}) // Move to point. + * .c({x1:100,y1:200,x2:200,y2:0,x:400,y:100}) // Create cubic bezier curve. + * .stroke() // Stroke path. + * .exe(ctx); // Render to canvas context. + */ + c({ x1, y1, x2, y2, x, y }) { return this.addCommand({ c: 'c', a: arguments[0] }); }, + + /** + * Draw arc with angular range to target point. + * @method + * @returns {object} g2 + * @param {object} - arc arguments object. + * @property {number} dw - angular range in radians. + * @property {number} x - x coordinate of target point. + * @property {number} y - y coordinate of target point. + * @example + * g2().p() // Begin path. + * .m({x:50,y:50}) // Move to point. + * .a({dw:2,x:300,y:100}) // Create arc segment. + * .stroke() // Stroke path. + * .exe(ctx); // Render to canvas context. + */ + a({ dw, x, y }) { + let prvcmd = this.commands[this.commands.length - 1]; + g2.cpyProp(prvcmd.a, 'x', arguments[0], '_xp'); + g2.cpyProp(prvcmd.a, 'y', arguments[0], '_yp'); + return this.addCommand({ c: 'a', a: arguments[0] }); + }, + + /** + * Stroke the current path or path object. + * @method + * @returns {object} g2 + * @param {object} - stroke arguments object. + * @property {string} [d = undefined] - SVG path definition string. Current path is ignored then. + */ + stroke({ d } = {}) { return this.addCommand({ c: 'stroke', a: arguments[0] }); }, + + /** + * Fill the current path or path object. + * @method + * @returns {object} g2 + * @param {object} - fill arguments object. + * @property {string} [d = undefined] - SVG path definition string. Current path is ignored then. + */ + fill({ d } = {}) { return this.addCommand({ c: 'fill', a: arguments[0] }); }, + + /** + * Shortcut for stroke and fill the current path or path object. + * In case of shadow style, only the path interior creates shadow, not also the path contour. + * @method + * @returns {object} g2 + * @param {object} - drw arguments object. + * @property {string} [d = undefined] - SVG path definition string. Current path is ignored then. + */ + drw({ d, lsh } = {}) { return this.addCommand({ c: 'drw', a: arguments[0] }); }, + + /** + * Delete all commands beginning from `idx` to end of command queue. + * @method + * @returns {object} g2 + */ + del(idx) { this.commands.length = idx || 0; return this; }, + + /** + * Call function between commands of the command queue. + * @method + * @returns {object} g2 + * @param {function} - ins argument function. + * @example + * const node = { + * fill:'lime', + * g2() { return g2().cir({x:160,y:50,r:15,fs:this.fill,lw:4,sh:[8,8,8,"gray"]}) } + * }; + * let color = 'red'; + * g2().cir({x:40,y:50,r:15,fs:color,lw:4,sh:[8,8,8,"gray"]}) // draw red circle. + * .ins(()=>{color='green'}) // color is now green. + * .cir({x:80,y:50,r:15,fs:color,lw:4,sh:[8,8,8,"gray"]}) // draw green circle. + * .ins((g) => // draw orange circle + * g.cir({x:120, y:50, r:15, fs:'orange', lw:4,sh:[8,8,8,"gray"]})) + * .ins(node) // draw node. + * .exe(ctx) // render to canvas context. + */ + ins(arg) { + return typeof arg === 'function' ? (arg(this) || this) // no further processing by handler ... + : typeof arg === 'object' ? (this.commands.push({ a: arg }), this) // no explicit command name .. ! + : this; + }, + /** + * Execute g2 commands. It does so automatically and recursively with 'use'ed commands. + * @method + * @returns {object} g2 + * @param {object} ctx Context. + */ + exe(ctx) { + let handler = g2.handler(ctx); + if (handler && handler.init(this)) + handler.exe(this.commands); + return this; + }, + // helpers ... + addCommand({ c, a }) { + if (a && Object.getPrototypeOf(a) === Object.prototype) { // modify only pure argument objects 'a' .. ! + for (const key in a) { + if (!Object.getOwnPropertyDescriptor(a, key).get // if 'key' is no getter ... + && key[0] !== '_' // and no private property ... + && typeof a[key] === 'function') { // and a function ... make it a getter + Object.defineProperty(a, key, { get: a[key], enumerable: true, configurable: true, writabel: false }); + } + if (typeof a[key] === 'string' && a[key][0] === '@') { // referring values by neighbor id's + const refidIdx = a[key].indexOf('.'); + const refid = refidIdx > 0 ? a[key].substr(1, refidIdx - 1) : ''; + const refkey = refid ? a[key].substr(refidIdx + 1) : ''; + const refcmd = refid ? () => this.commands.find((cmd) => cmd.a && cmd.a.id === refid) : undefined; + + if (refcmd) + Object.defineProperty(a, key, { + get: function () { + const rc = refcmd(); + return rc && (refkey in rc.a) ? rc.a[refkey] : 0; + }, + enumerable: true, + configurable: true, + writabel: false + }); + } + } + if (g2.prototype[c].prototype) Object.setPrototypeOf(a, g2.prototype[c].prototype); + } + this.commands.push(arguments[0]); + return this; + } +}; + +// statics +g2.defaultStyle = { fs: 'transparent', ls: '#000', lw: 1, lc: "butt", lj: "miter", ld: [], ml: 10, sh: [0, 0], lsh: false, font: '14px serif', thal: 'start', tval: 'alphabetic' }; +g2.symbol = { + unknown: g2().cir({ r: 12, fs: 'orange' }).txt({ str: '?', thal: 'center', tval: 'middle', font: 'bold 20pt serif' }) +}; +g2.handler = function (ctx) { + let hdl; + for (let h of g2.handler.factory) + if ((hdl = h(ctx)) !== false) + return hdl; + return false; +} +g2.handler.factory = []; + +// predefined polyline/spline point iterators +g2.pntIterator = { + "x,y": function (pts) { + function pitr(i) { return { x: pts[2 * i], y: pts[2 * i + 1] }; }; + Object.defineProperty(pitr, 'len', { get: () => pts.length / 2, enumerable: true, configurable: true, writabel: false }); + return pitr; + }, + "[x,y]": function (pts) { + function pitr(i) { return pts[i] ? { x: pts[i][0], y: pts[i][1] } : undefined; }; + Object.defineProperty(pitr, 'len', { get: () => pts.length, enumerable: true, configurable: true, writabel: false }); + return pitr; + }, + "{x,y}": function (pts) { + function pitr(i) { return pts[i]; }; + Object.defineProperty(pitr, 'len', { get: () => pts.length, enumerable: true, configurable: true, writabel: false }); + return pitr; + } +}; +g2.pntItrOf = function (pts) { + return !(pts && pts.length) ? undefined + : typeof pts[0] === "number" ? g2.pntIterator["x,y"](pts) + : Array.isArray(pts[0]) && pts[0].length >= 2 ? g2.pntIterator["[x,y]"](pts) + : typeof pts[0] === "object" && "x" in pts[0] && "y" in pts[0] ? g2.pntIterator["{x,y}"](pts) + : undefined; +}; +/** + * Get index of command resolving 'callbk' to 'true' starting from end of the queue walking back.
+ * Similar to 'Array.prototype.findIndex', only working reverse. + * @private + */ +g2.cmdIdxBy = function (cmds, callbk) { + for (let i = cmds.length - 1; i >= 0; i--) + if (callbk(cmds[i], i, cmds)) + return i; + return false; // command with index '0' signals 'failing' ... +}; + +/** + * Replacement for Object.assign, as it does not assign getters and setter properly ... + * See https://github.com/tc39/proposal-object-getownpropertydescriptors + * See https://medium.com/@benastontweet/mixins-in-javascript-700ec81f5e5c + * Shallow copy of prototypes (think interfaces) + * @private + */ +g2.mix = function mix(...protos) { + let mixture = {}; + for (const p of protos) + mixture = Object.defineProperties(mixture, Object.getOwnPropertyDescriptors(p)); + return mixture; +} + +/** + * Copy properties, even as getters .. a useful part of the above .. + * @private + */ +g2.cpyProp = function (from, fromKey, to, toKey) { Object.defineProperty(to, toKey, Object.getOwnPropertyDescriptor(from, fromKey)); } + +// Html canvas handler +g2.canvasHdl = function (ctx) { + if (this instanceof g2.canvasHdl) { + if (ctx instanceof CanvasRenderingContext2D) { + this.ctx = ctx; + this.cur = g2.defaultStyle; + this.stack = [this.cur]; + this.matrix = [[1, 0, 0, 1, 0.5, 0.5]]; + this.gridBase = 2; + this.gridExp = 1; + return this; + } + else + return null; + } + return g2.canvasHdl.apply(Object.create(g2.canvasHdl.prototype), arguments); +}; +g2.handler.factory.push((ctx) => ctx instanceof g2.canvasHdl ? ctx + : ctx instanceof CanvasRenderingContext2D ? g2.canvasHdl(ctx) : false); + +g2.canvasHdl.prototype = { + init(grp, style) { + this.stack.length = 1; + this.matrix.length = 1; + this.initStyle(style ? Object.assign({}, this.cur, style) : this.cur); + return true; + }, + async exe(commands) { + for (let cmd of commands) { + // cmd.a is an object offering a `g2` method, so call it and execute its returned commands array. + if (cmd.a && cmd.a.g2) { + const cmds = cmd.a.g2().commands; + // If false, ext was not applied to this cmd. But the command still renders + if (cmds) { + this.exe(cmds); + continue; + } + } + // cmd.a is a `g2` object, so directly execute its commands array. + else if (cmd.a && cmd.a.commands) { + this.exe(cmd.a.commands); + continue; + } + if (cmd.c && this[cmd.c]) { // explicit command name .. ! + const rx = this[cmd.c](cmd.a); + if (rx && rx instanceof Promise) { + await rx; + } + } + } + }, + view({ x = 0, y = 0, scl = 1, cartesian = false }) { + this.pushTrf(cartesian ? [scl, 0, 0, -scl, x, this.ctx.canvas.height - 1 - y] + : [scl, 0, 0, scl, x, y]); + }, + grid({ color = '#ccc', size } = {}) { + let ctx = this.ctx, b = ctx.canvas.width, h = ctx.canvas.height, + { x, y, scl } = this.uniTrf, + sz = size || this.gridSize(scl), + xoff = x % sz, yoff = y % sz; + ctx.save(); + ctx.setTransform(1, 0, 0, 1, 0, 0); + ctx.strokeStyle = color; + ctx.lineWidth = 1; + ctx.beginPath(); + for (let x = xoff, nx = b + 1; x < nx; x += sz) { ctx.moveTo(x, 0); ctx.lineTo(x, h); } + for (let y = yoff, ny = h + 1; y < ny; y += sz) { ctx.moveTo(0, y); ctx.lineTo(b, y); } + ctx.stroke(); + ctx.restore(); + }, + clr({ b, h } = {}) { + let ctx = this.ctx; + ctx.save(); + ctx.setTransform(1, 0, 0, 1, 0, 0); + ctx.clearRect(0, 0, b || ctx.canvas.width, h || ctx.canvas.height); + ctx.restore(); + }, + cir({ r }) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + this.ctx.beginPath(); + this.ctx.arc(x || 0, y || 0, Math.abs(r), 0, 2 * Math.PI, true); + this.drw(arguments[0]); + }, + arc({ r, w = 0, dw = 2 * Math.PI }) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + if (Math.abs(dw) > Number.EPSILON && Math.abs(r) > Number.EPSILON) { + this.ctx.beginPath(); + this.ctx.arc(x, y, Math.abs(r), w, w + dw, dw < 0); + this.drw(arguments[0]); + } + else if (Math.abs(dw) < Number.EPSILON && Math.abs(r) > Number.EPSILON) { + const cw = Math.cos(w), sw = Math.sin(w); + this.ctx.beginPath(); + this.ctx.moveTo(x - r * cw, y - r * sw); + this.ctx.lineTo(x + r * cw, y + r * sw); + } + // else // nothing to draw with r === 0 + }, + ell({ rx, ry, w = 0, dw = 2 * Math.PI, rot = 0 }) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + this.ctx.beginPath(); + this.ctx.ellipse(x, y, Math.abs(rx), Math.abs(ry), rot, w, w + dw, dw < 0); + this.drw(arguments[0]); + }, + rec({ b, h }) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + const tmp = this.setStyle(arguments[0]); + this.ctx.fillRect(x, y, b, h); + this.ctx.strokeRect(x, y, b, h); + this.resetStyle(tmp); + }, + lin(args) { + this.ctx.beginPath(); + this.ctx.moveTo(args.p1 && args.p1.x || args.x1 || 0, args.p1 && args.p1.y || args.y1 || 0); + this.ctx.lineTo(args.p2 && args.p2.x || args.x2 || 0, args.p2 && args.p2.y || args.y2 || 0); + this.stroke(args); + }, + ply({ pts, closed, w = 0, _itr }) { + if (_itr && _itr.len) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + let p, i, len = _itr.len, istrf = !!(x || y || w), cw, sw; + if (istrf) this.setTrf([cw = (w ? Math.cos(w) : 1), sw = (w ? Math.sin(w) : 0), -sw, cw, x, y]); + this.ctx.beginPath(); + this.ctx.moveTo((p = _itr(0)).x, p.y); + for (i = 1; i < len; i++) + this.ctx.lineTo((p = _itr(i)).x, p.y); + if (closed) // closed then .. + this.ctx.closePath(); + this.drw(arguments[0]); + if (istrf) this.resetTrf(); + return i - 1; // number of points .. + } + return 0; + }, + txt({ str, w = 0/*,unsizable*/ }) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + const tmp = this.setStyle(arguments[0]), + sw = w ? Math.sin(w) : 0, + cw = w ? Math.cos(w) : 1, + trf = this.isCartesian ? [cw, sw, sw, -cw, x, y] + : [cw, sw, -sw, cw, x, y]; + this.setTrf(trf); // this.setTrf(unsizable ? this.concatTrf(this.unscaleTrf({x,y}),trf) : trf); + if (this.ctx.fillStyle === 'rgba(0, 0, 0, 0)') { + this.ctx.fillStyle = this.ctx.strokeStyle; + tmp.fs = 'transparent'; + } + this.ctx.fillText(str, 0, 0); + this.resetTrf(); + this.resetStyle(tmp); + }, + errorImageStr: "", + images: Object.create(null), + async loadImage(uri) { + const download = async (xuri) => { + const pimg = new Promise((resolve, reject) => { + let img = new Image(); + img.src = xuri; + function error(err) { + img.removeEventListener('load', load); + img = undefined; + reject(err); + }; + function load() { + img.removeEventListener('error', error); + resolve(img); + img = undefined; + }; + img.addEventListener('error', error, { once: true }); + img.addEventListener('load', load, { once: true }); + }); + + try { + return await pimg; + } catch (err) { + // console.warn(`failed to (pre-)load image; '${xuri}'`, err); + if (xuri === this.errorImageStr) { + throw err; + } else { + return await download(this.errorImageStr); + } + } + } + + let img = this.images[uri]; + if (img !== undefined) { + return img instanceof Promise ? await img : img; + } + img = download(uri); + this.images[uri] = img; + try { + img = await img; + } finally { + this.images[uri] = img; + } + return img; + }, + async img({ uri, x = 0, y = 0, b, h, sx = 0, sy = 0, sb, sh, xoff = 0, yoff = 0, w = 0, scl = 1 }) { + const img_ = await this.loadImage(uri); + this.ctx.save(); + const cart = this.isCartesian ? -1 : 1; + sb = sb || img_.width; + b = b || img_.width; + sh = (sh || img_.height); + h = (h || img_.height) * cart; + yoff *= cart; + w *= cart; + y = this.isCartesian ? -(y / scl) + sy : y / scl; + const [cw, sw] = [Math.cos(w), Math.sin(w)]; + this.ctx.scale(scl, scl * cart); + this.ctx.transform(cw, sw, -sw, cw, x / scl, y); + this.ctx.drawImage(img_, sx, sy, sb, sh, xoff, yoff, b, h); + this.ctx.restore(); + }, + use({ grp }) { + this.beg(arguments[0]); + this.exe(grp.commands); + this.end(); + }, + beg({ w = 0, scl = 1, matrix/*,unsizable*/ } = {}) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + let trf = matrix; + if (!trf) { + let ssw, scw; + ssw = w ? Math.sin(w) * scl : 0; + scw = w ? Math.cos(w) * scl : scl; + trf = [scw, ssw, -ssw, scw, x, y]; + } + this.pushStyle(arguments[0]); + this.pushTrf(trf); // this.pushTrf(unsizable ? this.concatTrf(this.unscaleTrf({x,y}),trf) : trf); + }, + end() { + this.popTrf(); + this.popStyle(); + }, + p() { this.ctx.beginPath(); }, + z() { this.ctx.closePath(); }, + m({ x, y }) { this.ctx.moveTo(x, y); }, + l({ x, y }) { this.ctx.lineTo(x, y); }, + q({ x, y, x1, y1 }) { this.ctx.quadraticCurveTo(x1, y1, x, y); }, + c({ x, y, x1, y1, x2, y2 }) { this.ctx.bezierCurveTo(x1, y1, x2, y2, x, y); }, + a({ dw, k, phi, _xp, _yp }) { // todo: fix elliptical arc bug ... + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + if (k === undefined) k = 1; // ratio r1/r2 + if (Math.abs(dw) > Number.EPSILON) { + if (k === 1) { // circular arc ... + let x12 = x - _xp, y12 = y - _yp; + let tdw_2 = Math.tan(dw / 2), + rx = (x12 - y12 / tdw_2) / 2, ry = (y12 + x12 / tdw_2) / 2, + R = Math.hypot(rx, ry), + w = Math.atan2(-ry, -rx); + this.ctx.ellipse(_xp + rx, _yp + ry, R, R, 0, w, w + dw, this.cartesian ? dw > 0 : dw < 0); + } + else { // elliptical arc .. still buggy .. ! + if (phi === undefined) phi = 0; + let x1 = dw > 0 ? _xp : x, + y1 = dw > 0 ? _yp : y, + x2 = dw > 0 ? x : _xp, + y2 = dw > 0 ? y : _yp; + let x12 = x2 - x1, y12 = y2 - y1, + _dw = (dw < 0) ? dw : -dw; + // if (dw < 0) dw = -dw; // test for bugs .. ! + let cp = phi ? Math.cos(phi) : 1, sp = phi ? Math.sin(phi) : 0, + dx = -x12 * cp - y12 * sp, dy = -x12 * sp - y12 * cp, + sdw_2 = Math.sin(_dw / 2), + R = Math.sqrt((dx * dx + dy * dy / (k * k)) / (4 * sdw_2 * sdw_2)), + w = Math.atan2(k * dx, dy) - _dw / 2, + x0 = x1 - R * Math.cos(w), + y0 = y1 - R * k * Math.sin(w); + this.ctx.ellipse(x0, y0, R, R * k, phi, w, w + dw, this.cartesian ? dw > 0 : dw < 0); + } + } + else + this.ctx.lineTo(x, y); + }, + + stroke({ d } = {}) { + let tmp = this.setStyle(arguments[0]); + d ? this.ctx.stroke(new Path2D(d)) : this.ctx.stroke(); // SVG path syntax + this.resetStyle(tmp); + }, + fill({ d } = {}) { + let tmp = this.setStyle(arguments[0]); + d ? this.ctx.fill(new Path2D(d)) : this.ctx.fill(); // SVG path syntax + this.resetStyle(tmp); + }, + drw({ d, lsh } = {}) { + let ctx = this.ctx, + tmp = this.setStyle(arguments[0]), + p = d && new Path2D(d); // SVG path syntax + d ? ctx.fill(p) : ctx.fill(); + if (ctx.shadowColor !== 'rgba(0, 0, 0, 0)' && ctx.fillStyle !== 'rgba(0, 0, 0, 0)' && !lsh) { + let shc = ctx.shadowColor; // usually avoid stroke shadow when filling ... + ctx.shadowColor = 'rgba(0, 0, 0, 0)'; + d ? ctx.stroke(p) : ctx.stroke(); + ctx.shadowColor = shc; + } + else + d ? ctx.stroke(p) : ctx.stroke(); + this.resetStyle(tmp); + }, + + // State management (transform & style) + // getters & setters + get: { + fs: (ctx) => ctx.fillStyle, + ls: (ctx) => ctx.strokeStyle, + lw: (ctx) => ctx.lineWidth, + lc: (ctx) => ctx.lineCap, + lj: (ctx) => ctx.lineJoin, + ld: (ctx) => ctx.getLineDash(), + ldoff: (ctx) => ctx.lineDashOffset, + ml: (ctx) => ctx.miterLimit, + sh: (ctx) => [ctx.shadowOffsetX || 0, ctx.shadowOffsetY || 0, + ctx.shadowBlur || 0, ctx.shadowColor || 'black'], + font: (ctx) => ctx.font, + thal: (ctx) => ctx.textAlign, + tval: (ctx) => ctx.textBaseline, + }, + set: { + fs: (ctx, q) => { ctx.fillStyle = q; }, + ls: (ctx, q) => { ctx.strokeStyle = q; }, + lw: (ctx, q) => { ctx.lineWidth = q; }, + lc: (ctx, q) => { ctx.lineCap = q; }, + lj: (ctx, q) => { ctx.lineJoin = q; }, + ld: (ctx, q) => { ctx.setLineDash(q); }, + ldoff: (ctx, q) => { ctx.lineDashOffset = q; }, + ml: (ctx, q) => { ctx.miterLimit = q; }, + sh: (ctx, q) => { + if (q) { + ctx.shadowOffsetX = q[0] || 0; + ctx.shadowOffsetY = q[1] || 0; + ctx.shadowBlur = q[2] || 0; + ctx.shadowColor = q[3] || 'black'; + } + }, + font: (ctx, q) => { ctx.font = q; }, + thal: (ctx, q) => { ctx.textAlign = q; }, + tval: (ctx, q) => { ctx.textBaseline = q; } + }, + initStyle(style) { + for (const key in style) + if (this.get[key] && this.get[key](this.ctx) !== style[key]) + this.set[key](this.ctx, style[key]); + }, + setStyle(style) { // short circuit style setting + let q, prv = {}; + for (const key in style) { + if (this.get[key]) { // style keys only ... + if (typeof style[key] === 'string' && style[key][0] === '@') { + let ref = style[key].substr(1); + style[key] = g2.symbol[ref] || this.get[ref] && this.get[ref](this.ctx); + } + if ((q = this.get[key](this.ctx)) !== style[key]) { + prv[key] = q; + this.set[key](this.ctx, style[key]); + } + } + } + return prv; + }, + resetStyle(style) { // short circuit style reset + for (const key in style) + this.set[key](this.ctx, style[key]); + }, + pushStyle(style) { + let cur = {}; // hold changed properties ... + for (const key in style) + if (this.get[key]) { // style keys only ... + if (typeof style[key] === 'string' && style[key][0] === '@') { + let ref = style[key].substr(1); + style[key] = g2.symbol[ref] || this.get[ref] && this.get[ref](this.ctx); + } + if (this.cur[key] !== style[key]) + this.set[key](this.ctx, (cur[key] = style[key])); + } + this.stack.push(this.cur = Object.assign({}, this.cur, cur)); + }, + popStyle() { + let cur = this.stack.pop(); + this.cur = this.stack[this.stack.length - 1]; + for (const key in this.cur) + if (this.get[key] && this.cur[key] !== cur[key]) + this.set[key](this.ctx, this.cur[key]); + }, + concatTrf(q, t) { + return [ + q[0] * t[0] + q[2] * t[1], + q[1] * t[0] + q[3] * t[1], + q[0] * t[2] + q[2] * t[3], + q[1] * t[2] + q[3] * t[3], + q[0] * t[4] + q[2] * t[5] + q[4], + q[1] * t[4] + q[3] * t[5] + q[5] + ]; + }, + initTrf() { + this.ctx.setTransform(...this.matrix[0]); + }, + setTrf(t) { + this.ctx.setTransform(...this.concatTrf(this.matrix[this.matrix.length - 1], t)); + }, + resetTrf() { + this.ctx.setTransform(...this.matrix[this.matrix.length - 1]); + }, + pushTrf(t) { + let q_t = this.concatTrf(this.matrix[this.matrix.length - 1], t); + this.matrix.push(q_t); + this.ctx.setTransform(...q_t); + }, + popTrf() { + this.matrix.pop(); + this.ctx.setTransform(...this.matrix[this.matrix.length - 1]); + }, + get isCartesian() { // det of mat2x2 < 0 ! + let m = this.matrix[this.matrix.length - 1]; + return m[0] * m[3] - m[1] * m[2] < 0; + }, + get uniTrf() { + let m = this.matrix[this.matrix.length - 1]; + return { x: m[4], y: m[5], scl: Math.hypot(m[0], m[1]), cartesian: m[0] * m[3] - m[1] * m[2] < 0 }; + }, + unscaleTrf({ x, y }) { // remove scaling effect (make unzoomable with respect to (x,y)) + let m = this.matrix[this.matrix.length - 1], + invscl = 1 / Math.hypot(m[0], m[1]); + return [invscl, 0, 0, invscl, (1 - invscl) * x, (1 - invscl) * y]; + }, + gridSize(scl) { + let base = this.gridBase, exp = this.gridExp, sz; + while ((sz = scl * base * Math.pow(10, exp)) < 14 || sz > 35) { + if (sz < 14) { + if (base == 1) base = 2; + else if (base == 2) base = 5; + else if (base == 5) { base = 1; exp++; } + } + else { + if (base == 1) { base = 5; exp--; } + else if (base == 2) base = 1; + else if (base == 5) base = 2; + } + } + this.gridBase = base; + this.gridExp = exp; + return sz; + } +} + +// use it with node.js ... ? +if (typeof module !== 'undefined') module.exports = g2; + +/** + * g2.io (c) 2017-18 Stefan Goessner + * @license MIT License + */ +"use strict"; + +g2.io = function() { + if (this instanceof g2.io) { + this.model = null; + this.grpidx = 0; + return this; + } + return g2.io.apply(Object.create(g2.io.prototype)); +}; +g2.handler.factory.push((ctx) => ctx instanceof g2.io ? ctx : false); + +g2.io.parseGrp = function(model, id, onErr) { + let g; + onErr = onErr || console.error; + if (id in model) { + g = g2({id}); + for (let cmd of model[id]) { + if (cmd.c === 'use') { + cmd.a.grp = g2.io.parseGrp(model, cmd.a.grp); + g[cmd.c](cmd.a); + } + else if (g[cmd.c]) + cmd.a ? g[cmd.c](cmd.a) : g[cmd.c](); + else // invalid g2 command ! + onErr(`g2.io: Unable to handle command '${cmd.c}'`) + } + return g; + } + else if (id in g2.symbol) + return g2.symbol[id]; + else + onErr(`g2.io: Unable to find group with id '${id}'!`); + return false; +} + +g2.io.prototype = { + init: function(grp,style) { + this.model = {'main':[]}; + this.curgrp = this.model.main; + this.grpidx = 0; + return true; + }, + exe: function(commands) { + for (let cmd of commands) { + if (this[cmd.c]) + cmd.a ? this[cmd.c](cmd.a) : this[cmd.c](); + else + this.out(cmd.c,cmd.a); + } + }, + out: function(c,a) { + if (a) { + let args = {}; + Object.keys(a).forEach(k => { + if (k[0] !== '_') // no private key ... + args[k] = a[k]; + }); + this.curgrp.push({c:c,a:args}); + } + else + this.curgrp.push({c:c}); + }, + stringify: function(space) { return space ? JSON.stringify(this.model,null,space) : JSON.stringify(this.model); }, + toString: function() { return JSON.stringify(this.model); }, + + // customized commands ... + use: function(args) { + let grp = args.grp instanceof g2 ? args.grp + : typeof args.grp === 'string' && g2.symbol.includes(args.grp) ? g2.symbol[args.grp] + : null; + if (grp) { + if (!grp.id) grp.id = `$grp${++this.grpidx}`; + if (!(grp.id in this.model)) { // meet first time .. + let curgrp = this.curgrp; + this.curgrp = this.model[grp.id] = []; + for (let command of grp.commands) + this.out(command.c,command.a); + this.curgrp = curgrp; + } + args.grp = grp.id; + + this.out("use", args); + } + } +} + +/** + * g2.lib (c) 2013-17 Stefan Goessner + * geometric constants and higher functions + * @license MIT License + * @link https://github.com/goessner/g2 + */ +"use strict" + +var g2 = g2 || {}; // for standalone usage ... + +g2 = Object.assign(g2, { + EPS: Number.EPSILON, + PI: Math.PI, + PI2: 2*Math.PI, + SQRT2: Math.SQRT2, + SQRT2_2: Math.SQRT2/2, + /** + * Map angle to the range [0 .. 2*pi]. + * @param {number} w Angle in radians. + * @returns {number} Angle in radians in interval [0 .. 2*pi]. + */ + toPi2(w) { return (w % g2.PI2 + g2.PI2) % g2.PI2; }, + /** + * Map angle to the range [-pi .. pi]. + * @param {number} w Angle in radians. + * @returns {number} Angle in radians in interval [-pi .. pi]. + */ + toPi(w) { return (w = (w % g2.PI2 + g2.PI2) % g2.PI2) > g2.PI ? w - g2.PI2 : w; }, + /** + * Map angle to arc sector [w0 .. w0+dw]. + * @param {number} w Angle in range [0 .. 2*pi]. + * @param {number} w0 Start angle in range [0 .. 2*pi]. + * @param {number} dw angular range in radians. Can be positive or negative. + * @returns {number} Normalised angular parameter lambda. + * '0' corresponds to w0 and '1' to w0+dw. To reconstruct an angle from + * the return parameter lambda use: w = w0 + lambda*dw. + */ + toArc: function(w,w0,dw) { + if (dw > g2.EPS || dw < -g2.EPS) { + if (w0 > w && w0+dw > g2.PI2) w0 -= g2.PI2; + else if (w0 < w && w0+dw < 0) w0 += g2.PI2; + return (w-w0)/dw; + + } + return 0; + }, + /** + * Test, if point is located on line. + * @param {x,y} point to test. + * @param {x1,y1} start point of line. + * @param {x2,y2} end point of line. + * @param {number} eps. + * @return {boolean} the test result. + */ + isPntOnLin({x,y},p1,p2,eps=Number.EPSILON) { + let dx = p2.x - p1.x, dy = p2.y - p1.y, dx2 = x - p1.x, dy2 = y - p1.y, + dot = dx*dx2 + dy*dy2, perp = dx*dy2 - dy*dx2, len = Math.hypot(dx,dy), epslen = eps*len; + return -epslen < perp && perp < epslen && -epslen < dot && dot < len*(len+eps); + }, + /** + * Test, if point is located on circle circumference. + * @param {x,y} point to test. + * @param {x,y,r} circle. + * @param {number} eps. + * @return {boolean} the test result. + */ + isPntOnCir({x:xp,y:yp},{x,y,r},eps=Number.EPSILON) { + let dx = xp - x, dy = yp - y, + ddis = dx*dx + dy*dy - r*r, reps = eps*r; + return -reps < ddis && ddis < reps; + }, + /** + * Test, if point is located on a circular arc. + * @param {x,y} point to test. + * @param {x,y,r} circle. + * @param {number} eps. + * @return {boolean} the test result. + */ + isPntOnArc({x:xp,y:yp},{x,y,r,w,dw},eps=Number.EPSILON) { + var dx = xp - x, dy = yp - y, dist = Math.hypot(dx,dy), + mu = g2.toArc(g2.toPi2(Math.atan2(dy,dx)),g2.toPi2(w),dw); + return r*Math.abs(dw) > eps && Math.abs(dist-r) < eps && mu >= 0 && mu <= 1; + }, + /** + * Test, if point is located on a polygon line. + * @param {x,y} point to test. + * @param {pts,closed} polygon. + * @param {number} eps. + * @return {boolean} the test result. + */ + isPntOnPly({x,y},{pts,closed},eps=Number.EPSILON) { + // console.log(pts) + for (var i=0,n=pts.length; i<(closed ? n : n-1); i++) + if (g2.isPntOnLin({x,y},pts[i],pts[(i+1)%n],eps)) + return true; + return false; + }, + /** + * Test, if point is located on a box. A box in contrast to a rectangle + * is always aligned parallel to coordinate system axes, with its + * local origin `{x,y}` located in the center. The dimensions `{b,h}` are + * half size dimensions (so upper right corner is {x+b,y+h}). + * @param {x,y} point to test. + * @param {x,y,r} circle. + * @param {number} eps. + * @return {boolean} the test result. + */ + isPntOnBox({x:xp,y:yp},{x,y,b,h},eps=Number.EPSILON) { + var dx = x.p - x, dy = yp - y; + return dx >= b-eps && dx <= b+eps && dy <= h+eps && dy >= -h-eps + || dx >= -b-eps && dx <= b+eps && dy <= h+eps && dy >= h-eps + || dx >= -b-eps && dx <= -b+eps && dy <= h+eps && dy >= -h-eps + || dx >= -b-eps && dx <= b+eps && dy <= -h+eps && dy >= -h-eps; + }, + /** + * Test, if point is located inside of a circle. + * @param {x,y} point to test. + * @param {x,y,r} circle. + * @return {boolean} the test result. + */ + isPntInCir({x:xp,y:yp},{x,y,r}) { + return (x - xp)**2 + (y - yp)**2 < r*r; + }, + /** + * Test, if point is located inside of a closed polygon. + * (see http://paulbourke.net/geometry/polygonmesh/) + * @param {x,y} point to test. + * @param {pnts,closed} polygon. + * @returns {boolean} point is on polygon lines. + */ + isPntInPly({x,y},{pts,closed},eps=Number.EPSILON) { + let match = 0; + for (let n=pts.length,i=0,pi=pts[i],pj=pts[n-1]; i pi.y || y > pj.y) + && (y <= pi.y || y <= pj.y) + && (x <= pi.x || x <= pj.x) + && pi.y !== pj.y + && (pi.x === pj.x || x <= pj.x + (y-pj.y)*(pi.x-pj.x)/(pi.y-pj.y))) + match++; + return match%2 != 0; // even matches required for being outside .. + }, + /** + * Test, if point is located inside of a box. A box in contrast to a rectangle + * is always aligned parallel to coordinate system axes, with its + * local origin `{x,y}` located in the center. The dimensions `{b,h}` are + * half size dimensions (so upper right corner is {x+b,y+h}). + * @param {x,y} point to test. + * @param {x,y,r} circle. + * @return {boolean} the test result. + */ + isPntInBox({x:xp,y:yp},{x,y,b,h}) { + var dx = xp - x, dy = yp - y; + return dx >= -b && dx <= b && dy >= -h && dy <= h; + }, + + arc3pts(x1,y1,x2,y2,x3,y3) { + const dx1 = x2 - x1, dy1 = y2 - y1; + const dx2 = x3 - x2, dy2 = y3 - y2; + const den = dx1*dy2 - dy1*dx2; + const lam = Math.abs(den) > Number.EPSILON + ? 0.5*((dx1 + dx2)*dx2 + (dy1 + dy2)*dy2)/den + : 0; + const x0 = lam ? x1 + 0.5*dx1 - lam*dy1 : x1 + 0.5*(dx1 + dx2); + const y0 = lam ? y1 + 0.5*dy1 + lam*dx1 : y1 + 0.5*(dy1 + dy2); + const dx01 = x1 - x0, dy01 = y1 - y0; + const dx03 = x3 - x0, dy03 = y3 - y0; + const dw = lam ? Math.atan2(dx01*dy03-dy01*dx03,dx01*dx03+dy01*dy03) : 0; + const r = dw ? Math.hypot(dy01,dx01) : 0.5*Math.hypot(dy1+dy2,dx1+dx2); + + return {x:x0,y:y0,r:r,w:Math.atan2(dy01,dx01),dw}; + } +}) + +"use strict" + +/** + * g2.ext (c) 2015-20 Stefan Goessner + * @author Stefan Goessner + * @license MIT License + * @requires g2.core.js + * @typedef {g2} + * @description Additional methods for g2. + * @returns {g2} + */ + +var g2 = g2 || { prototype: {} }; // for jsdoc only ... + +// constants for element selection / editing +g2.NONE = 0x0; g2.OVER = 0x1; g2.DRAG = 0x2; g2.EDIT = 0x4; + +/** + * Extended style values. + * Not really meant to get overwritten. But if you actually want, proceed.
+ * These styles can be referenced using the comfortable '@' syntax. + * @namespace + * @property {object} symbol `g2` symbol namespace. + * @property {object} [symbol.tick] Predefined symbol: a little tick + * @property {object} [symbol.dot] Predefined symbol: a little dot + * @property {object} [symbol.sqr] Predefined symbol: a little square + * @property {string} [symbol.nodcolor=#333] node color. + * @property {string} [symbol.nodfill=#dedede] node fill color. + * @property {string} [symbol.nodfill2=#aeaeae] alternate node fill color, somewhat darker. + * @property {string} [symbol.linkcolor=#666] link color. + * @property {string} [symbol.linkfill=rgba(225,225,225,0.75)] link fill color, semi-transparent. + * @property {string} [symbol.dimcolor=darkslategray] dimension color. + * @property {array} [symbol.solid=[]] solid line style. + * @property {array} [symbol.dash=[15,10]] dashed line style. + * @property {array} [symbol.dot=[4,4]] dotted line style. + * @property {array} [symbol.dashdot=[25,6.5,2,6.5]] dashdotted line style. + * @property {number} [symbol.labelOffset=5] default label offset distance. + * @property {number} [symbol.labelSignificantDigits=3] default label's significant digits after numbering point. + */ +g2.symbol = g2.symbol || {}; +g2.symbol.tick = g2().p().m({ x: 0, y: -2 }).l({ x: 0, y: 2 }).stroke({ lc: "round", lwnosc: true }); +g2.symbol.dot = g2().cir({ x: 0, y: 0, r: 2, ls: "transparent" }); +g2.symbol.sqr = g2().rec({ x: -1.5, y: -1.5, b: 3, h: 3, ls: "transparent" }); + +g2.symbol.nodcolor = "#333"; +g2.symbol.nodfill = "#dedede"; +g2.symbol.nodfill2 = "#aeaeae"; +g2.symbol.linkcolor = "#666"; +g2.symbol.linkfill = "rgba(225,225,225,0.75)"; +g2.symbol.dimcolor = "darkslategray"; +g2.symbol.solid = []; +g2.symbol.dash = [15, 10]; +g2.symbol.dot = [4, 4]; +g2.symbol.dashdot = [25, 6.5, 2, 6.5]; +g2.symbol.labelSignificantDigits = 3; // 0.1234 => 0.123, 0.01234 => 0.0123, 1.234 => 1.23, 12.34 => 12.3, 123.4 => 123, 1234 => 1234 + +/** +* Flatten object properties (evaluate getters) +*/ +g2.flatten = function (obj) { + const args = Object.create(null); // important ! + for (let p in obj) + if (typeof obj[p] !== 'function') + args[p] = obj[p]; + return args; +} +/* +g2.strip = function(obj,prop) { + const clone = Object.create(Object.getPrototypeOf(obj),Object.getOwnPropertyDescriptors(obj)); + Object.defineProperty(clone, prop, { get:undefined, enumerable:true, configurable:true, writabel:false }); + return clone; +} +*/ +g2.pointIfc = { + // p vector notation ! ... helps to avoid object destruction + get p() { return { x: this.x, y: this.y }; }, // visible if 'p' is *not* explicite given. + get x() { return Object.getOwnPropertyDescriptor(this, 'p') ? this.p.x : 0; }, + get y() { return Object.getOwnPropertyDescriptor(this, 'p') ? this.p.y : 0; }, + set x(q) { if (Object.getOwnPropertyDescriptor(this, 'p')) this.p.x = q; }, + set y(q) { if (Object.getOwnPropertyDescriptor(this, 'p')) this.p.y = q; }, +} + +g2.labelIfc = { + getLabelOffset() { const off = this.label.off !== undefined ? +this.label.off : 1; return off + Math.sign(off) * (this.lw || 2) / 2; }, + getLabelString() { + let s = typeof this.label === 'object' ? this.label.str : typeof this.label === 'string' ? this.label : '?'; + if (s && s[0] === "@" && this[s.substr(1)]) { + s = s.substr(1); + let val = this[s]; + val = Number.isInteger(val) ? val + : Number(val).toFixed(Math.max(g2.symbol.labelSignificantDigits - Math.log10(val), 0)); + + s = `${val}${s === 'angle' ? "°" : ""}`; + } + return s; + }, + drawLabel(g) { + const lbl = this.label; + const font = lbl.font || g2.defaultStyle.font; + const h = parseInt(font); // font height (px assumed !) + const str = this.getLabelString(); + const rx = (str.length || 1) * 0.75 * h / 2, ry = 1.25 * h / 2; // ellipse semi-axes length + const pos = this.pointAt(lbl.loc || this.lbloc || 'se'); + const off = this.getLabelOffset(); + const p = { + x: pos.x + pos.nx * (off + Math.sign(off) * rx), + y: pos.y + pos.ny * (off + Math.sign(off) * ry) + }; + + if (lbl.border) g.ell({ x: p.x, y: p.y, rx, ry, ls: lbl.fs || 'black', fs: lbl.fs2 || '#ffc' }); + g.txt({ + str, x: p.x, y: p.y, + thal: "center", tval: "middle", + fs: lbl.fs || 'black', font: lbl.font + }); + return g; + } +} + +g2.markIfc = { + markAt(loc) { + const p = this.pointAt(loc); + const w = Math.atan2(p.ny, p.nx) + Math.PI / 2; + return { + grp: this.getMarkSymbol(), x: p.x, y: p.y, w: w, scl: this.lw || 1, + ls: this.ls || '#000', fs: this.fs || this.ls || '#000' + } + }, + getMarkSymbol() { + // Use tick as default + const mrk = this.mark + if (typeof mrk === 'number' || !mrk) return g2.symbol.tick; + if (typeof mrk.symbol === 'object') return mrk.symbol; + if (typeof mrk.symbol === 'string') return g2.symbol[mrk.symbol] + }, + // loop is for elements that close, e.g. rec or cir => loc at 0 === loc at 1 + drawMark(g, closed = false) { + let loc; + if (Array.isArray(this.mark)) { + loc = this.mark; + } + else { + const count = typeof this.mark === 'object' ? this.mark.count : this.mark; + loc = count ? + Array.from(Array(count)).map((_, i) => i / (count - !closed)) : + this.mark.loc; + } + for (let l of loc) { + g.use(this.markAt(l)); + } + return g; + } +} + +g2.prototype.cir.prototype = g2.mix(g2.pointIfc, g2.labelIfc, g2.markIfc, { + w: 0, // default start angle (used for dash-dot orgin and editing) + lbloc: 'c', + get isSolid() { return this.fs && this.fs !== 'transparent' }, + get len() { return 2 * Math.PI * this.r; }, + get lsh() { return this.state & g2.OVER; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false }, + get g2() { // dynamically switch existence of method via getter ... cool ! + const e = g2(); // hand object stripped from `g2` + this.label && e.ins((g) => this.drawLabel(g)); + this.mark && e.ins((g) => this.drawMark(g, true)); + return () => g2().cir(g2.flatten(this)).ins(e); // avoiding infinite recursion ! + }, + pointAt(loc) { + const Q = Math.SQRT2 / 2; + const LOC = { c: [0, 0], e: [1, 0], ne: [Q, Q], n: [0, 1], nw: [-Q, Q], w: [-1, 0], sw: [-Q, -Q], s: [0, -1], se: [Q, -Q] }; + const q = (loc + 0 === loc) ? [Math.cos(loc * 2 * Math.PI), Math.sin(loc * 2 * Math.PI)] + : (LOC[loc || "c"] || [0, 0]); + return { + x: this.x + q[0] * this.r, + y: this.y + q[1] * this.r, + nx: q[0], + ny: q[1] + }; + }, + hit({ x, y, eps }) { + return this.isSolid ? g2.isPntInCir({ x, y }, this, eps) + : g2.isPntOnCir({ x, y }, this, eps); + }, + drag({ dx, dy }) { this.x += dx; this.y += dy }, +}); + +g2.prototype.lin.prototype = g2.mix(g2.labelIfc, g2.markIfc, { + // p1 vector notation ! + get p1() { return { x1: this.x1, y1: this.y1 }; }, // relevant if 'p1' is *not* explicite given. + get x1() { return Object.getOwnPropertyDescriptor(this, 'p1') ? this.p1.x : 0; }, + get y1() { return Object.getOwnPropertyDescriptor(this, 'p1') ? this.p1.y : 0; }, + set x1(q) { if (Object.getOwnPropertyDescriptor(this, 'p1')) this.p1.x = q; }, + set y1(q) { if (Object.getOwnPropertyDescriptor(this, 'p1')) this.p1.y = q; }, + // p2 vector notation ! + get p2() { return { x2: this.x2, y2: this.y2 }; }, // relevant if 'p2' is *not* explicite given. + get x2() { return Object.getOwnPropertyDescriptor(this, 'p2') ? this.p2.x : 0; }, + get y2() { return Object.getOwnPropertyDescriptor(this, 'p2') ? this.p2.y : 0; }, + set x2(q) { if (Object.getOwnPropertyDescriptor(this, 'p2')) this.p2.x = q; }, + set y2(q) { if (Object.getOwnPropertyDescriptor(this, 'p2')) this.p2.y = q; }, + + isSolid: false, + get len() { return Math.hypot(this.x2 - this.x1, this.y2 - this.y1); }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false }, + get g2() { // dynamically switch existence of method via getter ... ! + const e = g2(); + this.label && e.ins(e => this.drawLabel(e)); + this.mark && e.ins(e => this.drawMark(e)); + return () => g2().lin(g2.flatten(this)).ins(e); + }, + + pointAt(loc) { + let t = loc === "beg" ? 0 + : loc === "end" ? 1 + : (loc + 0 === loc) ? loc // numerical arg .. + : 0.5, // 'mid' .. + dx = this.x2 - this.x1, + dy = this.y2 - this.y1, + len = Math.hypot(dx, dy); + return { + x: this.x1 + dx * t, + y: this.y1 + dy * t, + nx: len ? dy / len : 0, + ny: len ? -dx / len : -1 + }; + }, + hit({ x, y, eps }) { + return g2.isPntOnLin({ x, y }, { x: this.x1, y: this.y1 }, { x: this.x2, y: this.y2 }, eps); + }, + drag({ dx, dy }) { + this.x1 += dx; this.x2 += dx; + this.y1 += dy; this.y2 += dy; + } +}); + +g2.prototype.rec.prototype = g2.mix(g2.pointIfc, g2.labelIfc, g2.markIfc, { + get len() { return 2 * (this.b + this.h); }, + get isSolid() { return this.fs && this.fs !== 'transparent' }, + get lsh() { return this.state & g2.OVER; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false; }, + get g2() { // dynamically switch existence of method via getter ... ! + const e = g2(); + this.label && e.ins(e => this.drawLabel(e)); + this.mark && e.ins(e => this.drawMark(e, true)); + return () => g2().rec(g2.flatten(this)).ins(e); + }, + lbloc: 'c', + pointAt(loc) { + const locAt = (loc) => { + const o = { c: [0, 0], e: [1, 0], ne: [0.95, 0.95], n: [0, 1], nw: [-0.95, 0.95], w: [-1, 0], sw: [-0.95, -0.95], s: [0, -1], se: [0.95, -0.95] }; + + if (o[loc]) return o[loc]; + + const w = 2 * Math.PI * loc + pi / 4; + if (loc <= 0.25) return [1 / Math.tan(w), 1]; + if (loc <= 0.50) return [-1, -Math.tan(w)]; + if (loc <= 0.75) return [- 1 / Math.tan(w), -1]; + if (loc <= 1.00) return [1, Math.tan(w)]; + } + const q = locAt(loc); + return { + x: this.x + (1 + q[0]) * this.b / 2, + y: this.y + (1 + q[1]) * this.h / 2, + nx: 1 - Math.abs(q[0]) < 0.01 ? q[0] : 0, + ny: 1 - Math.abs(q[1]) < 0.01 ? q[1] : 0 + }; + }, + hit({ x, y, eps }) { + return this.isSolid ? g2.isPntInBox({ x, y }, { x: this.x + this.b / 2, y: this.y + this.h / 2, b: this.b / 2, h: this.h / 2 }, eps) + : g2.isPntOnBox({ x, y }, { x: this.x + this.b / 2, y: this.y + this.h / 2, b: this.b / 2, h: this.h / 2 }, eps); + }, + drag({ dx, dy }) { this.x += dx; this.y += dy } +}); + +g2.prototype.arc.prototype = g2.mix(g2.pointIfc, g2.labelIfc, g2.markIfc, { + get len() { return Math.abs(this.r * this.dw); }, + isSolid: false, + get angle() { return this.dw / Math.PI * 180; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false }, + get g2() { // dynamically switch existence of method via getter ... ! + const e = g2(); + this.label && e.ins(e => this.drawLabel(e)); + this.mark && e.ins(e => this.drawMark(e)); + return () => g2().arc(g2.flatten(this)).ins(e); + }, + lbloc: 'mid', + pointAt(loc) { + let t = loc === "beg" ? 0 + : loc === "end" ? 1 + : loc === "mid" ? 0.5 + : loc + 0 === loc ? loc + : 0.5, + ang = (this.w || 0) + t * (this.dw || Math.PI * 2), cang = Math.cos(ang), sang = Math.sin(ang), r = loc === "c" ? 0 : this.r; + return { + x: this.x + r * cang, + y: this.y + r * sang, + nx: cang, + ny: sang + }; + }, + hit({ x, y, eps }) { return g2.isPntOnArc({ x, y }, this, eps) }, + drag({ dx, dy }) { this.x += dx; this.y += dy; }, +}); + +/** +* Draw interactive handle. +* @method +* @returns {object} g2 +* @param {object} - handle object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().hdl({x:100,y:80}) // Draw handle. +*/ +g2.prototype.hdl = function (args) { return this.addCommand({ c: 'hdl', a: args }); } +g2.prototype.hdl.prototype = g2.mix(g2.prototype.cir.prototype, { + r: 5, + isSolid: true, + draggable: true, + lbloc: 'se', + get lsh() { return this.state & g2.OVER; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false }, + g2() { + const { x, y, r, b = 4, shape = 'cir', ls = 'black', fs = '#ccc', sh } = this; + return shape === 'cir' ? g2().cir({ x, y, r, ls, fs, sh }).ins((g) => this.label && this.drawLabel(g)) + : g2().rec({ x: x - b, y: y - b, b: 2 * b, h: 2 * b, ls, fs, sh }).ins((g) => this.label && this.drawLabel(g)); + } +}); + +/** +* Node symbol. +* @constructor +* @param {object} - symbol arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().nod({x:10,y:10}) +*/ + +g2.prototype.nod = function (args = {}) { return this.addCommand({ c: 'nod', a: args }); } +g2.prototype.nod.prototype = g2.mix(g2.prototype.cir.prototype, { + r: 5, + ls: '@nodcolor', + fs: g2.symbol.nodfill, + isSolid: true, + lbloc: 'se', + g2() { // in contrast to `g2.prototype.cir.prototype`, `g2()` is called always ! + return g2() + .cir({ ...g2.flatten(this), r: this.r * (this.scl !== undefined ? this.scl : 1) }) + .ins((g) => this.label && this.drawLabel(g)) + } +}); + +/** + * Double nod symbol + * @constructor + * @returns {object} g2 + * @param {object} - symbol arguments object. + * @property {number} x - x-value center. + * @property {number} y - y-value center. + * @example + * g2().dblnod({x:10,y:10}) +*/ +g2.prototype.dblnod = function ({ x = 0, y = 0 }) { return this.addCommand({ c: 'dblnod', a: arguments[0] }); } +g2.prototype.dblnod.prototype = g2.mix(g2.prototype.cir.prototype, { + get r() { return 6; }, + get isSolid() { return true; }, + g2() { + return g2() + .beg({ x: this.x, y: this.y }) + .cir({ r: 6, ls: '@nodcolor', fs: '@nodfill', sh: this.sh }) + .cir({ r: 3, ls: '@nodcolor', fs: '@nodfill2' }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) + +/** +* Pole symbol. +* @constructor +* @returns {object} g2 +* @param {object} - symbol arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().pol({x:10,y:10}) +*/ +g2.prototype.pol = function (args = {}) { return this.addCommand({ c: 'pol', a: args }); } +g2.prototype.pol.prototype = g2.mix(g2.prototype.nod.prototype, { + g2() { + return g2() + .beg(g2.flatten(this)) + .cir({ r: 6, fs: g2.symbol.nodfill }) + .cir({ r: 2.5, fs: '@ls', ls: 'transparent' }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) + +/** +* Ground symbol. +* @constructor +* @param {object} - arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().gnd({x:10,y:10}) +*/ +g2.prototype.gnd = function (args = {}) { return this.addCommand({ c: 'gnd', a: args }); } +g2.prototype.gnd.prototype = g2.mix(g2.prototype.nod.prototype, { + g2() { + return g2() + .beg(g2.flatten(this)) + .cir({ x: 0, y: 0, r: 6 }) + .p() + .m({ x: 0, y: 6 }) + .a({ dw: Math.PI / 2, x: -6, y: 0 }) + .l({ x: 6, y: 0 }) + .a({ dw: -Math.PI / 2, x: 0, y: -6 }) + .z() + .fill({ fs: g2.symbol.nodcolor }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) + +g2.prototype.nodfix = function (args = {}) { return this.addCommand({ c: 'nodfix', a: args }); } +g2.prototype.nodfix.prototype = g2.mix(g2.prototype.nod.prototype, { + g2() { + return g2() + .beg(g2.flatten(this)) + .p() + .m({ x: -8, y: -12 }) + .l({ x: 0, y: 0 }) + .l({ x: 8, y: -12 }) + .drw({ fs: g2.symbol.nodfill2 }) + .cir({ x: 0, y: 0, r: this.r }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) +/** +* @method +* @returns {object} g2 +* @param {object} - symbol arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().view({cartesian:true}) + * .nodflt({x:10,y:10}) +*/ +g2.prototype.nodflt = function (args = {}) { return this.addCommand({ c: 'nodflt', a: args }); } +g2.prototype.nodflt.prototype = g2.mix(g2.prototype.nod.prototype, { + g2() { + return g2() + .beg(g2.flatten(this)) + .p() + .m({ x: -8, y: -12 }) + .l({ x: 0, y: 0 }) + .l({ x: 8, y: -12 }) + .drw({ ls: g2.symbol.nodcolor, fs: g2.symbol.nodfill2 }) + .cir({ x: 0, y: 0, r: this.r, ls: g2.symbol.nodcolor, fs: g2.symbol.nodfill }) + .lin({ x1: -9, y1: -19, x2: 9, y2: -19, ls: g2.symbol.nodfill2, lw: 5 }) + .lin({ x1: -9, y1: -15.5, x2: 9, y2: -15.5, ls: g2.symbol.nodcolor, lw: 2 }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) + +/** +* Draw vector arrow. +* @method +* @returns {object} g2 +* @param {object} - vector arguments object. +* @property {number} x1 - start x coordinate. +* @property {number} y1 - start y coordinate. +* @property {number} x2 - end x coordinate. +* @property {number} y2 - end y coordinate. +* @example +* g2().vec({x1:50,y1:20,x2:250,y2:120}) +*/ +g2.prototype.vec = function vec(args) { return this.addCommand({ c: 'vec', a: args }); } +g2.prototype.vec.prototype = g2.mix(g2.prototype.lin.prototype, { + g2() { + const { x1, y1, x2, y2, lw = 1, ls = '#000', ld = [], fs = ls || '#000', lc = 'round', lj = 'round', } = this; + const dx = x2 - x1, dy = y2 - y1, r = Math.hypot(dx, dy); + const b = 3 * (1 + lw) > r ? r / 3 : (1 + lw); + const arrowHead = () => g2().p().m({ x: 0, y: 0 }).l({ x: -5 * b, y: b }).a({ dw: -Math.PI / 3, x: -5 * b, y: -b }).z().drw({ ls, fs, lc, lj }); + return g2() + .beg({ x: x1, y: y1, w: Math.atan2(dy, dx), lc, lj }) + .p().m({ x: 0, y: 0 }) + .l({ x: r - 3 * b, y: 0 }) + .stroke({ ls, lw, ld }) + .use({ grp: arrowHead, x: r, y: 0 }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}); + +/** +* Arc as Vector +* @method +* @returns {object} g2 +* @param {object} - angular dimension arguments. +* @property {number} x - start x coordinate. +* @property {number} y - start y coordinate. +* @property {number} r - radius +* @property {number} [w=0] - start angle (in radian). +* @property {number} [dw=Math.PI/2] - angular range in radian. In case of positive values it is running counterclockwise with + * right handed (cartesian) coordinate system. +* @example +* g2().avec({x:100,y:70,r:50,w:pi/3,dw:4*pi/3}) +*/ +g2.prototype.avec = function avec(args) { return this.addCommand({ c: 'avec', a: args }); } +g2.prototype.avec.prototype = g2.mix(g2.prototype.arc.prototype, { + g2() { + const { x, y, r, w, dw = 0, lw = 1, lc = 'round', lj = 'round', ls, fs = ls || "#000", label } = this; + const b = 3 * (1 + lw) > r ? r / 3 : (1 + lw), bw = 5 * b / r; + const arrowHead = () => g2().p().m({ x: 0, y: 2 * b }).l({ x: 0, y: -2 * b }).m({ x: 0, y: 0 }).l({ x: -5 * b, y: b }) + .a({ dw: -Math.PI / 3, x: -5 * b, y: -b }).z().drw({ ls, fs }); + + return g2() + .beg({ x, y, w, ls, lw, lc, lj }) + .arc({ r, w: 0, dw }) + .use({ + grp: arrowHead, x: r * Math.cos(dw), y: r * Math.sin(dw), + w: (dw >= 0 ? dw + Math.PI / 2 - bw / 2 : dw - Math.PI / 2 + bw / 2) + }) + .end() + .ins((g) => label && this.drawLabel(g)); + } +}); + +/** +* Linear Dimension +* @method +* @returns {object} g2 +* @param {object} - dimension arguments object. +* @property {number} x1 - start x coordinate. +* @property {number} y1 - start y coordinate. +* @property {number} x2 - end x coordinate. +* @property {number} y2 - end y coordinate. +* @property {number} off - offset. +* @property {boolean} [inside=true] - draw dimension arrows between or outside of ticks. +* @example +* g2().dim({x1:60,y1:40,x2:190,y2:120}) +*/ +g2.prototype.dim = function dim(args) { return this.addCommand({ c: 'dim', a: args }); } +g2.prototype.dim.prototype = g2.mix(g2.prototype.lin.prototype, { + pointAt(loc) { + const pnt = g2.prototype.lin.prototype.pointAt.call(this, loc); + if (this.off) { + pnt.x += this.off * pnt.nx; + pnt.y += this.off * pnt.ny; + } + return pnt; + }, + g2() { + const { x1, y1, x2, y2, lw = 1, lc = 'round', lj = 'round', off = 0, inside = true, ls, fs = ls || "#000", label } = this; + const dx = x2 - x1, dy = y2 - y1, r = Math.hypot(dx, dy); + const b = 3 * (1 + lw) > r ? r / 3 : (1 + lw); + const arrowHead = () => g2().p().m({ x: 0, y: 2 * b }).l({ x: 0, y: -2 * b }).m({ x: 0, y: 0 }).l({ x: -5 * b, y: b }) + .a({ dw: -Math.PI / 3, x: -5 * b, y: -b }).z().drw({ ls, fs }); + return g2() + .beg({ x: x1 + off / r * dy, y: y1 - off / r * dx, w: Math.atan2(dy, dx), ls, fs, lw, lc, lj }) + .lin({ x1: (inside ? 4 * b : 0), y1: 0, x2: (inside ? r - 4 * b : r), y2: 0 }) + .use({ grp: arrowHead, x: r, y: 0, w: (inside ? 0 : Math.PI) }) + .use({ grp: arrowHead, x: 0, y: 0, w: (inside ? Math.PI : 0) }) + .lin({ x1: 0, y1: off, x2: 0, y2: 0 }) + .lin({ x1: r, y1: off, x2: r, y2: 0 }) + .end() + .ins((g) => label && this.drawLabel(g)); + } +}); + +/** +* Angular dimension +* @method +* @returns {object} g2 +* @param {object} - angular dimension arguments. +* @property {number} x - start x coordinate. +* @property {number} y - start y coordinate. +* @property {number} r - radius +* @property {number} [w=0] - start angle (in radian). +* @property {number} [dw=Math.PI/2] - angular range in radian. In case of positive values it is running counterclockwise with + * right handed (cartesian) coordinate system. +* @property {boolean} [outside=false] - draw dimension arrows outside of ticks. +* @depricated {boolean} [inside] - draw dimension arrows between ticks. +* @example +* g2().adim({x:100,y:70,r:50,w:pi/3,dw:4*pi/3}) +*/ +g2.prototype.adim = function adim(args) { return this.addCommand({ c: 'adim', a: args }); } +g2.prototype.adim.prototype = g2.mix(g2.prototype.arc.prototype, { + g2() { + const { x, y, r, w, dw, lw = 1, lc = 'round', lj = 'round', ls, fs = ls || "#000", label } = this; + const b = 3 * (1 + lw) > r ? r / 3 : (1 + lw), bw = 5 * b / r; + const arrowHead = () => g2().p().m({ x: 0, y: 2 * b }).l({ x: 0, y: -2 * b }).m({ x: 0, y: 0 }).l({ x: -5 * b, y: b }) + .a({ dw: -Math.PI / 3, x: -5 * b, y: -b }).z().drw({ ls, fs }); + + const outside = (this.inside !== undefined && this.outside === undefined) ? !this.inside : !!this.outside; // still support depricated property ! + + return g2() + .beg({ x, y, w, ls, lw, lc, lj }) + .arc({ r, w: 0, dw }) + .use({ grp: arrowHead, x: r, y: 0, w: (!outside && dw > 0 || outside && dw < 0 ? -Math.PI / 2 + bw / 2 : Math.PI / 2 - bw / 2) }) + .use({ grp: arrowHead, x: r * Math.cos(dw), y: r * Math.sin(dw), w: (!outside && dw > 0 || outside && dw < 0 ? dw + Math.PI / 2 - bw / 2 : dw - Math.PI / 2 + bw / 2) }) + .end() + .ins((g) => label && this.drawLabel(g)); + } +}); + +/** +* Origin symbol +* @constructor +* @returns {object} g2 +* @param {object} - symbol arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @property {number} w - angle in radians. +* @example +* g2().view({cartesian:true}) + * .origin({x:10,y:10}) +*/ +g2.prototype.origin = function (args = {}) { return this.addCommand({ c: 'origin', a: args }); } +g2.prototype.origin.prototype = g2.mix(g2.prototype.nod.prototype, { + lbloc: 'sw', + g2() { + const { x, y, w, ls = '#000', lw = 1 } = this; + return g2() + .beg({ x, y, w, ls }) + .vec({ x1: 0, y1: 0, x2: 40, y2: 0, lw, fs: '#ccc' }) + .vec({ x1: 0, y1: 0, x2: 0, y2: 40, lw, fs: '#ccc' }) + .cir({ x: 0, y: 0, r: lw + 1, fs: '#ccc' }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}); + +g2.prototype.ply.prototype = g2.mix(g2.labelIfc, g2.markIfc, { + get isSolid() { return this.closed && this.fs && this.fs !== 'transparent'; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false; }, + // get len() { + // let len_itr = 0; + // let last_pt = {x:0,y:0}; + // g2.pntItrOf(this.pts).map(pt => { + // len_itr += Math.hypot(pt.x-last_pt.x, pt.y-last_pt.y); + // last_pt = pt; + // }); + // return len_itr; + // }, + pointAt(loc) { + const t = loc === "beg" ? 0 + : loc === "end" ? 1 + : (loc + 0 === loc) ? loc // numerical arg .. + : 0.5, // 'mid' .. + pitr = g2.pntItrOf(this.pts), + pts = [], + len = []; + + for (let itr = 0; itr < pitr.len; itr++) { + const next = pitr((itr + 1) % pitr.len); + pts.push(pitr(itr)); + len.push(Math.hypot( + next.x - pitr(itr).x, + next.y - pitr(itr).y)); + } + this.closed || len.pop(); + const { t2, x, y, dx, dy } = (() => { + const target = t * len.reduce((a, b) => a + b); + for (let itr = 0, tmp = 0; itr < pts.length; itr++) { + tmp += len[itr]; + const next = pitr(itr + 1).x ? pitr(itr + 1) : pitr(0); + if (tmp >= target) { + return { + t2: 1 - (tmp - target) / len[itr], + x: pts[itr].x, + y: pts[itr].y, + dx: next.x - pts[itr].x, + dy: next.y - pts[itr].y + } + } + } + })(); + const len2 = Math.hypot(dx, dy); + return { + x: (this.x || 0) + x + dx * t2, + y: (this.y || 0) + y + dy * t2, + nx: len2 ? dy / len2 : 1, + ny: len2 ? dx / len2 : 0, + }; + }, + hit({ x, y, eps }) { + return this.isSolid ? g2.isPntInPly({ x: x - this.x, y: y - this.y }, this, eps) // translational transformation only .. at current .. ! + : g2.isPntOnPly({ x: x - this.x, y: y - this.y }, this, eps); + }, + drag({ dx, dy }) { this.x += dx; this.y += dy; }, + get g2() { + const e = g2(); + this.label && e.ins(e => this.drawLabel(e)); + this.mark && e.ins(e => this.drawMark(e, this.closed)); + return () => g2().ply(g2.flatten(this)).ins(e); + } +}); + +g2.prototype.use.prototype = { + // p vector notation ! + get p() { return { x: this.x, y: this.y }; }, // relevant if 'p' is *not* explicite given. + get x() { return Object.getOwnPropertyDescriptor(this, 'p') ? this.p.x : 0; }, + get y() { return Object.getOwnPropertyDescriptor(this, 'p') ? this.p.y : 0; }, + set x(q) { if (Object.getOwnPropertyDescriptor(this, 'p')) this.p.x = q; }, + set y(q) { if (Object.getOwnPropertyDescriptor(this, 'p')) this.p.y = q; }, + + isSolid: false, + /* + hit(at) { + for (const cmd of this.grp.commands) { + if (cmd.a.hit && cmd.a.hit(at)) + return true; + } + return false; + }, + + pointAt: g2.prototype.cir.prototype.pointAt, + */ +}; +// complex macros / add prototypes to argument objects + +/** +* Draw spline by points. +* Implementing a centripetal Catmull-Rom spline (thus avoiding cusps and self-intersections). +* Using iterator function for getting points from array by index. +* It must return current point object {x,y} or object {done:true}. +* Default iterator expects sequence of x/y-coordinates as a flat array [x,y,...], +* array of [[x,y],...] arrays or array of [{x,y},...] objects. +* @see https://pomax.github.io/bezierinfo +* @see https://de.wikipedia.org/wiki/Kubisch_Hermitescher_Spline +* @method +* @returns {object} g2 +* @param {object} - spline arguments object. +* @property {object[] | number[][] | number[]} pts - array of points. +* @property {bool} [closed=false] - closed spline. +* @example +* g2().spline({pts:[100,50,50,150,150,150,100,50]}) +*/ +g2.prototype.spline = function spline({ pts, closed, x, y, w }) { + arguments[0]._itr = g2.pntItrOf(pts); + return this.addCommand({ c: 'spline', a: arguments[0] }); +} +g2.prototype.spline.prototype = g2.mix(g2.prototype.ply.prototype, { + g2: function () { + let { pts, closed, x, y, w, ls, lw, fs, sh } = this, itr = this._itr, gbez; + if (itr) { + let b = [], i, n = itr.len, + p1, p2, p3, p4, d1, d2, d3, + d1d2, d2d3, scl2, scl3, + den2, den3, istrf = x || y || w; + + gbez = g2(); + if (istrf) gbez.beg({ x, y, w }); + gbez.p().m(itr(0)); + for (let i = 0; i < (closed ? n : n - 1); i++) { + if (i === 0) { + p1 = closed ? itr(n - 1) : { x: 2 * itr(0).x - itr(1).x, y: 2 * itr(0).y - itr(1).y }; + p2 = itr(0); + p3 = itr(1); + p4 = n === 2 ? (closed ? itr(0) : { x: 2 * itr(1).x - itr(0).x, y: 2 * itr(1).y - itr(0).y }) : itr(2); + d1 = Math.max(Math.hypot(p2.x - p1.x, p2.y - p1.y), Number.EPSILON); // don't allow .. + d2 = Math.max(Math.hypot(p3.x - p2.x, p3.y - p2.y), Number.EPSILON); // zero point distances .. + } else { + p1 = p2; + p2 = p3; + p3 = p4; + p4 = (i === n - 2) ? (closed ? itr(0) : { x: 2 * itr(n - 1).x - itr(n - 2).x, y: 2 * itr(n - 1).y - itr(n - 2).y }) + : (i === n - 1) ? itr(1) + : itr(i + 2); + d1 = d2; + d2 = d3; + } + d3 = Math.max(Math.hypot(p4.x - p3.x, p4.y - p3.y), Number.EPSILON); + d1d2 = Math.sqrt(d1 * d2), d2d3 = Math.sqrt(d2 * d3), + scl2 = 2 * d1 + 3 * d1d2 + d2, + scl3 = 2 * d3 + 3 * d2d3 + d2, + den2 = 3 * (d1 + d1d2), + den3 = 3 * (d3 + d2d3); + gbez.c({ + x: p3.x, y: p3.y, + x1: (-d2 * p1.x + scl2 * p2.x + d1 * p3.x) / den2, + y1: (-d2 * p1.y + scl2 * p2.y + d1 * p3.y) / den2, + x2: (-d2 * p4.x + scl3 * p3.x + d3 * p2.x) / den3, + y2: (-d2 * p4.y + scl3 * p3.y + d3 * p2.y) / den3 + }); + } + gbez.c(closed ? { x: itr(0).x, y: itr(0).y } : { x: itr(n - 1).x, y: itr(n - 1).y }) + if (closed) gbez.z(); + gbez.drw({ ls, lw, fs, sh }); + if (istrf) gbez.end(); + } + return gbez; + } +}); +"use strict" + +/** + * g2.chart (c) 2015-18 Stefan Goessner + * @author Stefan Goessner + * @license MIT License + * @requires g2.core.js + * @requires g2.ext.js + * @typedef g2 + * @returns {object} chart + * @param {object} args - Chart arguments object or + * @property {float} x - x-position of lower left corner of chart rectangle. + * @property {float} y - y-position of lower left corner of chart rectangle. + * @property {float} [b=150] - width of chart rectangle. + * @property {float} [h=100] - height of chart rectangle. + * @property {string} [ls] - border color. + * @property {string} [fs] - fill color. + * @property {(string|object)} [title] - chart title. + * @property {string} [title.text] - chart title text string. + * @property {float} [title.offset=0] - chart title vertical offset. + * @property {object} [title.style] - chart title style. + * @property {string} [title.style.font=14px serif] - chart title font. + * @property {string} [title.style.thal=center] - chart title horizontal align. + * @property {string} [title.style.tval=bottom] - chart title vertical align. + * @property {array} [funcs=[]] - array of dataset `data` and/or function `fn` objects. + * @property {object} [funcs[item]] - dataset or function object. + * @property {array} [funcs[item].data] - data points as flat array `[x,y,..]`, array of point arrays `[[x,y],..]` or array of point objects `[{x,y},..]`. + * @property {function} [funcs[item].fn] - function `y = f(x)` recieving x-value returning y-value. + * @property {float} [funcs[item].dx] - x increment to apply to function `fn`. Ignored with data points. + * @property {boolean} [funcs[item].fill] - fill region between function graph and x-origin line. + * @property {boolean} [funcs[item].dots] - place circular dots at data points (Avoid with `fn`s). + * @property {boolean|object} [xaxis=false] - x-axis. + * @property {boolean|object} [xaxis.grid=false] - x-axis grid lines. + * @property {string} [xaxis.grid.ls] - x-axis grid line style (color). + * @property {string} [xaxis.grid.lw] - x-axis grid line width. + * @property {string} [xaxis.grid.ld] - x-axis grid line dash style. + * @property {boolean} [xaxis.line=true] - display x-axis base line. + * @property {boolean} [xaxis.origin=false] - display x-axis origin line. + * @property {boolean|object} [yaxis=false] - y-axis. + * @property {boolean|object} [yaxis.grid=false] - y-axis grid lines. + * @property {string} [yaxis.grid.ls] - y-axis grid line style color. + * @property {string} [yaxis.grid.lw] - y-axis grid line width. + * @property {string} [yaxis.grid.ld] - y-axis grid line dash style. + * @property {boolean} [yaxis.line=true] - display y-axis base line. + * @property {boolean} [yaxis.origin=false] - display y-axis origin line. + * @property {float} [xmin] - minimal x-axis value. If not given it is calculated from chart data values. + * @property {float} [xmax] - maximal x-axis value. If not given it is calculated from chart data values. + * @property {float} [ymin] - minimal y-axis value. If not given it is calculated from chart data values. + * @property {float} [ymax] - maximal y-axis value. If not given it is calculated from chart data values. + */ +g2.prototype.chart = function chart({x,y,b,h,style,title,funcs,xaxis,xmin,xmax,yaxis,ymin,ymax}) { + return this.addCommand({c:'chart',a:arguments[0]}); +} +g2.prototype.chart.prototype = { + g2() { + const g = g2(), + funcs = this.get('funcs'), + title = this.title && this.get('title'); + + if (!this.b) this.b = this.defaults.b; + if (!this.h) this.h = this.defaults.h; + // initialize function graphs (only once ...) + if (funcs && funcs.length) { // init all funcs ... + const tmp = [ + this.xmin===undefined, + this.xmax===undefined, + this.ymin===undefined, + this.ymax===undefined + ]; + funcs.forEach(f => this.initFunc(f,...tmp)); + } + // if (this.xaxis) + this.xAxis = this.autoAxis(this.get('xmin'),this.get('xmax'),0,this.b); + // if (this.yaxis) + this.yAxis = this.autoAxis(this.get('ymin'),this.get('ymax'),0,this.h); + + // draw background & border ... + g.rec({ + x:this.x,y:this.y,b:this.b,h:this.h, + fs:this.get("fs"),ls:this.get("ls") + }); + + // draw title & axes ... + g.beg(Object.assign({x:this.x,y:this.y,lw:1}, this.defaults.style,this.style)); + + if (title) + g.txt(Object.assign({ + str: this.title && this.title.text || this.title, + x: this.get('b')/2, + y: this.get('h') + this.get("title","offset"), + w: 0 + }, this.defaults.title.style, + (this.title && this.title.style || {}) + )); + if (this.xaxis) this.drawXAxis(g); + if (this.yaxis) this.drawYAxis(g); + + g.end(); + + // draw funcs ... + if (funcs) + funcs.forEach((fnc,i) => { this.drawFunc(g,fnc,this.defaults.colors[i%this.defaults.colors.length]); }); + + return g; + }, + /** + * Initialize chart function. + * @private + */ + initFunc(fn,setXmin,setXmax,setYmin,setYmax) { + // Install func iterator. + let itr; + if (fn.data && fn.data.length) { // data must have a polyline conform array structure + itr = fn.itr = g2.pntItrOf(fn.data); // get iterator ... + } + else if (fn.fn && fn.dx) { + const xmin = +this.xmin || this.defaults.xmin; + const xmax = +this.xmax || this.defaults.xmax; + itr = fn.itr = (i) => { let x = xmin + i*fn.dx; return { x:x, y:fn.fn(x) }; } + itr.len = (xmax - xmin)/fn.dx + 1; + } + // Get func's bounding box + if (itr && (setXmin || setXmax || setYmin || setYmax)) { + const xarr = []; + const yarr = []; + for (let i=0; i < itr.len; ++i) { + xarr.push(itr(i).x); + yarr.push(itr(i).y); + } + if (setXmin) { + const xmin = Math.min(...xarr); + if (!this.xmin || xmin < this.xmin) this.xmin = xmin; + } + if (setXmax) { + const xmax = Math.max(...xarr); + if (!this.xmax || xmax > this.xmax) this.xmax = xmax; + } + if (setYmin) { + const ymin = Math.min(...yarr); + if (!this.ymin || ymin < this.ymin) this.ymin = ymin; + } + if (setYmax) { + const ymax = Math.max(...yarr); + if (!this.ymax || ymax > this.ymax) this.ymax = ymax; + } + + if (fn.color && typeof fn.color === "number") // color index [0..n] + fn.color = this.defaults.colors[fn.color % this.defaults.colors.length]; + } + }, + autoAxis(zmin,zmax,tmin,tmax) { + let base = 2, exp = 1, eps = Math.sqrt(Number.EPSILON), + Dz = zmax - zmin || 1, // value range + Dt = tmax - tmin || 1, // area range + scl = Dz > eps ? Dt/Dz : 1, // scale [usr]->[pix] + dz = base*Math.pow(10,exp), // tick size [usr] + dt = Math.floor(scl*dz), // tick size [pix] + N, // # segments + dt01, // reminder segment + i0, j0, jth, t0, res; + + while (dt < 14 || dt > 35) { + if (dt < 14) { + if (base == 1) base = 2; + else if (base == 2) base = 5; + else if (base == 5) { base = 1; exp++; } + } + else { // dtick > 35 + if (base == 1) { base = 5; exp--; } + else if (base == 2) base = 1; + else if (base == 5) base = 2; + } + dz = base*Math.pow(10,exp); + dt = scl*dz; + } + i0 = (scl*Math.abs(zmin) + eps/2)%dt < eps + ? Math.floor(zmin/dz) + : Math.floor(zmin/dz) + 1; + let z0 = i0*dz; + t0 = Math.round(scl*(z0 - zmin)); + // console.log("Dt="+Dt+",N="+(Dt - t0)/ dt) + // console.log("DT="+Dt+",N="+(Dt - t0)/ dt) + N = Math.floor((Dt - t0)/ dt) + 1; + j0 = base % 2 && i0 % 2 ? i0 + 1 : i0; + jth = exp === 0 && N < 11 ? 1 : base===2 && N > 9 ? 5 : 2; + + return { + zmin, // min usr value + zmax, // max usr value + base, // one of [1,2,5] + exp, // 10^exp + scl, // scale [usr]->[pix] + dt, // tick range [pix] + dz, // tick range [usr] + N, // # of ticks + t0, // start tick position [pix] + z0, // start tick position [usr] + i0, // first tick index relative to tick origin (can be negative) + j0, // first labeled tick + jth, // # of ticks between two major ticks + itr(i) { // tick iterator + return { t: this.t0 + i*this.dt, + z: parseFloat((this.z0 + i*this.dz).toFixed(Math.abs(this.exp))), + maj: (this.j0 - this.i0 + i)%this.jth === 0 }; + } + } + }, + /** + * Draw x-axis. + * @private + */ + drawXAxis(g) { + let tick, + showgrid = this.xaxis && this.xaxis.grid, + gridstyle = showgrid && Object.assign({}, this.defaults.xaxis.grid, this.xaxis.grid), + showaxis = this.xaxis || this.xAxis, + axisstyle = showaxis && Object.assign({}, this.defaults.xaxis.style, this.defaults.xaxis.labels.style, (this.xaxis && this.xaxis.style || {}) ), + showline = showaxis && this.get("xaxis","line"), + showlabels = this.xAxis && showaxis && this.get("xaxis","labels"), + showticks = this.xAxis && showaxis && this.get("xaxis","ticks"), + ticklen = showticks ? this.get("xaxis","ticks","len") : 0, + showorigin = showaxis && this.get("xaxis","origin"), + title = this.xaxis && (this.get("xaxis","title","text") || this.xaxis.title) || ''; + // console.log(this.xAxis) + // draw tick/grid lines + g.beg(axisstyle); + for (let i=0; i= 0) + g.lin({x1:-this.xAxis.zmin*this.xAxis.scl,y1:0,x2:-this.xAxis.zmin*this.xAxis.scl,y2:this.h}); // origin line emphasized ... + if (title) + g.txt(Object.assign({ + str:title.text || title, + x:this.b/2, + y:-( this.get("xaxis","title","offset") + +(showticks && this.get("xaxis","ticks","len") || 0) + +(showlabels && this.get("xaxis","labels","offset") || 0) + +(showlabels && parseFloat(this.get("xaxis","labels","style","font")) || 0)), + w:0 + }, (this.get('xaxis','title','style')))); + g.end(); + }, + /** + * Draw y-axis. + * @private + */ + drawYAxis(g) { + let tick, + showgrid = this.yaxis && this.yaxis.grid, + gridstyle = showgrid && Object.assign({}, this.defaults.yaxis.grid,this.yaxis.grid), + showaxis = this.yaxis || this.yAxis, + axisstyle = showaxis && Object.assign({},this.defaults.yaxis.style, this.defaults.yaxis.labels.style, (this.yaxis && this.yaxis.style || {})), + showline = showaxis && this.get("yaxis","line"), + showlabels = this.yAxis && showaxis && this.get("yaxis","labels"), + showticks = this.yAxis && showaxis && this.get("yaxis","ticks"), + ticklen = showticks ? this.get("yaxis","ticks","len") : 0, + showorigin = showaxis && this.get("yaxis","origin"), + title = this.yaxis && (this.get("yaxis","title","text") || this.yaxis.title) || ''; + + // draw tick/grid lines + g.beg(axisstyle); + for (let i=0; i= 0) + g.lin({x1:0,y1:-this.yAxis.zmin*this.yAxis.scl,x2:this.b,y2:-this.yAxis.zmin*this.yAxis.scl}); // origin line emphasized ... + if (title) + g.txt(Object.assign({ + str: title.text || title, + x:-( this.get("yaxis","title","offset") + +(showticks && this.get("yaxis","ticks","len") || 0) + +(showlabels && this.get("yaxis","labels","offset") || 0) + +(showlabels && parseFloat(this.get("yaxis","labels","style","font")) || 0)), + y:this.h/2, + w:Math.PI/2 + }, (this.get('yaxis','title','style')))); + g.end(); + }, + /** + * Draw chart function. + * @private + */ + drawFunc(g,fn,defaultcolor) { + let itr = fn.itr; + + if (itr) { + let fill = fn.fill || fn.style && fn.style.fs && fn.style.fs !== "transparent", + color = fn.color = fn.color || fn.style && fn.style.ls || defaultcolor, + plydata = [], + args = Object.assign({ + pts:plydata, + closed:false, + ls:color, + fs:(fill?g2.color.rgbaStr(color,0.125):'transparent'), + lw:1 + }, fn.style); + + if (fill) // start from base line (y=0) + plydata.push(this.pntOf({x:itr(0).x,y:0})); + for (let i=0, n=itr.len; i ctx instanceof g2.selector ? ctx : false); + +// g2.selector.state = ['NONE','OVER','DRAG','OVER+DRAG','EDIT','OVER+EDIT']; + +g2.selector.prototype = { + init(grp) { return true; }, + exe(commands) { + for (let elm=false, i=commands.length; i && !elm; i--) // stop after first hit .. starting from list end ! + elm = this.hit(commands[i-1].a) + }, + selectable(elm) { + return elm && elm.draggable && elm.hit; + }, + hit(elm) { + if (!this.evt.inside // pointer not inside of canvas .. + || !this.selectable(elm) ) // no selectable elm .. + return false; + + if (!elm.state && this.elementHit(elm) && elm.draggable) { // no mode + if (!this.selection || this.selection && !(this.selection.state & g2.DRAG)) { + if (this.selection) this.selection.state ^= g2.OVER; + this.selection = elm; + elm.state = g2.OVER; // enter OVER mode .. + this.evt.hit = true; + } + } + else if (elm.state & g2.DRAG) { // in DRAG mode + if (!this.evt.btn) // leave DRAG mode .. + this.elementDragEnd(elm); + } + else if (elm.state & g2.OVER) { // in OVER mode + if (!this.elementHit(elm)) { // leave OVER mode .. + elm.state ^= g2.OVER; + this.evt.hit = false; + this.selection = false; + } + else if (this.evt.btn) // enter DRAG mode + this.elementDragBeg(elm); + } + + return elm.state && elm; // we definitely have a valid elm here ... + }, // ... but only return it depending on its state. + elementDragBeg(elm) { + elm.state |= g2.DRAG; + if (elm.dragBeg) elm.dragBeg(e); + }, + elementDragEnd(elm) { + elm.state ^= (g2.OVER | g2.DRAG); + this.selection = false; + if (elm.dragEnd) elm.dragEnd(e); + }, + elementHit(elm) { + return elm.hit && elm.hit({x:this.evt.xusr,y:this.evt.yusr,eps:this.evt.eps}); + } +}; + +/** + * canvasInteractor.js (c) 2018 Stefan Goessner + * @file interaction manager for html `canvas`. + * @author Stefan Goessner + * @license MIT License + */ +/* jshint -W014 */ +// Managing multiple canvases per static interactor as singleton ... +// .. using a single requestAnimationFrame loop ! +const canvasInteractor = { + create() { + const o = Object.create(this.prototype); + o.constructor.apply(o,arguments); + return o; + }, + // global static tickTimer properties + fps: '?', + fpsOrigin: 0, + frames: 0, + rafid: 0, + instances: [], + // global static timer methods + tick(time) { + canvasInteractor.fpsCount(time); + for (const instance of canvasInteractor.instances) { + instance.notify('tick',{t:time,dt:(time-instance.t)/1000,dirty:instance.dirty}); // notify listeners .. + instance.t = time; + instance.dirty = false; + } + canvasInteractor.rafid = requestAnimationFrame(canvasInteractor.tick); // request next animation frame ... + }, + add(instance) { + canvasInteractor.instances.push(instance); + if (canvasInteractor.instances.length === 1) // first instance added ... + canvasInteractor.tick(canvasInteractor.fpsOrigin = performance.now()); + }, + remove(instance) { + canvasInteractor.instances.splice(canvasInteractor.instances.indexOf(instance),1); + if (canvasInteractor.instances.length === 0) // last instance removed ... + cancelAnimationFrame(canvasInteractor.rafid); + }, + fpsCount(time) { + if (time - canvasInteractor.fpsOrigin > 1000) { // one second interval reached ... + const fps = ~~(canvasInteractor.frames*1000/(time - canvasInteractor.fpsOrigin) + 0.5); // ~~ as Math.floor() + if (fps !== canvasInteractor.fps) + for (const instance of canvasInteractor.instances) + instance.notify('fps',canvasInteractor.fps=fps); + canvasInteractor.fpsOrigin = time; + canvasInteractor.frames = 0; + } + canvasInteractor.frames++; + }, + + prototype: { + constructor(ctx, {x,y,scl,cartesian}) { + // canvas interaction properties + this.ctx = ctx; + this.view = {x:x||0,y:y||0,scl:scl||1,cartesian:cartesian||false}; + this.evt = { + type: false, + basetype: false, + x: -2, y:-2, + xi: 0, yi:0, + dx: 0, dy: 0, + btn: 0, + xbtn: 0, ybtn: 0, + xusr: -2, yusr: -2, + dxusr: 0, dyusr: 0, + delta: 0, + inside: false, + hit: false, // something hit by pointer ... + dscl: 1, // for zooming ... + eps: 5 // some pixel tolerance ... + }; + this.dirty = true; + // event handler registration + const canvas = ctx.canvas; + canvas.addEventListener("pointermove", this, false); + canvas.addEventListener("pointerdown", this, false); + canvas.addEventListener("pointerup", this, false); + canvas.addEventListener("pointerenter", this, false); + canvas.addEventListener("pointerleave", this, false); + canvas.addEventListener("wheel", this, false); + canvas.addEventListener("pointercancel", this, false); + }, + deinit() { + const canvas = this.ctx.canvas; + + canvas.removeEventListener("pointermove", this, false); + canvas.removeEventListener("pointerdown", this, false); + canvas.removeEventListener("pointerup", this, false); + canvas.removeEventListener("pointerenter", this, false); + canvas.removeEventListener("pointerleave", this, false); + canvas.removeEventListener("wheel", this, false); + canvas.removeEventListener("pointercancel", this, false); + + this.endTimer(); + + delete this.signals; + delete this.evt; + delete this.ctx; + + return this; + }, + // canvas interaction interface + handleEvent(e) { + if (e.type in this && (e.isPrimary || e.type === 'wheel')) { // can I handle events of type e.type .. ? + const bbox = e.target.getBoundingClientRect && e.target.getBoundingClientRect() || {left:0, top:0}, + x = e.clientX - Math.floor(bbox.left), + y = e.clientY - Math.floor(bbox.top), + btn = e.buttons !== undefined ? e.buttons : e.button || e.which; + + this.evt.type = e.type; + this.evt.basetype = e.type; // obsolete now ... ? + this.evt.xi = this.evt.x; // interim coordinates ... + this.evt.yi = this.evt.y; // ... of previous event. + this.evt.dx = this.evt.dy = 0; + this.evt.x = x; + this.evt.y = this.view.cartesian ? this.ctx.canvas.height - y : y; + this.evt.xusr = (this.evt.x - this.view.x)/this.view.scl; + this.evt.yusr = (this.evt.y - this.view.y)/this.view.scl; + this.evt.dxusr = this.evt.dyusr = 0; + this.evt.dbtn = btn - this.evt.btn; + this.evt.btn = btn; + this.evt.delta = Math.max(-1,Math.min(1,e.deltaY||e.wheelDelta)) || 0; + + if (this.isDefaultPreventer(e.type)) + e.preventDefault(); + this[e.type](); // handle specific event .. ! + this.notify(this.evt.type,this.evt); // .. tell the world .. ! + } + else + console.log(e) + }, + pointermove() { + this.evt.dx = this.evt.x - this.evt.xi; + this.evt.dy = this.evt.y - this.evt.yi; + if (this.evt.btn === 1) { // pointerdown state ... + this.evt.dxusr = this.evt.dx/this.view.scl; // correct usr coordinates ... + this.evt.dyusr = this.evt.dy/this.view.scl; + this.evt.xusr -= this.evt.dxusr; // correct usr coordinates ... + this.evt.yusr -= this.evt.dyusr; + if (!this.evt.hit) { // let outer app perform panning ... + this.evt.type = 'pan'; + } + else + this.evt.type = 'drag'; + } + // view, geometry or graphics might be modified ... + this.dirty = true; + }, + pointerdown() { + this.evt.xbtn = this.evt.x; + this.evt.ybtn = this.evt.y; + }, + pointerup() { + this.evt.type = this.evt.x===this.evt.xbtn && this.evt.y===this.evt.ybtn ? 'click' : 'pointerup'; + this.evt.xbtn = this.evt.x; + this.evt.ybtn = this.evt.y; + }, + pointerleave() { + this.evt.inside = false; + }, + pointerenter() { + this.evt.inside = true; + }, + wheel() { + this.evt.dscl = this.evt.delta>0?8/10:10/8; + this.evt.eps /= this.evt.dscl; + this.dirty = true; + }, + isDefaultPreventer(type) { + return ['pointermove','pointerdown','pointerup','wheel'].includes(type); + }, + pntToUsr: function(p) { + let vw = this.view; + p.x = (p.x - vw.x)/vw.scl; + p.y = (p.y - vw.y)/vw.scl; + return p; + }, + // tickTimer interface + startTimer() { // shouldn't there be a global startTimer method ? + canvasInteractor.add(this); + this.notify('timerStart',this); // notify potential listeners .. + return this; + }, + endTimer() { + this.notify('timerEnd',this.t/1000); // notify potential listeners .. + canvasInteractor.remove(this); + return this; + }, + // observable interface + notify(key,val) { + if (this.signals && this.signals[key]) + for (let hdl of this.signals[key]) + hdl(val); + return this; + }, + on(key,handler) { // support array of keys as first argument. + if (Array.isArray(key)) + for (let k of key) + this.on(k,handler); + else + ((this.signals || (this.signals = {})) && this.signals[key] || (this.signals[key]=[])).push(handler); + + return this; + }, + remove(key,handler) { + const idx = (this.signals && this.signals[key]) ? this.signals[key].indexOf(handler) : -1; + if (idx >= 0) + this.signals[key].splice(idx,1); + } + } +}; + +"use strict"; + +class G2ChartElement extends HTMLElement { + static get observedAttributes() { + return [ + 'width', + 'height', + 'xmin', + 'xmax', + 'ymin', + 'ymax', + 'title', + ]; + } + + constructor() { + super(); + this._root = this.attachShadow({ mode: 'open' }); + } + + get width() { return +this.getAttribute('width') || 301; } + set width(q) { q && this.setAttribute('width', q) } + get height() { return +this.getAttribute('height') || 201; } + set height(q) { q && this.setAttribute('height', q) } + get xmin() { return +this.getAttribute('xmin') || undefined; } + set xmin(q) { return q && +this.setAttribute('xmin', q) } + get xmax() { return +this.getAttribute('xmax') || undefined; } + set xmax(q) { return q && +this.setAttribute('xmax', q) } + get ymin() { return +this.getAttribute('ymin') || undefined; } + set ymin(q) { return q && +this.setAttribute('ymin', q) } + get ymax() { return +this.getAttribute('ymax') || undefined; } + set ymax(q) { return q && +this.setAttribute('ymax', q) } + get title() { return this.getAttribute('title') || ''; } + set title(q) { return q && this.setAttribute('title', q) } + + connectedCallback() { + this._root.innerHTML = G2ChartElement.template({ + width: this.width, height: this.height + }); + + this._ctx = this._root.getElementById('cnv').getContext('2d'); + + const t = 35; + this._chart = { + x: t, + y: t, + xmin: this.xmin, + xmax: this.xmax, + ymin: this.ymin, + ymax: this.ymax, + title: this.title, + b: this.width - t * 2, + h: this.height - t * 2, + xaxis: () => this.xaxis || {}, + yaxis: () => this.yaxis || {}, + title: () => this.title || "", + funcs: () => this.funcs + }; + + this._g = g2().del().clr().view({ cartesian: true }).chart(this._chart); + + try { + // If not true, the element should be referenced by another module. + if (this.innerHTML !== '') { + // Remove all functions (declared by "fn": fn) and fetch them before parsing. + // Then bring them back in using Function: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/eval#Never_use_eval + + // Find all functions declared by "fn:" like here: https://goessner.github.io/g2/g2.chart.html#example-multiple-functions + const funcRegEx = /(("|')fn("|'):)([^(,|})]+)/g; + const funcs = JSON.parse(this.innerHTML.replace(funcRegEx, '"fn":"PLACEHOLDER"').trim()); + let itr = 0; + for (const a of this.innerHTML.matchAll(funcRegEx)) { + funcs[itr].fn = (() => Function('"use strict"; return (' + a[4] + ')')())(); + itr++; + } + this.funcs = funcs; + } + } + catch (e) { + console.warn(e); + this._g.txt({ str: e, y: 5 }); + } + finally { + this._g.nod({ + x: () => this.nod && this.nod().x, + y: () => this.nod && this.nod().y, + scl: () => this.nod && this.nod().scl || 0 + }); + this.render(); + } + } + + render() { + this._g.exe(this._ctx); + } + + setFuncs(funcs) { + this.funcs = funcs; + return this; + } + + disconnectedCallback() { + // TODO + } + + static template({ width, height }) { + return `` + } +} +customElements.define('g2-chart', G2ChartElement); + diff --git a/dist/g2.full.js b/dist/g2.full.js new file mode 100644 index 0000000..1ab148c --- /dev/null +++ b/dist/g2.full.js @@ -0,0 +1,3009 @@ + +"use strict" + +/** + * g2.core (c) 2013-20 Stefan Goessner + * @author Stefan Goessner + * @license MIT License + * @link https://github.com/goessner/g2 + * @typedef {g2} + * @param {object} [opts] Custom options object. + * @description Create a 2D graphics command queue object. Call without using 'new'. + * @returns {g2} + * @example + * const ctx = document.getElementById("c").getContext("2d"); + * g2() // Create 'g2' instance. + * .lin({x1:50,y1:50,x2:100,y2:100}) // Append ... + * .lin({x1:100,y1:100,x2:200,y2:50}) // ... commands. + * .exe(ctx); // Execute commands addressing canvas context. + */ + +function g2(opts) { + let o = Object.create(g2.prototype); + o.commands = []; + if (opts) Object.assign(o, opts); + return o; +} + +g2.prototype = { + /** + * Clear viewport region.
+ * @method + * @returns {object} g2 + */ + clr() { return this.addCommand({ c: 'clr' }); }, + + /** + * Set the view by placing origin coordinates and scaling factor in device units + * and make viewport cartesian. + * @method + * @returns {object} g2 + * @param {object} - view arguments object. + * @property {number} [scl=1] - absolute scaling factor. + * @property {number} [x=0] - x-origin in device units. + * @property {number} [y=0] - y-origin in device units. + * @property {boolean} [cartesian=false] - set cartesian flag. + */ + view({ scl, x, y, cartesian }) { return this.addCommand({ c: 'view', a: arguments[0] }); }, + + /** + * Draw grid. + * @method + * @returns {object} g2 + * @param {object} - grid arguments object. + * @property {string} [color=#ccc] - change color. + * @property {number} [size=20] - change space between lines. + */ + grid({ color, size } = {}) { return this.addCommand({ c: 'grid', a: arguments[0] }); }, + + /** + * Draw circle by center and radius. + * @method + * @returns {object} g2 + * @param {object} - circle arguments object. + * @property {number} x - x-value center. + * @property {number} y - y-value center. + * @property {number} r - radius. + * @property {number} w - angle. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {array} [sh=[0,0,0,'transparent']] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @example + * g2().cir({x:100,y:80,r:20}) // Draw circle. + */ + cir({ x, y, r, w }) { return this.addCommand({ c: 'cir', a: arguments[0] }); }, + + /** + * Draw ellipse by center and radius for x and y. + * @method + * @returns {object} g2 + * @param {object} - ellispe argument object. + * @property {number} x - x-value center. + * @property {number} y - y-value center. + * @property {number} rx - radius x-axys. + * @property {number} ry - radius y-axys. + * @property {number} w - start angle. + * @property {number} dw - angular range. + * @property {number} rot - rotation. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @example + * g2().ell({x:100,y:80,rx:20,ry:30,w:0,dw:2*Math.PI/4,rot:1}) // Draw circle. + */ + ell({ x, y, rx, ry, w, dw, rot }) { return this.addCommand({ c: 'ell', a: arguments[0] }); }, + + /** + * Draw arc by center point, radius, start angle and angular range. + * @method + * @returns {object} g2 + * @param {object} - arc arguments object. + * @property {number} x - x-value center. + * @property {number} y - y-value center. + * @property {number} r - radius. + * @property {number} [w=0] - start angle (in radian). + * @property {number} [dw=2*pi] - angular range in Radians. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {string} [lc=butt] - line cap [`butt`, `round`, `square`]. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @example + * g2().arc({x:300,y:400,r:390,w:-Math.PI/4,dw:-Math.PI/2}) + * .exe(ctx); + */ + arc({ x, y, r, w, dw }) { return this.addCommand({ c: 'arc', a: arguments[0] }); }, + + /** + * Draw rectangle by anchor point and dimensions. + * @method + * @returns {object} g2 + * @param {object} - rectangle arguments object. + * @property {number} x - x-value upper left corner. + * @property {number} y - y-value upper left corner. + * @property {number} b - width. + * @property {number} h - height. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {string} [lj='miter'] - line join [`round`, `bevel`, `miter`]. + * @property {number} [ml=10] - miter limit. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @example + * g2().rec({x:100,y:80,b:40,h:30}) // Draw rectangle. + */ + rec({ x, y, b, h }) { return this.addCommand({ c: 'rec', a: arguments[0] }); }, + + /** + * Draw line by start point and end point. + * @method + * @returns {object} g2 + * @param {object} - line arguments object. + * @property {number} x1 - start x coordinate. + * @property {number} y1 - start y coordinate. + * @property {number} x2 - end x coordinate. + * @property {number} y2 - end y coordinate. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {string} [lc=butt] - line cap [`butt`, `round`, `square`]. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @example + * g2().lin({x1:10,x2:10,y1:190,y2:10}) // Draw line. + */ + lin({ x1, y1, x2, y2 }) { return this.addCommand({ c: 'lin', a: arguments[0] }); }, + + /** + * Draw polygon by points. + * Using iterator function for getting points from array by index. + * It must return current point object {x,y} or object {done:true}. + * Default iterator expects sequence of x/y-coordinates as a flat array [x,y,...], + * array of [[x,y],...] arrays or array of [{x,y},...] objects. + * @method + * @returns {object} g2 + * @param {object} - polygon arguments object. + * @property {array} pts - array of points. + * @property {string} [format] - format string of points array structure. Useful for handing over initial empty points array. One of `['x,y','[x,y]','{x,y}']`. Has precedence over `pts` content. + * @property {boolean} [closed = false] + * @property {number} x - start x coordinate. + * @property {number} y - start y coordinate. + * @property {number} w - angle. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {string} [lc=butt] - line cap [`butt`, `round`, `square`]. + * @property {string} [lj='miter'] - line join [`round`, `bevel`, `miter`]. + * @property {number} [ml=10] - miter limit. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @example + * g2().ply({pts:[100,50,120,60,80,70]}), + * .ply({pts:[150,60],[170,70],[130,80]],closed:true}), + * .ply({pts:[{x:160,y:70},{x:180,y:80},{x:140,y:90}]}), + * .exe(ctx); + */ + ply({ pts, format, closed, x, y, w }) { + arguments[0]._itr = format && g2.pntIterator[format](pts) || g2.pntItrOf(pts); + return this.addCommand({ c: 'ply', a: arguments[0] }); + }, + + /** + * Draw text string at anchor point. + * @method + * @returns {object} g2 + * @param {object} - text arguments object. + * @property {string} str - text string. + * @property {number} [x=0] - x coordinate of text anchor position. + * @property {number} [y=0] - y coordinate of text anchor position. + * @property {number} [w=0] - w Rotation angle about anchor point with respect to positive x-axis. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @property {string} [thal='start'] + * - Text horizontal alignment [`'start'`,`'end'`,`'left'`,`'right'`,`'center'`] + * @property {string} [tval='alphabetic'] + * - Text vertival alignment [`'top'`,`'hanging'`,`'middle'`,`'alphabetic'`,`'ideographic'`,`'bottom'`] + * @property {string} [font='normal 14px serif'] - + * [Font]{@link https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/font} + * [styling]{@link https://html.spec.whatwg.org/multipage/canvas.html#dom-context-2d-font} + */ + txt({ str, x, y, w }) { return this.addCommand({ c: 'txt', a: arguments[0] }); }, + + /** + * Reference g2 graphics commands from another g2 object or a predefined g2.symbol. + * With this command you can reuse instances of grouped graphics commands + * while applying a similarity transformation and style properties on them. + * In fact you might want to build custom graphics libraries on top of that feature. + * @method + * @returns {object} g2 + * @param {object} - use arguments object. + * @see {@link https://github.com/goessner/g2/blob/master/docs/api/g2.ext.md#g2symbol--object predefined symbols in g2.ext} + * @property {object | string} grp - g2 source object or symbol name found in 'g2.symbol' namespace. + * @property {number} [x=0] - translation value x. + * @property {number} [y=0] - translation value y. + * @property {number} [w=0] - rotation angle (in radians). + * @property {number} [scl=1] - scale factor. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {string} [lc=butt] - line cap [`butt`, `round`, `square`]. + * @property {string} [lj='miter'] - line join [`round`, `bevel`, `miter`]. + * @property {number} [ml=10] - miter limit. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @property {string} [thal='start'] + * - Text horizontal alignment [`'start'`,`'end'`,`'left'`,`'right'`,`'center'`] + * @property {string} [tval='alphabetic'] + * - Text vertival alignment [`'top'`,`'hanging'`,`'middle'`,`'alphabetic'`,`'ideographic'`,`'bottom'`] + * @property {string} [font='normal 14px serif'] - + * [Font]{@link https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/font} + * [styling]{@link https://html.spec.whatwg.org/multipage/canvas.html#dom-context-2d-font} + * @example + * g2.symbol.cross = g2().lin({x1:5,y1:5,x2:-5,y2:-5}).lin({x1:5,y1:-5,x2:-5,y2:5}); // Define symbol. + * g2().use({grp:"cross",x:100,y:100}) // Draw cross at position 100,100. + */ + use({ grp, x, y, w, scl }) { + if (grp && grp !== this) { // avoid self reference .. + if (typeof grp === "string") // must be a member name of the 'g2.symbol' namespace + arguments[0].grp = g2.symbol[(grp in g2.symbol) ? grp : 'unknown']; + this.addCommand({ c: 'use', a: arguments[0] }); + } + return this; + }, + + /** + * Draw image. + * This also applies to images of reused g2 objects. If an image can not be loaded, it will be replaced by a broken-image symbol. + * @method + * @returns {object} g2 + * @param {object} - image arguments object. + * @property {string} uri - image uri or data:url. + * @property {number} [x = 0] - x-coordinate of image (upper left). + * @property {number} [y = 0] - y-coordinate of image (upper left). + * @property {number} [b = image.width] - width. + * @property {number} [h = image.height] - height. + * @property {number} [sx = 0] - source x-offset. + * @property {number} [sy = 0] - source y-offset. + * @property {number} [sb = image.width] - source width. + * @property {number} [sh = image.height] - source height. + * @property {number} [xoff = 0] - x-offset. + * @property {number} [yoff = 0] - y-offset. + * @property {number} [w = 0] - rotation angle (about upper left, in radians). + * @property {number} [scl = 1] - image scaling. + */ + img({ uri, x, y, b, h, sx, sy, sb, sh, xoff, yoff, w, scl }) { return this.addCommand({ c: 'img', a: arguments[0] }); }, + + /** + * Begin subcommands. Current state is saved. + * Optionally apply transformation or style properties. + * @method + * @returns {object} g2 + * @param {object} - beg arguments object. + * @property {number} [x = 0] - translation value x. + * @property {number} [y = 0] - translation value y. + * @property {number} [w = 0] - rotation angle (in radians). + * @property {number} [scl = 1] - scale factor. + * @property {array} [matrix] - matrix instead of single transform arguments (SVG-structure [a,b,c,d,x,y]). + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {string} [lc=butt] - line cap [`butt`, `round`, `square`]. + * @property {string} [lj='miter'] - line join [`round`, `bevel`, `miter`]. + * @property {number} [ml=10] - miter limit. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @property {string} [thal='start'] + * - text horizontal alignment [`'start'`,`'end'`,`'left'`,`'right'`,`'center'`] + * @property {string} [tval='alphabetic'] + * - text vertival alignment [`'top'`,`'hanging'`,`'middle'`,`'alphabetic'`,`'ideographic'`,`'bottom'`] + * @property {string} [font='normal 14px serif'] - + * [Font]{@link https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/font} + * [styling]{@link https://html.spec.whatwg.org/multipage/canvas.html#dom-context-2d-font} + */ + beg({ x, y, w, scl, matrix } = {}) { return this.addCommand({ c: 'beg', a: arguments[0] }); }, + + /** + * End subcommands. Previous state is restored. + * @method + * @returns {object} g2 + * @param {object} - end arguments object. + */ + end() { // ignore 'end' commands without matching 'beg' + let myBeg = 1, + findMyBeg = (cmd) => { // care about nested beg...end blocks ... + if (cmd.c === 'beg') myBeg--; + else if (cmd.c === 'end') myBeg++; + return myBeg === 0; + } + return g2.cmdIdxBy(this.commands, findMyBeg) !== false ? this.addCommand({ c: 'end' }) : this; + }, + + /** + * Begin new path. + * @method + * @returns {object} g2 + */ + p() { return this.addCommand({ c: 'p' }); }, + + /** + * Close current path by straight line. + * @method + * @returns {object} g2 + */ + z() { return this.addCommand({ c: 'z' }); }, + + /** + * Move to point. + * @method + * @returns {object} g2 + * @param {object} - move arguments object. + * @property {number} x - move to x coordinate + * @property {number} y - move to y coordinate + */ + m({ x, y }) { return this.addCommand({ c: 'm', a: arguments[0] }); }, + + /** + * Create line segment to point. + * @method + * @returns {object} g2 + * @param {object} - line segment argument object. + * @property {number} x - x coordinate of target point. + * @property {number} y - y coordinate of target point. + * @example + * g2().p() // Begin path. + * .m({x:0,y:50}) // Move to point. + * .l({x:300,y:0}) // Line segment to point. + * .l({x:400,y:100}) // ... + * .stroke() // Stroke path. + */ + l({ x, y }) { return this.addCommand({ c: 'l', a: arguments[0] }); }, + + /** + * Create quadratic bezier curve segment to point. + * @method + * @returns {object} g2 + * @param {object} - quadratic curve arguments object. + * @property {number} x1 - x coordinate of control point. + * @property {number} y1 - y coordinate of control point. + * @property {number} x - x coordinate of target point. + * @property {number} y - y coordinate of target point. + * @example + * g2().p() // Begin path. + * .m({x:0,y:0}) // Move to point. + * .q({x1:200,y1:200,x:400,y:0}) // Quadratic bezier curve segment. + * .stroke() // Stroke path. + */ + q({ x1, y1, x, y }) { return this.addCommand({ c: 'q', a: arguments[0] }); }, + + /** + * Create cubic bezier curve to point. + * @method + * @returns {object} g2 + * @param {object} - cubic curve arguments object. + * @property {number} x1 - x coordinate of first control point. + * @property {number} y1 - y coordinate of first control point. + * @property {number} x2 - x coordinate of second control point. + * @property {number} y2 - y coordinate of second control point. + * @property {number} x - x coordinate of target point. + * @property {number} y - y coordinate of target point. + * @example + * g2().p() // Begin path. + * .m({x:0,y:100}) // Move to point. + * .c({x1:100,y1:200,x2:200,y2:0,x:400,y:100}) // Create cubic bezier curve. + * .stroke() // Stroke path. + * .exe(ctx); // Render to canvas context. + */ + c({ x1, y1, x2, y2, x, y }) { return this.addCommand({ c: 'c', a: arguments[0] }); }, + + /** + * Draw arc with angular range to target point. + * @method + * @returns {object} g2 + * @param {object} - arc arguments object. + * @property {number} dw - angular range in radians. + * @property {number} x - x coordinate of target point. + * @property {number} y - y coordinate of target point. + * @example + * g2().p() // Begin path. + * .m({x:50,y:50}) // Move to point. + * .a({dw:2,x:300,y:100}) // Create arc segment. + * .stroke() // Stroke path. + * .exe(ctx); // Render to canvas context. + */ + a({ dw, x, y }) { + let prvcmd = this.commands[this.commands.length - 1]; + g2.cpyProp(prvcmd.a, 'x', arguments[0], '_xp'); + g2.cpyProp(prvcmd.a, 'y', arguments[0], '_yp'); + return this.addCommand({ c: 'a', a: arguments[0] }); + }, + + /** + * Stroke the current path or path object. + * @method + * @returns {object} g2 + * @param {object} - stroke arguments object. + * @property {string} [d = undefined] - SVG path definition string. Current path is ignored then. + */ + stroke({ d } = {}) { return this.addCommand({ c: 'stroke', a: arguments[0] }); }, + + /** + * Fill the current path or path object. + * @method + * @returns {object} g2 + * @param {object} - fill arguments object. + * @property {string} [d = undefined] - SVG path definition string. Current path is ignored then. + */ + fill({ d } = {}) { return this.addCommand({ c: 'fill', a: arguments[0] }); }, + + /** + * Shortcut for stroke and fill the current path or path object. + * In case of shadow style, only the path interior creates shadow, not also the path contour. + * @method + * @returns {object} g2 + * @param {object} - drw arguments object. + * @property {string} [d = undefined] - SVG path definition string. Current path is ignored then. + */ + drw({ d, lsh } = {}) { return this.addCommand({ c: 'drw', a: arguments[0] }); }, + + /** + * Delete all commands beginning from `idx` to end of command queue. + * @method + * @returns {object} g2 + */ + del(idx) { this.commands.length = idx || 0; return this; }, + + /** + * Call function between commands of the command queue. + * @method + * @returns {object} g2 + * @param {function} - ins argument function. + * @example + * const node = { + * fill:'lime', + * g2() { return g2().cir({x:160,y:50,r:15,fs:this.fill,lw:4,sh:[8,8,8,"gray"]}) } + * }; + * let color = 'red'; + * g2().cir({x:40,y:50,r:15,fs:color,lw:4,sh:[8,8,8,"gray"]}) // draw red circle. + * .ins(()=>{color='green'}) // color is now green. + * .cir({x:80,y:50,r:15,fs:color,lw:4,sh:[8,8,8,"gray"]}) // draw green circle. + * .ins((g) => // draw orange circle + * g.cir({x:120, y:50, r:15, fs:'orange', lw:4,sh:[8,8,8,"gray"]})) + * .ins(node) // draw node. + * .exe(ctx) // render to canvas context. + */ + ins(arg) { + return typeof arg === 'function' ? (arg(this) || this) // no further processing by handler ... + : typeof arg === 'object' ? (this.commands.push({ a: arg }), this) // no explicit command name .. ! + : this; + }, + /** + * Execute g2 commands. It does so automatically and recursively with 'use'ed commands. + * @method + * @returns {object} g2 + * @param {object} ctx Context. + */ + exe(ctx) { + let handler = g2.handler(ctx); + if (handler && handler.init(this)) + handler.exe(this.commands); + return this; + }, + // helpers ... + addCommand({ c, a }) { + if (a && Object.getPrototypeOf(a) === Object.prototype) { // modify only pure argument objects 'a' .. ! + for (const key in a) { + if (!Object.getOwnPropertyDescriptor(a, key).get // if 'key' is no getter ... + && key[0] !== '_' // and no private property ... + && typeof a[key] === 'function') { // and a function ... make it a getter + Object.defineProperty(a, key, { get: a[key], enumerable: true, configurable: true, writabel: false }); + } + if (typeof a[key] === 'string' && a[key][0] === '@') { // referring values by neighbor id's + const refidIdx = a[key].indexOf('.'); + const refid = refidIdx > 0 ? a[key].substr(1, refidIdx - 1) : ''; + const refkey = refid ? a[key].substr(refidIdx + 1) : ''; + const refcmd = refid ? () => this.commands.find((cmd) => cmd.a && cmd.a.id === refid) : undefined; + + if (refcmd) + Object.defineProperty(a, key, { + get: function () { + const rc = refcmd(); + return rc && (refkey in rc.a) ? rc.a[refkey] : 0; + }, + enumerable: true, + configurable: true, + writabel: false + }); + } + } + if (g2.prototype[c].prototype) Object.setPrototypeOf(a, g2.prototype[c].prototype); + } + this.commands.push(arguments[0]); + return this; + } +}; + +// statics +g2.defaultStyle = { fs: 'transparent', ls: '#000', lw: 1, lc: "butt", lj: "miter", ld: [], ml: 10, sh: [0, 0], lsh: false, font: '14px serif', thal: 'start', tval: 'alphabetic' }; +g2.symbol = { + unknown: g2().cir({ r: 12, fs: 'orange' }).txt({ str: '?', thal: 'center', tval: 'middle', font: 'bold 20pt serif' }) +}; +g2.handler = function (ctx) { + let hdl; + for (let h of g2.handler.factory) + if ((hdl = h(ctx)) !== false) + return hdl; + return false; +} +g2.handler.factory = []; + +// predefined polyline/spline point iterators +g2.pntIterator = { + "x,y": function (pts) { + function pitr(i) { return { x: pts[2 * i], y: pts[2 * i + 1] }; }; + Object.defineProperty(pitr, 'len', { get: () => pts.length / 2, enumerable: true, configurable: true, writabel: false }); + return pitr; + }, + "[x,y]": function (pts) { + function pitr(i) { return pts[i] ? { x: pts[i][0], y: pts[i][1] } : undefined; }; + Object.defineProperty(pitr, 'len', { get: () => pts.length, enumerable: true, configurable: true, writabel: false }); + return pitr; + }, + "{x,y}": function (pts) { + function pitr(i) { return pts[i]; }; + Object.defineProperty(pitr, 'len', { get: () => pts.length, enumerable: true, configurable: true, writabel: false }); + return pitr; + } +}; +g2.pntItrOf = function (pts) { + return !(pts && pts.length) ? undefined + : typeof pts[0] === "number" ? g2.pntIterator["x,y"](pts) + : Array.isArray(pts[0]) && pts[0].length >= 2 ? g2.pntIterator["[x,y]"](pts) + : typeof pts[0] === "object" && "x" in pts[0] && "y" in pts[0] ? g2.pntIterator["{x,y}"](pts) + : undefined; +}; +/** + * Get index of command resolving 'callbk' to 'true' starting from end of the queue walking back.
+ * Similar to 'Array.prototype.findIndex', only working reverse. + * @private + */ +g2.cmdIdxBy = function (cmds, callbk) { + for (let i = cmds.length - 1; i >= 0; i--) + if (callbk(cmds[i], i, cmds)) + return i; + return false; // command with index '0' signals 'failing' ... +}; + +/** + * Replacement for Object.assign, as it does not assign getters and setter properly ... + * See https://github.com/tc39/proposal-object-getownpropertydescriptors + * See https://medium.com/@benastontweet/mixins-in-javascript-700ec81f5e5c + * Shallow copy of prototypes (think interfaces) + * @private + */ +g2.mix = function mix(...protos) { + let mixture = {}; + for (const p of protos) + mixture = Object.defineProperties(mixture, Object.getOwnPropertyDescriptors(p)); + return mixture; +} + +/** + * Copy properties, even as getters .. a useful part of the above .. + * @private + */ +g2.cpyProp = function (from, fromKey, to, toKey) { Object.defineProperty(to, toKey, Object.getOwnPropertyDescriptor(from, fromKey)); } + +// Html canvas handler +g2.canvasHdl = function (ctx) { + if (this instanceof g2.canvasHdl) { + if (ctx instanceof CanvasRenderingContext2D) { + this.ctx = ctx; + this.cur = g2.defaultStyle; + this.stack = [this.cur]; + this.matrix = [[1, 0, 0, 1, 0.5, 0.5]]; + this.gridBase = 2; + this.gridExp = 1; + return this; + } + else + return null; + } + return g2.canvasHdl.apply(Object.create(g2.canvasHdl.prototype), arguments); +}; +g2.handler.factory.push((ctx) => ctx instanceof g2.canvasHdl ? ctx + : ctx instanceof CanvasRenderingContext2D ? g2.canvasHdl(ctx) : false); + +g2.canvasHdl.prototype = { + init(grp, style) { + this.stack.length = 1; + this.matrix.length = 1; + this.initStyle(style ? Object.assign({}, this.cur, style) : this.cur); + return true; + }, + async exe(commands) { + for (let cmd of commands) { + // cmd.a is an object offering a `g2` method, so call it and execute its returned commands array. + if (cmd.a && cmd.a.g2) { + const cmds = cmd.a.g2().commands; + // If false, ext was not applied to this cmd. But the command still renders + if (cmds) { + this.exe(cmds); + continue; + } + } + // cmd.a is a `g2` object, so directly execute its commands array. + else if (cmd.a && cmd.a.commands) { + this.exe(cmd.a.commands); + continue; + } + if (cmd.c && this[cmd.c]) { // explicit command name .. ! + const rx = this[cmd.c](cmd.a); + if (rx && rx instanceof Promise) { + await rx; + } + } + } + }, + view({ x = 0, y = 0, scl = 1, cartesian = false }) { + this.pushTrf(cartesian ? [scl, 0, 0, -scl, x, this.ctx.canvas.height - 1 - y] + : [scl, 0, 0, scl, x, y]); + }, + grid({ color = '#ccc', size } = {}) { + let ctx = this.ctx, b = ctx.canvas.width, h = ctx.canvas.height, + { x, y, scl } = this.uniTrf, + sz = size || this.gridSize(scl), + xoff = x % sz, yoff = y % sz; + ctx.save(); + ctx.setTransform(1, 0, 0, 1, 0, 0); + ctx.strokeStyle = color; + ctx.lineWidth = 1; + ctx.beginPath(); + for (let x = xoff, nx = b + 1; x < nx; x += sz) { ctx.moveTo(x, 0); ctx.lineTo(x, h); } + for (let y = yoff, ny = h + 1; y < ny; y += sz) { ctx.moveTo(0, y); ctx.lineTo(b, y); } + ctx.stroke(); + ctx.restore(); + }, + clr({ b, h } = {}) { + let ctx = this.ctx; + ctx.save(); + ctx.setTransform(1, 0, 0, 1, 0, 0); + ctx.clearRect(0, 0, b || ctx.canvas.width, h || ctx.canvas.height); + ctx.restore(); + }, + cir({ r }) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + this.ctx.beginPath(); + this.ctx.arc(x || 0, y || 0, Math.abs(r), 0, 2 * Math.PI, true); + this.drw(arguments[0]); + }, + arc({ r, w = 0, dw = 2 * Math.PI }) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + if (Math.abs(dw) > Number.EPSILON && Math.abs(r) > Number.EPSILON) { + this.ctx.beginPath(); + this.ctx.arc(x, y, Math.abs(r), w, w + dw, dw < 0); + this.drw(arguments[0]); + } + else if (Math.abs(dw) < Number.EPSILON && Math.abs(r) > Number.EPSILON) { + const cw = Math.cos(w), sw = Math.sin(w); + this.ctx.beginPath(); + this.ctx.moveTo(x - r * cw, y - r * sw); + this.ctx.lineTo(x + r * cw, y + r * sw); + } + // else // nothing to draw with r === 0 + }, + ell({ rx, ry, w = 0, dw = 2 * Math.PI, rot = 0 }) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + this.ctx.beginPath(); + this.ctx.ellipse(x, y, Math.abs(rx), Math.abs(ry), rot, w, w + dw, dw < 0); + this.drw(arguments[0]); + }, + rec({ b, h }) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + const tmp = this.setStyle(arguments[0]); + this.ctx.fillRect(x, y, b, h); + this.ctx.strokeRect(x, y, b, h); + this.resetStyle(tmp); + }, + lin(args) { + this.ctx.beginPath(); + this.ctx.moveTo(args.p1 && args.p1.x || args.x1 || 0, args.p1 && args.p1.y || args.y1 || 0); + this.ctx.lineTo(args.p2 && args.p2.x || args.x2 || 0, args.p2 && args.p2.y || args.y2 || 0); + this.stroke(args); + }, + ply({ pts, closed, w = 0, _itr }) { + if (_itr && _itr.len) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + let p, i, len = _itr.len, istrf = !!(x || y || w), cw, sw; + if (istrf) this.setTrf([cw = (w ? Math.cos(w) : 1), sw = (w ? Math.sin(w) : 0), -sw, cw, x, y]); + this.ctx.beginPath(); + this.ctx.moveTo((p = _itr(0)).x, p.y); + for (i = 1; i < len; i++) + this.ctx.lineTo((p = _itr(i)).x, p.y); + if (closed) // closed then .. + this.ctx.closePath(); + this.drw(arguments[0]); + if (istrf) this.resetTrf(); + return i - 1; // number of points .. + } + return 0; + }, + txt({ str, w = 0/*,unsizable*/ }) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + const tmp = this.setStyle(arguments[0]), + sw = w ? Math.sin(w) : 0, + cw = w ? Math.cos(w) : 1, + trf = this.isCartesian ? [cw, sw, sw, -cw, x, y] + : [cw, sw, -sw, cw, x, y]; + this.setTrf(trf); // this.setTrf(unsizable ? this.concatTrf(this.unscaleTrf({x,y}),trf) : trf); + if (this.ctx.fillStyle === 'rgba(0, 0, 0, 0)') { + this.ctx.fillStyle = this.ctx.strokeStyle; + tmp.fs = 'transparent'; + } + this.ctx.fillText(str, 0, 0); + this.resetTrf(); + this.resetStyle(tmp); + }, + errorImageStr: "", + images: Object.create(null), + async loadImage(uri) { + const download = async (xuri) => { + const pimg = new Promise((resolve, reject) => { + let img = new Image(); + img.src = xuri; + function error(err) { + img.removeEventListener('load', load); + img = undefined; + reject(err); + }; + function load() { + img.removeEventListener('error', error); + resolve(img); + img = undefined; + }; + img.addEventListener('error', error, { once: true }); + img.addEventListener('load', load, { once: true }); + }); + + try { + return await pimg; + } catch (err) { + // console.warn(`failed to (pre-)load image; '${xuri}'`, err); + if (xuri === this.errorImageStr) { + throw err; + } else { + return await download(this.errorImageStr); + } + } + } + + let img = this.images[uri]; + if (img !== undefined) { + return img instanceof Promise ? await img : img; + } + img = download(uri); + this.images[uri] = img; + try { + img = await img; + } finally { + this.images[uri] = img; + } + return img; + }, + async img({ uri, x = 0, y = 0, b, h, sx = 0, sy = 0, sb, sh, xoff = 0, yoff = 0, w = 0, scl = 1 }) { + const img_ = await this.loadImage(uri); + this.ctx.save(); + const cart = this.isCartesian ? -1 : 1; + sb = sb || img_.width; + b = b || img_.width; + sh = (sh || img_.height); + h = (h || img_.height) * cart; + yoff *= cart; + w *= cart; + y = this.isCartesian ? -(y / scl) + sy : y / scl; + const [cw, sw] = [Math.cos(w), Math.sin(w)]; + this.ctx.scale(scl, scl * cart); + this.ctx.transform(cw, sw, -sw, cw, x / scl, y); + this.ctx.drawImage(img_, sx, sy, sb, sh, xoff, yoff, b, h); + this.ctx.restore(); + }, + use({ grp }) { + this.beg(arguments[0]); + this.exe(grp.commands); + this.end(); + }, + beg({ w = 0, scl = 1, matrix/*,unsizable*/ } = {}) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + let trf = matrix; + if (!trf) { + let ssw, scw; + ssw = w ? Math.sin(w) * scl : 0; + scw = w ? Math.cos(w) * scl : scl; + trf = [scw, ssw, -ssw, scw, x, y]; + } + this.pushStyle(arguments[0]); + this.pushTrf(trf); // this.pushTrf(unsizable ? this.concatTrf(this.unscaleTrf({x,y}),trf) : trf); + }, + end() { + this.popTrf(); + this.popStyle(); + }, + p() { this.ctx.beginPath(); }, + z() { this.ctx.closePath(); }, + m({ x, y }) { this.ctx.moveTo(x, y); }, + l({ x, y }) { this.ctx.lineTo(x, y); }, + q({ x, y, x1, y1 }) { this.ctx.quadraticCurveTo(x1, y1, x, y); }, + c({ x, y, x1, y1, x2, y2 }) { this.ctx.bezierCurveTo(x1, y1, x2, y2, x, y); }, + a({ dw, k, phi, _xp, _yp }) { // todo: fix elliptical arc bug ... + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + if (k === undefined) k = 1; // ratio r1/r2 + if (Math.abs(dw) > Number.EPSILON) { + if (k === 1) { // circular arc ... + let x12 = x - _xp, y12 = y - _yp; + let tdw_2 = Math.tan(dw / 2), + rx = (x12 - y12 / tdw_2) / 2, ry = (y12 + x12 / tdw_2) / 2, + R = Math.hypot(rx, ry), + w = Math.atan2(-ry, -rx); + this.ctx.ellipse(_xp + rx, _yp + ry, R, R, 0, w, w + dw, this.cartesian ? dw > 0 : dw < 0); + } + else { // elliptical arc .. still buggy .. ! + if (phi === undefined) phi = 0; + let x1 = dw > 0 ? _xp : x, + y1 = dw > 0 ? _yp : y, + x2 = dw > 0 ? x : _xp, + y2 = dw > 0 ? y : _yp; + let x12 = x2 - x1, y12 = y2 - y1, + _dw = (dw < 0) ? dw : -dw; + // if (dw < 0) dw = -dw; // test for bugs .. ! + let cp = phi ? Math.cos(phi) : 1, sp = phi ? Math.sin(phi) : 0, + dx = -x12 * cp - y12 * sp, dy = -x12 * sp - y12 * cp, + sdw_2 = Math.sin(_dw / 2), + R = Math.sqrt((dx * dx + dy * dy / (k * k)) / (4 * sdw_2 * sdw_2)), + w = Math.atan2(k * dx, dy) - _dw / 2, + x0 = x1 - R * Math.cos(w), + y0 = y1 - R * k * Math.sin(w); + this.ctx.ellipse(x0, y0, R, R * k, phi, w, w + dw, this.cartesian ? dw > 0 : dw < 0); + } + } + else + this.ctx.lineTo(x, y); + }, + + stroke({ d } = {}) { + let tmp = this.setStyle(arguments[0]); + d ? this.ctx.stroke(new Path2D(d)) : this.ctx.stroke(); // SVG path syntax + this.resetStyle(tmp); + }, + fill({ d } = {}) { + let tmp = this.setStyle(arguments[0]); + d ? this.ctx.fill(new Path2D(d)) : this.ctx.fill(); // SVG path syntax + this.resetStyle(tmp); + }, + drw({ d, lsh } = {}) { + let ctx = this.ctx, + tmp = this.setStyle(arguments[0]), + p = d && new Path2D(d); // SVG path syntax + d ? ctx.fill(p) : ctx.fill(); + if (ctx.shadowColor !== 'rgba(0, 0, 0, 0)' && ctx.fillStyle !== 'rgba(0, 0, 0, 0)' && !lsh) { + let shc = ctx.shadowColor; // usually avoid stroke shadow when filling ... + ctx.shadowColor = 'rgba(0, 0, 0, 0)'; + d ? ctx.stroke(p) : ctx.stroke(); + ctx.shadowColor = shc; + } + else + d ? ctx.stroke(p) : ctx.stroke(); + this.resetStyle(tmp); + }, + + // State management (transform & style) + // getters & setters + get: { + fs: (ctx) => ctx.fillStyle, + ls: (ctx) => ctx.strokeStyle, + lw: (ctx) => ctx.lineWidth, + lc: (ctx) => ctx.lineCap, + lj: (ctx) => ctx.lineJoin, + ld: (ctx) => ctx.getLineDash(), + ldoff: (ctx) => ctx.lineDashOffset, + ml: (ctx) => ctx.miterLimit, + sh: (ctx) => [ctx.shadowOffsetX || 0, ctx.shadowOffsetY || 0, + ctx.shadowBlur || 0, ctx.shadowColor || 'black'], + font: (ctx) => ctx.font, + thal: (ctx) => ctx.textAlign, + tval: (ctx) => ctx.textBaseline, + }, + set: { + fs: (ctx, q) => { ctx.fillStyle = q; }, + ls: (ctx, q) => { ctx.strokeStyle = q; }, + lw: (ctx, q) => { ctx.lineWidth = q; }, + lc: (ctx, q) => { ctx.lineCap = q; }, + lj: (ctx, q) => { ctx.lineJoin = q; }, + ld: (ctx, q) => { ctx.setLineDash(q); }, + ldoff: (ctx, q) => { ctx.lineDashOffset = q; }, + ml: (ctx, q) => { ctx.miterLimit = q; }, + sh: (ctx, q) => { + if (q) { + ctx.shadowOffsetX = q[0] || 0; + ctx.shadowOffsetY = q[1] || 0; + ctx.shadowBlur = q[2] || 0; + ctx.shadowColor = q[3] || 'black'; + } + }, + font: (ctx, q) => { ctx.font = q; }, + thal: (ctx, q) => { ctx.textAlign = q; }, + tval: (ctx, q) => { ctx.textBaseline = q; } + }, + initStyle(style) { + for (const key in style) + if (this.get[key] && this.get[key](this.ctx) !== style[key]) + this.set[key](this.ctx, style[key]); + }, + setStyle(style) { // short circuit style setting + let q, prv = {}; + for (const key in style) { + if (this.get[key]) { // style keys only ... + if (typeof style[key] === 'string' && style[key][0] === '@') { + let ref = style[key].substr(1); + style[key] = g2.symbol[ref] || this.get[ref] && this.get[ref](this.ctx); + } + if ((q = this.get[key](this.ctx)) !== style[key]) { + prv[key] = q; + this.set[key](this.ctx, style[key]); + } + } + } + return prv; + }, + resetStyle(style) { // short circuit style reset + for (const key in style) + this.set[key](this.ctx, style[key]); + }, + pushStyle(style) { + let cur = {}; // hold changed properties ... + for (const key in style) + if (this.get[key]) { // style keys only ... + if (typeof style[key] === 'string' && style[key][0] === '@') { + let ref = style[key].substr(1); + style[key] = g2.symbol[ref] || this.get[ref] && this.get[ref](this.ctx); + } + if (this.cur[key] !== style[key]) + this.set[key](this.ctx, (cur[key] = style[key])); + } + this.stack.push(this.cur = Object.assign({}, this.cur, cur)); + }, + popStyle() { + let cur = this.stack.pop(); + this.cur = this.stack[this.stack.length - 1]; + for (const key in this.cur) + if (this.get[key] && this.cur[key] !== cur[key]) + this.set[key](this.ctx, this.cur[key]); + }, + concatTrf(q, t) { + return [ + q[0] * t[0] + q[2] * t[1], + q[1] * t[0] + q[3] * t[1], + q[0] * t[2] + q[2] * t[3], + q[1] * t[2] + q[3] * t[3], + q[0] * t[4] + q[2] * t[5] + q[4], + q[1] * t[4] + q[3] * t[5] + q[5] + ]; + }, + initTrf() { + this.ctx.setTransform(...this.matrix[0]); + }, + setTrf(t) { + this.ctx.setTransform(...this.concatTrf(this.matrix[this.matrix.length - 1], t)); + }, + resetTrf() { + this.ctx.setTransform(...this.matrix[this.matrix.length - 1]); + }, + pushTrf(t) { + let q_t = this.concatTrf(this.matrix[this.matrix.length - 1], t); + this.matrix.push(q_t); + this.ctx.setTransform(...q_t); + }, + popTrf() { + this.matrix.pop(); + this.ctx.setTransform(...this.matrix[this.matrix.length - 1]); + }, + get isCartesian() { // det of mat2x2 < 0 ! + let m = this.matrix[this.matrix.length - 1]; + return m[0] * m[3] - m[1] * m[2] < 0; + }, + get uniTrf() { + let m = this.matrix[this.matrix.length - 1]; + return { x: m[4], y: m[5], scl: Math.hypot(m[0], m[1]), cartesian: m[0] * m[3] - m[1] * m[2] < 0 }; + }, + unscaleTrf({ x, y }) { // remove scaling effect (make unzoomable with respect to (x,y)) + let m = this.matrix[this.matrix.length - 1], + invscl = 1 / Math.hypot(m[0], m[1]); + return [invscl, 0, 0, invscl, (1 - invscl) * x, (1 - invscl) * y]; + }, + gridSize(scl) { + let base = this.gridBase, exp = this.gridExp, sz; + while ((sz = scl * base * Math.pow(10, exp)) < 14 || sz > 35) { + if (sz < 14) { + if (base == 1) base = 2; + else if (base == 2) base = 5; + else if (base == 5) { base = 1; exp++; } + } + else { + if (base == 1) { base = 5; exp--; } + else if (base == 2) base = 1; + else if (base == 5) base = 2; + } + } + this.gridBase = base; + this.gridExp = exp; + return sz; + } +} + +// use it with node.js ... ? +if (typeof module !== 'undefined') module.exports = g2; + +/** + * g2.lib (c) 2013-17 Stefan Goessner + * geometric constants and higher functions + * @license MIT License + * @link https://github.com/goessner/g2 + */ +"use strict" + +var g2 = g2 || {}; // for standalone usage ... + +g2 = Object.assign(g2, { + EPS: Number.EPSILON, + PI: Math.PI, + PI2: 2*Math.PI, + SQRT2: Math.SQRT2, + SQRT2_2: Math.SQRT2/2, + /** + * Map angle to the range [0 .. 2*pi]. + * @param {number} w Angle in radians. + * @returns {number} Angle in radians in interval [0 .. 2*pi]. + */ + toPi2(w) { return (w % g2.PI2 + g2.PI2) % g2.PI2; }, + /** + * Map angle to the range [-pi .. pi]. + * @param {number} w Angle in radians. + * @returns {number} Angle in radians in interval [-pi .. pi]. + */ + toPi(w) { return (w = (w % g2.PI2 + g2.PI2) % g2.PI2) > g2.PI ? w - g2.PI2 : w; }, + /** + * Map angle to arc sector [w0 .. w0+dw]. + * @param {number} w Angle in range [0 .. 2*pi]. + * @param {number} w0 Start angle in range [0 .. 2*pi]. + * @param {number} dw angular range in radians. Can be positive or negative. + * @returns {number} Normalised angular parameter lambda. + * '0' corresponds to w0 and '1' to w0+dw. To reconstruct an angle from + * the return parameter lambda use: w = w0 + lambda*dw. + */ + toArc: function(w,w0,dw) { + if (dw > g2.EPS || dw < -g2.EPS) { + if (w0 > w && w0+dw > g2.PI2) w0 -= g2.PI2; + else if (w0 < w && w0+dw < 0) w0 += g2.PI2; + return (w-w0)/dw; + + } + return 0; + }, + /** + * Test, if point is located on line. + * @param {x,y} point to test. + * @param {x1,y1} start point of line. + * @param {x2,y2} end point of line. + * @param {number} eps. + * @return {boolean} the test result. + */ + isPntOnLin({x,y},p1,p2,eps=Number.EPSILON) { + let dx = p2.x - p1.x, dy = p2.y - p1.y, dx2 = x - p1.x, dy2 = y - p1.y, + dot = dx*dx2 + dy*dy2, perp = dx*dy2 - dy*dx2, len = Math.hypot(dx,dy), epslen = eps*len; + return -epslen < perp && perp < epslen && -epslen < dot && dot < len*(len+eps); + }, + /** + * Test, if point is located on circle circumference. + * @param {x,y} point to test. + * @param {x,y,r} circle. + * @param {number} eps. + * @return {boolean} the test result. + */ + isPntOnCir({x:xp,y:yp},{x,y,r},eps=Number.EPSILON) { + let dx = xp - x, dy = yp - y, + ddis = dx*dx + dy*dy - r*r, reps = eps*r; + return -reps < ddis && ddis < reps; + }, + /** + * Test, if point is located on a circular arc. + * @param {x,y} point to test. + * @param {x,y,r} circle. + * @param {number} eps. + * @return {boolean} the test result. + */ + isPntOnArc({x:xp,y:yp},{x,y,r,w,dw},eps=Number.EPSILON) { + var dx = xp - x, dy = yp - y, dist = Math.hypot(dx,dy), + mu = g2.toArc(g2.toPi2(Math.atan2(dy,dx)),g2.toPi2(w),dw); + return r*Math.abs(dw) > eps && Math.abs(dist-r) < eps && mu >= 0 && mu <= 1; + }, + /** + * Test, if point is located on a polygon line. + * @param {x,y} point to test. + * @param {pts,closed} polygon. + * @param {number} eps. + * @return {boolean} the test result. + */ + isPntOnPly({x,y},{pts,closed},eps=Number.EPSILON) { + // console.log(pts) + for (var i=0,n=pts.length; i<(closed ? n : n-1); i++) + if (g2.isPntOnLin({x,y},pts[i],pts[(i+1)%n],eps)) + return true; + return false; + }, + /** + * Test, if point is located on a box. A box in contrast to a rectangle + * is always aligned parallel to coordinate system axes, with its + * local origin `{x,y}` located in the center. The dimensions `{b,h}` are + * half size dimensions (so upper right corner is {x+b,y+h}). + * @param {x,y} point to test. + * @param {x,y,r} circle. + * @param {number} eps. + * @return {boolean} the test result. + */ + isPntOnBox({x:xp,y:yp},{x,y,b,h},eps=Number.EPSILON) { + var dx = x.p - x, dy = yp - y; + return dx >= b-eps && dx <= b+eps && dy <= h+eps && dy >= -h-eps + || dx >= -b-eps && dx <= b+eps && dy <= h+eps && dy >= h-eps + || dx >= -b-eps && dx <= -b+eps && dy <= h+eps && dy >= -h-eps + || dx >= -b-eps && dx <= b+eps && dy <= -h+eps && dy >= -h-eps; + }, + /** + * Test, if point is located inside of a circle. + * @param {x,y} point to test. + * @param {x,y,r} circle. + * @return {boolean} the test result. + */ + isPntInCir({x:xp,y:yp},{x,y,r}) { + return (x - xp)**2 + (y - yp)**2 < r*r; + }, + /** + * Test, if point is located inside of a closed polygon. + * (see http://paulbourke.net/geometry/polygonmesh/) + * @param {x,y} point to test. + * @param {pnts,closed} polygon. + * @returns {boolean} point is on polygon lines. + */ + isPntInPly({x,y},{pts,closed},eps=Number.EPSILON) { + let match = 0; + for (let n=pts.length,i=0,pi=pts[i],pj=pts[n-1]; i pi.y || y > pj.y) + && (y <= pi.y || y <= pj.y) + && (x <= pi.x || x <= pj.x) + && pi.y !== pj.y + && (pi.x === pj.x || x <= pj.x + (y-pj.y)*(pi.x-pj.x)/(pi.y-pj.y))) + match++; + return match%2 != 0; // even matches required for being outside .. + }, + /** + * Test, if point is located inside of a box. A box in contrast to a rectangle + * is always aligned parallel to coordinate system axes, with its + * local origin `{x,y}` located in the center. The dimensions `{b,h}` are + * half size dimensions (so upper right corner is {x+b,y+h}). + * @param {x,y} point to test. + * @param {x,y,r} circle. + * @return {boolean} the test result. + */ + isPntInBox({x:xp,y:yp},{x,y,b,h}) { + var dx = xp - x, dy = yp - y; + return dx >= -b && dx <= b && dy >= -h && dy <= h; + }, + + arc3pts(x1,y1,x2,y2,x3,y3) { + const dx1 = x2 - x1, dy1 = y2 - y1; + const dx2 = x3 - x2, dy2 = y3 - y2; + const den = dx1*dy2 - dy1*dx2; + const lam = Math.abs(den) > Number.EPSILON + ? 0.5*((dx1 + dx2)*dx2 + (dy1 + dy2)*dy2)/den + : 0; + const x0 = lam ? x1 + 0.5*dx1 - lam*dy1 : x1 + 0.5*(dx1 + dx2); + const y0 = lam ? y1 + 0.5*dy1 + lam*dx1 : y1 + 0.5*(dy1 + dy2); + const dx01 = x1 - x0, dy01 = y1 - y0; + const dx03 = x3 - x0, dy03 = y3 - y0; + const dw = lam ? Math.atan2(dx01*dy03-dy01*dx03,dx01*dx03+dy01*dy03) : 0; + const r = dw ? Math.hypot(dy01,dx01) : 0.5*Math.hypot(dy1+dy2,dx1+dx2); + + return {x:x0,y:y0,r:r,w:Math.atan2(dy01,dx01),dw}; + } +}) + +"use strict" + +/** + * g2.ext (c) 2015-20 Stefan Goessner + * @author Stefan Goessner + * @license MIT License + * @requires g2.core.js + * @typedef {g2} + * @description Additional methods for g2. + * @returns {g2} + */ + +var g2 = g2 || { prototype: {} }; // for jsdoc only ... + +// constants for element selection / editing +g2.NONE = 0x0; g2.OVER = 0x1; g2.DRAG = 0x2; g2.EDIT = 0x4; + +/** + * Extended style values. + * Not really meant to get overwritten. But if you actually want, proceed.
+ * These styles can be referenced using the comfortable '@' syntax. + * @namespace + * @property {object} symbol `g2` symbol namespace. + * @property {object} [symbol.tick] Predefined symbol: a little tick + * @property {object} [symbol.dot] Predefined symbol: a little dot + * @property {object} [symbol.sqr] Predefined symbol: a little square + * @property {string} [symbol.nodcolor=#333] node color. + * @property {string} [symbol.nodfill=#dedede] node fill color. + * @property {string} [symbol.nodfill2=#aeaeae] alternate node fill color, somewhat darker. + * @property {string} [symbol.linkcolor=#666] link color. + * @property {string} [symbol.linkfill=rgba(225,225,225,0.75)] link fill color, semi-transparent. + * @property {string} [symbol.dimcolor=darkslategray] dimension color. + * @property {array} [symbol.solid=[]] solid line style. + * @property {array} [symbol.dash=[15,10]] dashed line style. + * @property {array} [symbol.dot=[4,4]] dotted line style. + * @property {array} [symbol.dashdot=[25,6.5,2,6.5]] dashdotted line style. + * @property {number} [symbol.labelOffset=5] default label offset distance. + * @property {number} [symbol.labelSignificantDigits=3] default label's significant digits after numbering point. + */ +g2.symbol = g2.symbol || {}; +g2.symbol.tick = g2().p().m({ x: 0, y: -2 }).l({ x: 0, y: 2 }).stroke({ lc: "round", lwnosc: true }); +g2.symbol.dot = g2().cir({ x: 0, y: 0, r: 2, ls: "transparent" }); +g2.symbol.sqr = g2().rec({ x: -1.5, y: -1.5, b: 3, h: 3, ls: "transparent" }); + +g2.symbol.nodcolor = "#333"; +g2.symbol.nodfill = "#dedede"; +g2.symbol.nodfill2 = "#aeaeae"; +g2.symbol.linkcolor = "#666"; +g2.symbol.linkfill = "rgba(225,225,225,0.75)"; +g2.symbol.dimcolor = "darkslategray"; +g2.symbol.solid = []; +g2.symbol.dash = [15, 10]; +g2.symbol.dot = [4, 4]; +g2.symbol.dashdot = [25, 6.5, 2, 6.5]; +g2.symbol.labelSignificantDigits = 3; // 0.1234 => 0.123, 0.01234 => 0.0123, 1.234 => 1.23, 12.34 => 12.3, 123.4 => 123, 1234 => 1234 + +/** +* Flatten object properties (evaluate getters) +*/ +g2.flatten = function (obj) { + const args = Object.create(null); // important ! + for (let p in obj) + if (typeof obj[p] !== 'function') + args[p] = obj[p]; + return args; +} +/* +g2.strip = function(obj,prop) { + const clone = Object.create(Object.getPrototypeOf(obj),Object.getOwnPropertyDescriptors(obj)); + Object.defineProperty(clone, prop, { get:undefined, enumerable:true, configurable:true, writabel:false }); + return clone; +} +*/ +g2.pointIfc = { + // p vector notation ! ... helps to avoid object destruction + get p() { return { x: this.x, y: this.y }; }, // visible if 'p' is *not* explicite given. + get x() { return Object.getOwnPropertyDescriptor(this, 'p') ? this.p.x : 0; }, + get y() { return Object.getOwnPropertyDescriptor(this, 'p') ? this.p.y : 0; }, + set x(q) { if (Object.getOwnPropertyDescriptor(this, 'p')) this.p.x = q; }, + set y(q) { if (Object.getOwnPropertyDescriptor(this, 'p')) this.p.y = q; }, +} + +g2.labelIfc = { + getLabelOffset() { const off = this.label.off !== undefined ? +this.label.off : 1; return off + Math.sign(off) * (this.lw || 2) / 2; }, + getLabelString() { + let s = typeof this.label === 'object' ? this.label.str : typeof this.label === 'string' ? this.label : '?'; + if (s && s[0] === "@" && this[s.substr(1)]) { + s = s.substr(1); + let val = this[s]; + val = Number.isInteger(val) ? val + : Number(val).toFixed(Math.max(g2.symbol.labelSignificantDigits - Math.log10(val), 0)); + + s = `${val}${s === 'angle' ? "°" : ""}`; + } + return s; + }, + drawLabel(g) { + const lbl = this.label; + const font = lbl.font || g2.defaultStyle.font; + const h = parseInt(font); // font height (px assumed !) + const str = this.getLabelString(); + const rx = (str.length || 1) * 0.75 * h / 2, ry = 1.25 * h / 2; // ellipse semi-axes length + const pos = this.pointAt(lbl.loc || this.lbloc || 'se'); + const off = this.getLabelOffset(); + const p = { + x: pos.x + pos.nx * (off + Math.sign(off) * rx), + y: pos.y + pos.ny * (off + Math.sign(off) * ry) + }; + + if (lbl.border) g.ell({ x: p.x, y: p.y, rx, ry, ls: lbl.fs || 'black', fs: lbl.fs2 || '#ffc' }); + g.txt({ + str, x: p.x, y: p.y, + thal: "center", tval: "middle", + fs: lbl.fs || 'black', font: lbl.font + }); + return g; + } +} + +g2.markIfc = { + markAt(loc) { + const p = this.pointAt(loc); + const w = Math.atan2(p.ny, p.nx) + Math.PI / 2; + return { + grp: this.getMarkSymbol(), x: p.x, y: p.y, w: w, scl: this.lw || 1, + ls: this.ls || '#000', fs: this.fs || this.ls || '#000' + } + }, + getMarkSymbol() { + // Use tick as default + const mrk = this.mark + if (typeof mrk === 'number' || !mrk) return g2.symbol.tick; + if (typeof mrk.symbol === 'object') return mrk.symbol; + if (typeof mrk.symbol === 'string') return g2.symbol[mrk.symbol] + }, + // loop is for elements that close, e.g. rec or cir => loc at 0 === loc at 1 + drawMark(g, closed = false) { + let loc; + if (Array.isArray(this.mark)) { + loc = this.mark; + } + else { + const count = typeof this.mark === 'object' ? this.mark.count : this.mark; + loc = count ? + Array.from(Array(count)).map((_, i) => i / (count - !closed)) : + this.mark.loc; + } + for (let l of loc) { + g.use(this.markAt(l)); + } + return g; + } +} + +g2.prototype.cir.prototype = g2.mix(g2.pointIfc, g2.labelIfc, g2.markIfc, { + w: 0, // default start angle (used for dash-dot orgin and editing) + lbloc: 'c', + get isSolid() { return this.fs && this.fs !== 'transparent' }, + get len() { return 2 * Math.PI * this.r; }, + get lsh() { return this.state & g2.OVER; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false }, + get g2() { // dynamically switch existence of method via getter ... cool ! + const e = g2(); // hand object stripped from `g2` + this.label && e.ins((g) => this.drawLabel(g)); + this.mark && e.ins((g) => this.drawMark(g, true)); + return () => g2().cir(g2.flatten(this)).ins(e); // avoiding infinite recursion ! + }, + pointAt(loc) { + const Q = Math.SQRT2 / 2; + const LOC = { c: [0, 0], e: [1, 0], ne: [Q, Q], n: [0, 1], nw: [-Q, Q], w: [-1, 0], sw: [-Q, -Q], s: [0, -1], se: [Q, -Q] }; + const q = (loc + 0 === loc) ? [Math.cos(loc * 2 * Math.PI), Math.sin(loc * 2 * Math.PI)] + : (LOC[loc || "c"] || [0, 0]); + return { + x: this.x + q[0] * this.r, + y: this.y + q[1] * this.r, + nx: q[0], + ny: q[1] + }; + }, + hit({ x, y, eps }) { + return this.isSolid ? g2.isPntInCir({ x, y }, this, eps) + : g2.isPntOnCir({ x, y }, this, eps); + }, + drag({ dx, dy }) { this.x += dx; this.y += dy }, +}); + +g2.prototype.lin.prototype = g2.mix(g2.labelIfc, g2.markIfc, { + // p1 vector notation ! + get p1() { return { x1: this.x1, y1: this.y1 }; }, // relevant if 'p1' is *not* explicite given. + get x1() { return Object.getOwnPropertyDescriptor(this, 'p1') ? this.p1.x : 0; }, + get y1() { return Object.getOwnPropertyDescriptor(this, 'p1') ? this.p1.y : 0; }, + set x1(q) { if (Object.getOwnPropertyDescriptor(this, 'p1')) this.p1.x = q; }, + set y1(q) { if (Object.getOwnPropertyDescriptor(this, 'p1')) this.p1.y = q; }, + // p2 vector notation ! + get p2() { return { x2: this.x2, y2: this.y2 }; }, // relevant if 'p2' is *not* explicite given. + get x2() { return Object.getOwnPropertyDescriptor(this, 'p2') ? this.p2.x : 0; }, + get y2() { return Object.getOwnPropertyDescriptor(this, 'p2') ? this.p2.y : 0; }, + set x2(q) { if (Object.getOwnPropertyDescriptor(this, 'p2')) this.p2.x = q; }, + set y2(q) { if (Object.getOwnPropertyDescriptor(this, 'p2')) this.p2.y = q; }, + + isSolid: false, + get len() { return Math.hypot(this.x2 - this.x1, this.y2 - this.y1); }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false }, + get g2() { // dynamically switch existence of method via getter ... ! + const e = g2(); + this.label && e.ins(e => this.drawLabel(e)); + this.mark && e.ins(e => this.drawMark(e)); + return () => g2().lin(g2.flatten(this)).ins(e); + }, + + pointAt(loc) { + let t = loc === "beg" ? 0 + : loc === "end" ? 1 + : (loc + 0 === loc) ? loc // numerical arg .. + : 0.5, // 'mid' .. + dx = this.x2 - this.x1, + dy = this.y2 - this.y1, + len = Math.hypot(dx, dy); + return { + x: this.x1 + dx * t, + y: this.y1 + dy * t, + nx: len ? dy / len : 0, + ny: len ? -dx / len : -1 + }; + }, + hit({ x, y, eps }) { + return g2.isPntOnLin({ x, y }, { x: this.x1, y: this.y1 }, { x: this.x2, y: this.y2 }, eps); + }, + drag({ dx, dy }) { + this.x1 += dx; this.x2 += dx; + this.y1 += dy; this.y2 += dy; + } +}); + +g2.prototype.rec.prototype = g2.mix(g2.pointIfc, g2.labelIfc, g2.markIfc, { + get len() { return 2 * (this.b + this.h); }, + get isSolid() { return this.fs && this.fs !== 'transparent' }, + get lsh() { return this.state & g2.OVER; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false; }, + get g2() { // dynamically switch existence of method via getter ... ! + const e = g2(); + this.label && e.ins(e => this.drawLabel(e)); + this.mark && e.ins(e => this.drawMark(e, true)); + return () => g2().rec(g2.flatten(this)).ins(e); + }, + lbloc: 'c', + pointAt(loc) { + const locAt = (loc) => { + const o = { c: [0, 0], e: [1, 0], ne: [0.95, 0.95], n: [0, 1], nw: [-0.95, 0.95], w: [-1, 0], sw: [-0.95, -0.95], s: [0, -1], se: [0.95, -0.95] }; + + if (o[loc]) return o[loc]; + + const w = 2 * Math.PI * loc + pi / 4; + if (loc <= 0.25) return [1 / Math.tan(w), 1]; + if (loc <= 0.50) return [-1, -Math.tan(w)]; + if (loc <= 0.75) return [- 1 / Math.tan(w), -1]; + if (loc <= 1.00) return [1, Math.tan(w)]; + } + const q = locAt(loc); + return { + x: this.x + (1 + q[0]) * this.b / 2, + y: this.y + (1 + q[1]) * this.h / 2, + nx: 1 - Math.abs(q[0]) < 0.01 ? q[0] : 0, + ny: 1 - Math.abs(q[1]) < 0.01 ? q[1] : 0 + }; + }, + hit({ x, y, eps }) { + return this.isSolid ? g2.isPntInBox({ x, y }, { x: this.x + this.b / 2, y: this.y + this.h / 2, b: this.b / 2, h: this.h / 2 }, eps) + : g2.isPntOnBox({ x, y }, { x: this.x + this.b / 2, y: this.y + this.h / 2, b: this.b / 2, h: this.h / 2 }, eps); + }, + drag({ dx, dy }) { this.x += dx; this.y += dy } +}); + +g2.prototype.arc.prototype = g2.mix(g2.pointIfc, g2.labelIfc, g2.markIfc, { + get len() { return Math.abs(this.r * this.dw); }, + isSolid: false, + get angle() { return this.dw / Math.PI * 180; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false }, + get g2() { // dynamically switch existence of method via getter ... ! + const e = g2(); + this.label && e.ins(e => this.drawLabel(e)); + this.mark && e.ins(e => this.drawMark(e)); + return () => g2().arc(g2.flatten(this)).ins(e); + }, + lbloc: 'mid', + pointAt(loc) { + let t = loc === "beg" ? 0 + : loc === "end" ? 1 + : loc === "mid" ? 0.5 + : loc + 0 === loc ? loc + : 0.5, + ang = (this.w || 0) + t * (this.dw || Math.PI * 2), cang = Math.cos(ang), sang = Math.sin(ang), r = loc === "c" ? 0 : this.r; + return { + x: this.x + r * cang, + y: this.y + r * sang, + nx: cang, + ny: sang + }; + }, + hit({ x, y, eps }) { return g2.isPntOnArc({ x, y }, this, eps) }, + drag({ dx, dy }) { this.x += dx; this.y += dy; }, +}); + +/** +* Draw interactive handle. +* @method +* @returns {object} g2 +* @param {object} - handle object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().hdl({x:100,y:80}) // Draw handle. +*/ +g2.prototype.hdl = function (args) { return this.addCommand({ c: 'hdl', a: args }); } +g2.prototype.hdl.prototype = g2.mix(g2.prototype.cir.prototype, { + r: 5, + isSolid: true, + draggable: true, + lbloc: 'se', + get lsh() { return this.state & g2.OVER; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false }, + g2() { + const { x, y, r, b = 4, shape = 'cir', ls = 'black', fs = '#ccc', sh } = this; + return shape === 'cir' ? g2().cir({ x, y, r, ls, fs, sh }).ins((g) => this.label && this.drawLabel(g)) + : g2().rec({ x: x - b, y: y - b, b: 2 * b, h: 2 * b, ls, fs, sh }).ins((g) => this.label && this.drawLabel(g)); + } +}); + +/** +* Node symbol. +* @constructor +* @param {object} - symbol arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().nod({x:10,y:10}) +*/ + +g2.prototype.nod = function (args = {}) { return this.addCommand({ c: 'nod', a: args }); } +g2.prototype.nod.prototype = g2.mix(g2.prototype.cir.prototype, { + r: 5, + ls: '@nodcolor', + fs: g2.symbol.nodfill, + isSolid: true, + lbloc: 'se', + g2() { // in contrast to `g2.prototype.cir.prototype`, `g2()` is called always ! + return g2() + .cir({ ...g2.flatten(this), r: this.r * (this.scl !== undefined ? this.scl : 1) }) + .ins((g) => this.label && this.drawLabel(g)) + } +}); + +/** + * Double nod symbol + * @constructor + * @returns {object} g2 + * @param {object} - symbol arguments object. + * @property {number} x - x-value center. + * @property {number} y - y-value center. + * @example + * g2().dblnod({x:10,y:10}) +*/ +g2.prototype.dblnod = function ({ x = 0, y = 0 }) { return this.addCommand({ c: 'dblnod', a: arguments[0] }); } +g2.prototype.dblnod.prototype = g2.mix(g2.prototype.cir.prototype, { + get r() { return 6; }, + get isSolid() { return true; }, + g2() { + return g2() + .beg({ x: this.x, y: this.y }) + .cir({ r: 6, ls: '@nodcolor', fs: '@nodfill', sh: this.sh }) + .cir({ r: 3, ls: '@nodcolor', fs: '@nodfill2' }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) + +/** +* Pole symbol. +* @constructor +* @returns {object} g2 +* @param {object} - symbol arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().pol({x:10,y:10}) +*/ +g2.prototype.pol = function (args = {}) { return this.addCommand({ c: 'pol', a: args }); } +g2.prototype.pol.prototype = g2.mix(g2.prototype.nod.prototype, { + g2() { + return g2() + .beg(g2.flatten(this)) + .cir({ r: 6, fs: g2.symbol.nodfill }) + .cir({ r: 2.5, fs: '@ls', ls: 'transparent' }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) + +/** +* Ground symbol. +* @constructor +* @param {object} - arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().gnd({x:10,y:10}) +*/ +g2.prototype.gnd = function (args = {}) { return this.addCommand({ c: 'gnd', a: args }); } +g2.prototype.gnd.prototype = g2.mix(g2.prototype.nod.prototype, { + g2() { + return g2() + .beg(g2.flatten(this)) + .cir({ x: 0, y: 0, r: 6 }) + .p() + .m({ x: 0, y: 6 }) + .a({ dw: Math.PI / 2, x: -6, y: 0 }) + .l({ x: 6, y: 0 }) + .a({ dw: -Math.PI / 2, x: 0, y: -6 }) + .z() + .fill({ fs: g2.symbol.nodcolor }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) + +g2.prototype.nodfix = function (args = {}) { return this.addCommand({ c: 'nodfix', a: args }); } +g2.prototype.nodfix.prototype = g2.mix(g2.prototype.nod.prototype, { + g2() { + return g2() + .beg(g2.flatten(this)) + .p() + .m({ x: -8, y: -12 }) + .l({ x: 0, y: 0 }) + .l({ x: 8, y: -12 }) + .drw({ fs: g2.symbol.nodfill2 }) + .cir({ x: 0, y: 0, r: this.r }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) +/** +* @method +* @returns {object} g2 +* @param {object} - symbol arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().view({cartesian:true}) + * .nodflt({x:10,y:10}) +*/ +g2.prototype.nodflt = function (args = {}) { return this.addCommand({ c: 'nodflt', a: args }); } +g2.prototype.nodflt.prototype = g2.mix(g2.prototype.nod.prototype, { + g2() { + return g2() + .beg(g2.flatten(this)) + .p() + .m({ x: -8, y: -12 }) + .l({ x: 0, y: 0 }) + .l({ x: 8, y: -12 }) + .drw({ ls: g2.symbol.nodcolor, fs: g2.symbol.nodfill2 }) + .cir({ x: 0, y: 0, r: this.r, ls: g2.symbol.nodcolor, fs: g2.symbol.nodfill }) + .lin({ x1: -9, y1: -19, x2: 9, y2: -19, ls: g2.symbol.nodfill2, lw: 5 }) + .lin({ x1: -9, y1: -15.5, x2: 9, y2: -15.5, ls: g2.symbol.nodcolor, lw: 2 }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) + +/** +* Draw vector arrow. +* @method +* @returns {object} g2 +* @param {object} - vector arguments object. +* @property {number} x1 - start x coordinate. +* @property {number} y1 - start y coordinate. +* @property {number} x2 - end x coordinate. +* @property {number} y2 - end y coordinate. +* @example +* g2().vec({x1:50,y1:20,x2:250,y2:120}) +*/ +g2.prototype.vec = function vec(args) { return this.addCommand({ c: 'vec', a: args }); } +g2.prototype.vec.prototype = g2.mix(g2.prototype.lin.prototype, { + g2() { + const { x1, y1, x2, y2, lw = 1, ls = '#000', ld = [], fs = ls || '#000', lc = 'round', lj = 'round', } = this; + const dx = x2 - x1, dy = y2 - y1, r = Math.hypot(dx, dy); + const b = 3 * (1 + lw) > r ? r / 3 : (1 + lw); + const arrowHead = () => g2().p().m({ x: 0, y: 0 }).l({ x: -5 * b, y: b }).a({ dw: -Math.PI / 3, x: -5 * b, y: -b }).z().drw({ ls, fs, lc, lj }); + return g2() + .beg({ x: x1, y: y1, w: Math.atan2(dy, dx), lc, lj }) + .p().m({ x: 0, y: 0 }) + .l({ x: r - 3 * b, y: 0 }) + .stroke({ ls, lw, ld }) + .use({ grp: arrowHead, x: r, y: 0 }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}); + +/** +* Arc as Vector +* @method +* @returns {object} g2 +* @param {object} - angular dimension arguments. +* @property {number} x - start x coordinate. +* @property {number} y - start y coordinate. +* @property {number} r - radius +* @property {number} [w=0] - start angle (in radian). +* @property {number} [dw=Math.PI/2] - angular range in radian. In case of positive values it is running counterclockwise with + * right handed (cartesian) coordinate system. +* @example +* g2().avec({x:100,y:70,r:50,w:pi/3,dw:4*pi/3}) +*/ +g2.prototype.avec = function avec(args) { return this.addCommand({ c: 'avec', a: args }); } +g2.prototype.avec.prototype = g2.mix(g2.prototype.arc.prototype, { + g2() { + const { x, y, r, w, dw = 0, lw = 1, lc = 'round', lj = 'round', ls, fs = ls || "#000", label } = this; + const b = 3 * (1 + lw) > r ? r / 3 : (1 + lw), bw = 5 * b / r; + const arrowHead = () => g2().p().m({ x: 0, y: 2 * b }).l({ x: 0, y: -2 * b }).m({ x: 0, y: 0 }).l({ x: -5 * b, y: b }) + .a({ dw: -Math.PI / 3, x: -5 * b, y: -b }).z().drw({ ls, fs }); + + return g2() + .beg({ x, y, w, ls, lw, lc, lj }) + .arc({ r, w: 0, dw }) + .use({ + grp: arrowHead, x: r * Math.cos(dw), y: r * Math.sin(dw), + w: (dw >= 0 ? dw + Math.PI / 2 - bw / 2 : dw - Math.PI / 2 + bw / 2) + }) + .end() + .ins((g) => label && this.drawLabel(g)); + } +}); + +/** +* Linear Dimension +* @method +* @returns {object} g2 +* @param {object} - dimension arguments object. +* @property {number} x1 - start x coordinate. +* @property {number} y1 - start y coordinate. +* @property {number} x2 - end x coordinate. +* @property {number} y2 - end y coordinate. +* @property {number} off - offset. +* @property {boolean} [inside=true] - draw dimension arrows between or outside of ticks. +* @example +* g2().dim({x1:60,y1:40,x2:190,y2:120}) +*/ +g2.prototype.dim = function dim(args) { return this.addCommand({ c: 'dim', a: args }); } +g2.prototype.dim.prototype = g2.mix(g2.prototype.lin.prototype, { + pointAt(loc) { + const pnt = g2.prototype.lin.prototype.pointAt.call(this, loc); + if (this.off) { + pnt.x += this.off * pnt.nx; + pnt.y += this.off * pnt.ny; + } + return pnt; + }, + g2() { + const { x1, y1, x2, y2, lw = 1, lc = 'round', lj = 'round', off = 0, inside = true, ls, fs = ls || "#000", label } = this; + const dx = x2 - x1, dy = y2 - y1, r = Math.hypot(dx, dy); + const b = 3 * (1 + lw) > r ? r / 3 : (1 + lw); + const arrowHead = () => g2().p().m({ x: 0, y: 2 * b }).l({ x: 0, y: -2 * b }).m({ x: 0, y: 0 }).l({ x: -5 * b, y: b }) + .a({ dw: -Math.PI / 3, x: -5 * b, y: -b }).z().drw({ ls, fs }); + return g2() + .beg({ x: x1 + off / r * dy, y: y1 - off / r * dx, w: Math.atan2(dy, dx), ls, fs, lw, lc, lj }) + .lin({ x1: (inside ? 4 * b : 0), y1: 0, x2: (inside ? r - 4 * b : r), y2: 0 }) + .use({ grp: arrowHead, x: r, y: 0, w: (inside ? 0 : Math.PI) }) + .use({ grp: arrowHead, x: 0, y: 0, w: (inside ? Math.PI : 0) }) + .lin({ x1: 0, y1: off, x2: 0, y2: 0 }) + .lin({ x1: r, y1: off, x2: r, y2: 0 }) + .end() + .ins((g) => label && this.drawLabel(g)); + } +}); + +/** +* Angular dimension +* @method +* @returns {object} g2 +* @param {object} - angular dimension arguments. +* @property {number} x - start x coordinate. +* @property {number} y - start y coordinate. +* @property {number} r - radius +* @property {number} [w=0] - start angle (in radian). +* @property {number} [dw=Math.PI/2] - angular range in radian. In case of positive values it is running counterclockwise with + * right handed (cartesian) coordinate system. +* @property {boolean} [outside=false] - draw dimension arrows outside of ticks. +* @depricated {boolean} [inside] - draw dimension arrows between ticks. +* @example +* g2().adim({x:100,y:70,r:50,w:pi/3,dw:4*pi/3}) +*/ +g2.prototype.adim = function adim(args) { return this.addCommand({ c: 'adim', a: args }); } +g2.prototype.adim.prototype = g2.mix(g2.prototype.arc.prototype, { + g2() { + const { x, y, r, w, dw, lw = 1, lc = 'round', lj = 'round', ls, fs = ls || "#000", label } = this; + const b = 3 * (1 + lw) > r ? r / 3 : (1 + lw), bw = 5 * b / r; + const arrowHead = () => g2().p().m({ x: 0, y: 2 * b }).l({ x: 0, y: -2 * b }).m({ x: 0, y: 0 }).l({ x: -5 * b, y: b }) + .a({ dw: -Math.PI / 3, x: -5 * b, y: -b }).z().drw({ ls, fs }); + + const outside = (this.inside !== undefined && this.outside === undefined) ? !this.inside : !!this.outside; // still support depricated property ! + + return g2() + .beg({ x, y, w, ls, lw, lc, lj }) + .arc({ r, w: 0, dw }) + .use({ grp: arrowHead, x: r, y: 0, w: (!outside && dw > 0 || outside && dw < 0 ? -Math.PI / 2 + bw / 2 : Math.PI / 2 - bw / 2) }) + .use({ grp: arrowHead, x: r * Math.cos(dw), y: r * Math.sin(dw), w: (!outside && dw > 0 || outside && dw < 0 ? dw + Math.PI / 2 - bw / 2 : dw - Math.PI / 2 + bw / 2) }) + .end() + .ins((g) => label && this.drawLabel(g)); + } +}); + +/** +* Origin symbol +* @constructor +* @returns {object} g2 +* @param {object} - symbol arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @property {number} w - angle in radians. +* @example +* g2().view({cartesian:true}) + * .origin({x:10,y:10}) +*/ +g2.prototype.origin = function (args = {}) { return this.addCommand({ c: 'origin', a: args }); } +g2.prototype.origin.prototype = g2.mix(g2.prototype.nod.prototype, { + lbloc: 'sw', + g2() { + const { x, y, w, ls = '#000', lw = 1 } = this; + return g2() + .beg({ x, y, w, ls }) + .vec({ x1: 0, y1: 0, x2: 40, y2: 0, lw, fs: '#ccc' }) + .vec({ x1: 0, y1: 0, x2: 0, y2: 40, lw, fs: '#ccc' }) + .cir({ x: 0, y: 0, r: lw + 1, fs: '#ccc' }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}); + +g2.prototype.ply.prototype = g2.mix(g2.labelIfc, g2.markIfc, { + get isSolid() { return this.closed && this.fs && this.fs !== 'transparent'; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false; }, + // get len() { + // let len_itr = 0; + // let last_pt = {x:0,y:0}; + // g2.pntItrOf(this.pts).map(pt => { + // len_itr += Math.hypot(pt.x-last_pt.x, pt.y-last_pt.y); + // last_pt = pt; + // }); + // return len_itr; + // }, + pointAt(loc) { + const t = loc === "beg" ? 0 + : loc === "end" ? 1 + : (loc + 0 === loc) ? loc // numerical arg .. + : 0.5, // 'mid' .. + pitr = g2.pntItrOf(this.pts), + pts = [], + len = []; + + for (let itr = 0; itr < pitr.len; itr++) { + const next = pitr((itr + 1) % pitr.len); + pts.push(pitr(itr)); + len.push(Math.hypot( + next.x - pitr(itr).x, + next.y - pitr(itr).y)); + } + this.closed || len.pop(); + const { t2, x, y, dx, dy } = (() => { + const target = t * len.reduce((a, b) => a + b); + for (let itr = 0, tmp = 0; itr < pts.length; itr++) { + tmp += len[itr]; + const next = pitr(itr + 1).x ? pitr(itr + 1) : pitr(0); + if (tmp >= target) { + return { + t2: 1 - (tmp - target) / len[itr], + x: pts[itr].x, + y: pts[itr].y, + dx: next.x - pts[itr].x, + dy: next.y - pts[itr].y + } + } + } + })(); + const len2 = Math.hypot(dx, dy); + return { + x: (this.x || 0) + x + dx * t2, + y: (this.y || 0) + y + dy * t2, + nx: len2 ? dy / len2 : 1, + ny: len2 ? dx / len2 : 0, + }; + }, + hit({ x, y, eps }) { + return this.isSolid ? g2.isPntInPly({ x: x - this.x, y: y - this.y }, this, eps) // translational transformation only .. at current .. ! + : g2.isPntOnPly({ x: x - this.x, y: y - this.y }, this, eps); + }, + drag({ dx, dy }) { this.x += dx; this.y += dy; }, + get g2() { + const e = g2(); + this.label && e.ins(e => this.drawLabel(e)); + this.mark && e.ins(e => this.drawMark(e, this.closed)); + return () => g2().ply(g2.flatten(this)).ins(e); + } +}); + +g2.prototype.use.prototype = { + // p vector notation ! + get p() { return { x: this.x, y: this.y }; }, // relevant if 'p' is *not* explicite given. + get x() { return Object.getOwnPropertyDescriptor(this, 'p') ? this.p.x : 0; }, + get y() { return Object.getOwnPropertyDescriptor(this, 'p') ? this.p.y : 0; }, + set x(q) { if (Object.getOwnPropertyDescriptor(this, 'p')) this.p.x = q; }, + set y(q) { if (Object.getOwnPropertyDescriptor(this, 'p')) this.p.y = q; }, + + isSolid: false, + /* + hit(at) { + for (const cmd of this.grp.commands) { + if (cmd.a.hit && cmd.a.hit(at)) + return true; + } + return false; + }, + + pointAt: g2.prototype.cir.prototype.pointAt, + */ +}; +// complex macros / add prototypes to argument objects + +/** +* Draw spline by points. +* Implementing a centripetal Catmull-Rom spline (thus avoiding cusps and self-intersections). +* Using iterator function for getting points from array by index. +* It must return current point object {x,y} or object {done:true}. +* Default iterator expects sequence of x/y-coordinates as a flat array [x,y,...], +* array of [[x,y],...] arrays or array of [{x,y},...] objects. +* @see https://pomax.github.io/bezierinfo +* @see https://de.wikipedia.org/wiki/Kubisch_Hermitescher_Spline +* @method +* @returns {object} g2 +* @param {object} - spline arguments object. +* @property {object[] | number[][] | number[]} pts - array of points. +* @property {bool} [closed=false] - closed spline. +* @example +* g2().spline({pts:[100,50,50,150,150,150,100,50]}) +*/ +g2.prototype.spline = function spline({ pts, closed, x, y, w }) { + arguments[0]._itr = g2.pntItrOf(pts); + return this.addCommand({ c: 'spline', a: arguments[0] }); +} +g2.prototype.spline.prototype = g2.mix(g2.prototype.ply.prototype, { + g2: function () { + let { pts, closed, x, y, w, ls, lw, fs, sh } = this, itr = this._itr, gbez; + if (itr) { + let b = [], i, n = itr.len, + p1, p2, p3, p4, d1, d2, d3, + d1d2, d2d3, scl2, scl3, + den2, den3, istrf = x || y || w; + + gbez = g2(); + if (istrf) gbez.beg({ x, y, w }); + gbez.p().m(itr(0)); + for (let i = 0; i < (closed ? n : n - 1); i++) { + if (i === 0) { + p1 = closed ? itr(n - 1) : { x: 2 * itr(0).x - itr(1).x, y: 2 * itr(0).y - itr(1).y }; + p2 = itr(0); + p3 = itr(1); + p4 = n === 2 ? (closed ? itr(0) : { x: 2 * itr(1).x - itr(0).x, y: 2 * itr(1).y - itr(0).y }) : itr(2); + d1 = Math.max(Math.hypot(p2.x - p1.x, p2.y - p1.y), Number.EPSILON); // don't allow .. + d2 = Math.max(Math.hypot(p3.x - p2.x, p3.y - p2.y), Number.EPSILON); // zero point distances .. + } else { + p1 = p2; + p2 = p3; + p3 = p4; + p4 = (i === n - 2) ? (closed ? itr(0) : { x: 2 * itr(n - 1).x - itr(n - 2).x, y: 2 * itr(n - 1).y - itr(n - 2).y }) + : (i === n - 1) ? itr(1) + : itr(i + 2); + d1 = d2; + d2 = d3; + } + d3 = Math.max(Math.hypot(p4.x - p3.x, p4.y - p3.y), Number.EPSILON); + d1d2 = Math.sqrt(d1 * d2), d2d3 = Math.sqrt(d2 * d3), + scl2 = 2 * d1 + 3 * d1d2 + d2, + scl3 = 2 * d3 + 3 * d2d3 + d2, + den2 = 3 * (d1 + d1d2), + den3 = 3 * (d3 + d2d3); + gbez.c({ + x: p3.x, y: p3.y, + x1: (-d2 * p1.x + scl2 * p2.x + d1 * p3.x) / den2, + y1: (-d2 * p1.y + scl2 * p2.y + d1 * p3.y) / den2, + x2: (-d2 * p4.x + scl3 * p3.x + d3 * p2.x) / den3, + y2: (-d2 * p4.y + scl3 * p3.y + d3 * p2.y) / den3 + }); + } + gbez.c(closed ? { x: itr(0).x, y: itr(0).y } : { x: itr(n - 1).x, y: itr(n - 1).y }) + if (closed) gbez.z(); + gbez.drw({ ls, lw, fs, sh }); + if (istrf) gbez.end(); + } + return gbez; + } +}); + +"use strict" + +/** + * g2.mec (c) 2013-18 Stefan Goessner + * @author Stefan Goessner + * @license MIT License + * @requires g2.core.js + * @requires g2.ext.js + * @typedef {g2} + * @description Mechanical extensions. (Requires cartesian coordinates) + * @returns {g2} + */ + +var g2 = g2 || { prototype:{} }; // for jsdoc only ... + +/** + * Draw slider. + * @method + * @returns {object} g2 + * @param {object} - slider arguments object. + * @property {number} x - start x coordinate. + * @property {number} y - start y coordinate. + * @property {number} [b=32] - slider breadth. + * @property {number} [h=16] - slider height. + * @property {number} [w=0] - rotation. + * @example + * g2().slider({x:150,y:75,w:Math.PI/4,b:64,h:32}) + */ +g2.prototype.slider = function () { return this.addCommand({c:'slider',a:arguments[0]}); } +g2.prototype.slider.prototype = g2.mix(g2.prototype.rec.prototype,{ + g2() { + const args = Object.assign({b:32,h:16,fs:'@linkfill'}, this); + return g2() + .beg({x:args.x,y:args.y,w:args.w,fs:args.fs}) + .rec({x:-args.b/2,y:-args.h/2,b:args.b,h:args.h}) + .end(); + } +}) + +/** + * Draw linear spring + * @method + * @returns {object} g2 + * @param {object} - linear spring arguments object. + * @property {number} x1 - start x coordinate. + * @property {number} y1 - start y coordinate. + * @property {number} x2 - end x coordinate. + * @property {number} y2 - end y coordinate. + * @property {number} [h=16] Spring height. + * @example + * g2().spring({x1:50,y1:100,x2:200,y2:75}) + */ +g2.prototype.spring = function () { return this.addCommand({c:'spring',a:arguments[0]}); } +g2.prototype.spring.prototype = g2.mix(g2.prototype.lin.prototype,{ + g2() { + const args = Object.assign({h:16}, this); + const len = Math.hypot(args.x2-args.x1, args.y2-args.y1); + const xm = (args.x2+args.x1)/2; + const ym = (args.y2+args.y1)/2; + const h = args.h; + const ux = (args.x2-args.x1)/len; + const uy = (args.y2-args.y1)/len; + return g2() + .p() + .m({x:args.x1,y:args.y1}) + .l({x:xm-ux*h/2,y:ym-uy*h/2}) + .l({x:xm+(-ux/6+uy/2)*h,y:ym+(-uy/6-ux/2)*h}) + .l({x:xm+( ux/6-uy/2)*h,y:ym+( uy/6+ux/2)*h}) + .l({x:xm+ux*h/2,y:ym+uy*h/2}) + .l({x:args.x2,y:args.y2}) + .stroke(Object.assign({}, {ls:'@nodcolor'},this,{fs:'transparent',lc:'round',lj:'round'})); + } +}) + +/** + * Draw line with centered square damper symbol. + * @method + * @returns {object} g2 + * @param {object} - damper arguments object. + * @property {number} x1 - start x coordinate. + * @property {number} y1 - start y coordinate. + * @property {number} x2 - end x coordinate. + * @property {number} y2 - end y coordinate. + * @property {number} [h=16] - damper height. + * * g2().damper({x1:60,y1:120,x2:200,y2:75}) + */ +g2.prototype.damper = function () { return this.addCommand({c:'damper',a:arguments[0]}); } +g2.prototype.damper.prototype = g2.mix(g2.prototype.lin.prototype,{ + g2() { + const args = Object.assign({h:16}, this); + const len = Math.hypot(args.x2-args.x1, args.y2-args.y1); + const xm = (args.x2+args.x1)/2; + const ym = (args.y2+args.y1)/2; + const h = args.h; + const ux = (args.x2-args.x1)/len; + const uy = (args.y2-args.y1)/len; + return g2() + .p() + .m({x:args.x1,y:args.y1}) + .l({x:xm-ux*h/2,y:ym-uy*h/2}) + .m({x:xm+( ux-uy)*h/2,y:ym+( uy+ux)*h/2}) + .l({x:xm+(-ux-uy)*h/2,y:ym+(-uy+ux)*h/2}) + .l({x:xm+(-ux+uy)*h/2,y:ym+(-uy-ux)*h/2}) + .l({x:xm+( ux+uy)*h/2,y:ym+( uy-ux)*h/2}) + .m({x:xm,y:ym}) + .l({x:args.x2,y:args.y2}) + .stroke(Object.assign({}, {ls:'@nodcolor'},this,{fs:'transparent',lc:'round',lj:'round'})); + } +}) + +/** + * Draw polygonial link. + * @method + * @returns {object} g2 + * @param {object} - link arguments object. + * @property {object[] | number[][] | number[]} pts - array of points. + * @property {bool} [closed = false] - closed link. + * @property {number} x - start x coordinate. + * @property {number} y - start y coordinate. + * @property {number} [w=0] - angle. + * @example + * let A = {x:50,y:25},B = {x:150,y:25}, + * C = {x:50,y:75}, D = {x:100,y:75}, + * E = {x:50,y:125}; + * g2().view({cartesian:true}) + * .link({pts:[A,B,E,A,D,C]}) + */ +g2.prototype.link = function () { return this.addCommand({c:'link',a:arguments[0]}); } +g2.prototype.link.prototype = g2.mix(g2.prototype.ply.prototype,{ + g2() { + const args = Object.assign({ls:'@linkcolor',fs:'transparent'}, this); + return g2().ply(Object.assign({}, this, {closed:true,ls:args.ls,fs:args.fs,lw:7,lc:'round',lj:'round'})); + } +}) + +/** + * Draw alternate glossy polygonial link. + * @method + * @returns {object} g2 + * @param {object} - link2 arguments object. + * @property {object[] | number[][] | number[]} pts - array of points. + * @property {bool} [closed = false] - closed link. + * @property {number} x - start x coordinate. + * @property {number} y - start y coordinate. + * @property {number} [w=0] - angle. + * @example + * let A = {x:50,y:25},B = {x:150,y:25}, + * C = {x:50,y:75}, D = {x:100,y:75}, + * E = {x:50,y:125}; + * g2().view({cartesian:true}) + * .link({pts:[A,B,E,A,D,C]}) + */ +g2.prototype.link2 = function () { return this.addCommand({c:'link2',a:arguments[0]}); } +g2.prototype.link2.prototype = g2.mix(g2.prototype.ply.prototype,{ + g2() { + return g2() + .ply(Object.assign({closed:true,ls:'@nodcolor',fs:'transparent',lw:7,lc:'round',lj:'round'},this)) + .ply(Object.assign({closed:true,ls:'@nodfill2',fs:'transparent',lw:4.5,lc:'round',lj:'round'},this)) + .ply(Object.assign({closed:true,ls:'@nodfill',fs:'transparent',lw:2,lc:'round',lj:'round'},this)); + } +}) + +/** + * Draw polygonial beam. + * @method + * @returns {object} g2 + * @param {object} - beam arguments object. + * @property {object[] | number[][] | number[]} pts - array of points. + * @property {number} x - start x coordinate. + * @property {number} y - start y coordinate. + * @property {number} [w=0] - angle. + * @example + * g2().view({cartesian}) + * .beam({pts:[[200,125][50,125][50,50][200,50]]}) + */ +g2.prototype.beam = function () { return this.addCommand({c:'beam',a:arguments[0]}); } +g2.prototype.beam.prototype = g2.mix(g2.prototype.ply.prototype,{ + g2() { + return g2().ply(Object.assign({closed:false,ls:'@linkcolor',fs:'transparent',lw:7,lc:'round',lj:'round'},this)); + } +}) + +/** + * Draw alternate glossy polygonial beam. + * @method + * @returns {object} g2 + * @param {object} - beam2 arguments object. + * @property {object[] | number[][] | number[]} pts - array of points. + * @property {number} x - start x coordinate. + * @property {number} y - start y coordinate. + * @property {number} [w=0] - angle. + * @example + * g2().view({cartesian}) + * .beam2({pts:[[200,125][50,125][50,50][200,50]]}) + */ +g2.prototype.beam2 = function () { return this.addCommand({c:'beam2',a:arguments[0]}); } +g2.prototype.beam2.prototype = g2.mix(g2.prototype.ply.prototype,{ + g2() { + return g2() + .ply(Object.assign({closed:false,ls:'@nodcolor',fs:'transparent',lw:7,lc:'round',lj:'round'},this)) + .ply(Object.assign({closed:false,ls:'@nodfill2',fs:'transparent',lw:4.5,lc:'round',lj:'round'},this)) + .ply(Object.assign({closed:false,ls:'@nodfill',fs:'transparent',lw:2,lc:'round',lj:'round'},this)); + } +}) + +/** + * Draw bar. + * @method + * @returns {object} g2 + * @param {object} - bar arguments object. + * @property {number} x1 - start x coordinate. + * @property {number} y1 - start y coordinate. + * @property {number} x2 - end x coordinate. + * @property {number} y2 - end y coordinate. + * @example + * g2().bar({x1:50,y1:20,x2:250,y2:120}) + */ +g2.prototype.bar = function () { return this.addCommand({c:'bar',a:arguments[0]}); } +g2.prototype.bar.prototype = g2.mix(g2.prototype.lin.prototype,{ + g2() { + return g2().lin(Object.assign({ls:'@linkcolor',lw:6,lc:'round'},this)); + } +}) + +/** + * Draw alternate glossy bar. + * @method + * @returns {object} g2 + * @param {object} - bar2 arguments object. + * @property {number} x1 - start x coordinate. + * @property {number} y1 - start y coordinate. + * @property {number} x2 - end x coordinate. + * @property {number} y2 - end y coordinate. + * @example + * g2().bar2({x1:50,y1:20,x2:250,y2:120}) + */ +g2.prototype.bar2 = function () { return this.addCommand({c:'bar2',a:arguments[0]}); } +g2.prototype.bar2.prototype = g2.mix(g2.prototype.lin.prototype,{ + g2() { + const args = Object.assign({}, this); + return g2() + .lin({x1:args.x1,y1:args.y1,x2:args.x2,y2:args.y2,ls:'@nodcolor',lw:7,lc:'round'}) + .lin({x1:args.x1,y1:args.y1,x2:args.x2,y2:args.y2,ls:'@nodfill2',lw:4.5,lc:'round'}) + .lin({x1:args.x1,y1:args.y1,x2:args.x2,y2:args.y2,ls:'@nodfill',lw:2,lc:'round'}); + } +}) + +/** + * Draw pulley. + * @method + * @returns {object} g2 + * @param {object} - pulley arguments object. + * @property {number} x - x-value center. + * @property {number} y - y-value center. + * @property {number} r - radius. + * @property {number} w - angle. + * @example + * g2().pulley({x:100,y:75,r:50}) + */ +g2.prototype.pulley = function () { return this.addCommand({c:'pulley',a:arguments[0]}); } +g2.prototype.pulley.prototype = g2.mix(g2.prototype.cir.prototype,{ + g2() { + const args = Object.assign({}, this); + return g2() + .beg({x:args.x,y:args.y,w:args.w}) + .cir({x:0,y:0,r:args.r,ls:'@nodcolor',fs:'#e6e6e6',lw:1}) + .cir({x:0,y:0,r:args.r-5,ls:'@nodcolor',fs:'#e6e6e6',lw:1}) + .cir({x:0,y:0,r:args.r-6,ls:'#8e8e8e',fs:'transparent',lw:2}) + .cir({x:0,y:0,r:args.r-8,ls:'#aeaeae',fs:'transparent',lw:2}) + .cir({x:0,y:0,r:args.r-10,ls:'#cecece',fs:'transparent',lw:2}) + .end(); + } +}) + +/** + * Draw alternate pulley. + * @method + * @returns {object} g2 + * @param {object} - pulley2 arguments object. + * @property {number} x - x-value center. + * @property {number} y - y-value center. + * @property {number} r - radius. + * @property {number} w - angle. + * @example + * g2().pulley2({x:50,y:30,r:25}) + */ +g2.prototype.pulley2 = function () { return this.addCommand({c:'pulley2',a:arguments[0]}); } +g2.prototype.pulley2.prototype = g2.mix(g2.prototype.cir.prototype,{ + g2() { + const args = Object.assign({}, this); + return g2() + .beg({x:args.x,y:args.y,w:args.w}) + .bar2({x1:0,y1:-args.r+4,x2:0,y2:args.r-4}) + .bar2({x1:-args.r+4,y1:0,x2:args.r-4,y2:0}) + .cir({x:0,y:0,r:args.r-2.5,ls:'#e6e6e6',fs:'transparent',lw:5}) + .cir({x:0,y:0,r:args.r,ls:'@nodcolor',fs:'transparent',lw:1}) + .cir({x:0,y:0,r:args.r-5,ls:'@nodcolor',fs:'transparent',lw:1}) + .end(); + } +}) + +/** + * Draw rope. Amount of pulley radii must be greater than 10 units. They are forced to zero otherwise. + * @method + * @returns {object} g2 + * @param {object} - rope arguments object. + * @property {object | number} p1 - starting point or Coordinate. + * @property {object | number} p2 - end point or Coordinate. + * @property {number} r - radius of parent element. + * @example + * let A = {x:50,y:30}, B = {x:200,y:75}; + * g2().view({cartesian:true}) + * .pulley({...A,r:20}) + * .pulley2({...B,r:40}) + * .rope({p1:A,r1:20,p2:B,r2:40}) + */ +g2.prototype.rope = function () { return this.addCommand({c:'rope',a:arguments[0]}); } +g2.prototype.rope.prototype = g2.mix(g2.prototype.lin.prototype,{ + g2() { + const args = Object.assign({w:0}, this); + let x1 = 'p1' in args ? args.p1.x + : 'x1' in args ? args.x1 + : 'x' in args ? args.x + : 0; + let y1 = 'p1' in args ? args.p1.y + : 'y1' in args ? args.y1 + : 'y' in args ? args.y + : 0; + let x2 = 'p2' in args ? args.p2.x + : 'x2' in args ? args.x2 + : 'dx' in args ? (x1 + args.dx) + : 'r' in args ? x1 + args.r*Math.cos(args.w) + : x1+10; + let y2 = 'p2' in args ? args.p2.y + : 'y2' in args ? args.y2 + : 'dy' in args ? (y1 + args.dy) + : 'r' in args ? y1 + args.r*Math.sin(args.w) + : y1; + let Rmin = 10; + let R1 = args.r1 > Rmin ? args.r1 - 2.5 + : args.r1 <-Rmin ? args.r1 + 2.5 + : 0; + let R2 = args.r2 > Rmin ? args.r2 - 2.5 + : args.r2 < Rmin ? args.r2 + 2.5 + : 0; + let dx = x2-x1, dy = y2-y1, dd = dx**2 + dy**2; + let R12 = R1 + R2, l = Math.sqrt(dd - R12**2); + let cpsi = (R12*dx + l*dy)/dd; + let spsi = (R12*dy - l*dx)/dd; + x1 = x1 + cpsi*R1, + y1 = y1 + spsi*R1, + x2 = x2 - cpsi*R2, + y2 = y2 - spsi*R2; + return g2().lin({x1:x1,x2:x2,y1:y1,y2:y2,ls:'#888',lw:4}); + } +}) + + +/** + * Polygon ground. + * @method + * @returns {object} g2 + * @param {object} - ground arguments object. + * @property {object[] | number[][] | number[]} pts - array of points. + * @property {bool} [closed=false] - closed polygon. + * @property {number} [h=4] - ground shade line width. + * @property {string} [pos=right] - ground shade position ['left','right']. + * @example + * g2().ground({pts:[25,25,25,75,75,75,75,25,125,25],pos:'left'}) + */ +g2.prototype.ground = function () { return this.addCommand({c:'ground',a:arguments[0]}); } +g2.prototype.ground.prototype = g2.mix(g2.prototype.ply.prototype,{ + g2() { + const args = Object.assign({h:4}, this); // , {closed: this.closed || false}); + const itr = g2.pntItrOf(args.pts); + let pn, en, lam, i; + let pp = itr(i=0); + let p0 = pp; + let h = args.h; + let p = itr(++i); + let dx = p.x - pp.x; + let dy = p.y - pp.y; + let len = Math.hypot(dx,dy) || 1; + let ep = {x:dx/len,y:dy/len}; + let e0 = ep; + let eq = [p0]; + let sign = args.pos === 'left' ? 1 : -1; + for (pn = itr(++i); i < itr.len; pn = itr(++i)) { + dx = pn.x - p.x; dy = pn.y - p.y; len = Math.hypot(dx,dy) || 1; + len = Math.hypot(dx,dy) || 1; + en = {x:dx/len,y:dy/len}; + lam = (1 - en.x*ep.x - en.y*ep.y) / (ep.y*en.x - ep.x*en.y); + eq.push({x:p.x+sign*(h+1)*(lam*ep.x - ep.y), y:p.y+sign*(h+1)*(lam*ep.y + ep.x)}); + ep = en; + pp = p; + p = pn; + } + if (args.closed) { + dx = p0.x-p.x; dy = p0.y-p.y; len = Math.hypot(dx,dy) || 1; + en = {x:dx/len,y:dy/len}; + lam = (1 - en.x*ep.x - en.y*ep.y) / (ep.y*en.x - ep.x*en.y); + eq.push({x:p.x+sign*(h+1)*(lam*ep.x - ep.y), y:p.y+sign*(h+1)*(lam*ep.y + ep.x)}); + lam = (1 - e0.x*en.x - e0.y*en.y) / (en.y*e0.x - en.x*e0.y); + eq[0] = {x:p0.x+sign*(h+1)*(-lam*e0.x - e0.y), y:p0.y+sign*(h+1)*(-lam*e0.y + e0.x)}; + } else { + eq[0] = {x:p0.x-sign*(h+1)*e0.y, y:p0.y+sign*(h+1)*e0.x}; + eq.push({x:p.x -sign*(h+1)*ep.y, y:p.y +sign*(h+1)*ep.x}); + } + return g2() + .beg({x:-0.5,y:-0.5,ls:'@linkcolor',lw:2,fs:'transparent',lc:'butt',lj:'miter'}) + .ply(Object.assign({}, args,{pts:eq,ls:'@nodfill2',lw:2*h})) + .ply(Object.assign({}, args)) + .end(); + } +}); + +/** + * Polygonial line load. The first and last point define the base line onto which + * the load is acting orthogonal. + * @method + * @returns {object} g2 + * @param {object} - load arguments object. + * @property {object[] | number[][] | number[]} pts - array of points. + * @property {number} w - angle of vectors. + * @property {number} spacing - spacing of the vectors drawn as a positive real number, interprete as
+ * * spacing < 1: spacing = 1/m with a partition of m.
+ * * spacing > 1: length of spacing. + */ +g2.prototype.load = function () { return this.addCommand({c:'load',a:arguments[0]}); } +g2.prototype.load.prototype = g2.mix(g2.prototype.ply.prototype,{ + g2() { + const args = Object.assign({ pointAt: this.pointAt, spacing: 20, w: -Math.PI/2 }, this); + const pitr = g2.pntItrOf(args.pts), startLoc = [], arr = []; + let arrLen = 0; + for (let itr = 0; itr < pitr.len ; itr++) { + arr.push(pitr(itr)); + } + if (arr[arr.length-1] !== arr[0]) { + arr.push(arr[0]); + } + for (let itr = 1; itr < arr.length; itr++) { + arrLen += Math.hypot(arr[itr].y-arr[itr-1].y,arr[itr].x-arr[itr-1].x); + } + for(let itr=0;itr*args.spacing < arrLen; itr++) { + startLoc.push((itr*args.spacing)/arrLen); + } + args.pts = arr; // for args.pointsAt(...)... + /*-----------------------------------stolen from g2.lib-----------------------------------*/ + function isPntInPly({x,y}) { + let match = 0; + for (let n=arr.length,i=0,pi=arr[i],pj=arr[n-1]; i= pi.y || y >= pj.y) + && (y <= pi.y || y <= pj.y) + && (x <= pi.x || x <= pj.x) + && pi.y !== pj.y + && (pi.x === pj.x || x <= pj.x + (y-pj.y)*(pi.x-pj.x)/(pi.y-pj.y))) { + match++; + } + } + return match%2 != 0; + }; + /*----------------------------------------------------------------------------------------*/ + return g2() + .ply({pts:args.pts,closed:true,ls:'transparent',fs:'@linkfill'}) + .ins(g => { + for (const pts of startLoc) { + let dist = (10*args.lw||10); // minimum distance a vector has to be + const {x,y} = args.pointAt(pts); + const t = { + x:x+Math.cos(args.w)*dist, + y:y+Math.sin(args.w)*dist + }; + if (isPntInPly(t,{pts:arr})) { + while(isPntInPly(t,{pts:arr})) { + dist++; + t.x = x+Math.cos(args.w)*dist, + t.y = y+Math.sin(args.w)*dist + }; + g.vec({ + x1:x, y1:y, + x2:t.x, y2:t.y, + ls: args.ls || "darkred", + lw: args.lw || 1 + }); + } + } + }); + } +}); + +"use strict" + +/** + * g2.chart (c) 2015-18 Stefan Goessner + * @author Stefan Goessner + * @license MIT License + * @requires g2.core.js + * @requires g2.ext.js + * @typedef g2 + * @returns {object} chart + * @param {object} args - Chart arguments object or + * @property {float} x - x-position of lower left corner of chart rectangle. + * @property {float} y - y-position of lower left corner of chart rectangle. + * @property {float} [b=150] - width of chart rectangle. + * @property {float} [h=100] - height of chart rectangle. + * @property {string} [ls] - border color. + * @property {string} [fs] - fill color. + * @property {(string|object)} [title] - chart title. + * @property {string} [title.text] - chart title text string. + * @property {float} [title.offset=0] - chart title vertical offset. + * @property {object} [title.style] - chart title style. + * @property {string} [title.style.font=14px serif] - chart title font. + * @property {string} [title.style.thal=center] - chart title horizontal align. + * @property {string} [title.style.tval=bottom] - chart title vertical align. + * @property {array} [funcs=[]] - array of dataset `data` and/or function `fn` objects. + * @property {object} [funcs[item]] - dataset or function object. + * @property {array} [funcs[item].data] - data points as flat array `[x,y,..]`, array of point arrays `[[x,y],..]` or array of point objects `[{x,y},..]`. + * @property {function} [funcs[item].fn] - function `y = f(x)` recieving x-value returning y-value. + * @property {float} [funcs[item].dx] - x increment to apply to function `fn`. Ignored with data points. + * @property {boolean} [funcs[item].fill] - fill region between function graph and x-origin line. + * @property {boolean} [funcs[item].dots] - place circular dots at data points (Avoid with `fn`s). + * @property {boolean|object} [xaxis=false] - x-axis. + * @property {boolean|object} [xaxis.grid=false] - x-axis grid lines. + * @property {string} [xaxis.grid.ls] - x-axis grid line style (color). + * @property {string} [xaxis.grid.lw] - x-axis grid line width. + * @property {string} [xaxis.grid.ld] - x-axis grid line dash style. + * @property {boolean} [xaxis.line=true] - display x-axis base line. + * @property {boolean} [xaxis.origin=false] - display x-axis origin line. + * @property {boolean|object} [yaxis=false] - y-axis. + * @property {boolean|object} [yaxis.grid=false] - y-axis grid lines. + * @property {string} [yaxis.grid.ls] - y-axis grid line style color. + * @property {string} [yaxis.grid.lw] - y-axis grid line width. + * @property {string} [yaxis.grid.ld] - y-axis grid line dash style. + * @property {boolean} [yaxis.line=true] - display y-axis base line. + * @property {boolean} [yaxis.origin=false] - display y-axis origin line. + * @property {float} [xmin] - minimal x-axis value. If not given it is calculated from chart data values. + * @property {float} [xmax] - maximal x-axis value. If not given it is calculated from chart data values. + * @property {float} [ymin] - minimal y-axis value. If not given it is calculated from chart data values. + * @property {float} [ymax] - maximal y-axis value. If not given it is calculated from chart data values. + */ +g2.prototype.chart = function chart({x,y,b,h,style,title,funcs,xaxis,xmin,xmax,yaxis,ymin,ymax}) { + return this.addCommand({c:'chart',a:arguments[0]}); +} +g2.prototype.chart.prototype = { + g2() { + const g = g2(), + funcs = this.get('funcs'), + title = this.title && this.get('title'); + + if (!this.b) this.b = this.defaults.b; + if (!this.h) this.h = this.defaults.h; + // initialize function graphs (only once ...) + if (funcs && funcs.length) { // init all funcs ... + const tmp = [ + this.xmin===undefined, + this.xmax===undefined, + this.ymin===undefined, + this.ymax===undefined + ]; + funcs.forEach(f => this.initFunc(f,...tmp)); + } + // if (this.xaxis) + this.xAxis = this.autoAxis(this.get('xmin'),this.get('xmax'),0,this.b); + // if (this.yaxis) + this.yAxis = this.autoAxis(this.get('ymin'),this.get('ymax'),0,this.h); + + // draw background & border ... + g.rec({ + x:this.x,y:this.y,b:this.b,h:this.h, + fs:this.get("fs"),ls:this.get("ls") + }); + + // draw title & axes ... + g.beg(Object.assign({x:this.x,y:this.y,lw:1}, this.defaults.style,this.style)); + + if (title) + g.txt(Object.assign({ + str: this.title && this.title.text || this.title, + x: this.get('b')/2, + y: this.get('h') + this.get("title","offset"), + w: 0 + }, this.defaults.title.style, + (this.title && this.title.style || {}) + )); + if (this.xaxis) this.drawXAxis(g); + if (this.yaxis) this.drawYAxis(g); + + g.end(); + + // draw funcs ... + if (funcs) + funcs.forEach((fnc,i) => { this.drawFunc(g,fnc,this.defaults.colors[i%this.defaults.colors.length]); }); + + return g; + }, + /** + * Initialize chart function. + * @private + */ + initFunc(fn,setXmin,setXmax,setYmin,setYmax) { + // Install func iterator. + let itr; + if (fn.data && fn.data.length) { // data must have a polyline conform array structure + itr = fn.itr = g2.pntItrOf(fn.data); // get iterator ... + } + else if (fn.fn && fn.dx) { + const xmin = +this.xmin || this.defaults.xmin; + const xmax = +this.xmax || this.defaults.xmax; + itr = fn.itr = (i) => { let x = xmin + i*fn.dx; return { x:x, y:fn.fn(x) }; } + itr.len = (xmax - xmin)/fn.dx + 1; + } + // Get func's bounding box + if (itr && (setXmin || setXmax || setYmin || setYmax)) { + const xarr = []; + const yarr = []; + for (let i=0; i < itr.len; ++i) { + xarr.push(itr(i).x); + yarr.push(itr(i).y); + } + if (setXmin) { + const xmin = Math.min(...xarr); + if (!this.xmin || xmin < this.xmin) this.xmin = xmin; + } + if (setXmax) { + const xmax = Math.max(...xarr); + if (!this.xmax || xmax > this.xmax) this.xmax = xmax; + } + if (setYmin) { + const ymin = Math.min(...yarr); + if (!this.ymin || ymin < this.ymin) this.ymin = ymin; + } + if (setYmax) { + const ymax = Math.max(...yarr); + if (!this.ymax || ymax > this.ymax) this.ymax = ymax; + } + + if (fn.color && typeof fn.color === "number") // color index [0..n] + fn.color = this.defaults.colors[fn.color % this.defaults.colors.length]; + } + }, + autoAxis(zmin,zmax,tmin,tmax) { + let base = 2, exp = 1, eps = Math.sqrt(Number.EPSILON), + Dz = zmax - zmin || 1, // value range + Dt = tmax - tmin || 1, // area range + scl = Dz > eps ? Dt/Dz : 1, // scale [usr]->[pix] + dz = base*Math.pow(10,exp), // tick size [usr] + dt = Math.floor(scl*dz), // tick size [pix] + N, // # segments + dt01, // reminder segment + i0, j0, jth, t0, res; + + while (dt < 14 || dt > 35) { + if (dt < 14) { + if (base == 1) base = 2; + else if (base == 2) base = 5; + else if (base == 5) { base = 1; exp++; } + } + else { // dtick > 35 + if (base == 1) { base = 5; exp--; } + else if (base == 2) base = 1; + else if (base == 5) base = 2; + } + dz = base*Math.pow(10,exp); + dt = scl*dz; + } + i0 = (scl*Math.abs(zmin) + eps/2)%dt < eps + ? Math.floor(zmin/dz) + : Math.floor(zmin/dz) + 1; + let z0 = i0*dz; + t0 = Math.round(scl*(z0 - zmin)); + // console.log("Dt="+Dt+",N="+(Dt - t0)/ dt) + // console.log("DT="+Dt+",N="+(Dt - t0)/ dt) + N = Math.floor((Dt - t0)/ dt) + 1; + j0 = base % 2 && i0 % 2 ? i0 + 1 : i0; + jth = exp === 0 && N < 11 ? 1 : base===2 && N > 9 ? 5 : 2; + + return { + zmin, // min usr value + zmax, // max usr value + base, // one of [1,2,5] + exp, // 10^exp + scl, // scale [usr]->[pix] + dt, // tick range [pix] + dz, // tick range [usr] + N, // # of ticks + t0, // start tick position [pix] + z0, // start tick position [usr] + i0, // first tick index relative to tick origin (can be negative) + j0, // first labeled tick + jth, // # of ticks between two major ticks + itr(i) { // tick iterator + return { t: this.t0 + i*this.dt, + z: parseFloat((this.z0 + i*this.dz).toFixed(Math.abs(this.exp))), + maj: (this.j0 - this.i0 + i)%this.jth === 0 }; + } + } + }, + /** + * Draw x-axis. + * @private + */ + drawXAxis(g) { + let tick, + showgrid = this.xaxis && this.xaxis.grid, + gridstyle = showgrid && Object.assign({}, this.defaults.xaxis.grid, this.xaxis.grid), + showaxis = this.xaxis || this.xAxis, + axisstyle = showaxis && Object.assign({}, this.defaults.xaxis.style, this.defaults.xaxis.labels.style, (this.xaxis && this.xaxis.style || {}) ), + showline = showaxis && this.get("xaxis","line"), + showlabels = this.xAxis && showaxis && this.get("xaxis","labels"), + showticks = this.xAxis && showaxis && this.get("xaxis","ticks"), + ticklen = showticks ? this.get("xaxis","ticks","len") : 0, + showorigin = showaxis && this.get("xaxis","origin"), + title = this.xaxis && (this.get("xaxis","title","text") || this.xaxis.title) || ''; + // console.log(this.xAxis) + // draw tick/grid lines + g.beg(axisstyle); + for (let i=0; i= 0) + g.lin({x1:-this.xAxis.zmin*this.xAxis.scl,y1:0,x2:-this.xAxis.zmin*this.xAxis.scl,y2:this.h}); // origin line emphasized ... + if (title) + g.txt(Object.assign({ + str:title.text || title, + x:this.b/2, + y:-( this.get("xaxis","title","offset") + +(showticks && this.get("xaxis","ticks","len") || 0) + +(showlabels && this.get("xaxis","labels","offset") || 0) + +(showlabels && parseFloat(this.get("xaxis","labels","style","font")) || 0)), + w:0 + }, (this.get('xaxis','title','style')))); + g.end(); + }, + /** + * Draw y-axis. + * @private + */ + drawYAxis(g) { + let tick, + showgrid = this.yaxis && this.yaxis.grid, + gridstyle = showgrid && Object.assign({}, this.defaults.yaxis.grid,this.yaxis.grid), + showaxis = this.yaxis || this.yAxis, + axisstyle = showaxis && Object.assign({},this.defaults.yaxis.style, this.defaults.yaxis.labels.style, (this.yaxis && this.yaxis.style || {})), + showline = showaxis && this.get("yaxis","line"), + showlabels = this.yAxis && showaxis && this.get("yaxis","labels"), + showticks = this.yAxis && showaxis && this.get("yaxis","ticks"), + ticklen = showticks ? this.get("yaxis","ticks","len") : 0, + showorigin = showaxis && this.get("yaxis","origin"), + title = this.yaxis && (this.get("yaxis","title","text") || this.yaxis.title) || ''; + + // draw tick/grid lines + g.beg(axisstyle); + for (let i=0; i= 0) + g.lin({x1:0,y1:-this.yAxis.zmin*this.yAxis.scl,x2:this.b,y2:-this.yAxis.zmin*this.yAxis.scl}); // origin line emphasized ... + if (title) + g.txt(Object.assign({ + str: title.text || title, + x:-( this.get("yaxis","title","offset") + +(showticks && this.get("yaxis","ticks","len") || 0) + +(showlabels && this.get("yaxis","labels","offset") || 0) + +(showlabels && parseFloat(this.get("yaxis","labels","style","font")) || 0)), + y:this.h/2, + w:Math.PI/2 + }, (this.get('yaxis','title','style')))); + g.end(); + }, + /** + * Draw chart function. + * @private + */ + drawFunc(g,fn,defaultcolor) { + let itr = fn.itr; + + if (itr) { + let fill = fn.fill || fn.style && fn.style.fs && fn.style.fs !== "transparent", + color = fn.color = fn.color || fn.style && fn.style.ls || defaultcolor, + plydata = [], + args = Object.assign({ + pts:plydata, + closed:false, + ls:color, + fs:(fill?g2.color.rgbaStr(color,0.125):'transparent'), + lw:1 + }, fn.style); + + if (fill) // start from base line (y=0) + plydata.push(this.pntOf({x:itr(0).x,y:0})); + for (let i=0, n=itr.len; i + * @method + * @returns {object} g2 + */ + clr() { return this.addCommand({ c: 'clr' }); }, + + /** + * Set the view by placing origin coordinates and scaling factor in device units + * and make viewport cartesian. + * @method + * @returns {object} g2 + * @param {object} - view arguments object. + * @property {number} [scl=1] - absolute scaling factor. + * @property {number} [x=0] - x-origin in device units. + * @property {number} [y=0] - y-origin in device units. + * @property {boolean} [cartesian=false] - set cartesian flag. + */ + view({ scl, x, y, cartesian }) { return this.addCommand({ c: 'view', a: arguments[0] }); }, + + /** + * Draw grid. + * @method + * @returns {object} g2 + * @param {object} - grid arguments object. + * @property {string} [color=#ccc] - change color. + * @property {number} [size=20] - change space between lines. + */ + grid({ color, size } = {}) { return this.addCommand({ c: 'grid', a: arguments[0] }); }, + + /** + * Draw circle by center and radius. + * @method + * @returns {object} g2 + * @param {object} - circle arguments object. + * @property {number} x - x-value center. + * @property {number} y - y-value center. + * @property {number} r - radius. + * @property {number} w - angle. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {array} [sh=[0,0,0,'transparent']] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @example + * g2().cir({x:100,y:80,r:20}) // Draw circle. + */ + cir({ x, y, r, w }) { return this.addCommand({ c: 'cir', a: arguments[0] }); }, + + /** + * Draw ellipse by center and radius for x and y. + * @method + * @returns {object} g2 + * @param {object} - ellispe argument object. + * @property {number} x - x-value center. + * @property {number} y - y-value center. + * @property {number} rx - radius x-axys. + * @property {number} ry - radius y-axys. + * @property {number} w - start angle. + * @property {number} dw - angular range. + * @property {number} rot - rotation. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @example + * g2().ell({x:100,y:80,rx:20,ry:30,w:0,dw:2*Math.PI/4,rot:1}) // Draw circle. + */ + ell({ x, y, rx, ry, w, dw, rot }) { return this.addCommand({ c: 'ell', a: arguments[0] }); }, + + /** + * Draw arc by center point, radius, start angle and angular range. + * @method + * @returns {object} g2 + * @param {object} - arc arguments object. + * @property {number} x - x-value center. + * @property {number} y - y-value center. + * @property {number} r - radius. + * @property {number} [w=0] - start angle (in radian). + * @property {number} [dw=2*pi] - angular range in Radians. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {string} [lc=butt] - line cap [`butt`, `round`, `square`]. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @example + * g2().arc({x:300,y:400,r:390,w:-Math.PI/4,dw:-Math.PI/2}) + * .exe(ctx); + */ + arc({ x, y, r, w, dw }) { return this.addCommand({ c: 'arc', a: arguments[0] }); }, + + /** + * Draw rectangle by anchor point and dimensions. + * @method + * @returns {object} g2 + * @param {object} - rectangle arguments object. + * @property {number} x - x-value upper left corner. + * @property {number} y - y-value upper left corner. + * @property {number} b - width. + * @property {number} h - height. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {string} [lj='miter'] - line join [`round`, `bevel`, `miter`]. + * @property {number} [ml=10] - miter limit. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @example + * g2().rec({x:100,y:80,b:40,h:30}) // Draw rectangle. + */ + rec({ x, y, b, h }) { return this.addCommand({ c: 'rec', a: arguments[0] }); }, + + /** + * Draw line by start point and end point. + * @method + * @returns {object} g2 + * @param {object} - line arguments object. + * @property {number} x1 - start x coordinate. + * @property {number} y1 - start y coordinate. + * @property {number} x2 - end x coordinate. + * @property {number} y2 - end y coordinate. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {string} [lc=butt] - line cap [`butt`, `round`, `square`]. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @example + * g2().lin({x1:10,x2:10,y1:190,y2:10}) // Draw line. + */ + lin({ x1, y1, x2, y2 }) { return this.addCommand({ c: 'lin', a: arguments[0] }); }, + + /** + * Draw polygon by points. + * Using iterator function for getting points from array by index. + * It must return current point object {x,y} or object {done:true}. + * Default iterator expects sequence of x/y-coordinates as a flat array [x,y,...], + * array of [[x,y],...] arrays or array of [{x,y},...] objects. + * @method + * @returns {object} g2 + * @param {object} - polygon arguments object. + * @property {array} pts - array of points. + * @property {string} [format] - format string of points array structure. Useful for handing over initial empty points array. One of `['x,y','[x,y]','{x,y}']`. Has precedence over `pts` content. + * @property {boolean} [closed = false] + * @property {number} x - start x coordinate. + * @property {number} y - start y coordinate. + * @property {number} w - angle. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {string} [lc=butt] - line cap [`butt`, `round`, `square`]. + * @property {string} [lj='miter'] - line join [`round`, `bevel`, `miter`]. + * @property {number} [ml=10] - miter limit. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @example + * g2().ply({pts:[100,50,120,60,80,70]}), + * .ply({pts:[150,60],[170,70],[130,80]],closed:true}), + * .ply({pts:[{x:160,y:70},{x:180,y:80},{x:140,y:90}]}), + * .exe(ctx); + */ + ply({ pts, format, closed, x, y, w }) { + arguments[0]._itr = format && g2.pntIterator[format](pts) || g2.pntItrOf(pts); + return this.addCommand({ c: 'ply', a: arguments[0] }); + }, + + /** + * Draw text string at anchor point. + * @method + * @returns {object} g2 + * @param {object} - text arguments object. + * @property {string} str - text string. + * @property {number} [x=0] - x coordinate of text anchor position. + * @property {number} [y=0] - y coordinate of text anchor position. + * @property {number} [w=0] - w Rotation angle about anchor point with respect to positive x-axis. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @property {string} [thal='start'] + * - Text horizontal alignment [`'start'`,`'end'`,`'left'`,`'right'`,`'center'`] + * @property {string} [tval='alphabetic'] + * - Text vertival alignment [`'top'`,`'hanging'`,`'middle'`,`'alphabetic'`,`'ideographic'`,`'bottom'`] + * @property {string} [font='normal 14px serif'] - + * [Font]{@link https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/font} + * [styling]{@link https://html.spec.whatwg.org/multipage/canvas.html#dom-context-2d-font} + */ + txt({ str, x, y, w }) { return this.addCommand({ c: 'txt', a: arguments[0] }); }, + + /** + * Reference g2 graphics commands from another g2 object or a predefined g2.symbol. + * With this command you can reuse instances of grouped graphics commands + * while applying a similarity transformation and style properties on them. + * In fact you might want to build custom graphics libraries on top of that feature. + * @method + * @returns {object} g2 + * @param {object} - use arguments object. + * @see {@link https://github.com/goessner/g2/blob/master/docs/api/g2.ext.md#g2symbol--object predefined symbols in g2.ext} + * @property {object | string} grp - g2 source object or symbol name found in 'g2.symbol' namespace. + * @property {number} [x=0] - translation value x. + * @property {number} [y=0] - translation value y. + * @property {number} [w=0] - rotation angle (in radians). + * @property {number} [scl=1] - scale factor. + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {string} [lc=butt] - line cap [`butt`, `round`, `square`]. + * @property {string} [lj='miter'] - line join [`round`, `bevel`, `miter`]. + * @property {number} [ml=10] - miter limit. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @property {string} [thal='start'] + * - Text horizontal alignment [`'start'`,`'end'`,`'left'`,`'right'`,`'center'`] + * @property {string} [tval='alphabetic'] + * - Text vertival alignment [`'top'`,`'hanging'`,`'middle'`,`'alphabetic'`,`'ideographic'`,`'bottom'`] + * @property {string} [font='normal 14px serif'] - + * [Font]{@link https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/font} + * [styling]{@link https://html.spec.whatwg.org/multipage/canvas.html#dom-context-2d-font} + * @example + * g2.symbol.cross = g2().lin({x1:5,y1:5,x2:-5,y2:-5}).lin({x1:5,y1:-5,x2:-5,y2:5}); // Define symbol. + * g2().use({grp:"cross",x:100,y:100}) // Draw cross at position 100,100. + */ + use({ grp, x, y, w, scl }) { + if (grp && grp !== this) { // avoid self reference .. + if (typeof grp === "string") // must be a member name of the 'g2.symbol' namespace + arguments[0].grp = g2.symbol[(grp in g2.symbol) ? grp : 'unknown']; + this.addCommand({ c: 'use', a: arguments[0] }); + } + return this; + }, + + /** + * Draw image. + * This also applies to images of reused g2 objects. If an image can not be loaded, it will be replaced by a broken-image symbol. + * @method + * @returns {object} g2 + * @param {object} - image arguments object. + * @property {string} uri - image uri or data:url. + * @property {number} [x = 0] - x-coordinate of image (upper left). + * @property {number} [y = 0] - y-coordinate of image (upper left). + * @property {number} [b = image.width] - width. + * @property {number} [h = image.height] - height. + * @property {number} [sx = 0] - source x-offset. + * @property {number} [sy = 0] - source y-offset. + * @property {number} [sb = image.width] - source width. + * @property {number} [sh = image.height] - source height. + * @property {number} [xoff = 0] - x-offset. + * @property {number} [yoff = 0] - y-offset. + * @property {number} [w = 0] - rotation angle (about upper left, in radians). + * @property {number} [scl = 1] - image scaling. + */ + img({ uri, x, y, b, h, sx, sy, sb, sh, xoff, yoff, w, scl }) { return this.addCommand({ c: 'img', a: arguments[0] }); }, + + /** + * Begin subcommands. Current state is saved. + * Optionally apply transformation or style properties. + * @method + * @returns {object} g2 + * @param {object} - beg arguments object. + * @property {number} [x = 0] - translation value x. + * @property {number} [y = 0] - translation value y. + * @property {number} [w = 0] - rotation angle (in radians). + * @property {number} [scl = 1] - scale factor. + * @property {array} [matrix] - matrix instead of single transform arguments (SVG-structure [a,b,c,d,x,y]). + * @property {string} [fs=transparent] - fill color. + * @property {string} [ls=black] - stroke color. + * @property {string} [lw=1] - line width. + * @property {string} [lc=butt] - line cap [`butt`, `round`, `square`]. + * @property {string} [lj='miter'] - line join [`round`, `bevel`, `miter`]. + * @property {number} [ml=10] - miter limit. + * @property {array} [ld=[]] - line dash array. + * @property {array} [sh=[0,0,0,"transparent"]] + * shadow values [`x-offset`,`y-offset`,`blur`,`color`], + * @property {string} [thal='start'] + * - text horizontal alignment [`'start'`,`'end'`,`'left'`,`'right'`,`'center'`] + * @property {string} [tval='alphabetic'] + * - text vertival alignment [`'top'`,`'hanging'`,`'middle'`,`'alphabetic'`,`'ideographic'`,`'bottom'`] + * @property {string} [font='normal 14px serif'] - + * [Font]{@link https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/font} + * [styling]{@link https://html.spec.whatwg.org/multipage/canvas.html#dom-context-2d-font} + */ + beg({ x, y, w, scl, matrix } = {}) { return this.addCommand({ c: 'beg', a: arguments[0] }); }, + + /** + * End subcommands. Previous state is restored. + * @method + * @returns {object} g2 + * @param {object} - end arguments object. + */ + end() { // ignore 'end' commands without matching 'beg' + let myBeg = 1, + findMyBeg = (cmd) => { // care about nested beg...end blocks ... + if (cmd.c === 'beg') myBeg--; + else if (cmd.c === 'end') myBeg++; + return myBeg === 0; + } + return g2.cmdIdxBy(this.commands, findMyBeg) !== false ? this.addCommand({ c: 'end' }) : this; + }, + + /** + * Begin new path. + * @method + * @returns {object} g2 + */ + p() { return this.addCommand({ c: 'p' }); }, + + /** + * Close current path by straight line. + * @method + * @returns {object} g2 + */ + z() { return this.addCommand({ c: 'z' }); }, + + /** + * Move to point. + * @method + * @returns {object} g2 + * @param {object} - move arguments object. + * @property {number} x - move to x coordinate + * @property {number} y - move to y coordinate + */ + m({ x, y }) { return this.addCommand({ c: 'm', a: arguments[0] }); }, + + /** + * Create line segment to point. + * @method + * @returns {object} g2 + * @param {object} - line segment argument object. + * @property {number} x - x coordinate of target point. + * @property {number} y - y coordinate of target point. + * @example + * g2().p() // Begin path. + * .m({x:0,y:50}) // Move to point. + * .l({x:300,y:0}) // Line segment to point. + * .l({x:400,y:100}) // ... + * .stroke() // Stroke path. + */ + l({ x, y }) { return this.addCommand({ c: 'l', a: arguments[0] }); }, + + /** + * Create quadratic bezier curve segment to point. + * @method + * @returns {object} g2 + * @param {object} - quadratic curve arguments object. + * @property {number} x1 - x coordinate of control point. + * @property {number} y1 - y coordinate of control point. + * @property {number} x - x coordinate of target point. + * @property {number} y - y coordinate of target point. + * @example + * g2().p() // Begin path. + * .m({x:0,y:0}) // Move to point. + * .q({x1:200,y1:200,x:400,y:0}) // Quadratic bezier curve segment. + * .stroke() // Stroke path. + */ + q({ x1, y1, x, y }) { return this.addCommand({ c: 'q', a: arguments[0] }); }, + + /** + * Create cubic bezier curve to point. + * @method + * @returns {object} g2 + * @param {object} - cubic curve arguments object. + * @property {number} x1 - x coordinate of first control point. + * @property {number} y1 - y coordinate of first control point. + * @property {number} x2 - x coordinate of second control point. + * @property {number} y2 - y coordinate of second control point. + * @property {number} x - x coordinate of target point. + * @property {number} y - y coordinate of target point. + * @example + * g2().p() // Begin path. + * .m({x:0,y:100}) // Move to point. + * .c({x1:100,y1:200,x2:200,y2:0,x:400,y:100}) // Create cubic bezier curve. + * .stroke() // Stroke path. + * .exe(ctx); // Render to canvas context. + */ + c({ x1, y1, x2, y2, x, y }) { return this.addCommand({ c: 'c', a: arguments[0] }); }, + + /** + * Draw arc with angular range to target point. + * @method + * @returns {object} g2 + * @param {object} - arc arguments object. + * @property {number} dw - angular range in radians. + * @property {number} x - x coordinate of target point. + * @property {number} y - y coordinate of target point. + * @example + * g2().p() // Begin path. + * .m({x:50,y:50}) // Move to point. + * .a({dw:2,x:300,y:100}) // Create arc segment. + * .stroke() // Stroke path. + * .exe(ctx); // Render to canvas context. + */ + a({ dw, x, y }) { + let prvcmd = this.commands[this.commands.length - 1]; + g2.cpyProp(prvcmd.a, 'x', arguments[0], '_xp'); + g2.cpyProp(prvcmd.a, 'y', arguments[0], '_yp'); + return this.addCommand({ c: 'a', a: arguments[0] }); + }, + + /** + * Stroke the current path or path object. + * @method + * @returns {object} g2 + * @param {object} - stroke arguments object. + * @property {string} [d = undefined] - SVG path definition string. Current path is ignored then. + */ + stroke({ d } = {}) { return this.addCommand({ c: 'stroke', a: arguments[0] }); }, + + /** + * Fill the current path or path object. + * @method + * @returns {object} g2 + * @param {object} - fill arguments object. + * @property {string} [d = undefined] - SVG path definition string. Current path is ignored then. + */ + fill({ d } = {}) { return this.addCommand({ c: 'fill', a: arguments[0] }); }, + + /** + * Shortcut for stroke and fill the current path or path object. + * In case of shadow style, only the path interior creates shadow, not also the path contour. + * @method + * @returns {object} g2 + * @param {object} - drw arguments object. + * @property {string} [d = undefined] - SVG path definition string. Current path is ignored then. + */ + drw({ d, lsh } = {}) { return this.addCommand({ c: 'drw', a: arguments[0] }); }, + + /** + * Delete all commands beginning from `idx` to end of command queue. + * @method + * @returns {object} g2 + */ + del(idx) { this.commands.length = idx || 0; return this; }, + + /** + * Call function between commands of the command queue. + * @method + * @returns {object} g2 + * @param {function} - ins argument function. + * @example + * const node = { + * fill:'lime', + * g2() { return g2().cir({x:160,y:50,r:15,fs:this.fill,lw:4,sh:[8,8,8,"gray"]}) } + * }; + * let color = 'red'; + * g2().cir({x:40,y:50,r:15,fs:color,lw:4,sh:[8,8,8,"gray"]}) // draw red circle. + * .ins(()=>{color='green'}) // color is now green. + * .cir({x:80,y:50,r:15,fs:color,lw:4,sh:[8,8,8,"gray"]}) // draw green circle. + * .ins((g) => // draw orange circle + * g.cir({x:120, y:50, r:15, fs:'orange', lw:4,sh:[8,8,8,"gray"]})) + * .ins(node) // draw node. + * .exe(ctx) // render to canvas context. + */ + ins(arg) { + return typeof arg === 'function' ? (arg(this) || this) // no further processing by handler ... + : typeof arg === 'object' ? (this.commands.push({ a: arg }), this) // no explicit command name .. ! + : this; + }, + /** + * Execute g2 commands. It does so automatically and recursively with 'use'ed commands. + * @method + * @returns {object} g2 + * @param {object} ctx Context. + */ + exe(ctx) { + let handler = g2.handler(ctx); + if (handler && handler.init(this)) + handler.exe(this.commands); + return this; + }, + // helpers ... + addCommand({ c, a }) { + if (a && Object.getPrototypeOf(a) === Object.prototype) { // modify only pure argument objects 'a' .. ! + for (const key in a) { + if (!Object.getOwnPropertyDescriptor(a, key).get // if 'key' is no getter ... + && key[0] !== '_' // and no private property ... + && typeof a[key] === 'function') { // and a function ... make it a getter + Object.defineProperty(a, key, { get: a[key], enumerable: true, configurable: true, writabel: false }); + } + if (typeof a[key] === 'string' && a[key][0] === '@') { // referring values by neighbor id's + const refidIdx = a[key].indexOf('.'); + const refid = refidIdx > 0 ? a[key].substr(1, refidIdx - 1) : ''; + const refkey = refid ? a[key].substr(refidIdx + 1) : ''; + const refcmd = refid ? () => this.commands.find((cmd) => cmd.a && cmd.a.id === refid) : undefined; + + if (refcmd) + Object.defineProperty(a, key, { + get: function () { + const rc = refcmd(); + return rc && (refkey in rc.a) ? rc.a[refkey] : 0; + }, + enumerable: true, + configurable: true, + writabel: false + }); + } + } + if (g2.prototype[c].prototype) Object.setPrototypeOf(a, g2.prototype[c].prototype); + } + this.commands.push(arguments[0]); + return this; + } +}; + +// statics +g2.defaultStyle = { fs: 'transparent', ls: '#000', lw: 1, lc: "butt", lj: "miter", ld: [], ml: 10, sh: [0, 0], lsh: false, font: '14px serif', thal: 'start', tval: 'alphabetic' }; +g2.symbol = { + unknown: g2().cir({ r: 12, fs: 'orange' }).txt({ str: '?', thal: 'center', tval: 'middle', font: 'bold 20pt serif' }) +}; +g2.handler = function (ctx) { + let hdl; + for (let h of g2.handler.factory) + if ((hdl = h(ctx)) !== false) + return hdl; + return false; +} +g2.handler.factory = []; + +// predefined polyline/spline point iterators +g2.pntIterator = { + "x,y": function (pts) { + function pitr(i) { return { x: pts[2 * i], y: pts[2 * i + 1] }; }; + Object.defineProperty(pitr, 'len', { get: () => pts.length / 2, enumerable: true, configurable: true, writabel: false }); + return pitr; + }, + "[x,y]": function (pts) { + function pitr(i) { return pts[i] ? { x: pts[i][0], y: pts[i][1] } : undefined; }; + Object.defineProperty(pitr, 'len', { get: () => pts.length, enumerable: true, configurable: true, writabel: false }); + return pitr; + }, + "{x,y}": function (pts) { + function pitr(i) { return pts[i]; }; + Object.defineProperty(pitr, 'len', { get: () => pts.length, enumerable: true, configurable: true, writabel: false }); + return pitr; + } +}; +g2.pntItrOf = function (pts) { + return !(pts && pts.length) ? undefined + : typeof pts[0] === "number" ? g2.pntIterator["x,y"](pts) + : Array.isArray(pts[0]) && pts[0].length >= 2 ? g2.pntIterator["[x,y]"](pts) + : typeof pts[0] === "object" && "x" in pts[0] && "y" in pts[0] ? g2.pntIterator["{x,y}"](pts) + : undefined; +}; +/** + * Get index of command resolving 'callbk' to 'true' starting from end of the queue walking back.
+ * Similar to 'Array.prototype.findIndex', only working reverse. + * @private + */ +g2.cmdIdxBy = function (cmds, callbk) { + for (let i = cmds.length - 1; i >= 0; i--) + if (callbk(cmds[i], i, cmds)) + return i; + return false; // command with index '0' signals 'failing' ... +}; + +/** + * Replacement for Object.assign, as it does not assign getters and setter properly ... + * See https://github.com/tc39/proposal-object-getownpropertydescriptors + * See https://medium.com/@benastontweet/mixins-in-javascript-700ec81f5e5c + * Shallow copy of prototypes (think interfaces) + * @private + */ +g2.mix = function mix(...protos) { + let mixture = {}; + for (const p of protos) + mixture = Object.defineProperties(mixture, Object.getOwnPropertyDescriptors(p)); + return mixture; +} + +/** + * Copy properties, even as getters .. a useful part of the above .. + * @private + */ +g2.cpyProp = function (from, fromKey, to, toKey) { Object.defineProperty(to, toKey, Object.getOwnPropertyDescriptor(from, fromKey)); } + +// Html canvas handler +g2.canvasHdl = function (ctx) { + if (this instanceof g2.canvasHdl) { + if (ctx instanceof CanvasRenderingContext2D) { + this.ctx = ctx; + this.cur = g2.defaultStyle; + this.stack = [this.cur]; + this.matrix = [[1, 0, 0, 1, 0.5, 0.5]]; + this.gridBase = 2; + this.gridExp = 1; + return this; + } + else + return null; + } + return g2.canvasHdl.apply(Object.create(g2.canvasHdl.prototype), arguments); +}; +g2.handler.factory.push((ctx) => ctx instanceof g2.canvasHdl ? ctx + : ctx instanceof CanvasRenderingContext2D ? g2.canvasHdl(ctx) : false); + +g2.canvasHdl.prototype = { + init(grp, style) { + this.stack.length = 1; + this.matrix.length = 1; + this.initStyle(style ? Object.assign({}, this.cur, style) : this.cur); + return true; + }, + async exe(commands) { + for (let cmd of commands) { + // cmd.a is an object offering a `g2` method, so call it and execute its returned commands array. + if (cmd.a && cmd.a.g2) { + const cmds = cmd.a.g2().commands; + // If false, ext was not applied to this cmd. But the command still renders + if (cmds) { + this.exe(cmds); + continue; + } + } + // cmd.a is a `g2` object, so directly execute its commands array. + else if (cmd.a && cmd.a.commands) { + this.exe(cmd.a.commands); + continue; + } + if (cmd.c && this[cmd.c]) { // explicit command name .. ! + const rx = this[cmd.c](cmd.a); + if (rx && rx instanceof Promise) { + await rx; + } + } + } + }, + view({ x = 0, y = 0, scl = 1, cartesian = false }) { + this.pushTrf(cartesian ? [scl, 0, 0, -scl, x, this.ctx.canvas.height - 1 - y] + : [scl, 0, 0, scl, x, y]); + }, + grid({ color = '#ccc', size } = {}) { + let ctx = this.ctx, b = ctx.canvas.width, h = ctx.canvas.height, + { x, y, scl } = this.uniTrf, + sz = size || this.gridSize(scl), + xoff = x % sz, yoff = y % sz; + ctx.save(); + ctx.setTransform(1, 0, 0, 1, 0, 0); + ctx.strokeStyle = color; + ctx.lineWidth = 1; + ctx.beginPath(); + for (let x = xoff, nx = b + 1; x < nx; x += sz) { ctx.moveTo(x, 0); ctx.lineTo(x, h); } + for (let y = yoff, ny = h + 1; y < ny; y += sz) { ctx.moveTo(0, y); ctx.lineTo(b, y); } + ctx.stroke(); + ctx.restore(); + }, + clr({ b, h } = {}) { + let ctx = this.ctx; + ctx.save(); + ctx.setTransform(1, 0, 0, 1, 0, 0); + ctx.clearRect(0, 0, b || ctx.canvas.width, h || ctx.canvas.height); + ctx.restore(); + }, + cir({ r }) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + this.ctx.beginPath(); + this.ctx.arc(x || 0, y || 0, Math.abs(r), 0, 2 * Math.PI, true); + this.drw(arguments[0]); + }, + arc({ r, w = 0, dw = 2 * Math.PI }) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + if (Math.abs(dw) > Number.EPSILON && Math.abs(r) > Number.EPSILON) { + this.ctx.beginPath(); + this.ctx.arc(x, y, Math.abs(r), w, w + dw, dw < 0); + this.drw(arguments[0]); + } + else if (Math.abs(dw) < Number.EPSILON && Math.abs(r) > Number.EPSILON) { + const cw = Math.cos(w), sw = Math.sin(w); + this.ctx.beginPath(); + this.ctx.moveTo(x - r * cw, y - r * sw); + this.ctx.lineTo(x + r * cw, y + r * sw); + } + // else // nothing to draw with r === 0 + }, + ell({ rx, ry, w = 0, dw = 2 * Math.PI, rot = 0 }) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + this.ctx.beginPath(); + this.ctx.ellipse(x, y, Math.abs(rx), Math.abs(ry), rot, w, w + dw, dw < 0); + this.drw(arguments[0]); + }, + rec({ b, h }) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + const tmp = this.setStyle(arguments[0]); + this.ctx.fillRect(x, y, b, h); + this.ctx.strokeRect(x, y, b, h); + this.resetStyle(tmp); + }, + lin(args) { + this.ctx.beginPath(); + this.ctx.moveTo(args.p1 && args.p1.x || args.x1 || 0, args.p1 && args.p1.y || args.y1 || 0); + this.ctx.lineTo(args.p2 && args.p2.x || args.x2 || 0, args.p2 && args.p2.y || args.y2 || 0); + this.stroke(args); + }, + ply({ pts, closed, w = 0, _itr }) { + if (_itr && _itr.len) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + let p, i, len = _itr.len, istrf = !!(x || y || w), cw, sw; + if (istrf) this.setTrf([cw = (w ? Math.cos(w) : 1), sw = (w ? Math.sin(w) : 0), -sw, cw, x, y]); + this.ctx.beginPath(); + this.ctx.moveTo((p = _itr(0)).x, p.y); + for (i = 1; i < len; i++) + this.ctx.lineTo((p = _itr(i)).x, p.y); + if (closed) // closed then .. + this.ctx.closePath(); + this.drw(arguments[0]); + if (istrf) this.resetTrf(); + return i - 1; // number of points .. + } + return 0; + }, + txt({ str, w = 0/*,unsizable*/ }) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + const tmp = this.setStyle(arguments[0]), + sw = w ? Math.sin(w) : 0, + cw = w ? Math.cos(w) : 1, + trf = this.isCartesian ? [cw, sw, sw, -cw, x, y] + : [cw, sw, -sw, cw, x, y]; + this.setTrf(trf); // this.setTrf(unsizable ? this.concatTrf(this.unscaleTrf({x,y}),trf) : trf); + if (this.ctx.fillStyle === 'rgba(0, 0, 0, 0)') { + this.ctx.fillStyle = this.ctx.strokeStyle; + tmp.fs = 'transparent'; + } + this.ctx.fillText(str, 0, 0); + this.resetTrf(); + this.resetStyle(tmp); + }, + errorImageStr: "", + images: Object.create(null), + async loadImage(uri) { + const download = async (xuri) => { + const pimg = new Promise((resolve, reject) => { + let img = new Image(); + img.src = xuri; + function error(err) { + img.removeEventListener('load', load); + img = undefined; + reject(err); + }; + function load() { + img.removeEventListener('error', error); + resolve(img); + img = undefined; + }; + img.addEventListener('error', error, { once: true }); + img.addEventListener('load', load, { once: true }); + }); + + try { + return await pimg; + } catch (err) { + // console.warn(`failed to (pre-)load image; '${xuri}'`, err); + if (xuri === this.errorImageStr) { + throw err; + } else { + return await download(this.errorImageStr); + } + } + } + + let img = this.images[uri]; + if (img !== undefined) { + return img instanceof Promise ? await img : img; + } + img = download(uri); + this.images[uri] = img; + try { + img = await img; + } finally { + this.images[uri] = img; + } + return img; + }, + async img({ uri, x = 0, y = 0, b, h, sx = 0, sy = 0, sb, sh, xoff = 0, yoff = 0, w = 0, scl = 1 }) { + const img_ = await this.loadImage(uri); + this.ctx.save(); + const cart = this.isCartesian ? -1 : 1; + sb = sb || img_.width; + b = b || img_.width; + sh = (sh || img_.height); + h = (h || img_.height) * cart; + yoff *= cart; + w *= cart; + y = this.isCartesian ? -(y / scl) + sy : y / scl; + const [cw, sw] = [Math.cos(w), Math.sin(w)]; + this.ctx.scale(scl, scl * cart); + this.ctx.transform(cw, sw, -sw, cw, x / scl, y); + this.ctx.drawImage(img_, sx, sy, sb, sh, xoff, yoff, b, h); + this.ctx.restore(); + }, + use({ grp }) { + this.beg(arguments[0]); + this.exe(grp.commands); + this.end(); + }, + beg({ w = 0, scl = 1, matrix/*,unsizable*/ } = {}) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + let trf = matrix; + if (!trf) { + let ssw, scw; + ssw = w ? Math.sin(w) * scl : 0; + scw = w ? Math.cos(w) * scl : scl; + trf = [scw, ssw, -ssw, scw, x, y]; + } + this.pushStyle(arguments[0]); + this.pushTrf(trf); // this.pushTrf(unsizable ? this.concatTrf(this.unscaleTrf({x,y}),trf) : trf); + }, + end() { + this.popTrf(); + this.popStyle(); + }, + p() { this.ctx.beginPath(); }, + z() { this.ctx.closePath(); }, + m({ x, y }) { this.ctx.moveTo(x, y); }, + l({ x, y }) { this.ctx.lineTo(x, y); }, + q({ x, y, x1, y1 }) { this.ctx.quadraticCurveTo(x1, y1, x, y); }, + c({ x, y, x1, y1, x2, y2 }) { this.ctx.bezierCurveTo(x1, y1, x2, y2, x, y); }, + a({ dw, k, phi, _xp, _yp }) { // todo: fix elliptical arc bug ... + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + if (k === undefined) k = 1; // ratio r1/r2 + if (Math.abs(dw) > Number.EPSILON) { + if (k === 1) { // circular arc ... + let x12 = x - _xp, y12 = y - _yp; + let tdw_2 = Math.tan(dw / 2), + rx = (x12 - y12 / tdw_2) / 2, ry = (y12 + x12 / tdw_2) / 2, + R = Math.hypot(rx, ry), + w = Math.atan2(-ry, -rx); + this.ctx.ellipse(_xp + rx, _yp + ry, R, R, 0, w, w + dw, this.cartesian ? dw > 0 : dw < 0); + } + else { // elliptical arc .. still buggy .. ! + if (phi === undefined) phi = 0; + let x1 = dw > 0 ? _xp : x, + y1 = dw > 0 ? _yp : y, + x2 = dw > 0 ? x : _xp, + y2 = dw > 0 ? y : _yp; + let x12 = x2 - x1, y12 = y2 - y1, + _dw = (dw < 0) ? dw : -dw; + // if (dw < 0) dw = -dw; // test for bugs .. ! + let cp = phi ? Math.cos(phi) : 1, sp = phi ? Math.sin(phi) : 0, + dx = -x12 * cp - y12 * sp, dy = -x12 * sp - y12 * cp, + sdw_2 = Math.sin(_dw / 2), + R = Math.sqrt((dx * dx + dy * dy / (k * k)) / (4 * sdw_2 * sdw_2)), + w = Math.atan2(k * dx, dy) - _dw / 2, + x0 = x1 - R * Math.cos(w), + y0 = y1 - R * k * Math.sin(w); + this.ctx.ellipse(x0, y0, R, R * k, phi, w, w + dw, this.cartesian ? dw > 0 : dw < 0); + } + } + else + this.ctx.lineTo(x, y); + }, + + stroke({ d } = {}) { + let tmp = this.setStyle(arguments[0]); + d ? this.ctx.stroke(new Path2D(d)) : this.ctx.stroke(); // SVG path syntax + this.resetStyle(tmp); + }, + fill({ d } = {}) { + let tmp = this.setStyle(arguments[0]); + d ? this.ctx.fill(new Path2D(d)) : this.ctx.fill(); // SVG path syntax + this.resetStyle(tmp); + }, + drw({ d, lsh } = {}) { + let ctx = this.ctx, + tmp = this.setStyle(arguments[0]), + p = d && new Path2D(d); // SVG path syntax + d ? ctx.fill(p) : ctx.fill(); + if (ctx.shadowColor !== 'rgba(0, 0, 0, 0)' && ctx.fillStyle !== 'rgba(0, 0, 0, 0)' && !lsh) { + let shc = ctx.shadowColor; // usually avoid stroke shadow when filling ... + ctx.shadowColor = 'rgba(0, 0, 0, 0)'; + d ? ctx.stroke(p) : ctx.stroke(); + ctx.shadowColor = shc; + } + else + d ? ctx.stroke(p) : ctx.stroke(); + this.resetStyle(tmp); + }, + + // State management (transform & style) + // getters & setters + get: { + fs: (ctx) => ctx.fillStyle, + ls: (ctx) => ctx.strokeStyle, + lw: (ctx) => ctx.lineWidth, + lc: (ctx) => ctx.lineCap, + lj: (ctx) => ctx.lineJoin, + ld: (ctx) => ctx.getLineDash(), + ldoff: (ctx) => ctx.lineDashOffset, + ml: (ctx) => ctx.miterLimit, + sh: (ctx) => [ctx.shadowOffsetX || 0, ctx.shadowOffsetY || 0, + ctx.shadowBlur || 0, ctx.shadowColor || 'black'], + font: (ctx) => ctx.font, + thal: (ctx) => ctx.textAlign, + tval: (ctx) => ctx.textBaseline, + }, + set: { + fs: (ctx, q) => { ctx.fillStyle = q; }, + ls: (ctx, q) => { ctx.strokeStyle = q; }, + lw: (ctx, q) => { ctx.lineWidth = q; }, + lc: (ctx, q) => { ctx.lineCap = q; }, + lj: (ctx, q) => { ctx.lineJoin = q; }, + ld: (ctx, q) => { ctx.setLineDash(q); }, + ldoff: (ctx, q) => { ctx.lineDashOffset = q; }, + ml: (ctx, q) => { ctx.miterLimit = q; }, + sh: (ctx, q) => { + if (q) { + ctx.shadowOffsetX = q[0] || 0; + ctx.shadowOffsetY = q[1] || 0; + ctx.shadowBlur = q[2] || 0; + ctx.shadowColor = q[3] || 'black'; + } + }, + font: (ctx, q) => { ctx.font = q; }, + thal: (ctx, q) => { ctx.textAlign = q; }, + tval: (ctx, q) => { ctx.textBaseline = q; } + }, + initStyle(style) { + for (const key in style) + if (this.get[key] && this.get[key](this.ctx) !== style[key]) + this.set[key](this.ctx, style[key]); + }, + setStyle(style) { // short circuit style setting + let q, prv = {}; + for (const key in style) { + if (this.get[key]) { // style keys only ... + if (typeof style[key] === 'string' && style[key][0] === '@') { + let ref = style[key].substr(1); + style[key] = g2.symbol[ref] || this.get[ref] && this.get[ref](this.ctx); + } + if ((q = this.get[key](this.ctx)) !== style[key]) { + prv[key] = q; + this.set[key](this.ctx, style[key]); + } + } + } + return prv; + }, + resetStyle(style) { // short circuit style reset + for (const key in style) + this.set[key](this.ctx, style[key]); + }, + pushStyle(style) { + let cur = {}; // hold changed properties ... + for (const key in style) + if (this.get[key]) { // style keys only ... + if (typeof style[key] === 'string' && style[key][0] === '@') { + let ref = style[key].substr(1); + style[key] = g2.symbol[ref] || this.get[ref] && this.get[ref](this.ctx); + } + if (this.cur[key] !== style[key]) + this.set[key](this.ctx, (cur[key] = style[key])); + } + this.stack.push(this.cur = Object.assign({}, this.cur, cur)); + }, + popStyle() { + let cur = this.stack.pop(); + this.cur = this.stack[this.stack.length - 1]; + for (const key in this.cur) + if (this.get[key] && this.cur[key] !== cur[key]) + this.set[key](this.ctx, this.cur[key]); + }, + concatTrf(q, t) { + return [ + q[0] * t[0] + q[2] * t[1], + q[1] * t[0] + q[3] * t[1], + q[0] * t[2] + q[2] * t[3], + q[1] * t[2] + q[3] * t[3], + q[0] * t[4] + q[2] * t[5] + q[4], + q[1] * t[4] + q[3] * t[5] + q[5] + ]; + }, + initTrf() { + this.ctx.setTransform(...this.matrix[0]); + }, + setTrf(t) { + this.ctx.setTransform(...this.concatTrf(this.matrix[this.matrix.length - 1], t)); + }, + resetTrf() { + this.ctx.setTransform(...this.matrix[this.matrix.length - 1]); + }, + pushTrf(t) { + let q_t = this.concatTrf(this.matrix[this.matrix.length - 1], t); + this.matrix.push(q_t); + this.ctx.setTransform(...q_t); + }, + popTrf() { + this.matrix.pop(); + this.ctx.setTransform(...this.matrix[this.matrix.length - 1]); + }, + get isCartesian() { // det of mat2x2 < 0 ! + let m = this.matrix[this.matrix.length - 1]; + return m[0] * m[3] - m[1] * m[2] < 0; + }, + get uniTrf() { + let m = this.matrix[this.matrix.length - 1]; + return { x: m[4], y: m[5], scl: Math.hypot(m[0], m[1]), cartesian: m[0] * m[3] - m[1] * m[2] < 0 }; + }, + unscaleTrf({ x, y }) { // remove scaling effect (make unzoomable with respect to (x,y)) + let m = this.matrix[this.matrix.length - 1], + invscl = 1 / Math.hypot(m[0], m[1]); + return [invscl, 0, 0, invscl, (1 - invscl) * x, (1 - invscl) * y]; + }, + gridSize(scl) { + let base = this.gridBase, exp = this.gridExp, sz; + while ((sz = scl * base * Math.pow(10, exp)) < 14 || sz > 35) { + if (sz < 14) { + if (base == 1) base = 2; + else if (base == 2) base = 5; + else if (base == 5) { base = 1; exp++; } + } + else { + if (base == 1) { base = 5; exp--; } + else if (base == 2) base = 1; + else if (base == 5) base = 2; + } + } + this.gridBase = base; + this.gridExp = exp; + return sz; + } +} + +// use it with node.js ... ? +if (typeof module !== 'undefined') module.exports = g2; + +/** + * g2.io (c) 2017-18 Stefan Goessner + * @license MIT License + */ +"use strict"; + +g2.io = function() { + if (this instanceof g2.io) { + this.model = null; + this.grpidx = 0; + return this; + } + return g2.io.apply(Object.create(g2.io.prototype)); +}; +g2.handler.factory.push((ctx) => ctx instanceof g2.io ? ctx : false); + +g2.io.parseGrp = function(model, id, onErr) { + let g; + onErr = onErr || console.error; + if (id in model) { + g = g2({id}); + for (let cmd of model[id]) { + if (cmd.c === 'use') { + cmd.a.grp = g2.io.parseGrp(model, cmd.a.grp); + g[cmd.c](cmd.a); + } + else if (g[cmd.c]) + cmd.a ? g[cmd.c](cmd.a) : g[cmd.c](); + else // invalid g2 command ! + onErr(`g2.io: Unable to handle command '${cmd.c}'`) + } + return g; + } + else if (id in g2.symbol) + return g2.symbol[id]; + else + onErr(`g2.io: Unable to find group with id '${id}'!`); + return false; +} + +g2.io.prototype = { + init: function(grp,style) { + this.model = {'main':[]}; + this.curgrp = this.model.main; + this.grpidx = 0; + return true; + }, + exe: function(commands) { + for (let cmd of commands) { + if (this[cmd.c]) + cmd.a ? this[cmd.c](cmd.a) : this[cmd.c](); + else + this.out(cmd.c,cmd.a); + } + }, + out: function(c,a) { + if (a) { + let args = {}; + Object.keys(a).forEach(k => { + if (k[0] !== '_') // no private key ... + args[k] = a[k]; + }); + this.curgrp.push({c:c,a:args}); + } + else + this.curgrp.push({c:c}); + }, + stringify: function(space) { return space ? JSON.stringify(this.model,null,space) : JSON.stringify(this.model); }, + toString: function() { return JSON.stringify(this.model); }, + + // customized commands ... + use: function(args) { + let grp = args.grp instanceof g2 ? args.grp + : typeof args.grp === 'string' && g2.symbol.includes(args.grp) ? g2.symbol[args.grp] + : null; + if (grp) { + if (!grp.id) grp.id = `$grp${++this.grpidx}`; + if (!(grp.id in this.model)) { // meet first time .. + let curgrp = this.curgrp; + this.curgrp = this.model[grp.id] = []; + for (let command of grp.commands) + this.out(command.c,command.a); + this.curgrp = curgrp; + } + args.grp = grp.id; + + this.out("use", args); + } + } +} + +/** + * g2.lib (c) 2013-17 Stefan Goessner + * geometric constants and higher functions + * @license MIT License + * @link https://github.com/goessner/g2 + */ +"use strict" + +var g2 = g2 || {}; // for standalone usage ... + +g2 = Object.assign(g2, { + EPS: Number.EPSILON, + PI: Math.PI, + PI2: 2*Math.PI, + SQRT2: Math.SQRT2, + SQRT2_2: Math.SQRT2/2, + /** + * Map angle to the range [0 .. 2*pi]. + * @param {number} w Angle in radians. + * @returns {number} Angle in radians in interval [0 .. 2*pi]. + */ + toPi2(w) { return (w % g2.PI2 + g2.PI2) % g2.PI2; }, + /** + * Map angle to the range [-pi .. pi]. + * @param {number} w Angle in radians. + * @returns {number} Angle in radians in interval [-pi .. pi]. + */ + toPi(w) { return (w = (w % g2.PI2 + g2.PI2) % g2.PI2) > g2.PI ? w - g2.PI2 : w; }, + /** + * Map angle to arc sector [w0 .. w0+dw]. + * @param {number} w Angle in range [0 .. 2*pi]. + * @param {number} w0 Start angle in range [0 .. 2*pi]. + * @param {number} dw angular range in radians. Can be positive or negative. + * @returns {number} Normalised angular parameter lambda. + * '0' corresponds to w0 and '1' to w0+dw. To reconstruct an angle from + * the return parameter lambda use: w = w0 + lambda*dw. + */ + toArc: function(w,w0,dw) { + if (dw > g2.EPS || dw < -g2.EPS) { + if (w0 > w && w0+dw > g2.PI2) w0 -= g2.PI2; + else if (w0 < w && w0+dw < 0) w0 += g2.PI2; + return (w-w0)/dw; + + } + return 0; + }, + /** + * Test, if point is located on line. + * @param {x,y} point to test. + * @param {x1,y1} start point of line. + * @param {x2,y2} end point of line. + * @param {number} eps. + * @return {boolean} the test result. + */ + isPntOnLin({x,y},p1,p2,eps=Number.EPSILON) { + let dx = p2.x - p1.x, dy = p2.y - p1.y, dx2 = x - p1.x, dy2 = y - p1.y, + dot = dx*dx2 + dy*dy2, perp = dx*dy2 - dy*dx2, len = Math.hypot(dx,dy), epslen = eps*len; + return -epslen < perp && perp < epslen && -epslen < dot && dot < len*(len+eps); + }, + /** + * Test, if point is located on circle circumference. + * @param {x,y} point to test. + * @param {x,y,r} circle. + * @param {number} eps. + * @return {boolean} the test result. + */ + isPntOnCir({x:xp,y:yp},{x,y,r},eps=Number.EPSILON) { + let dx = xp - x, dy = yp - y, + ddis = dx*dx + dy*dy - r*r, reps = eps*r; + return -reps < ddis && ddis < reps; + }, + /** + * Test, if point is located on a circular arc. + * @param {x,y} point to test. + * @param {x,y,r} circle. + * @param {number} eps. + * @return {boolean} the test result. + */ + isPntOnArc({x:xp,y:yp},{x,y,r,w,dw},eps=Number.EPSILON) { + var dx = xp - x, dy = yp - y, dist = Math.hypot(dx,dy), + mu = g2.toArc(g2.toPi2(Math.atan2(dy,dx)),g2.toPi2(w),dw); + return r*Math.abs(dw) > eps && Math.abs(dist-r) < eps && mu >= 0 && mu <= 1; + }, + /** + * Test, if point is located on a polygon line. + * @param {x,y} point to test. + * @param {pts,closed} polygon. + * @param {number} eps. + * @return {boolean} the test result. + */ + isPntOnPly({x,y},{pts,closed},eps=Number.EPSILON) { + // console.log(pts) + for (var i=0,n=pts.length; i<(closed ? n : n-1); i++) + if (g2.isPntOnLin({x,y},pts[i],pts[(i+1)%n],eps)) + return true; + return false; + }, + /** + * Test, if point is located on a box. A box in contrast to a rectangle + * is always aligned parallel to coordinate system axes, with its + * local origin `{x,y}` located in the center. The dimensions `{b,h}` are + * half size dimensions (so upper right corner is {x+b,y+h}). + * @param {x,y} point to test. + * @param {x,y,r} circle. + * @param {number} eps. + * @return {boolean} the test result. + */ + isPntOnBox({x:xp,y:yp},{x,y,b,h},eps=Number.EPSILON) { + var dx = x.p - x, dy = yp - y; + return dx >= b-eps && dx <= b+eps && dy <= h+eps && dy >= -h-eps + || dx >= -b-eps && dx <= b+eps && dy <= h+eps && dy >= h-eps + || dx >= -b-eps && dx <= -b+eps && dy <= h+eps && dy >= -h-eps + || dx >= -b-eps && dx <= b+eps && dy <= -h+eps && dy >= -h-eps; + }, + /** + * Test, if point is located inside of a circle. + * @param {x,y} point to test. + * @param {x,y,r} circle. + * @return {boolean} the test result. + */ + isPntInCir({x:xp,y:yp},{x,y,r}) { + return (x - xp)**2 + (y - yp)**2 < r*r; + }, + /** + * Test, if point is located inside of a closed polygon. + * (see http://paulbourke.net/geometry/polygonmesh/) + * @param {x,y} point to test. + * @param {pnts,closed} polygon. + * @returns {boolean} point is on polygon lines. + */ + isPntInPly({x,y},{pts,closed},eps=Number.EPSILON) { + let match = 0; + for (let n=pts.length,i=0,pi=pts[i],pj=pts[n-1]; i pi.y || y > pj.y) + && (y <= pi.y || y <= pj.y) + && (x <= pi.x || x <= pj.x) + && pi.y !== pj.y + && (pi.x === pj.x || x <= pj.x + (y-pj.y)*(pi.x-pj.x)/(pi.y-pj.y))) + match++; + return match%2 != 0; // even matches required for being outside .. + }, + /** + * Test, if point is located inside of a box. A box in contrast to a rectangle + * is always aligned parallel to coordinate system axes, with its + * local origin `{x,y}` located in the center. The dimensions `{b,h}` are + * half size dimensions (so upper right corner is {x+b,y+h}). + * @param {x,y} point to test. + * @param {x,y,r} circle. + * @return {boolean} the test result. + */ + isPntInBox({x:xp,y:yp},{x,y,b,h}) { + var dx = xp - x, dy = yp - y; + return dx >= -b && dx <= b && dy >= -h && dy <= h; + }, + + arc3pts(x1,y1,x2,y2,x3,y3) { + const dx1 = x2 - x1, dy1 = y2 - y1; + const dx2 = x3 - x2, dy2 = y3 - y2; + const den = dx1*dy2 - dy1*dx2; + const lam = Math.abs(den) > Number.EPSILON + ? 0.5*((dx1 + dx2)*dx2 + (dy1 + dy2)*dy2)/den + : 0; + const x0 = lam ? x1 + 0.5*dx1 - lam*dy1 : x1 + 0.5*(dx1 + dx2); + const y0 = lam ? y1 + 0.5*dy1 + lam*dx1 : y1 + 0.5*(dy1 + dy2); + const dx01 = x1 - x0, dy01 = y1 - y0; + const dx03 = x3 - x0, dy03 = y3 - y0; + const dw = lam ? Math.atan2(dx01*dy03-dy01*dx03,dx01*dx03+dy01*dy03) : 0; + const r = dw ? Math.hypot(dy01,dx01) : 0.5*Math.hypot(dy1+dy2,dx1+dx2); + + return {x:x0,y:y0,r:r,w:Math.atan2(dy01,dx01),dw}; + } +}) + +"use strict" + +/** + * g2.ext (c) 2015-20 Stefan Goessner + * @author Stefan Goessner + * @license MIT License + * @requires g2.core.js + * @typedef {g2} + * @description Additional methods for g2. + * @returns {g2} + */ + +var g2 = g2 || { prototype: {} }; // for jsdoc only ... + +// constants for element selection / editing +g2.NONE = 0x0; g2.OVER = 0x1; g2.DRAG = 0x2; g2.EDIT = 0x4; + +/** + * Extended style values. + * Not really meant to get overwritten. But if you actually want, proceed.
+ * These styles can be referenced using the comfortable '@' syntax. + * @namespace + * @property {object} symbol `g2` symbol namespace. + * @property {object} [symbol.tick] Predefined symbol: a little tick + * @property {object} [symbol.dot] Predefined symbol: a little dot + * @property {object} [symbol.sqr] Predefined symbol: a little square + * @property {string} [symbol.nodcolor=#333] node color. + * @property {string} [symbol.nodfill=#dedede] node fill color. + * @property {string} [symbol.nodfill2=#aeaeae] alternate node fill color, somewhat darker. + * @property {string} [symbol.linkcolor=#666] link color. + * @property {string} [symbol.linkfill=rgba(225,225,225,0.75)] link fill color, semi-transparent. + * @property {string} [symbol.dimcolor=darkslategray] dimension color. + * @property {array} [symbol.solid=[]] solid line style. + * @property {array} [symbol.dash=[15,10]] dashed line style. + * @property {array} [symbol.dot=[4,4]] dotted line style. + * @property {array} [symbol.dashdot=[25,6.5,2,6.5]] dashdotted line style. + * @property {number} [symbol.labelOffset=5] default label offset distance. + * @property {number} [symbol.labelSignificantDigits=3] default label's significant digits after numbering point. + */ +g2.symbol = g2.symbol || {}; +g2.symbol.tick = g2().p().m({ x: 0, y: -2 }).l({ x: 0, y: 2 }).stroke({ lc: "round", lwnosc: true }); +g2.symbol.dot = g2().cir({ x: 0, y: 0, r: 2, ls: "transparent" }); +g2.symbol.sqr = g2().rec({ x: -1.5, y: -1.5, b: 3, h: 3, ls: "transparent" }); + +g2.symbol.nodcolor = "#333"; +g2.symbol.nodfill = "#dedede"; +g2.symbol.nodfill2 = "#aeaeae"; +g2.symbol.linkcolor = "#666"; +g2.symbol.linkfill = "rgba(225,225,225,0.75)"; +g2.symbol.dimcolor = "darkslategray"; +g2.symbol.solid = []; +g2.symbol.dash = [15, 10]; +g2.symbol.dot = [4, 4]; +g2.symbol.dashdot = [25, 6.5, 2, 6.5]; +g2.symbol.labelSignificantDigits = 3; // 0.1234 => 0.123, 0.01234 => 0.0123, 1.234 => 1.23, 12.34 => 12.3, 123.4 => 123, 1234 => 1234 + +/** +* Flatten object properties (evaluate getters) +*/ +g2.flatten = function (obj) { + const args = Object.create(null); // important ! + for (let p in obj) + if (typeof obj[p] !== 'function') + args[p] = obj[p]; + return args; +} +/* +g2.strip = function(obj,prop) { + const clone = Object.create(Object.getPrototypeOf(obj),Object.getOwnPropertyDescriptors(obj)); + Object.defineProperty(clone, prop, { get:undefined, enumerable:true, configurable:true, writabel:false }); + return clone; +} +*/ +g2.pointIfc = { + // p vector notation ! ... helps to avoid object destruction + get p() { return { x: this.x, y: this.y }; }, // visible if 'p' is *not* explicite given. + get x() { return Object.getOwnPropertyDescriptor(this, 'p') ? this.p.x : 0; }, + get y() { return Object.getOwnPropertyDescriptor(this, 'p') ? this.p.y : 0; }, + set x(q) { if (Object.getOwnPropertyDescriptor(this, 'p')) this.p.x = q; }, + set y(q) { if (Object.getOwnPropertyDescriptor(this, 'p')) this.p.y = q; }, +} + +g2.labelIfc = { + getLabelOffset() { const off = this.label.off !== undefined ? +this.label.off : 1; return off + Math.sign(off) * (this.lw || 2) / 2; }, + getLabelString() { + let s = typeof this.label === 'object' ? this.label.str : typeof this.label === 'string' ? this.label : '?'; + if (s && s[0] === "@" && this[s.substr(1)]) { + s = s.substr(1); + let val = this[s]; + val = Number.isInteger(val) ? val + : Number(val).toFixed(Math.max(g2.symbol.labelSignificantDigits - Math.log10(val), 0)); + + s = `${val}${s === 'angle' ? "°" : ""}`; + } + return s; + }, + drawLabel(g) { + const lbl = this.label; + const font = lbl.font || g2.defaultStyle.font; + const h = parseInt(font); // font height (px assumed !) + const str = this.getLabelString(); + const rx = (str.length || 1) * 0.75 * h / 2, ry = 1.25 * h / 2; // ellipse semi-axes length + const pos = this.pointAt(lbl.loc || this.lbloc || 'se'); + const off = this.getLabelOffset(); + const p = { + x: pos.x + pos.nx * (off + Math.sign(off) * rx), + y: pos.y + pos.ny * (off + Math.sign(off) * ry) + }; + + if (lbl.border) g.ell({ x: p.x, y: p.y, rx, ry, ls: lbl.fs || 'black', fs: lbl.fs2 || '#ffc' }); + g.txt({ + str, x: p.x, y: p.y, + thal: "center", tval: "middle", + fs: lbl.fs || 'black', font: lbl.font + }); + return g; + } +} + +g2.markIfc = { + markAt(loc) { + const p = this.pointAt(loc); + const w = Math.atan2(p.ny, p.nx) + Math.PI / 2; + return { + grp: this.getMarkSymbol(), x: p.x, y: p.y, w: w, scl: this.lw || 1, + ls: this.ls || '#000', fs: this.fs || this.ls || '#000' + } + }, + getMarkSymbol() { + // Use tick as default + const mrk = this.mark + if (typeof mrk === 'number' || !mrk) return g2.symbol.tick; + if (typeof mrk.symbol === 'object') return mrk.symbol; + if (typeof mrk.symbol === 'string') return g2.symbol[mrk.symbol] + }, + // loop is for elements that close, e.g. rec or cir => loc at 0 === loc at 1 + drawMark(g, closed = false) { + let loc; + if (Array.isArray(this.mark)) { + loc = this.mark; + } + else { + const count = typeof this.mark === 'object' ? this.mark.count : this.mark; + loc = count ? + Array.from(Array(count)).map((_, i) => i / (count - !closed)) : + this.mark.loc; + } + for (let l of loc) { + g.use(this.markAt(l)); + } + return g; + } +} + +g2.prototype.cir.prototype = g2.mix(g2.pointIfc, g2.labelIfc, g2.markIfc, { + w: 0, // default start angle (used for dash-dot orgin and editing) + lbloc: 'c', + get isSolid() { return this.fs && this.fs !== 'transparent' }, + get len() { return 2 * Math.PI * this.r; }, + get lsh() { return this.state & g2.OVER; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false }, + get g2() { // dynamically switch existence of method via getter ... cool ! + const e = g2(); // hand object stripped from `g2` + this.label && e.ins((g) => this.drawLabel(g)); + this.mark && e.ins((g) => this.drawMark(g, true)); + return () => g2().cir(g2.flatten(this)).ins(e); // avoiding infinite recursion ! + }, + pointAt(loc) { + const Q = Math.SQRT2 / 2; + const LOC = { c: [0, 0], e: [1, 0], ne: [Q, Q], n: [0, 1], nw: [-Q, Q], w: [-1, 0], sw: [-Q, -Q], s: [0, -1], se: [Q, -Q] }; + const q = (loc + 0 === loc) ? [Math.cos(loc * 2 * Math.PI), Math.sin(loc * 2 * Math.PI)] + : (LOC[loc || "c"] || [0, 0]); + return { + x: this.x + q[0] * this.r, + y: this.y + q[1] * this.r, + nx: q[0], + ny: q[1] + }; + }, + hit({ x, y, eps }) { + return this.isSolid ? g2.isPntInCir({ x, y }, this, eps) + : g2.isPntOnCir({ x, y }, this, eps); + }, + drag({ dx, dy }) { this.x += dx; this.y += dy }, +}); + +g2.prototype.lin.prototype = g2.mix(g2.labelIfc, g2.markIfc, { + // p1 vector notation ! + get p1() { return { x1: this.x1, y1: this.y1 }; }, // relevant if 'p1' is *not* explicite given. + get x1() { return Object.getOwnPropertyDescriptor(this, 'p1') ? this.p1.x : 0; }, + get y1() { return Object.getOwnPropertyDescriptor(this, 'p1') ? this.p1.y : 0; }, + set x1(q) { if (Object.getOwnPropertyDescriptor(this, 'p1')) this.p1.x = q; }, + set y1(q) { if (Object.getOwnPropertyDescriptor(this, 'p1')) this.p1.y = q; }, + // p2 vector notation ! + get p2() { return { x2: this.x2, y2: this.y2 }; }, // relevant if 'p2' is *not* explicite given. + get x2() { return Object.getOwnPropertyDescriptor(this, 'p2') ? this.p2.x : 0; }, + get y2() { return Object.getOwnPropertyDescriptor(this, 'p2') ? this.p2.y : 0; }, + set x2(q) { if (Object.getOwnPropertyDescriptor(this, 'p2')) this.p2.x = q; }, + set y2(q) { if (Object.getOwnPropertyDescriptor(this, 'p2')) this.p2.y = q; }, + + isSolid: false, + get len() { return Math.hypot(this.x2 - this.x1, this.y2 - this.y1); }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false }, + get g2() { // dynamically switch existence of method via getter ... ! + const e = g2(); + this.label && e.ins(e => this.drawLabel(e)); + this.mark && e.ins(e => this.drawMark(e)); + return () => g2().lin(g2.flatten(this)).ins(e); + }, + + pointAt(loc) { + let t = loc === "beg" ? 0 + : loc === "end" ? 1 + : (loc + 0 === loc) ? loc // numerical arg .. + : 0.5, // 'mid' .. + dx = this.x2 - this.x1, + dy = this.y2 - this.y1, + len = Math.hypot(dx, dy); + return { + x: this.x1 + dx * t, + y: this.y1 + dy * t, + nx: len ? dy / len : 0, + ny: len ? -dx / len : -1 + }; + }, + hit({ x, y, eps }) { + return g2.isPntOnLin({ x, y }, { x: this.x1, y: this.y1 }, { x: this.x2, y: this.y2 }, eps); + }, + drag({ dx, dy }) { + this.x1 += dx; this.x2 += dx; + this.y1 += dy; this.y2 += dy; + } +}); + +g2.prototype.rec.prototype = g2.mix(g2.pointIfc, g2.labelIfc, g2.markIfc, { + get len() { return 2 * (this.b + this.h); }, + get isSolid() { return this.fs && this.fs !== 'transparent' }, + get lsh() { return this.state & g2.OVER; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false; }, + get g2() { // dynamically switch existence of method via getter ... ! + const e = g2(); + this.label && e.ins(e => this.drawLabel(e)); + this.mark && e.ins(e => this.drawMark(e, true)); + return () => g2().rec(g2.flatten(this)).ins(e); + }, + lbloc: 'c', + pointAt(loc) { + const locAt = (loc) => { + const o = { c: [0, 0], e: [1, 0], ne: [0.95, 0.95], n: [0, 1], nw: [-0.95, 0.95], w: [-1, 0], sw: [-0.95, -0.95], s: [0, -1], se: [0.95, -0.95] }; + + if (o[loc]) return o[loc]; + + const w = 2 * Math.PI * loc + pi / 4; + if (loc <= 0.25) return [1 / Math.tan(w), 1]; + if (loc <= 0.50) return [-1, -Math.tan(w)]; + if (loc <= 0.75) return [- 1 / Math.tan(w), -1]; + if (loc <= 1.00) return [1, Math.tan(w)]; + } + const q = locAt(loc); + return { + x: this.x + (1 + q[0]) * this.b / 2, + y: this.y + (1 + q[1]) * this.h / 2, + nx: 1 - Math.abs(q[0]) < 0.01 ? q[0] : 0, + ny: 1 - Math.abs(q[1]) < 0.01 ? q[1] : 0 + }; + }, + hit({ x, y, eps }) { + return this.isSolid ? g2.isPntInBox({ x, y }, { x: this.x + this.b / 2, y: this.y + this.h / 2, b: this.b / 2, h: this.h / 2 }, eps) + : g2.isPntOnBox({ x, y }, { x: this.x + this.b / 2, y: this.y + this.h / 2, b: this.b / 2, h: this.h / 2 }, eps); + }, + drag({ dx, dy }) { this.x += dx; this.y += dy } +}); + +g2.prototype.arc.prototype = g2.mix(g2.pointIfc, g2.labelIfc, g2.markIfc, { + get len() { return Math.abs(this.r * this.dw); }, + isSolid: false, + get angle() { return this.dw / Math.PI * 180; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false }, + get g2() { // dynamically switch existence of method via getter ... ! + const e = g2(); + this.label && e.ins(e => this.drawLabel(e)); + this.mark && e.ins(e => this.drawMark(e)); + return () => g2().arc(g2.flatten(this)).ins(e); + }, + lbloc: 'mid', + pointAt(loc) { + let t = loc === "beg" ? 0 + : loc === "end" ? 1 + : loc === "mid" ? 0.5 + : loc + 0 === loc ? loc + : 0.5, + ang = (this.w || 0) + t * (this.dw || Math.PI * 2), cang = Math.cos(ang), sang = Math.sin(ang), r = loc === "c" ? 0 : this.r; + return { + x: this.x + r * cang, + y: this.y + r * sang, + nx: cang, + ny: sang + }; + }, + hit({ x, y, eps }) { return g2.isPntOnArc({ x, y }, this, eps) }, + drag({ dx, dy }) { this.x += dx; this.y += dy; }, +}); + +/** +* Draw interactive handle. +* @method +* @returns {object} g2 +* @param {object} - handle object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().hdl({x:100,y:80}) // Draw handle. +*/ +g2.prototype.hdl = function (args) { return this.addCommand({ c: 'hdl', a: args }); } +g2.prototype.hdl.prototype = g2.mix(g2.prototype.cir.prototype, { + r: 5, + isSolid: true, + draggable: true, + lbloc: 'se', + get lsh() { return this.state & g2.OVER; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false }, + g2() { + const { x, y, r, b = 4, shape = 'cir', ls = 'black', fs = '#ccc', sh } = this; + return shape === 'cir' ? g2().cir({ x, y, r, ls, fs, sh }).ins((g) => this.label && this.drawLabel(g)) + : g2().rec({ x: x - b, y: y - b, b: 2 * b, h: 2 * b, ls, fs, sh }).ins((g) => this.label && this.drawLabel(g)); + } +}); + +/** +* Node symbol. +* @constructor +* @param {object} - symbol arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().nod({x:10,y:10}) +*/ + +g2.prototype.nod = function (args = {}) { return this.addCommand({ c: 'nod', a: args }); } +g2.prototype.nod.prototype = g2.mix(g2.prototype.cir.prototype, { + r: 5, + ls: '@nodcolor', + fs: g2.symbol.nodfill, + isSolid: true, + lbloc: 'se', + g2() { // in contrast to `g2.prototype.cir.prototype`, `g2()` is called always ! + return g2() + .cir({ ...g2.flatten(this), r: this.r * (this.scl !== undefined ? this.scl : 1) }) + .ins((g) => this.label && this.drawLabel(g)) + } +}); + +/** + * Double nod symbol + * @constructor + * @returns {object} g2 + * @param {object} - symbol arguments object. + * @property {number} x - x-value center. + * @property {number} y - y-value center. + * @example + * g2().dblnod({x:10,y:10}) +*/ +g2.prototype.dblnod = function ({ x = 0, y = 0 }) { return this.addCommand({ c: 'dblnod', a: arguments[0] }); } +g2.prototype.dblnod.prototype = g2.mix(g2.prototype.cir.prototype, { + get r() { return 6; }, + get isSolid() { return true; }, + g2() { + return g2() + .beg({ x: this.x, y: this.y }) + .cir({ r: 6, ls: '@nodcolor', fs: '@nodfill', sh: this.sh }) + .cir({ r: 3, ls: '@nodcolor', fs: '@nodfill2' }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) + +/** +* Pole symbol. +* @constructor +* @returns {object} g2 +* @param {object} - symbol arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().pol({x:10,y:10}) +*/ +g2.prototype.pol = function (args = {}) { return this.addCommand({ c: 'pol', a: args }); } +g2.prototype.pol.prototype = g2.mix(g2.prototype.nod.prototype, { + g2() { + return g2() + .beg(g2.flatten(this)) + .cir({ r: 6, fs: g2.symbol.nodfill }) + .cir({ r: 2.5, fs: '@ls', ls: 'transparent' }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) + +/** +* Ground symbol. +* @constructor +* @param {object} - arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().gnd({x:10,y:10}) +*/ +g2.prototype.gnd = function (args = {}) { return this.addCommand({ c: 'gnd', a: args }); } +g2.prototype.gnd.prototype = g2.mix(g2.prototype.nod.prototype, { + g2() { + return g2() + .beg(g2.flatten(this)) + .cir({ x: 0, y: 0, r: 6 }) + .p() + .m({ x: 0, y: 6 }) + .a({ dw: Math.PI / 2, x: -6, y: 0 }) + .l({ x: 6, y: 0 }) + .a({ dw: -Math.PI / 2, x: 0, y: -6 }) + .z() + .fill({ fs: g2.symbol.nodcolor }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) + +g2.prototype.nodfix = function (args = {}) { return this.addCommand({ c: 'nodfix', a: args }); } +g2.prototype.nodfix.prototype = g2.mix(g2.prototype.nod.prototype, { + g2() { + return g2() + .beg(g2.flatten(this)) + .p() + .m({ x: -8, y: -12 }) + .l({ x: 0, y: 0 }) + .l({ x: 8, y: -12 }) + .drw({ fs: g2.symbol.nodfill2 }) + .cir({ x: 0, y: 0, r: this.r }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) +/** +* @method +* @returns {object} g2 +* @param {object} - symbol arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().view({cartesian:true}) + * .nodflt({x:10,y:10}) +*/ +g2.prototype.nodflt = function (args = {}) { return this.addCommand({ c: 'nodflt', a: args }); } +g2.prototype.nodflt.prototype = g2.mix(g2.prototype.nod.prototype, { + g2() { + return g2() + .beg(g2.flatten(this)) + .p() + .m({ x: -8, y: -12 }) + .l({ x: 0, y: 0 }) + .l({ x: 8, y: -12 }) + .drw({ ls: g2.symbol.nodcolor, fs: g2.symbol.nodfill2 }) + .cir({ x: 0, y: 0, r: this.r, ls: g2.symbol.nodcolor, fs: g2.symbol.nodfill }) + .lin({ x1: -9, y1: -19, x2: 9, y2: -19, ls: g2.symbol.nodfill2, lw: 5 }) + .lin({ x1: -9, y1: -15.5, x2: 9, y2: -15.5, ls: g2.symbol.nodcolor, lw: 2 }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) + +/** +* Draw vector arrow. +* @method +* @returns {object} g2 +* @param {object} - vector arguments object. +* @property {number} x1 - start x coordinate. +* @property {number} y1 - start y coordinate. +* @property {number} x2 - end x coordinate. +* @property {number} y2 - end y coordinate. +* @example +* g2().vec({x1:50,y1:20,x2:250,y2:120}) +*/ +g2.prototype.vec = function vec(args) { return this.addCommand({ c: 'vec', a: args }); } +g2.prototype.vec.prototype = g2.mix(g2.prototype.lin.prototype, { + g2() { + const { x1, y1, x2, y2, lw = 1, ls = '#000', ld = [], fs = ls || '#000', lc = 'round', lj = 'round', } = this; + const dx = x2 - x1, dy = y2 - y1, r = Math.hypot(dx, dy); + const b = 3 * (1 + lw) > r ? r / 3 : (1 + lw); + const arrowHead = () => g2().p().m({ x: 0, y: 0 }).l({ x: -5 * b, y: b }).a({ dw: -Math.PI / 3, x: -5 * b, y: -b }).z().drw({ ls, fs, lc, lj }); + return g2() + .beg({ x: x1, y: y1, w: Math.atan2(dy, dx), lc, lj }) + .p().m({ x: 0, y: 0 }) + .l({ x: r - 3 * b, y: 0 }) + .stroke({ ls, lw, ld }) + .use({ grp: arrowHead, x: r, y: 0 }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}); + +/** +* Arc as Vector +* @method +* @returns {object} g2 +* @param {object} - angular dimension arguments. +* @property {number} x - start x coordinate. +* @property {number} y - start y coordinate. +* @property {number} r - radius +* @property {number} [w=0] - start angle (in radian). +* @property {number} [dw=Math.PI/2] - angular range in radian. In case of positive values it is running counterclockwise with + * right handed (cartesian) coordinate system. +* @example +* g2().avec({x:100,y:70,r:50,w:pi/3,dw:4*pi/3}) +*/ +g2.prototype.avec = function avec(args) { return this.addCommand({ c: 'avec', a: args }); } +g2.prototype.avec.prototype = g2.mix(g2.prototype.arc.prototype, { + g2() { + const { x, y, r, w, dw = 0, lw = 1, lc = 'round', lj = 'round', ls, fs = ls || "#000", label } = this; + const b = 3 * (1 + lw) > r ? r / 3 : (1 + lw), bw = 5 * b / r; + const arrowHead = () => g2().p().m({ x: 0, y: 2 * b }).l({ x: 0, y: -2 * b }).m({ x: 0, y: 0 }).l({ x: -5 * b, y: b }) + .a({ dw: -Math.PI / 3, x: -5 * b, y: -b }).z().drw({ ls, fs }); + + return g2() + .beg({ x, y, w, ls, lw, lc, lj }) + .arc({ r, w: 0, dw }) + .use({ + grp: arrowHead, x: r * Math.cos(dw), y: r * Math.sin(dw), + w: (dw >= 0 ? dw + Math.PI / 2 - bw / 2 : dw - Math.PI / 2 + bw / 2) + }) + .end() + .ins((g) => label && this.drawLabel(g)); + } +}); + +/** +* Linear Dimension +* @method +* @returns {object} g2 +* @param {object} - dimension arguments object. +* @property {number} x1 - start x coordinate. +* @property {number} y1 - start y coordinate. +* @property {number} x2 - end x coordinate. +* @property {number} y2 - end y coordinate. +* @property {number} off - offset. +* @property {boolean} [inside=true] - draw dimension arrows between or outside of ticks. +* @example +* g2().dim({x1:60,y1:40,x2:190,y2:120}) +*/ +g2.prototype.dim = function dim(args) { return this.addCommand({ c: 'dim', a: args }); } +g2.prototype.dim.prototype = g2.mix(g2.prototype.lin.prototype, { + pointAt(loc) { + const pnt = g2.prototype.lin.prototype.pointAt.call(this, loc); + if (this.off) { + pnt.x += this.off * pnt.nx; + pnt.y += this.off * pnt.ny; + } + return pnt; + }, + g2() { + const { x1, y1, x2, y2, lw = 1, lc = 'round', lj = 'round', off = 0, inside = true, ls, fs = ls || "#000", label } = this; + const dx = x2 - x1, dy = y2 - y1, r = Math.hypot(dx, dy); + const b = 3 * (1 + lw) > r ? r / 3 : (1 + lw); + const arrowHead = () => g2().p().m({ x: 0, y: 2 * b }).l({ x: 0, y: -2 * b }).m({ x: 0, y: 0 }).l({ x: -5 * b, y: b }) + .a({ dw: -Math.PI / 3, x: -5 * b, y: -b }).z().drw({ ls, fs }); + return g2() + .beg({ x: x1 + off / r * dy, y: y1 - off / r * dx, w: Math.atan2(dy, dx), ls, fs, lw, lc, lj }) + .lin({ x1: (inside ? 4 * b : 0), y1: 0, x2: (inside ? r - 4 * b : r), y2: 0 }) + .use({ grp: arrowHead, x: r, y: 0, w: (inside ? 0 : Math.PI) }) + .use({ grp: arrowHead, x: 0, y: 0, w: (inside ? Math.PI : 0) }) + .lin({ x1: 0, y1: off, x2: 0, y2: 0 }) + .lin({ x1: r, y1: off, x2: r, y2: 0 }) + .end() + .ins((g) => label && this.drawLabel(g)); + } +}); + +/** +* Angular dimension +* @method +* @returns {object} g2 +* @param {object} - angular dimension arguments. +* @property {number} x - start x coordinate. +* @property {number} y - start y coordinate. +* @property {number} r - radius +* @property {number} [w=0] - start angle (in radian). +* @property {number} [dw=Math.PI/2] - angular range in radian. In case of positive values it is running counterclockwise with + * right handed (cartesian) coordinate system. +* @property {boolean} [outside=false] - draw dimension arrows outside of ticks. +* @depricated {boolean} [inside] - draw dimension arrows between ticks. +* @example +* g2().adim({x:100,y:70,r:50,w:pi/3,dw:4*pi/3}) +*/ +g2.prototype.adim = function adim(args) { return this.addCommand({ c: 'adim', a: args }); } +g2.prototype.adim.prototype = g2.mix(g2.prototype.arc.prototype, { + g2() { + const { x, y, r, w, dw, lw = 1, lc = 'round', lj = 'round', ls, fs = ls || "#000", label } = this; + const b = 3 * (1 + lw) > r ? r / 3 : (1 + lw), bw = 5 * b / r; + const arrowHead = () => g2().p().m({ x: 0, y: 2 * b }).l({ x: 0, y: -2 * b }).m({ x: 0, y: 0 }).l({ x: -5 * b, y: b }) + .a({ dw: -Math.PI / 3, x: -5 * b, y: -b }).z().drw({ ls, fs }); + + const outside = (this.inside !== undefined && this.outside === undefined) ? !this.inside : !!this.outside; // still support depricated property ! + + return g2() + .beg({ x, y, w, ls, lw, lc, lj }) + .arc({ r, w: 0, dw }) + .use({ grp: arrowHead, x: r, y: 0, w: (!outside && dw > 0 || outside && dw < 0 ? -Math.PI / 2 + bw / 2 : Math.PI / 2 - bw / 2) }) + .use({ grp: arrowHead, x: r * Math.cos(dw), y: r * Math.sin(dw), w: (!outside && dw > 0 || outside && dw < 0 ? dw + Math.PI / 2 - bw / 2 : dw - Math.PI / 2 + bw / 2) }) + .end() + .ins((g) => label && this.drawLabel(g)); + } +}); + +/** +* Origin symbol +* @constructor +* @returns {object} g2 +* @param {object} - symbol arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @property {number} w - angle in radians. +* @example +* g2().view({cartesian:true}) + * .origin({x:10,y:10}) +*/ +g2.prototype.origin = function (args = {}) { return this.addCommand({ c: 'origin', a: args }); } +g2.prototype.origin.prototype = g2.mix(g2.prototype.nod.prototype, { + lbloc: 'sw', + g2() { + const { x, y, w, ls = '#000', lw = 1 } = this; + return g2() + .beg({ x, y, w, ls }) + .vec({ x1: 0, y1: 0, x2: 40, y2: 0, lw, fs: '#ccc' }) + .vec({ x1: 0, y1: 0, x2: 0, y2: 40, lw, fs: '#ccc' }) + .cir({ x: 0, y: 0, r: lw + 1, fs: '#ccc' }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}); + +g2.prototype.ply.prototype = g2.mix(g2.labelIfc, g2.markIfc, { + get isSolid() { return this.closed && this.fs && this.fs !== 'transparent'; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false; }, + // get len() { + // let len_itr = 0; + // let last_pt = {x:0,y:0}; + // g2.pntItrOf(this.pts).map(pt => { + // len_itr += Math.hypot(pt.x-last_pt.x, pt.y-last_pt.y); + // last_pt = pt; + // }); + // return len_itr; + // }, + pointAt(loc) { + const t = loc === "beg" ? 0 + : loc === "end" ? 1 + : (loc + 0 === loc) ? loc // numerical arg .. + : 0.5, // 'mid' .. + pitr = g2.pntItrOf(this.pts), + pts = [], + len = []; + + for (let itr = 0; itr < pitr.len; itr++) { + const next = pitr((itr + 1) % pitr.len); + pts.push(pitr(itr)); + len.push(Math.hypot( + next.x - pitr(itr).x, + next.y - pitr(itr).y)); + } + this.closed || len.pop(); + const { t2, x, y, dx, dy } = (() => { + const target = t * len.reduce((a, b) => a + b); + for (let itr = 0, tmp = 0; itr < pts.length; itr++) { + tmp += len[itr]; + const next = pitr(itr + 1).x ? pitr(itr + 1) : pitr(0); + if (tmp >= target) { + return { + t2: 1 - (tmp - target) / len[itr], + x: pts[itr].x, + y: pts[itr].y, + dx: next.x - pts[itr].x, + dy: next.y - pts[itr].y + } + } + } + })(); + const len2 = Math.hypot(dx, dy); + return { + x: (this.x || 0) + x + dx * t2, + y: (this.y || 0) + y + dy * t2, + nx: len2 ? dy / len2 : 1, + ny: len2 ? dx / len2 : 0, + }; + }, + hit({ x, y, eps }) { + return this.isSolid ? g2.isPntInPly({ x: x - this.x, y: y - this.y }, this, eps) // translational transformation only .. at current .. ! + : g2.isPntOnPly({ x: x - this.x, y: y - this.y }, this, eps); + }, + drag({ dx, dy }) { this.x += dx; this.y += dy; }, + get g2() { + const e = g2(); + this.label && e.ins(e => this.drawLabel(e)); + this.mark && e.ins(e => this.drawMark(e, this.closed)); + return () => g2().ply(g2.flatten(this)).ins(e); + } +}); + +g2.prototype.use.prototype = { + // p vector notation ! + get p() { return { x: this.x, y: this.y }; }, // relevant if 'p' is *not* explicite given. + get x() { return Object.getOwnPropertyDescriptor(this, 'p') ? this.p.x : 0; }, + get y() { return Object.getOwnPropertyDescriptor(this, 'p') ? this.p.y : 0; }, + set x(q) { if (Object.getOwnPropertyDescriptor(this, 'p')) this.p.x = q; }, + set y(q) { if (Object.getOwnPropertyDescriptor(this, 'p')) this.p.y = q; }, + + isSolid: false, + /* + hit(at) { + for (const cmd of this.grp.commands) { + if (cmd.a.hit && cmd.a.hit(at)) + return true; + } + return false; + }, + + pointAt: g2.prototype.cir.prototype.pointAt, + */ +}; +// complex macros / add prototypes to argument objects + +/** +* Draw spline by points. +* Implementing a centripetal Catmull-Rom spline (thus avoiding cusps and self-intersections). +* Using iterator function for getting points from array by index. +* It must return current point object {x,y} or object {done:true}. +* Default iterator expects sequence of x/y-coordinates as a flat array [x,y,...], +* array of [[x,y],...] arrays or array of [{x,y},...] objects. +* @see https://pomax.github.io/bezierinfo +* @see https://de.wikipedia.org/wiki/Kubisch_Hermitescher_Spline +* @method +* @returns {object} g2 +* @param {object} - spline arguments object. +* @property {object[] | number[][] | number[]} pts - array of points. +* @property {bool} [closed=false] - closed spline. +* @example +* g2().spline({pts:[100,50,50,150,150,150,100,50]}) +*/ +g2.prototype.spline = function spline({ pts, closed, x, y, w }) { + arguments[0]._itr = g2.pntItrOf(pts); + return this.addCommand({ c: 'spline', a: arguments[0] }); +} +g2.prototype.spline.prototype = g2.mix(g2.prototype.ply.prototype, { + g2: function () { + let { pts, closed, x, y, w, ls, lw, fs, sh } = this, itr = this._itr, gbez; + if (itr) { + let b = [], i, n = itr.len, + p1, p2, p3, p4, d1, d2, d3, + d1d2, d2d3, scl2, scl3, + den2, den3, istrf = x || y || w; + + gbez = g2(); + if (istrf) gbez.beg({ x, y, w }); + gbez.p().m(itr(0)); + for (let i = 0; i < (closed ? n : n - 1); i++) { + if (i === 0) { + p1 = closed ? itr(n - 1) : { x: 2 * itr(0).x - itr(1).x, y: 2 * itr(0).y - itr(1).y }; + p2 = itr(0); + p3 = itr(1); + p4 = n === 2 ? (closed ? itr(0) : { x: 2 * itr(1).x - itr(0).x, y: 2 * itr(1).y - itr(0).y }) : itr(2); + d1 = Math.max(Math.hypot(p2.x - p1.x, p2.y - p1.y), Number.EPSILON); // don't allow .. + d2 = Math.max(Math.hypot(p3.x - p2.x, p3.y - p2.y), Number.EPSILON); // zero point distances .. + } else { + p1 = p2; + p2 = p3; + p3 = p4; + p4 = (i === n - 2) ? (closed ? itr(0) : { x: 2 * itr(n - 1).x - itr(n - 2).x, y: 2 * itr(n - 1).y - itr(n - 2).y }) + : (i === n - 1) ? itr(1) + : itr(i + 2); + d1 = d2; + d2 = d3; + } + d3 = Math.max(Math.hypot(p4.x - p3.x, p4.y - p3.y), Number.EPSILON); + d1d2 = Math.sqrt(d1 * d2), d2d3 = Math.sqrt(d2 * d3), + scl2 = 2 * d1 + 3 * d1d2 + d2, + scl3 = 2 * d3 + 3 * d2d3 + d2, + den2 = 3 * (d1 + d1d2), + den3 = 3 * (d3 + d2d3); + gbez.c({ + x: p3.x, y: p3.y, + x1: (-d2 * p1.x + scl2 * p2.x + d1 * p3.x) / den2, + y1: (-d2 * p1.y + scl2 * p2.y + d1 * p3.y) / den2, + x2: (-d2 * p4.x + scl3 * p3.x + d3 * p2.x) / den3, + y2: (-d2 * p4.y + scl3 * p3.y + d3 * p2.y) / den3 + }); + } + gbez.c(closed ? { x: itr(0).x, y: itr(0).y } : { x: itr(n - 1).x, y: itr(n - 1).y }) + if (closed) gbez.z(); + gbez.drw({ ls, lw, fs, sh }); + if (istrf) gbez.end(); + } + return gbez; + } +}); + +"use strict" + +/** + * g2.mec (c) 2013-18 Stefan Goessner + * @author Stefan Goessner + * @license MIT License + * @requires g2.core.js + * @requires g2.ext.js + * @typedef {g2} + * @description Mechanical extensions. (Requires cartesian coordinates) + * @returns {g2} + */ + +var g2 = g2 || { prototype:{} }; // for jsdoc only ... + +/** + * Draw slider. + * @method + * @returns {object} g2 + * @param {object} - slider arguments object. + * @property {number} x - start x coordinate. + * @property {number} y - start y coordinate. + * @property {number} [b=32] - slider breadth. + * @property {number} [h=16] - slider height. + * @property {number} [w=0] - rotation. + * @example + * g2().slider({x:150,y:75,w:Math.PI/4,b:64,h:32}) + */ +g2.prototype.slider = function () { return this.addCommand({c:'slider',a:arguments[0]}); } +g2.prototype.slider.prototype = g2.mix(g2.prototype.rec.prototype,{ + g2() { + const args = Object.assign({b:32,h:16,fs:'@linkfill'}, this); + return g2() + .beg({x:args.x,y:args.y,w:args.w,fs:args.fs}) + .rec({x:-args.b/2,y:-args.h/2,b:args.b,h:args.h}) + .end(); + } +}) + +/** + * Draw linear spring + * @method + * @returns {object} g2 + * @param {object} - linear spring arguments object. + * @property {number} x1 - start x coordinate. + * @property {number} y1 - start y coordinate. + * @property {number} x2 - end x coordinate. + * @property {number} y2 - end y coordinate. + * @property {number} [h=16] Spring height. + * @example + * g2().spring({x1:50,y1:100,x2:200,y2:75}) + */ +g2.prototype.spring = function () { return this.addCommand({c:'spring',a:arguments[0]}); } +g2.prototype.spring.prototype = g2.mix(g2.prototype.lin.prototype,{ + g2() { + const args = Object.assign({h:16}, this); + const len = Math.hypot(args.x2-args.x1, args.y2-args.y1); + const xm = (args.x2+args.x1)/2; + const ym = (args.y2+args.y1)/2; + const h = args.h; + const ux = (args.x2-args.x1)/len; + const uy = (args.y2-args.y1)/len; + return g2() + .p() + .m({x:args.x1,y:args.y1}) + .l({x:xm-ux*h/2,y:ym-uy*h/2}) + .l({x:xm+(-ux/6+uy/2)*h,y:ym+(-uy/6-ux/2)*h}) + .l({x:xm+( ux/6-uy/2)*h,y:ym+( uy/6+ux/2)*h}) + .l({x:xm+ux*h/2,y:ym+uy*h/2}) + .l({x:args.x2,y:args.y2}) + .stroke(Object.assign({}, {ls:'@nodcolor'},this,{fs:'transparent',lc:'round',lj:'round'})); + } +}) + +/** + * Draw line with centered square damper symbol. + * @method + * @returns {object} g2 + * @param {object} - damper arguments object. + * @property {number} x1 - start x coordinate. + * @property {number} y1 - start y coordinate. + * @property {number} x2 - end x coordinate. + * @property {number} y2 - end y coordinate. + * @property {number} [h=16] - damper height. + * * g2().damper({x1:60,y1:120,x2:200,y2:75}) + */ +g2.prototype.damper = function () { return this.addCommand({c:'damper',a:arguments[0]}); } +g2.prototype.damper.prototype = g2.mix(g2.prototype.lin.prototype,{ + g2() { + const args = Object.assign({h:16}, this); + const len = Math.hypot(args.x2-args.x1, args.y2-args.y1); + const xm = (args.x2+args.x1)/2; + const ym = (args.y2+args.y1)/2; + const h = args.h; + const ux = (args.x2-args.x1)/len; + const uy = (args.y2-args.y1)/len; + return g2() + .p() + .m({x:args.x1,y:args.y1}) + .l({x:xm-ux*h/2,y:ym-uy*h/2}) + .m({x:xm+( ux-uy)*h/2,y:ym+( uy+ux)*h/2}) + .l({x:xm+(-ux-uy)*h/2,y:ym+(-uy+ux)*h/2}) + .l({x:xm+(-ux+uy)*h/2,y:ym+(-uy-ux)*h/2}) + .l({x:xm+( ux+uy)*h/2,y:ym+( uy-ux)*h/2}) + .m({x:xm,y:ym}) + .l({x:args.x2,y:args.y2}) + .stroke(Object.assign({}, {ls:'@nodcolor'},this,{fs:'transparent',lc:'round',lj:'round'})); + } +}) + +/** + * Draw polygonial link. + * @method + * @returns {object} g2 + * @param {object} - link arguments object. + * @property {object[] | number[][] | number[]} pts - array of points. + * @property {bool} [closed = false] - closed link. + * @property {number} x - start x coordinate. + * @property {number} y - start y coordinate. + * @property {number} [w=0] - angle. + * @example + * let A = {x:50,y:25},B = {x:150,y:25}, + * C = {x:50,y:75}, D = {x:100,y:75}, + * E = {x:50,y:125}; + * g2().view({cartesian:true}) + * .link({pts:[A,B,E,A,D,C]}) + */ +g2.prototype.link = function () { return this.addCommand({c:'link',a:arguments[0]}); } +g2.prototype.link.prototype = g2.mix(g2.prototype.ply.prototype,{ + g2() { + const args = Object.assign({ls:'@linkcolor',fs:'transparent'}, this); + return g2().ply(Object.assign({}, this, {closed:true,ls:args.ls,fs:args.fs,lw:7,lc:'round',lj:'round'})); + } +}) + +/** + * Draw alternate glossy polygonial link. + * @method + * @returns {object} g2 + * @param {object} - link2 arguments object. + * @property {object[] | number[][] | number[]} pts - array of points. + * @property {bool} [closed = false] - closed link. + * @property {number} x - start x coordinate. + * @property {number} y - start y coordinate. + * @property {number} [w=0] - angle. + * @example + * let A = {x:50,y:25},B = {x:150,y:25}, + * C = {x:50,y:75}, D = {x:100,y:75}, + * E = {x:50,y:125}; + * g2().view({cartesian:true}) + * .link({pts:[A,B,E,A,D,C]}) + */ +g2.prototype.link2 = function () { return this.addCommand({c:'link2',a:arguments[0]}); } +g2.prototype.link2.prototype = g2.mix(g2.prototype.ply.prototype,{ + g2() { + return g2() + .ply(Object.assign({closed:true,ls:'@nodcolor',fs:'transparent',lw:7,lc:'round',lj:'round'},this)) + .ply(Object.assign({closed:true,ls:'@nodfill2',fs:'transparent',lw:4.5,lc:'round',lj:'round'},this)) + .ply(Object.assign({closed:true,ls:'@nodfill',fs:'transparent',lw:2,lc:'round',lj:'round'},this)); + } +}) + +/** + * Draw polygonial beam. + * @method + * @returns {object} g2 + * @param {object} - beam arguments object. + * @property {object[] | number[][] | number[]} pts - array of points. + * @property {number} x - start x coordinate. + * @property {number} y - start y coordinate. + * @property {number} [w=0] - angle. + * @example + * g2().view({cartesian}) + * .beam({pts:[[200,125][50,125][50,50][200,50]]}) + */ +g2.prototype.beam = function () { return this.addCommand({c:'beam',a:arguments[0]}); } +g2.prototype.beam.prototype = g2.mix(g2.prototype.ply.prototype,{ + g2() { + return g2().ply(Object.assign({closed:false,ls:'@linkcolor',fs:'transparent',lw:7,lc:'round',lj:'round'},this)); + } +}) + +/** + * Draw alternate glossy polygonial beam. + * @method + * @returns {object} g2 + * @param {object} - beam2 arguments object. + * @property {object[] | number[][] | number[]} pts - array of points. + * @property {number} x - start x coordinate. + * @property {number} y - start y coordinate. + * @property {number} [w=0] - angle. + * @example + * g2().view({cartesian}) + * .beam2({pts:[[200,125][50,125][50,50][200,50]]}) + */ +g2.prototype.beam2 = function () { return this.addCommand({c:'beam2',a:arguments[0]}); } +g2.prototype.beam2.prototype = g2.mix(g2.prototype.ply.prototype,{ + g2() { + return g2() + .ply(Object.assign({closed:false,ls:'@nodcolor',fs:'transparent',lw:7,lc:'round',lj:'round'},this)) + .ply(Object.assign({closed:false,ls:'@nodfill2',fs:'transparent',lw:4.5,lc:'round',lj:'round'},this)) + .ply(Object.assign({closed:false,ls:'@nodfill',fs:'transparent',lw:2,lc:'round',lj:'round'},this)); + } +}) + +/** + * Draw bar. + * @method + * @returns {object} g2 + * @param {object} - bar arguments object. + * @property {number} x1 - start x coordinate. + * @property {number} y1 - start y coordinate. + * @property {number} x2 - end x coordinate. + * @property {number} y2 - end y coordinate. + * @example + * g2().bar({x1:50,y1:20,x2:250,y2:120}) + */ +g2.prototype.bar = function () { return this.addCommand({c:'bar',a:arguments[0]}); } +g2.prototype.bar.prototype = g2.mix(g2.prototype.lin.prototype,{ + g2() { + return g2().lin(Object.assign({ls:'@linkcolor',lw:6,lc:'round'},this)); + } +}) + +/** + * Draw alternate glossy bar. + * @method + * @returns {object} g2 + * @param {object} - bar2 arguments object. + * @property {number} x1 - start x coordinate. + * @property {number} y1 - start y coordinate. + * @property {number} x2 - end x coordinate. + * @property {number} y2 - end y coordinate. + * @example + * g2().bar2({x1:50,y1:20,x2:250,y2:120}) + */ +g2.prototype.bar2 = function () { return this.addCommand({c:'bar2',a:arguments[0]}); } +g2.prototype.bar2.prototype = g2.mix(g2.prototype.lin.prototype,{ + g2() { + const args = Object.assign({}, this); + return g2() + .lin({x1:args.x1,y1:args.y1,x2:args.x2,y2:args.y2,ls:'@nodcolor',lw:7,lc:'round'}) + .lin({x1:args.x1,y1:args.y1,x2:args.x2,y2:args.y2,ls:'@nodfill2',lw:4.5,lc:'round'}) + .lin({x1:args.x1,y1:args.y1,x2:args.x2,y2:args.y2,ls:'@nodfill',lw:2,lc:'round'}); + } +}) + +/** + * Draw pulley. + * @method + * @returns {object} g2 + * @param {object} - pulley arguments object. + * @property {number} x - x-value center. + * @property {number} y - y-value center. + * @property {number} r - radius. + * @property {number} w - angle. + * @example + * g2().pulley({x:100,y:75,r:50}) + */ +g2.prototype.pulley = function () { return this.addCommand({c:'pulley',a:arguments[0]}); } +g2.prototype.pulley.prototype = g2.mix(g2.prototype.cir.prototype,{ + g2() { + const args = Object.assign({}, this); + return g2() + .beg({x:args.x,y:args.y,w:args.w}) + .cir({x:0,y:0,r:args.r,ls:'@nodcolor',fs:'#e6e6e6',lw:1}) + .cir({x:0,y:0,r:args.r-5,ls:'@nodcolor',fs:'#e6e6e6',lw:1}) + .cir({x:0,y:0,r:args.r-6,ls:'#8e8e8e',fs:'transparent',lw:2}) + .cir({x:0,y:0,r:args.r-8,ls:'#aeaeae',fs:'transparent',lw:2}) + .cir({x:0,y:0,r:args.r-10,ls:'#cecece',fs:'transparent',lw:2}) + .end(); + } +}) + +/** + * Draw alternate pulley. + * @method + * @returns {object} g2 + * @param {object} - pulley2 arguments object. + * @property {number} x - x-value center. + * @property {number} y - y-value center. + * @property {number} r - radius. + * @property {number} w - angle. + * @example + * g2().pulley2({x:50,y:30,r:25}) + */ +g2.prototype.pulley2 = function () { return this.addCommand({c:'pulley2',a:arguments[0]}); } +g2.prototype.pulley2.prototype = g2.mix(g2.prototype.cir.prototype,{ + g2() { + const args = Object.assign({}, this); + return g2() + .beg({x:args.x,y:args.y,w:args.w}) + .bar2({x1:0,y1:-args.r+4,x2:0,y2:args.r-4}) + .bar2({x1:-args.r+4,y1:0,x2:args.r-4,y2:0}) + .cir({x:0,y:0,r:args.r-2.5,ls:'#e6e6e6',fs:'transparent',lw:5}) + .cir({x:0,y:0,r:args.r,ls:'@nodcolor',fs:'transparent',lw:1}) + .cir({x:0,y:0,r:args.r-5,ls:'@nodcolor',fs:'transparent',lw:1}) + .end(); + } +}) + +/** + * Draw rope. Amount of pulley radii must be greater than 10 units. They are forced to zero otherwise. + * @method + * @returns {object} g2 + * @param {object} - rope arguments object. + * @property {object | number} p1 - starting point or Coordinate. + * @property {object | number} p2 - end point or Coordinate. + * @property {number} r - radius of parent element. + * @example + * let A = {x:50,y:30}, B = {x:200,y:75}; + * g2().view({cartesian:true}) + * .pulley({...A,r:20}) + * .pulley2({...B,r:40}) + * .rope({p1:A,r1:20,p2:B,r2:40}) + */ +g2.prototype.rope = function () { return this.addCommand({c:'rope',a:arguments[0]}); } +g2.prototype.rope.prototype = g2.mix(g2.prototype.lin.prototype,{ + g2() { + const args = Object.assign({w:0}, this); + let x1 = 'p1' in args ? args.p1.x + : 'x1' in args ? args.x1 + : 'x' in args ? args.x + : 0; + let y1 = 'p1' in args ? args.p1.y + : 'y1' in args ? args.y1 + : 'y' in args ? args.y + : 0; + let x2 = 'p2' in args ? args.p2.x + : 'x2' in args ? args.x2 + : 'dx' in args ? (x1 + args.dx) + : 'r' in args ? x1 + args.r*Math.cos(args.w) + : x1+10; + let y2 = 'p2' in args ? args.p2.y + : 'y2' in args ? args.y2 + : 'dy' in args ? (y1 + args.dy) + : 'r' in args ? y1 + args.r*Math.sin(args.w) + : y1; + let Rmin = 10; + let R1 = args.r1 > Rmin ? args.r1 - 2.5 + : args.r1 <-Rmin ? args.r1 + 2.5 + : 0; + let R2 = args.r2 > Rmin ? args.r2 - 2.5 + : args.r2 < Rmin ? args.r2 + 2.5 + : 0; + let dx = x2-x1, dy = y2-y1, dd = dx**2 + dy**2; + let R12 = R1 + R2, l = Math.sqrt(dd - R12**2); + let cpsi = (R12*dx + l*dy)/dd; + let spsi = (R12*dy - l*dx)/dd; + x1 = x1 + cpsi*R1, + y1 = y1 + spsi*R1, + x2 = x2 - cpsi*R2, + y2 = y2 - spsi*R2; + return g2().lin({x1:x1,x2:x2,y1:y1,y2:y2,ls:'#888',lw:4}); + } +}) + + +/** + * Polygon ground. + * @method + * @returns {object} g2 + * @param {object} - ground arguments object. + * @property {object[] | number[][] | number[]} pts - array of points. + * @property {bool} [closed=false] - closed polygon. + * @property {number} [h=4] - ground shade line width. + * @property {string} [pos=right] - ground shade position ['left','right']. + * @example + * g2().ground({pts:[25,25,25,75,75,75,75,25,125,25],pos:'left'}) + */ +g2.prototype.ground = function () { return this.addCommand({c:'ground',a:arguments[0]}); } +g2.prototype.ground.prototype = g2.mix(g2.prototype.ply.prototype,{ + g2() { + const args = Object.assign({h:4}, this); // , {closed: this.closed || false}); + const itr = g2.pntItrOf(args.pts); + let pn, en, lam, i; + let pp = itr(i=0); + let p0 = pp; + let h = args.h; + let p = itr(++i); + let dx = p.x - pp.x; + let dy = p.y - pp.y; + let len = Math.hypot(dx,dy) || 1; + let ep = {x:dx/len,y:dy/len}; + let e0 = ep; + let eq = [p0]; + let sign = args.pos === 'left' ? 1 : -1; + for (pn = itr(++i); i < itr.len; pn = itr(++i)) { + dx = pn.x - p.x; dy = pn.y - p.y; len = Math.hypot(dx,dy) || 1; + len = Math.hypot(dx,dy) || 1; + en = {x:dx/len,y:dy/len}; + lam = (1 - en.x*ep.x - en.y*ep.y) / (ep.y*en.x - ep.x*en.y); + eq.push({x:p.x+sign*(h+1)*(lam*ep.x - ep.y), y:p.y+sign*(h+1)*(lam*ep.y + ep.x)}); + ep = en; + pp = p; + p = pn; + } + if (args.closed) { + dx = p0.x-p.x; dy = p0.y-p.y; len = Math.hypot(dx,dy) || 1; + en = {x:dx/len,y:dy/len}; + lam = (1 - en.x*ep.x - en.y*ep.y) / (ep.y*en.x - ep.x*en.y); + eq.push({x:p.x+sign*(h+1)*(lam*ep.x - ep.y), y:p.y+sign*(h+1)*(lam*ep.y + ep.x)}); + lam = (1 - e0.x*en.x - e0.y*en.y) / (en.y*e0.x - en.x*e0.y); + eq[0] = {x:p0.x+sign*(h+1)*(-lam*e0.x - e0.y), y:p0.y+sign*(h+1)*(-lam*e0.y + e0.x)}; + } else { + eq[0] = {x:p0.x-sign*(h+1)*e0.y, y:p0.y+sign*(h+1)*e0.x}; + eq.push({x:p.x -sign*(h+1)*ep.y, y:p.y +sign*(h+1)*ep.x}); + } + return g2() + .beg({x:-0.5,y:-0.5,ls:'@linkcolor',lw:2,fs:'transparent',lc:'butt',lj:'miter'}) + .ply(Object.assign({}, args,{pts:eq,ls:'@nodfill2',lw:2*h})) + .ply(Object.assign({}, args)) + .end(); + } +}); + +/** + * Polygonial line load. The first and last point define the base line onto which + * the load is acting orthogonal. + * @method + * @returns {object} g2 + * @param {object} - load arguments object. + * @property {object[] | number[][] | number[]} pts - array of points. + * @property {number} w - angle of vectors. + * @property {number} spacing - spacing of the vectors drawn as a positive real number, interprete as
+ * * spacing < 1: spacing = 1/m with a partition of m.
+ * * spacing > 1: length of spacing. + */ +g2.prototype.load = function () { return this.addCommand({c:'load',a:arguments[0]}); } +g2.prototype.load.prototype = g2.mix(g2.prototype.ply.prototype,{ + g2() { + const args = Object.assign({ pointAt: this.pointAt, spacing: 20, w: -Math.PI/2 }, this); + const pitr = g2.pntItrOf(args.pts), startLoc = [], arr = []; + let arrLen = 0; + for (let itr = 0; itr < pitr.len ; itr++) { + arr.push(pitr(itr)); + } + if (arr[arr.length-1] !== arr[0]) { + arr.push(arr[0]); + } + for (let itr = 1; itr < arr.length; itr++) { + arrLen += Math.hypot(arr[itr].y-arr[itr-1].y,arr[itr].x-arr[itr-1].x); + } + for(let itr=0;itr*args.spacing < arrLen; itr++) { + startLoc.push((itr*args.spacing)/arrLen); + } + args.pts = arr; // for args.pointsAt(...)... + /*-----------------------------------stolen from g2.lib-----------------------------------*/ + function isPntInPly({x,y}) { + let match = 0; + for (let n=arr.length,i=0,pi=arr[i],pj=arr[n-1]; i= pi.y || y >= pj.y) + && (y <= pi.y || y <= pj.y) + && (x <= pi.x || x <= pj.x) + && pi.y !== pj.y + && (pi.x === pj.x || x <= pj.x + (y-pj.y)*(pi.x-pj.x)/(pi.y-pj.y))) { + match++; + } + } + return match%2 != 0; + }; + /*----------------------------------------------------------------------------------------*/ + return g2() + .ply({pts:args.pts,closed:true,ls:'transparent',fs:'@linkfill'}) + .ins(g => { + for (const pts of startLoc) { + let dist = (10*args.lw||10); // minimum distance a vector has to be + const {x,y} = args.pointAt(pts); + const t = { + x:x+Math.cos(args.w)*dist, + y:y+Math.sin(args.w)*dist + }; + if (isPntInPly(t,{pts:arr})) { + while(isPntInPly(t,{pts:arr})) { + dist++; + t.x = x+Math.cos(args.w)*dist, + t.y = y+Math.sin(args.w)*dist + }; + g.vec({ + x1:x, y1:y, + x2:t.x, y2:t.y, + ls: args.ls || "darkred", + lw: args.lw || 1 + }); + } + } + }); + } +}); + +/** + * g2.selector.js (c) 2018 Stefan Goessner + * @file selector for `g2` elements. + * @author Stefan Goessner + * @license MIT License + */ +/* jshint -W014 */ + +/** + * Extensions. + * (Requires cartesian coordinate system) + * @namespace + */ +var g2 = g2 || { prototype:{} }; // for jsdoc only ... + +// extend prototypes for argument objects +g2.selector = function(evt) { + if (this instanceof g2.selector) { + this.selection = false; + this.evt = evt; // sharing evt object with canvasInteractor as owner ... important ! + return this; + } + return g2.selector.apply(Object.create(g2.selector.prototype), arguments); +}; +g2.handler.factory.push((ctx) => ctx instanceof g2.selector ? ctx : false); + +// g2.selector.state = ['NONE','OVER','DRAG','OVER+DRAG','EDIT','OVER+EDIT']; + +g2.selector.prototype = { + init(grp) { return true; }, + exe(commands) { + for (let elm=false, i=commands.length; i && !elm; i--) // stop after first hit .. starting from list end ! + elm = this.hit(commands[i-1].a) + }, + selectable(elm) { + return elm && elm.draggable && elm.hit; + }, + hit(elm) { + if (!this.evt.inside // pointer not inside of canvas .. + || !this.selectable(elm) ) // no selectable elm .. + return false; + + if (!elm.state && this.elementHit(elm) && elm.draggable) { // no mode + if (!this.selection || this.selection && !(this.selection.state & g2.DRAG)) { + if (this.selection) this.selection.state ^= g2.OVER; + this.selection = elm; + elm.state = g2.OVER; // enter OVER mode .. + this.evt.hit = true; + } + } + else if (elm.state & g2.DRAG) { // in DRAG mode + if (!this.evt.btn) // leave DRAG mode .. + this.elementDragEnd(elm); + } + else if (elm.state & g2.OVER) { // in OVER mode + if (!this.elementHit(elm)) { // leave OVER mode .. + elm.state ^= g2.OVER; + this.evt.hit = false; + this.selection = false; + } + else if (this.evt.btn) // enter DRAG mode + this.elementDragBeg(elm); + } + + return elm.state && elm; // we definitely have a valid elm here ... + }, // ... but only return it depending on its state. + elementDragBeg(elm) { + elm.state |= g2.DRAG; + if (elm.dragBeg) elm.dragBeg(e); + }, + elementDragEnd(elm) { + elm.state ^= (g2.OVER | g2.DRAG); + this.selection = false; + if (elm.dragEnd) elm.dragEnd(e); + }, + elementHit(elm) { + return elm.hit && elm.hit({x:this.evt.xusr,y:this.evt.yusr,eps:this.evt.eps}); + } +}; + +/** + * canvasInteractor.js (c) 2018 Stefan Goessner + * @file interaction manager for html `canvas`. + * @author Stefan Goessner + * @license MIT License + */ +/* jshint -W014 */ +// Managing multiple canvases per static interactor as singleton ... +// .. using a single requestAnimationFrame loop ! +const canvasInteractor = { + create() { + const o = Object.create(this.prototype); + o.constructor.apply(o,arguments); + return o; + }, + // global static tickTimer properties + fps: '?', + fpsOrigin: 0, + frames: 0, + rafid: 0, + instances: [], + // global static timer methods + tick(time) { + canvasInteractor.fpsCount(time); + for (const instance of canvasInteractor.instances) { + instance.notify('tick',{t:time,dt:(time-instance.t)/1000,dirty:instance.dirty}); // notify listeners .. + instance.t = time; + instance.dirty = false; + } + canvasInteractor.rafid = requestAnimationFrame(canvasInteractor.tick); // request next animation frame ... + }, + add(instance) { + canvasInteractor.instances.push(instance); + if (canvasInteractor.instances.length === 1) // first instance added ... + canvasInteractor.tick(canvasInteractor.fpsOrigin = performance.now()); + }, + remove(instance) { + canvasInteractor.instances.splice(canvasInteractor.instances.indexOf(instance),1); + if (canvasInteractor.instances.length === 0) // last instance removed ... + cancelAnimationFrame(canvasInteractor.rafid); + }, + fpsCount(time) { + if (time - canvasInteractor.fpsOrigin > 1000) { // one second interval reached ... + const fps = ~~(canvasInteractor.frames*1000/(time - canvasInteractor.fpsOrigin) + 0.5); // ~~ as Math.floor() + if (fps !== canvasInteractor.fps) + for (const instance of canvasInteractor.instances) + instance.notify('fps',canvasInteractor.fps=fps); + canvasInteractor.fpsOrigin = time; + canvasInteractor.frames = 0; + } + canvasInteractor.frames++; + }, + + prototype: { + constructor(ctx, {x,y,scl,cartesian}) { + // canvas interaction properties + this.ctx = ctx; + this.view = {x:x||0,y:y||0,scl:scl||1,cartesian:cartesian||false}; + this.evt = { + type: false, + basetype: false, + x: -2, y:-2, + xi: 0, yi:0, + dx: 0, dy: 0, + btn: 0, + xbtn: 0, ybtn: 0, + xusr: -2, yusr: -2, + dxusr: 0, dyusr: 0, + delta: 0, + inside: false, + hit: false, // something hit by pointer ... + dscl: 1, // for zooming ... + eps: 5 // some pixel tolerance ... + }; + this.dirty = true; + // event handler registration + const canvas = ctx.canvas; + canvas.addEventListener("pointermove", this, false); + canvas.addEventListener("pointerdown", this, false); + canvas.addEventListener("pointerup", this, false); + canvas.addEventListener("pointerenter", this, false); + canvas.addEventListener("pointerleave", this, false); + canvas.addEventListener("wheel", this, false); + canvas.addEventListener("pointercancel", this, false); + }, + deinit() { + const canvas = this.ctx.canvas; + + canvas.removeEventListener("pointermove", this, false); + canvas.removeEventListener("pointerdown", this, false); + canvas.removeEventListener("pointerup", this, false); + canvas.removeEventListener("pointerenter", this, false); + canvas.removeEventListener("pointerleave", this, false); + canvas.removeEventListener("wheel", this, false); + canvas.removeEventListener("pointercancel", this, false); + + this.endTimer(); + + delete this.signals; + delete this.evt; + delete this.ctx; + + return this; + }, + // canvas interaction interface + handleEvent(e) { + if (e.type in this && (e.isPrimary || e.type === 'wheel')) { // can I handle events of type e.type .. ? + const bbox = e.target.getBoundingClientRect && e.target.getBoundingClientRect() || {left:0, top:0}, + x = e.clientX - Math.floor(bbox.left), + y = e.clientY - Math.floor(bbox.top), + btn = e.buttons !== undefined ? e.buttons : e.button || e.which; + + this.evt.type = e.type; + this.evt.basetype = e.type; // obsolete now ... ? + this.evt.xi = this.evt.x; // interim coordinates ... + this.evt.yi = this.evt.y; // ... of previous event. + this.evt.dx = this.evt.dy = 0; + this.evt.x = x; + this.evt.y = this.view.cartesian ? this.ctx.canvas.height - y : y; + this.evt.xusr = (this.evt.x - this.view.x)/this.view.scl; + this.evt.yusr = (this.evt.y - this.view.y)/this.view.scl; + this.evt.dxusr = this.evt.dyusr = 0; + this.evt.dbtn = btn - this.evt.btn; + this.evt.btn = btn; + this.evt.delta = Math.max(-1,Math.min(1,e.deltaY||e.wheelDelta)) || 0; + + if (this.isDefaultPreventer(e.type)) + e.preventDefault(); + this[e.type](); // handle specific event .. ! + this.notify(this.evt.type,this.evt); // .. tell the world .. ! + } + else + console.log(e) + }, + pointermove() { + this.evt.dx = this.evt.x - this.evt.xi; + this.evt.dy = this.evt.y - this.evt.yi; + if (this.evt.btn === 1) { // pointerdown state ... + this.evt.dxusr = this.evt.dx/this.view.scl; // correct usr coordinates ... + this.evt.dyusr = this.evt.dy/this.view.scl; + this.evt.xusr -= this.evt.dxusr; // correct usr coordinates ... + this.evt.yusr -= this.evt.dyusr; + if (!this.evt.hit) { // let outer app perform panning ... + this.evt.type = 'pan'; + } + else + this.evt.type = 'drag'; + } + // view, geometry or graphics might be modified ... + this.dirty = true; + }, + pointerdown() { + this.evt.xbtn = this.evt.x; + this.evt.ybtn = this.evt.y; + }, + pointerup() { + this.evt.type = this.evt.x===this.evt.xbtn && this.evt.y===this.evt.ybtn ? 'click' : 'pointerup'; + this.evt.xbtn = this.evt.x; + this.evt.ybtn = this.evt.y; + }, + pointerleave() { + this.evt.inside = false; + }, + pointerenter() { + this.evt.inside = true; + }, + wheel() { + this.evt.dscl = this.evt.delta>0?8/10:10/8; + this.evt.eps /= this.evt.dscl; + this.dirty = true; + }, + isDefaultPreventer(type) { + return ['pointermove','pointerdown','pointerup','wheel'].includes(type); + }, + pntToUsr: function(p) { + let vw = this.view; + p.x = (p.x - vw.x)/vw.scl; + p.y = (p.y - vw.y)/vw.scl; + return p; + }, + // tickTimer interface + startTimer() { // shouldn't there be a global startTimer method ? + canvasInteractor.add(this); + this.notify('timerStart',this); // notify potential listeners .. + return this; + }, + endTimer() { + this.notify('timerEnd',this.t/1000); // notify potential listeners .. + canvasInteractor.remove(this); + return this; + }, + // observable interface + notify(key,val) { + if (this.signals && this.signals[key]) + for (let hdl of this.signals[key]) + hdl(val); + return this; + }, + on(key,handler) { // support array of keys as first argument. + if (Array.isArray(key)) + for (let k of key) + this.on(k,handler); + else + ((this.signals || (this.signals = {})) && this.signals[key] || (this.signals[key]=[])).push(handler); + + return this; + }, + remove(key,handler) { + const idx = (this.signals && this.signals[key]) ? this.signals[key].indexOf(handler) : -1; + if (idx >= 0) + this.signals[key].splice(idx,1); + } + } +}; + +/** + * g2.element.js (c) 2019-20 Stefan Goessner + * @license MIT License + */ +"use strict"; + +class G2Element extends HTMLElement { + static get observedAttributes() { + return ['width', 'height','cartesian','grid', 'x0', 'y0', 'darkmode', 'interactive','background']; + } + + constructor() { + super(); + this._root = this.attachShadow({ mode:'open' }); + } + + get width() { return +this.getAttribute('width') || 301; } + set width(q) { if (q) this.setAttribute('width',q); } + get height() { return +this.getAttribute('height') || 201; } + set height(q) { if (q) this.setAttribute('height',q); } + get x0() { return (+this.getAttribute('x0')) || 0; } + set x0(q) { if (q) this.setAttribute('x0',q); } + get y0() { return (+this.getAttribute('y0')) || 0; } + set y0(q) { if (q) this.setAttribute('y0',q); } + get cartesian() { return this.hasAttribute('cartesian'); } + set cartesian(q) { q ? this.setAttribute('cartesian','') : this.removeAttribute('cartesian'); } + get grid() { return this.hasAttribute('grid') || false; } + set grid(q) { q ? this.setAttribute('grid','') : this.removeAttribute('grid'); } + get darkmode() { return this.hasAttribute('darkmode') || false; } + get interactive() { return this.hasAttribute('interactive') || false; } + get background() { return this.getAttribute('background') || false; } + get g() { return this._g; } + get canvas() { return this._ctx && this._ctx.canvas || false } + get readyState() { return !!this._g; } + + update() { + if (!this.interactive && this._g && this._ctx) + this._g.exe(this._ctx); + } + + init() { + const state = {x:this.x0,y:this.y0,cartesian:this.cartesian}; + // add shadow dom + this._root.innerHTML = G2Element.template({width:this.width,height:this.height,darkmode:this.darkmode}); + // cache elements of shadow dom + this._logview = this._root.getElementById('logview'); + // set up canvas context + this._ctx = this._root.getElementById('cnv').getContext('2d'); + // set up canvas background + if (this.background && G2Element[this.background]) + this._ctx.canvas.style.backgroundImage = 'url('+G2Element[this.background]+')'; + // set up canvas interactor + if (this.interactive) { + this._interactor = canvasInteractor.create(this._ctx, state); + this._selector = g2.selector(this._interactor.evt); + this._interactor.on('tick', e => this.ontick(e)) + .on('pan', e => this.onpan(e)) + .on('drag', e => this.ondrag(e)) + .on('wheel', e => this.onwheel(e)); + } + this._g = g2().clr().view(this.interactive && this._interactor.view || state); + if (this.grid) this._g.grid({color:this.darkmode?'#999':'#ccc'}); + this.initContent(this.innerHTML.trim(), e => this.log(e)); + if (this.interactive) + this._interactor.startTimer(); + else // no tick timer .. ! + this._g.exe(this._ctx); + this.dispatchEvent(new CustomEvent('init')); + } + initContent(content, onErr) { + if (!content) return; + try { + content = JSON.parse(content); // is valid JSON string ? + content = g2.io.parseGrp(content, 'main', onErr); // is valid g2 json string + if (content && content.commands) { // content is a valid g2 object. + this._g.commands.push(...content.commands); // inject commands of `main` group ... + } // ... into `_g` with a little brute force. + } + catch(e) { + const w = this.width, h = this.height, x0 = this.x0, y0 = this.y0, dx = w/20, dy = h/20, + x = q => q*w-x0, y = q => q*h-y0; + + content = g2({id:'main'}).stroke({"d":`M${x(0.05)},${y(0.05)} ${x(0.45)},${y(0.45)}M${x(0.55)},${y(0.55)} ${x(0.95)},${y(0.95)}M${x(0.05)},${y(0.95)} ${x(0.45)},${y(0.55)}M${x(0.55)},${y(0.45)} ${x(0.95)},${y(0.05)}`,lw:10,ls:"red",lc:"round"}); + this._g.ins(content); + onErr(e.message); + } + return !!content; + } + deinit() { + delete this._selector; + if (this.interactive) { + delete this._interactor.deinit(); + delete this._interactor; + delete this._selector; + } + delete this._g; + // delete cached data + delete this._ctx; + } + log(str) { + this._logview.innerHTML += str; + } + + ontick(e) { + this.dispatchEvent(new CustomEvent('tick')); + if (this.interactive) + this._g.exe(this._selector); + this._g.exe(this._ctx); + } + onpan(e) { + this._interactor.view.x = this.x0 += e.dx; + this._interactor.view.y = this.y0 += e.dy; + } + ondrag(e) { // only modify selected geometry here .. do not redraw .. ! + if (this._selector.selection && this._selector.selection.drag) { + this._selector.selection.drag({x:e.xusr,y:e.yusr,dx:e.dxusr,dy:e.dyusr,mode:'drag'}); + this.dispatchEvent(new CustomEvent('drag')); + } + } + onwheel(e) { + this._interactor.view.x = e.x + e.dscl*(this._interactor.view.x - e.x); + this._interactor.view.y = e.y + e.dscl*(this._interactor.view.y - e.y); + this._interactor.view.scl *= e.dscl; + } + + on(hdl,fn) { } + + // standard lifecycle callbacks + // https://developer.mozilla.org/en-US/docs/Web/Web_Components/Using_custom_elements + connectedCallback() { + this.init(); + } + disconnectedCallback() { + this.deinit(); + } + attributeChangedCallback(name, oldval, val) { + if (this._root && this._root.getElementById('cnv')) { + if (name === 'width') { // todo: preserve minimum width + this._root.getElementById('cnv').setAttribute('width',val); + this._root.querySelector('.status').style.width = val+'px'; + } + if (name === 'height') // todo: preserve minimum height + this._root.getElementById('cnv').setAttribute('height',val); + } + } + + static template({width,height,darkmode}) { +return ` + +
+
+

+
+` + } + + static get paper() { return " "; } +} +customElements.define('g-2', G2Element); + diff --git a/dist/g2.js b/dist/g2.js new file mode 100644 index 0000000..a2e0a02 --- /dev/null +++ b/dist/g2.js @@ -0,0 +1,91 @@ +"use strict"; +/** + * g2.core (c) 2013-20 Stefan Goessner + * @author Stefan Goessner + * @license MIT License + * @link https://github.com/goessner/g2 + * @typedef {g2} + * @param {object} [opts] Custom options object. + * @description Create a 2D graphics command queue object. Call without using 'new'. + * @returns {g2} + * @example + * const ctx = document.getElementById("c").getContext("2d"); + * g2() // Create 'g2' instance. + * .lin({x1:50,y1:50,x2:100,y2:100}) // Append ... + * .lin({x1:100,y1:100,x2:200,y2:50}) // ... commands. + * .exe(ctx); // Execute commands addressing canvas context. + */function g2(opts){let o=Object.create(g2.prototype);o.commands=[];if(opts)Object.assign(o,opts);return o}g2.prototype={clr(){return this.addCommand({c:"clr"})},view({scl:scl,x:x,y:y,cartesian:cartesian}){return this.addCommand({c:"view",a:arguments[0]})},grid({color:color,size:size}={}){return this.addCommand({c:"grid",a:arguments[0]})},cir({x:x,y:y,r:r,w:w}){return this.addCommand({c:"cir",a:arguments[0]})},ell({x:x,y:y,rx:rx,ry:ry,w:w,dw:dw,rot:rot}){return this.addCommand({c:"ell",a:arguments[0]})},arc({x:x,y:y,r:r,w:w,dw:dw}){return this.addCommand({c:"arc",a:arguments[0]})},rec({x:x,y:y,b:b,h:h}){return this.addCommand({c:"rec",a:arguments[0]})},lin({x1:x1,y1:y1,x2:x2,y2:y2}){return this.addCommand({c:"lin",a:arguments[0]})},ply({pts:pts,format:format,closed:closed,x:x,y:y,w:w}){arguments[0]._itr=format&&g2.pntIterator[format](pts)||g2.pntItrOf(pts);return this.addCommand({c:"ply",a:arguments[0]})},txt({str:str,x:x,y:y,w:w}){return this.addCommand({c:"txt",a:arguments[0]})},use({grp:grp,x:x,y:y,w:w,scl:scl}){if(grp&&grp!==this){if(typeof grp==="string")arguments[0].grp=g2.symbol[grp in g2.symbol?grp:"unknown"];this.addCommand({c:"use",a:arguments[0]})}return this},img({uri:uri,x:x,y:y,b:b,h:h,sx:sx,sy:sy,sb:sb,sh:sh,xoff:xoff,yoff:yoff,w:w,scl:scl}){return this.addCommand({c:"img",a:arguments[0]})},beg({x:x,y:y,w:w,scl:scl,matrix:matrix}={}){return this.addCommand({c:"beg",a:arguments[0]})},end(){let myBeg=1,findMyBeg=cmd=>{if(cmd.c==="beg")myBeg--;else if(cmd.c==="end")myBeg++;return myBeg===0};return g2.cmdIdxBy(this.commands,findMyBeg)!==false?this.addCommand({c:"end"}):this},p(){return this.addCommand({c:"p"})},z(){return this.addCommand({c:"z"})},m({x:x,y:y}){return this.addCommand({c:"m",a:arguments[0]})},l({x:x,y:y}){return this.addCommand({c:"l",a:arguments[0]})},q({x1:x1,y1:y1,x:x,y:y}){return this.addCommand({c:"q",a:arguments[0]})},c({x1:x1,y1:y1,x2:x2,y2:y2,x:x,y:y}){return this.addCommand({c:"c",a:arguments[0]})},a({dw:dw,x:x,y:y}){let prvcmd=this.commands[this.commands.length-1];g2.cpyProp(prvcmd.a,"x",arguments[0],"_xp");g2.cpyProp(prvcmd.a,"y",arguments[0],"_yp");return this.addCommand({c:"a",a:arguments[0]})},stroke({d:d}={}){return this.addCommand({c:"stroke",a:arguments[0]})},fill({d:d}={}){return this.addCommand({c:"fill",a:arguments[0]})},drw({d:d,lsh:lsh}={}){return this.addCommand({c:"drw",a:arguments[0]})},del(idx){this.commands.length=idx||0;return this},ins(arg){return typeof arg==="function"?arg(this)||this:typeof arg==="object"?(this.commands.push({a:arg}),this):this},exe(ctx){let handler=g2.handler(ctx);if(handler&&handler.init(this))handler.exe(this.commands);return this},addCommand({c:c,a:a}){if(a&&Object.getPrototypeOf(a)===Object.prototype){for(const key in a){if(!Object.getOwnPropertyDescriptor(a,key).get&&key[0]!=="_"&&typeof a[key]==="function"){Object.defineProperty(a,key,{get:a[key],enumerable:true,configurable:true,writabel:false})}if(typeof a[key]==="string"&&a[key][0]==="@"){const refidIdx=a[key].indexOf(".");const refid=refidIdx>0?a[key].substr(1,refidIdx-1):"";const refkey=refid?a[key].substr(refidIdx+1):"";const refcmd=refid?()=>this.commands.find(cmd=>cmd.a&&cmd.a.id===refid):undefined;if(refcmd)Object.defineProperty(a,key,{get:function(){const rc=refcmd();return rc&&refkey in rc.a?rc.a[refkey]:0},enumerable:true,configurable:true,writabel:false})}}if(g2.prototype[c].prototype)Object.setPrototypeOf(a,g2.prototype[c].prototype)}this.commands.push(arguments[0]);return this}};g2.defaultStyle={fs:"transparent",ls:"#000",lw:1,lc:"butt",lj:"miter",ld:[],ml:10,sh:[0,0],lsh:false,font:"14px serif",thal:"start",tval:"alphabetic"};g2.symbol={unknown:g2().cir({r:12,fs:"orange"}).txt({str:"?",thal:"center",tval:"middle",font:"bold 20pt serif"})};g2.handler=function(ctx){let hdl;for(let h of g2.handler.factory)if((hdl=h(ctx))!==false)return hdl;return false};g2.handler.factory=[];g2.pntIterator={"x,y":function(pts){function pitr(i){return{x:pts[2*i],y:pts[2*i+1]}}Object.defineProperty(pitr,"len",{get:()=>pts.length/2,enumerable:true,configurable:true,writabel:false});return pitr},"[x,y]":function(pts){function pitr(i){return pts[i]?{x:pts[i][0],y:pts[i][1]}:undefined}Object.defineProperty(pitr,"len",{get:()=>pts.length,enumerable:true,configurable:true,writabel:false});return pitr},"{x,y}":function(pts){function pitr(i){return pts[i]}Object.defineProperty(pitr,"len",{get:()=>pts.length,enumerable:true,configurable:true,writabel:false});return pitr}};g2.pntItrOf=function(pts){return!(pts&&pts.length)?undefined:typeof pts[0]==="number"?g2.pntIterator["x,y"](pts):Array.isArray(pts[0])&&pts[0].length>=2?g2.pntIterator["[x,y]"](pts):typeof pts[0]==="object"&&"x"in pts[0]&&"y"in pts[0]?g2.pntIterator["{x,y}"](pts):undefined};g2.cmdIdxBy=function(cmds,callbk){for(let i=cmds.length-1;i>=0;i--)if(callbk(cmds[i],i,cmds))return i;return false};g2.mix=function mix(...protos){let mixture={};for(const p of protos)mixture=Object.defineProperties(mixture,Object.getOwnPropertyDescriptors(p));return mixture};g2.cpyProp=function(from,fromKey,to,toKey){Object.defineProperty(to,toKey,Object.getOwnPropertyDescriptor(from,fromKey))};g2.canvasHdl=function(ctx){if(this instanceof g2.canvasHdl){if(ctx instanceof CanvasRenderingContext2D){this.ctx=ctx;this.cur=g2.defaultStyle;this.stack=[this.cur];this.matrix=[[1,0,0,1,.5,.5]];this.gridBase=2;this.gridExp=1;return this}else return null}return g2.canvasHdl.apply(Object.create(g2.canvasHdl.prototype),arguments)};g2.handler.factory.push(ctx=>ctx instanceof g2.canvasHdl?ctx:ctx instanceof CanvasRenderingContext2D?g2.canvasHdl(ctx):false);g2.canvasHdl.prototype={init(grp,style){this.stack.length=1;this.matrix.length=1;this.initStyle(style?Object.assign({},this.cur,style):this.cur);return true},async exe(commands){for(let cmd of commands){if(cmd.a&&cmd.a.g2){const cmds=cmd.a.g2().commands;if(cmds){this.exe(cmds);continue}}else if(cmd.a&&cmd.a.commands){this.exe(cmd.a.commands);continue}if(cmd.c&&this[cmd.c]){const rx=this[cmd.c](cmd.a);if(rx&&rx instanceof Promise){await rx}}}},view({x:x=0,y:y=0,scl:scl=1,cartesian:cartesian=false}){this.pushTrf(cartesian?[scl,0,0,-scl,x,this.ctx.canvas.height-1-y]:[scl,0,0,scl,x,y])},grid({color:color="#ccc",size:size}={}){let ctx=this.ctx,b=ctx.canvas.width,h=ctx.canvas.height,{x:x,y:y,scl:scl}=this.uniTrf,sz=size||this.gridSize(scl),xoff=x%sz,yoff=y%sz;ctx.save();ctx.setTransform(1,0,0,1,0,0);ctx.strokeStyle=color;ctx.lineWidth=1;ctx.beginPath();for(let x=xoff,nx=b+1;xNumber.EPSILON&&Math.abs(r)>Number.EPSILON){this.ctx.beginPath();this.ctx.arc(x,y,Math.abs(r),w,w+dw,dw<0);this.drw(arguments[0])}else if(Math.abs(dw)Number.EPSILON){const cw=Math.cos(w),sw=Math.sin(w);this.ctx.beginPath();this.ctx.moveTo(x-r*cw,y-r*sw);this.ctx.lineTo(x+r*cw,y+r*sw)}},ell({rx:rx,ry:ry,w:w=0,dw:dw=2*Math.PI,rot:rot=0}){const{x:x=0,y:y=0}=arguments[0].p!==undefined?arguments[0].p:arguments[0];this.ctx.beginPath();this.ctx.ellipse(x,y,Math.abs(rx),Math.abs(ry),rot,w,w+dw,dw<0);this.drw(arguments[0])},rec({b:b,h:h}){const{x:x=0,y:y=0}=arguments[0].p!==undefined?arguments[0].p:arguments[0];const tmp=this.setStyle(arguments[0]);this.ctx.fillRect(x,y,b,h);this.ctx.strokeRect(x,y,b,h);this.resetStyle(tmp)},lin(args){this.ctx.beginPath();this.ctx.moveTo(args.p1&&args.p1.x||args.x1||0,args.p1&&args.p1.y||args.y1||0);this.ctx.lineTo(args.p2&&args.p2.x||args.x2||0,args.p2&&args.p2.y||args.y2||0);this.stroke(args)},ply({pts:pts,closed:closed,w:w=0,_itr:_itr}){if(_itr&&_itr.len){const{x:x=0,y:y=0}=arguments[0].p!==undefined?arguments[0].p:arguments[0];let p,i,len=_itr.len,istrf=!!(x||y||w),cw,sw;if(istrf)this.setTrf([cw=w?Math.cos(w):1,sw=w?Math.sin(w):0,-sw,cw,x,y]);this.ctx.beginPath();this.ctx.moveTo((p=_itr(0)).x,p.y);for(i=1;i{const pimg=new Promise((resolve,reject)=>{let img=new Image;img.src=xuri;function error(err){img.removeEventListener("load",load);img=undefined;reject(err)}function load(){img.removeEventListener("error",error);resolve(img);img=undefined}img.addEventListener("error",error,{once:true});img.addEventListener("load",load,{once:true})});try{return await pimg}catch(err){if(xuri===this.errorImageStr){throw err}else{return await download(this.errorImageStr)}}};let img=this.images[uri];if(img!==undefined){return img instanceof Promise?await img:img}img=download(uri);this.images[uri]=img;try{img=await img}finally{this.images[uri]=img}return img},async img({uri:uri,x:x=0,y:y=0,b:b,h:h,sx:sx=0,sy:sy=0,sb:sb,sh:sh,xoff:xoff=0,yoff:yoff=0,w:w=0,scl:scl=1}){const img_=await this.loadImage(uri);this.ctx.save();const cart=this.isCartesian?-1:1;sb=sb||img_.width;b=b||img_.width;sh=sh||img_.height;h=(h||img_.height)*cart;yoff*=cart;w*=cart;y=this.isCartesian?-(y/scl)+sy:y/scl;const[cw,sw]=[Math.cos(w),Math.sin(w)];this.ctx.scale(scl,scl*cart);this.ctx.transform(cw,sw,-sw,cw,x/scl,y);this.ctx.drawImage(img_,sx,sy,sb,sh,xoff,yoff,b,h);this.ctx.restore()},use({grp:grp}){this.beg(arguments[0]);this.exe(grp.commands);this.end()},beg({w:w=0,scl:scl=1,matrix:matrix}={}){const{x:x=0,y:y=0}=arguments[0].p!==undefined?arguments[0].p:arguments[0];let trf=matrix;if(!trf){let ssw,scw;ssw=w?Math.sin(w)*scl:0;scw=w?Math.cos(w)*scl:scl;trf=[scw,ssw,-ssw,scw,x,y]}this.pushStyle(arguments[0]);this.pushTrf(trf)},end(){this.popTrf();this.popStyle()},p(){this.ctx.beginPath()},z(){this.ctx.closePath()},m({x:x,y:y}){this.ctx.moveTo(x,y)},l({x:x,y:y}){this.ctx.lineTo(x,y)},q({x:x,y:y,x1:x1,y1:y1}){this.ctx.quadraticCurveTo(x1,y1,x,y)},c({x:x,y:y,x1:x1,y1:y1,x2:x2,y2:y2}){this.ctx.bezierCurveTo(x1,y1,x2,y2,x,y)},a({dw:dw,k:k,phi:phi,_xp:_xp,_yp:_yp}){const{x:x=0,y:y=0}=arguments[0].p!==undefined?arguments[0].p:arguments[0];if(k===undefined)k=1;if(Math.abs(dw)>Number.EPSILON){if(k===1){let x12=x-_xp,y12=y-_yp;let tdw_2=Math.tan(dw/2),rx=(x12-y12/tdw_2)/2,ry=(y12+x12/tdw_2)/2,R=Math.hypot(rx,ry),w=Math.atan2(-ry,-rx);this.ctx.ellipse(_xp+rx,_yp+ry,R,R,0,w,w+dw,this.cartesian?dw>0:dw<0)}else{if(phi===undefined)phi=0;let x1=dw>0?_xp:x,y1=dw>0?_yp:y,x2=dw>0?x:_xp,y2=dw>0?y:_yp;let x12=x2-x1,y12=y2-y1,_dw=dw<0?dw:-dw;let cp=phi?Math.cos(phi):1,sp=phi?Math.sin(phi):0,dx=-x12*cp-y12*sp,dy=-x12*sp-y12*cp,sdw_2=Math.sin(_dw/2),R=Math.sqrt((dx*dx+dy*dy/(k*k))/(4*sdw_2*sdw_2)),w=Math.atan2(k*dx,dy)-_dw/2,x0=x1-R*Math.cos(w),y0=y1-R*k*Math.sin(w);this.ctx.ellipse(x0,y0,R,R*k,phi,w,w+dw,this.cartesian?dw>0:dw<0)}}else this.ctx.lineTo(x,y)},stroke({d:d}={}){let tmp=this.setStyle(arguments[0]);d?this.ctx.stroke(new Path2D(d)):this.ctx.stroke();this.resetStyle(tmp)},fill({d:d}={}){let tmp=this.setStyle(arguments[0]);d?this.ctx.fill(new Path2D(d)):this.ctx.fill();this.resetStyle(tmp)},drw({d:d,lsh:lsh}={}){let ctx=this.ctx,tmp=this.setStyle(arguments[0]),p=d&&new Path2D(d);d?ctx.fill(p):ctx.fill();if(ctx.shadowColor!=="rgba(0, 0, 0, 0)"&&ctx.fillStyle!=="rgba(0, 0, 0, 0)"&&!lsh){let shc=ctx.shadowColor;ctx.shadowColor="rgba(0, 0, 0, 0)";d?ctx.stroke(p):ctx.stroke();ctx.shadowColor=shc}else d?ctx.stroke(p):ctx.stroke();this.resetStyle(tmp)},get:{fs:ctx=>ctx.fillStyle,ls:ctx=>ctx.strokeStyle,lw:ctx=>ctx.lineWidth,lc:ctx=>ctx.lineCap,lj:ctx=>ctx.lineJoin,ld:ctx=>ctx.getLineDash(),ldoff:ctx=>ctx.lineDashOffset,ml:ctx=>ctx.miterLimit,sh:ctx=>[ctx.shadowOffsetX||0,ctx.shadowOffsetY||0,ctx.shadowBlur||0,ctx.shadowColor||"black"],font:ctx=>ctx.font,thal:ctx=>ctx.textAlign,tval:ctx=>ctx.textBaseline},set:{fs:(ctx,q)=>{ctx.fillStyle=q},ls:(ctx,q)=>{ctx.strokeStyle=q},lw:(ctx,q)=>{ctx.lineWidth=q},lc:(ctx,q)=>{ctx.lineCap=q},lj:(ctx,q)=>{ctx.lineJoin=q},ld:(ctx,q)=>{ctx.setLineDash(q)},ldoff:(ctx,q)=>{ctx.lineDashOffset=q},ml:(ctx,q)=>{ctx.miterLimit=q},sh:(ctx,q)=>{if(q){ctx.shadowOffsetX=q[0]||0;ctx.shadowOffsetY=q[1]||0;ctx.shadowBlur=q[2]||0;ctx.shadowColor=q[3]||"black"}},font:(ctx,q)=>{ctx.font=q},thal:(ctx,q)=>{ctx.textAlign=q},tval:(ctx,q)=>{ctx.textBaseline=q}},initStyle(style){for(const key in style)if(this.get[key]&&this.get[key](this.ctx)!==style[key])this.set[key](this.ctx,style[key])},setStyle(style){let q,prv={};for(const key in style){if(this.get[key]){if(typeof style[key]==="string"&&style[key][0]==="@"){let ref=style[key].substr(1);style[key]=g2.symbol[ref]||this.get[ref]&&this.get[ref](this.ctx)}if((q=this.get[key](this.ctx))!==style[key]){prv[key]=q;this.set[key](this.ctx,style[key])}}}return prv},resetStyle(style){for(const key in style)this.set[key](this.ctx,style[key])},pushStyle(style){let cur={};for(const key in style)if(this.get[key]){if(typeof style[key]==="string"&&style[key][0]==="@"){let ref=style[key].substr(1);style[key]=g2.symbol[ref]||this.get[ref]&&this.get[ref](this.ctx)}if(this.cur[key]!==style[key])this.set[key](this.ctx,cur[key]=style[key])}this.stack.push(this.cur=Object.assign({},this.cur,cur))},popStyle(){let cur=this.stack.pop();this.cur=this.stack[this.stack.length-1];for(const key in this.cur)if(this.get[key]&&this.cur[key]!==cur[key])this.set[key](this.ctx,this.cur[key])},concatTrf(q,t){return[q[0]*t[0]+q[2]*t[1],q[1]*t[0]+q[3]*t[1],q[0]*t[2]+q[2]*t[3],q[1]*t[2]+q[3]*t[3],q[0]*t[4]+q[2]*t[5]+q[4],q[1]*t[4]+q[3]*t[5]+q[5]]},initTrf(){this.ctx.setTransform(...this.matrix[0])},setTrf(t){this.ctx.setTransform(...this.concatTrf(this.matrix[this.matrix.length-1],t))},resetTrf(){this.ctx.setTransform(...this.matrix[this.matrix.length-1])},pushTrf(t){let q_t=this.concatTrf(this.matrix[this.matrix.length-1],t);this.matrix.push(q_t);this.ctx.setTransform(...q_t)},popTrf(){this.matrix.pop();this.ctx.setTransform(...this.matrix[this.matrix.length-1])},get isCartesian(){let m=this.matrix[this.matrix.length-1];return m[0]*m[3]-m[1]*m[2]<0},get uniTrf(){let m=this.matrix[this.matrix.length-1];return{x:m[4],y:m[5],scl:Math.hypot(m[0],m[1]),cartesian:m[0]*m[3]-m[1]*m[2]<0}},unscaleTrf({x:x,y:y}){let m=this.matrix[this.matrix.length-1],invscl=1/Math.hypot(m[0],m[1]);return[invscl,0,0,invscl,(1-invscl)*x,(1-invscl)*y]},gridSize(scl){let base=this.gridBase,exp=this.gridExp,sz;while((sz=scl*base*Math.pow(10,exp))<14||sz>35){if(sz<14){if(base==1)base=2;else if(base==2)base=5;else if(base==5){base=1;exp++}}else{if(base==1){base=5;exp--}else if(base==2)base=1;else if(base==5)base=2}}this.gridBase=base;this.gridExp=exp;return sz}};if(typeof module!=="undefined")module.exports=g2; +/** + * g2.lib (c) 2013-17 Stefan Goessner + * geometric constants and higher functions + * @license MIT License + * @link https://github.com/goessner/g2 + */ +"use strict";var g2=g2||{};g2=Object.assign(g2,{EPS:Number.EPSILON,PI:Math.PI,PI2:2*Math.PI,SQRT2:Math.SQRT2,SQRT2_2:Math.SQRT2/2,toPi2(w){return(w%g2.PI2+g2.PI2)%g2.PI2},toPi(w){return(w=(w%g2.PI2+g2.PI2)%g2.PI2)>g2.PI?w-g2.PI2:w},toArc:function(w,w0,dw){if(dw>g2.EPS||dw<-g2.EPS){if(w0>w&&w0+dw>g2.PI2)w0-=g2.PI2;else if(w0eps&&Math.abs(dist-r)=0&&mu<=1},isPntOnPly({x:x,y:y},{pts:pts,closed:closed},eps=Number.EPSILON){for(var i=0,n=pts.length;i<(closed?n:n-1);i++)if(g2.isPntOnLin({x:x,y:y},pts[i],pts[(i+1)%n],eps))return true;return false},isPntOnBox({x:xp,y:yp},{x:x,y:y,b:b,h:h},eps=Number.EPSILON){var dx=x.p-x,dy=yp-y;return dx>=b-eps&&dx<=b+eps&&dy<=h+eps&&dy>=-h-eps||dx>=-b-eps&&dx<=b+eps&&dy<=h+eps&&dy>=h-eps||dx>=-b-eps&&dx<=-b+eps&&dy<=h+eps&&dy>=-h-eps||dx>=-b-eps&&dx<=b+eps&&dy<=-h+eps&&dy>=-h-eps},isPntInCir({x:xp,y:yp},{x:x,y:y,r:r}){return(x-xp)**2+(y-yp)**2pi.y||y>pj.y)&&(y<=pi.y||y<=pj.y)&&(x<=pi.x||x<=pj.x)&&pi.y!==pj.y&&(pi.x===pj.x||x<=pj.x+(y-pj.y)*(pi.x-pj.x)/(pi.y-pj.y)))match++;return match%2!=0},isPntInBox({x:xp,y:yp},{x:x,y:y,b:b,h:h}){var dx=xp-x,dy=yp-y;return dx>=-b&&dx<=b&&dy>=-h&&dy<=h},arc3pts(x1,y1,x2,y2,x3,y3){const dx1=x2-x1,dy1=y2-y1;const dx2=x3-x2,dy2=y3-y2;const den=dx1*dy2-dy1*dx2;const lam=Math.abs(den)>Number.EPSILON?.5*((dx1+dx2)*dx2+(dy1+dy2)*dy2)/den:0;const x0=lam?x1+.5*dx1-lam*dy1:x1+.5*(dx1+dx2);const y0=lam?y1+.5*dy1+lam*dx1:y1+.5*(dy1+dy2);const dx01=x1-x0,dy01=y1-y0;const dx03=x3-x0,dy03=y3-y0;const dw=lam?Math.atan2(dx01*dy03-dy01*dx03,dx01*dx03+dy01*dy03):0;const r=dw?Math.hypot(dy01,dx01):.5*Math.hypot(dy1+dy2,dx1+dx2);return{x:x0,y:y0,r:r,w:Math.atan2(dy01,dx01),dw:dw}}});"use strict"; +/** + * g2.ext (c) 2015-20 Stefan Goessner + * @author Stefan Goessner + * @license MIT License + * @requires g2.core.js + * @typedef {g2} + * @description Additional methods for g2. + * @returns {g2} + */var g2=g2||{prototype:{}};g2.NONE=0;g2.OVER=1;g2.DRAG=2;g2.EDIT=4;g2.symbol=g2.symbol||{};g2.symbol.tick=g2().p().m({x:0,y:-2}).l({x:0,y:2}).stroke({lc:"round",lwnosc:true});g2.symbol.dot=g2().cir({x:0,y:0,r:2,ls:"transparent"});g2.symbol.sqr=g2().rec({x:-1.5,y:-1.5,b:3,h:3,ls:"transparent"});g2.symbol.nodcolor="#333";g2.symbol.nodfill="#dedede";g2.symbol.nodfill2="#aeaeae";g2.symbol.linkcolor="#666";g2.symbol.linkfill="rgba(225,225,225,0.75)";g2.symbol.dimcolor="darkslategray";g2.symbol.solid=[];g2.symbol.dash=[15,10];g2.symbol.dot=[4,4];g2.symbol.dashdot=[25,6.5,2,6.5];g2.symbol.labelSignificantDigits=3;g2.flatten=function(obj){const args=Object.create(null);for(let p in obj)if(typeof obj[p]!=="function")args[p]=obj[p];return args};g2.pointIfc={get p(){return{x:this.x,y:this.y}},get x(){return Object.getOwnPropertyDescriptor(this,"p")?this.p.x:0},get y(){return Object.getOwnPropertyDescriptor(this,"p")?this.p.y:0},set x(q){if(Object.getOwnPropertyDescriptor(this,"p"))this.p.x=q},set y(q){if(Object.getOwnPropertyDescriptor(this,"p"))this.p.y=q}};g2.labelIfc={getLabelOffset(){const off=this.label.off!==undefined?+this.label.off:1;return off+Math.sign(off)*(this.lw||2)/2},getLabelString(){let s=typeof this.label==="object"?this.label.str:typeof this.label==="string"?this.label:"?";if(s&&s[0]==="@"&&this[s.substr(1)]){s=s.substr(1);let val=this[s];val=Number.isInteger(val)?val:Number(val).toFixed(Math.max(g2.symbol.labelSignificantDigits-Math.log10(val),0));s=`${val}${s==="angle"?"°":""}`}return s},drawLabel(g){const lbl=this.label;const font=lbl.font||g2.defaultStyle.font;const h=parseInt(font);const str=this.getLabelString();const rx=(str.length||1)*.75*h/2,ry=1.25*h/2;const pos=this.pointAt(lbl.loc||this.lbloc||"se");const off=this.getLabelOffset();const p={x:pos.x+pos.nx*(off+Math.sign(off)*rx),y:pos.y+pos.ny*(off+Math.sign(off)*ry)};if(lbl.border)g.ell({x:p.x,y:p.y,rx:rx,ry:ry,ls:lbl.fs||"black",fs:lbl.fs2||"#ffc"});g.txt({str:str,x:p.x,y:p.y,thal:"center",tval:"middle",fs:lbl.fs||"black",font:lbl.font});return g}};g2.markIfc={markAt(loc){const p=this.pointAt(loc);const w=Math.atan2(p.ny,p.nx)+Math.PI/2;return{grp:this.getMarkSymbol(),x:p.x,y:p.y,w:w,scl:this.lw||1,ls:this.ls||"#000",fs:this.fs||this.ls||"#000"}},getMarkSymbol(){const mrk=this.mark;if(typeof mrk==="number"||!mrk)return g2.symbol.tick;if(typeof mrk.symbol==="object")return mrk.symbol;if(typeof mrk.symbol==="string")return g2.symbol[mrk.symbol]},drawMark(g,closed=false){let loc;if(Array.isArray(this.mark)){loc=this.mark}else{const count=typeof this.mark==="object"?this.mark.count:this.mark;loc=count?Array.from(Array(count)).map((_,i)=>i/(count-!closed)):this.mark.loc}for(let l of loc){g.use(this.markAt(l))}return g}};g2.prototype.cir.prototype=g2.mix(g2.pointIfc,g2.labelIfc,g2.markIfc,{w:0,lbloc:"c",get isSolid(){return this.fs&&this.fs!=="transparent"},get len(){return 2*Math.PI*this.r},get lsh(){return this.state&g2.OVER},get sh(){return this.state&g2.OVER?[0,0,5,"black"]:false},get g2(){const e=g2();this.label&&e.ins(g=>this.drawLabel(g));this.mark&&e.ins(g=>this.drawMark(g,true));return()=>g2().cir(g2.flatten(this)).ins(e)},pointAt(loc){const Q=Math.SQRT2/2;const LOC={c:[0,0],e:[1,0],ne:[Q,Q],n:[0,1],nw:[-Q,Q],w:[-1,0],sw:[-Q,-Q],s:[0,-1],se:[Q,-Q]};const q=loc+0===loc?[Math.cos(loc*2*Math.PI),Math.sin(loc*2*Math.PI)]:LOC[loc||"c"]||[0,0];return{x:this.x+q[0]*this.r,y:this.y+q[1]*this.r,nx:q[0],ny:q[1]}},hit({x:x,y:y,eps:eps}){return this.isSolid?g2.isPntInCir({x:x,y:y},this,eps):g2.isPntOnCir({x:x,y:y},this,eps)},drag({dx:dx,dy:dy}){this.x+=dx;this.y+=dy}});g2.prototype.lin.prototype=g2.mix(g2.labelIfc,g2.markIfc,{get p1(){return{x1:this.x1,y1:this.y1}},get x1(){return Object.getOwnPropertyDescriptor(this,"p1")?this.p1.x:0},get y1(){return Object.getOwnPropertyDescriptor(this,"p1")?this.p1.y:0},set x1(q){if(Object.getOwnPropertyDescriptor(this,"p1"))this.p1.x=q},set y1(q){if(Object.getOwnPropertyDescriptor(this,"p1"))this.p1.y=q},get p2(){return{x2:this.x2,y2:this.y2}},get x2(){return Object.getOwnPropertyDescriptor(this,"p2")?this.p2.x:0},get y2(){return Object.getOwnPropertyDescriptor(this,"p2")?this.p2.y:0},set x2(q){if(Object.getOwnPropertyDescriptor(this,"p2"))this.p2.x=q},set y2(q){if(Object.getOwnPropertyDescriptor(this,"p2"))this.p2.y=q},isSolid:false,get len(){return Math.hypot(this.x2-this.x1,this.y2-this.y1)},get sh(){return this.state&g2.OVER?[0,0,5,"black"]:false},get g2(){const e=g2();this.label&&e.ins(e=>this.drawLabel(e));this.mark&&e.ins(e=>this.drawMark(e));return()=>g2().lin(g2.flatten(this)).ins(e)},pointAt(loc){let t=loc==="beg"?0:loc==="end"?1:loc+0===loc?loc:.5,dx=this.x2-this.x1,dy=this.y2-this.y1,len=Math.hypot(dx,dy);return{x:this.x1+dx*t,y:this.y1+dy*t,nx:len?dy/len:0,ny:len?-dx/len:-1}},hit({x:x,y:y,eps:eps}){return g2.isPntOnLin({x:x,y:y},{x:this.x1,y:this.y1},{x:this.x2,y:this.y2},eps)},drag({dx:dx,dy:dy}){this.x1+=dx;this.x2+=dx;this.y1+=dy;this.y2+=dy}});g2.prototype.rec.prototype=g2.mix(g2.pointIfc,g2.labelIfc,g2.markIfc,{get len(){return 2*(this.b+this.h)},get isSolid(){return this.fs&&this.fs!=="transparent"},get lsh(){return this.state&g2.OVER},get sh(){return this.state&g2.OVER?[0,0,5,"black"]:false},get g2(){const e=g2();this.label&&e.ins(e=>this.drawLabel(e));this.mark&&e.ins(e=>this.drawMark(e,true));return()=>g2().rec(g2.flatten(this)).ins(e)},lbloc:"c",pointAt(loc){const locAt=loc=>{const o={c:[0,0],e:[1,0],ne:[.95,.95],n:[0,1],nw:[-.95,.95],w:[-1,0],sw:[-.95,-.95],s:[0,-1],se:[.95,-.95]};if(o[loc])return o[loc];const w=2*Math.PI*loc+pi/4;if(loc<=.25)return[1/Math.tan(w),1];if(loc<=.5)return[-1,-Math.tan(w)];if(loc<=.75)return[-1/Math.tan(w),-1];if(loc<=1)return[1,Math.tan(w)]};const q=locAt(loc);return{x:this.x+(1+q[0])*this.b/2,y:this.y+(1+q[1])*this.h/2,nx:1-Math.abs(q[0])<.01?q[0]:0,ny:1-Math.abs(q[1])<.01?q[1]:0}},hit({x:x,y:y,eps:eps}){return this.isSolid?g2.isPntInBox({x:x,y:y},{x:this.x+this.b/2,y:this.y+this.h/2,b:this.b/2,h:this.h/2},eps):g2.isPntOnBox({x:x,y:y},{x:this.x+this.b/2,y:this.y+this.h/2,b:this.b/2,h:this.h/2},eps)},drag({dx:dx,dy:dy}){this.x+=dx;this.y+=dy}});g2.prototype.arc.prototype=g2.mix(g2.pointIfc,g2.labelIfc,g2.markIfc,{get len(){return Math.abs(this.r*this.dw)},isSolid:false,get angle(){return this.dw/Math.PI*180},get sh(){return this.state&g2.OVER?[0,0,5,"black"]:false},get g2(){const e=g2();this.label&&e.ins(e=>this.drawLabel(e));this.mark&&e.ins(e=>this.drawMark(e));return()=>g2().arc(g2.flatten(this)).ins(e)},lbloc:"mid",pointAt(loc){let t=loc==="beg"?0:loc==="end"?1:loc==="mid"?.5:loc+0===loc?loc:.5,ang=(this.w||0)+t*(this.dw||Math.PI*2),cang=Math.cos(ang),sang=Math.sin(ang),r=loc==="c"?0:this.r;return{x:this.x+r*cang,y:this.y+r*sang,nx:cang,ny:sang}},hit({x:x,y:y,eps:eps}){return g2.isPntOnArc({x:x,y:y},this,eps)},drag({dx:dx,dy:dy}){this.x+=dx;this.y+=dy}});g2.prototype.hdl=function(args){return this.addCommand({c:"hdl",a:args})};g2.prototype.hdl.prototype=g2.mix(g2.prototype.cir.prototype,{r:5,isSolid:true,draggable:true,lbloc:"se",get lsh(){return this.state&g2.OVER},get sh(){return this.state&g2.OVER?[0,0,5,"black"]:false},g2(){const{x:x,y:y,r:r,b:b=4,shape:shape="cir",ls:ls="black",fs:fs="#ccc",sh:sh}=this;return shape==="cir"?g2().cir({x:x,y:y,r:r,ls:ls,fs:fs,sh:sh}).ins(g=>this.label&&this.drawLabel(g)):g2().rec({x:x-b,y:y-b,b:2*b,h:2*b,ls:ls,fs:fs,sh:sh}).ins(g=>this.label&&this.drawLabel(g))}});g2.prototype.nod=function(args={}){return this.addCommand({c:"nod",a:args})};g2.prototype.nod.prototype=g2.mix(g2.prototype.cir.prototype,{r:5,ls:"@nodcolor",fs:g2.symbol.nodfill,isSolid:true,lbloc:"se",g2(){return g2().cir({...g2.flatten(this),r:this.r*(this.scl!==undefined?this.scl:1)}).ins(g=>this.label&&this.drawLabel(g))}});g2.prototype.dblnod=function({x:x=0,y:y=0}){return this.addCommand({c:"dblnod",a:arguments[0]})};g2.prototype.dblnod.prototype=g2.mix(g2.prototype.cir.prototype,{get r(){return 6},get isSolid(){return true},g2(){return g2().beg({x:this.x,y:this.y}).cir({r:6,ls:"@nodcolor",fs:"@nodfill",sh:this.sh}).cir({r:3,ls:"@nodcolor",fs:"@nodfill2"}).end().ins(g=>this.label&&this.drawLabel(g))}});g2.prototype.pol=function(args={}){return this.addCommand({c:"pol",a:args})};g2.prototype.pol.prototype=g2.mix(g2.prototype.nod.prototype,{g2(){return g2().beg(g2.flatten(this)).cir({r:6,fs:g2.symbol.nodfill}).cir({r:2.5,fs:"@ls",ls:"transparent"}).end().ins(g=>this.label&&this.drawLabel(g))}});g2.prototype.gnd=function(args={}){return this.addCommand({c:"gnd",a:args})};g2.prototype.gnd.prototype=g2.mix(g2.prototype.nod.prototype,{g2(){return g2().beg(g2.flatten(this)).cir({x:0,y:0,r:6}).p().m({x:0,y:6}).a({dw:Math.PI/2,x:-6,y:0}).l({x:6,y:0}).a({dw:-Math.PI/2,x:0,y:-6}).z().fill({fs:g2.symbol.nodcolor}).end().ins(g=>this.label&&this.drawLabel(g))}});g2.prototype.nodfix=function(args={}){return this.addCommand({c:"nodfix",a:args})};g2.prototype.nodfix.prototype=g2.mix(g2.prototype.nod.prototype,{g2(){return g2().beg(g2.flatten(this)).p().m({x:-8,y:-12}).l({x:0,y:0}).l({x:8,y:-12}).drw({fs:g2.symbol.nodfill2}).cir({x:0,y:0,r:this.r}).end().ins(g=>this.label&&this.drawLabel(g))}});g2.prototype.nodflt=function(args={}){return this.addCommand({c:"nodflt",a:args})};g2.prototype.nodflt.prototype=g2.mix(g2.prototype.nod.prototype,{g2(){return g2().beg(g2.flatten(this)).p().m({x:-8,y:-12}).l({x:0,y:0}).l({x:8,y:-12}).drw({ls:g2.symbol.nodcolor,fs:g2.symbol.nodfill2}).cir({x:0,y:0,r:this.r,ls:g2.symbol.nodcolor,fs:g2.symbol.nodfill}).lin({x1:-9,y1:-19,x2:9,y2:-19,ls:g2.symbol.nodfill2,lw:5}).lin({x1:-9,y1:-15.5,x2:9,y2:-15.5,ls:g2.symbol.nodcolor,lw:2}).end().ins(g=>this.label&&this.drawLabel(g))}});g2.prototype.vec=function vec(args){return this.addCommand({c:"vec",a:args})};g2.prototype.vec.prototype=g2.mix(g2.prototype.lin.prototype,{g2(){const{x1:x1,y1:y1,x2:x2,y2:y2,lw:lw=1,ls:ls="#000",ld:ld=[],fs:fs=ls||"#000",lc:lc="round",lj:lj="round"}=this;const dx=x2-x1,dy=y2-y1,r=Math.hypot(dx,dy);const b=3*(1+lw)>r?r/3:1+lw;const arrowHead=()=>g2().p().m({x:0,y:0}).l({x:-5*b,y:b}).a({dw:-Math.PI/3,x:-5*b,y:-b}).z().drw({ls:ls,fs:fs,lc:lc,lj:lj});return g2().beg({x:x1,y:y1,w:Math.atan2(dy,dx),lc:lc,lj:lj}).p().m({x:0,y:0}).l({x:r-3*b,y:0}).stroke({ls:ls,lw:lw,ld:ld}).use({grp:arrowHead,x:r,y:0}).end().ins(g=>this.label&&this.drawLabel(g))}});g2.prototype.avec=function avec(args){return this.addCommand({c:"avec",a:args})};g2.prototype.avec.prototype=g2.mix(g2.prototype.arc.prototype,{g2(){const{x:x,y:y,r:r,w:w,dw:dw=0,lw:lw=1,lc:lc="round",lj:lj="round",ls:ls,fs:fs=ls||"#000",label:label}=this;const b=3*(1+lw)>r?r/3:1+lw,bw=5*b/r;const arrowHead=()=>g2().p().m({x:0,y:2*b}).l({x:0,y:-2*b}).m({x:0,y:0}).l({x:-5*b,y:b}).a({dw:-Math.PI/3,x:-5*b,y:-b}).z().drw({ls:ls,fs:fs});return g2().beg({x:x,y:y,w:w,ls:ls,lw:lw,lc:lc,lj:lj}).arc({r:r,w:0,dw:dw}).use({grp:arrowHead,x:r*Math.cos(dw),y:r*Math.sin(dw),w:dw>=0?dw+Math.PI/2-bw/2:dw-Math.PI/2+bw/2}).end().ins(g=>label&&this.drawLabel(g))}});g2.prototype.dim=function dim(args){return this.addCommand({c:"dim",a:args})};g2.prototype.dim.prototype=g2.mix(g2.prototype.lin.prototype,{pointAt(loc){const pnt=g2.prototype.lin.prototype.pointAt.call(this,loc);if(this.off){pnt.x+=this.off*pnt.nx;pnt.y+=this.off*pnt.ny}return pnt},g2(){const{x1:x1,y1:y1,x2:x2,y2:y2,lw:lw=1,lc:lc="round",lj:lj="round",off:off=0,inside:inside=true,ls:ls,fs:fs=ls||"#000",label:label}=this;const dx=x2-x1,dy=y2-y1,r=Math.hypot(dx,dy);const b=3*(1+lw)>r?r/3:1+lw;const arrowHead=()=>g2().p().m({x:0,y:2*b}).l({x:0,y:-2*b}).m({x:0,y:0}).l({x:-5*b,y:b}).a({dw:-Math.PI/3,x:-5*b,y:-b}).z().drw({ls:ls,fs:fs});return g2().beg({x:x1+off/r*dy,y:y1-off/r*dx,w:Math.atan2(dy,dx),ls:ls,fs:fs,lw:lw,lc:lc,lj:lj}).lin({x1:inside?4*b:0,y1:0,x2:inside?r-4*b:r,y2:0}).use({grp:arrowHead,x:r,y:0,w:inside?0:Math.PI}).use({grp:arrowHead,x:0,y:0,w:inside?Math.PI:0}).lin({x1:0,y1:off,x2:0,y2:0}).lin({x1:r,y1:off,x2:r,y2:0}).end().ins(g=>label&&this.drawLabel(g))}});g2.prototype.adim=function adim(args){return this.addCommand({c:"adim",a:args})};g2.prototype.adim.prototype=g2.mix(g2.prototype.arc.prototype,{g2(){const{x:x,y:y,r:r,w:w,dw:dw,lw:lw=1,lc:lc="round",lj:lj="round",ls:ls,fs:fs=ls||"#000",label:label}=this;const b=3*(1+lw)>r?r/3:1+lw,bw=5*b/r;const arrowHead=()=>g2().p().m({x:0,y:2*b}).l({x:0,y:-2*b}).m({x:0,y:0}).l({x:-5*b,y:b}).a({dw:-Math.PI/3,x:-5*b,y:-b}).z().drw({ls:ls,fs:fs});const outside=this.inside!==undefined&&this.outside===undefined?!this.inside:!!this.outside;return g2().beg({x:x,y:y,w:w,ls:ls,lw:lw,lc:lc,lj:lj}).arc({r:r,w:0,dw:dw}).use({grp:arrowHead,x:r,y:0,w:!outside&&dw>0||outside&&dw<0?-Math.PI/2+bw/2:Math.PI/2-bw/2}).use({grp:arrowHead,x:r*Math.cos(dw),y:r*Math.sin(dw),w:!outside&&dw>0||outside&&dw<0?dw+Math.PI/2-bw/2:dw-Math.PI/2+bw/2}).end().ins(g=>label&&this.drawLabel(g))}});g2.prototype.origin=function(args={}){return this.addCommand({c:"origin",a:args})};g2.prototype.origin.prototype=g2.mix(g2.prototype.nod.prototype,{lbloc:"sw",g2(){const{x:x,y:y,w:w,ls:ls="#000",lw:lw=1}=this;return g2().beg({x:x,y:y,w:w,ls:ls}).vec({x1:0,y1:0,x2:40,y2:0,lw:lw,fs:"#ccc"}).vec({x1:0,y1:0,x2:0,y2:40,lw:lw,fs:"#ccc"}).cir({x:0,y:0,r:lw+1,fs:"#ccc"}).end().ins(g=>this.label&&this.drawLabel(g))}});g2.prototype.ply.prototype=g2.mix(g2.labelIfc,g2.markIfc,{get isSolid(){return this.closed&&this.fs&&this.fs!=="transparent"},get sh(){return this.state&g2.OVER?[0,0,5,"black"]:false},pointAt(loc){const t=loc==="beg"?0:loc==="end"?1:loc+0===loc?loc:.5,pitr=g2.pntItrOf(this.pts),pts=[],len=[];for(let itr=0;itr{const target=t*len.reduce((a,b)=>a+b);for(let itr=0,tmp=0;itr=target){return{t2:1-(tmp-target)/len[itr],x:pts[itr].x,y:pts[itr].y,dx:next.x-pts[itr].x,dy:next.y-pts[itr].y}}}})();const len2=Math.hypot(dx,dy);return{x:(this.x||0)+x+dx*t2,y:(this.y||0)+y+dy*t2,nx:len2?dy/len2:1,ny:len2?dx/len2:0}},hit({x:x,y:y,eps:eps}){return this.isSolid?g2.isPntInPly({x:x-this.x,y:y-this.y},this,eps):g2.isPntOnPly({x:x-this.x,y:y-this.y},this,eps)},drag({dx:dx,dy:dy}){this.x+=dx;this.y+=dy},get g2(){const e=g2();this.label&&e.ins(e=>this.drawLabel(e));this.mark&&e.ins(e=>this.drawMark(e,this.closed));return()=>g2().ply(g2.flatten(this)).ins(e)}});g2.prototype.use.prototype={get p(){return{x:this.x,y:this.y}},get x(){return Object.getOwnPropertyDescriptor(this,"p")?this.p.x:0},get y(){return Object.getOwnPropertyDescriptor(this,"p")?this.p.y:0},set x(q){if(Object.getOwnPropertyDescriptor(this,"p"))this.p.x=q},set y(q){if(Object.getOwnPropertyDescriptor(this,"p"))this.p.y=q},isSolid:false};g2.prototype.spline=function spline({pts:pts,closed:closed,x:x,y:y,w:w}){arguments[0]._itr=g2.pntItrOf(pts);return this.addCommand({c:"spline",a:arguments[0]})};g2.prototype.spline.prototype=g2.mix(g2.prototype.ply.prototype,{g2:function(){let{pts:pts,closed:closed,x:x,y:y,w:w,ls:ls,lw:lw,fs:fs,sh:sh}=this,itr=this._itr,gbez;if(itr){let b=[],i,n=itr.len,p1,p2,p3,p4,d1,d2,d3,d1d2,d2d3,scl2,scl3,den2,den3,istrf=x||y||w;gbez=g2();if(istrf)gbez.beg({x:x,y:y,w:w});gbez.p().m(itr(0));for(let i=0;i<(closed?n:n-1);i++){if(i===0){p1=closed?itr(n-1):{x:2*itr(0).x-itr(1).x,y:2*itr(0).y-itr(1).y};p2=itr(0);p3=itr(1);p4=n===2?closed?itr(0):{x:2*itr(1).x-itr(0).x,y:2*itr(1).y-itr(0).y}:itr(2);d1=Math.max(Math.hypot(p2.x-p1.x,p2.y-p1.y),Number.EPSILON);d2=Math.max(Math.hypot(p3.x-p2.x,p3.y-p2.y),Number.EPSILON)}else{p1=p2;p2=p3;p3=p4;p4=i===n-2?closed?itr(0):{x:2*itr(n-1).x-itr(n-2).x,y:2*itr(n-1).y-itr(n-2).y}:i===n-1?itr(1):itr(i+2);d1=d2;d2=d3}d3=Math.max(Math.hypot(p4.x-p3.x,p4.y-p3.y),Number.EPSILON);d1d2=Math.sqrt(d1*d2),d2d3=Math.sqrt(d2*d3),scl2=2*d1+3*d1d2+d2,scl3=2*d3+3*d2d3+d2,den2=3*(d1+d1d2),den3=3*(d3+d2d3);gbez.c({x:p3.x,y:p3.y,x1:(-d2*p1.x+scl2*p2.x+d1*p3.x)/den2,y1:(-d2*p1.y+scl2*p2.y+d1*p3.y)/den2,x2:(-d2*p4.x+scl3*p3.x+d3*p2.x)/den3,y2:(-d2*p4.y+scl3*p3.y+d3*p2.y)/den3})}gbez.c(closed?{x:itr(0).x,y:itr(0).y}:{x:itr(n-1).x,y:itr(n-1).y});if(closed)gbez.z();gbez.drw({ls:ls,lw:lw,fs:fs,sh:sh});if(istrf)gbez.end()}return gbez}});"use strict"; +/** + * g2.mec (c) 2013-18 Stefan Goessner + * @author Stefan Goessner + * @license MIT License + * @requires g2.core.js + * @requires g2.ext.js + * @typedef {g2} + * @description Mechanical extensions. (Requires cartesian coordinates) + * @returns {g2} + */var g2=g2||{prototype:{}};g2.prototype.slider=function(){return this.addCommand({c:"slider",a:arguments[0]})};g2.prototype.slider.prototype=g2.mix(g2.prototype.rec.prototype,{g2(){const args=Object.assign({b:32,h:16,fs:"@linkfill"},this);return g2().beg({x:args.x,y:args.y,w:args.w,fs:args.fs}).rec({x:-args.b/2,y:-args.h/2,b:args.b,h:args.h}).end()}});g2.prototype.spring=function(){return this.addCommand({c:"spring",a:arguments[0]})};g2.prototype.spring.prototype=g2.mix(g2.prototype.lin.prototype,{g2(){const args=Object.assign({h:16},this);const len=Math.hypot(args.x2-args.x1,args.y2-args.y1);const xm=(args.x2+args.x1)/2;const ym=(args.y2+args.y1)/2;const h=args.h;const ux=(args.x2-args.x1)/len;const uy=(args.y2-args.y1)/len;return g2().p().m({x:args.x1,y:args.y1}).l({x:xm-ux*h/2,y:ym-uy*h/2}).l({x:xm+(-ux/6+uy/2)*h,y:ym+(-uy/6-ux/2)*h}).l({x:xm+(ux/6-uy/2)*h,y:ym+(uy/6+ux/2)*h}).l({x:xm+ux*h/2,y:ym+uy*h/2}).l({x:args.x2,y:args.y2}).stroke(Object.assign({},{ls:"@nodcolor"},this,{fs:"transparent",lc:"round",lj:"round"}))}});g2.prototype.damper=function(){return this.addCommand({c:"damper",a:arguments[0]})};g2.prototype.damper.prototype=g2.mix(g2.prototype.lin.prototype,{g2(){const args=Object.assign({h:16},this);const len=Math.hypot(args.x2-args.x1,args.y2-args.y1);const xm=(args.x2+args.x1)/2;const ym=(args.y2+args.y1)/2;const h=args.h;const ux=(args.x2-args.x1)/len;const uy=(args.y2-args.y1)/len;return g2().p().m({x:args.x1,y:args.y1}).l({x:xm-ux*h/2,y:ym-uy*h/2}).m({x:xm+(ux-uy)*h/2,y:ym+(uy+ux)*h/2}).l({x:xm+(-ux-uy)*h/2,y:ym+(-uy+ux)*h/2}).l({x:xm+(-ux+uy)*h/2,y:ym+(-uy-ux)*h/2}).l({x:xm+(ux+uy)*h/2,y:ym+(uy-ux)*h/2}).m({x:xm,y:ym}).l({x:args.x2,y:args.y2}).stroke(Object.assign({},{ls:"@nodcolor"},this,{fs:"transparent",lc:"round",lj:"round"}))}});g2.prototype.link=function(){return this.addCommand({c:"link",a:arguments[0]})};g2.prototype.link.prototype=g2.mix(g2.prototype.ply.prototype,{g2(){const args=Object.assign({ls:"@linkcolor",fs:"transparent"},this);return g2().ply(Object.assign({},this,{closed:true,ls:args.ls,fs:args.fs,lw:7,lc:"round",lj:"round"}))}});g2.prototype.link2=function(){return this.addCommand({c:"link2",a:arguments[0]})};g2.prototype.link2.prototype=g2.mix(g2.prototype.ply.prototype,{g2(){return g2().ply(Object.assign({closed:true,ls:"@nodcolor",fs:"transparent",lw:7,lc:"round",lj:"round"},this)).ply(Object.assign({closed:true,ls:"@nodfill2",fs:"transparent",lw:4.5,lc:"round",lj:"round"},this)).ply(Object.assign({closed:true,ls:"@nodfill",fs:"transparent",lw:2,lc:"round",lj:"round"},this))}});g2.prototype.beam=function(){return this.addCommand({c:"beam",a:arguments[0]})};g2.prototype.beam.prototype=g2.mix(g2.prototype.ply.prototype,{g2(){return g2().ply(Object.assign({closed:false,ls:"@linkcolor",fs:"transparent",lw:7,lc:"round",lj:"round"},this))}});g2.prototype.beam2=function(){return this.addCommand({c:"beam2",a:arguments[0]})};g2.prototype.beam2.prototype=g2.mix(g2.prototype.ply.prototype,{g2(){return g2().ply(Object.assign({closed:false,ls:"@nodcolor",fs:"transparent",lw:7,lc:"round",lj:"round"},this)).ply(Object.assign({closed:false,ls:"@nodfill2",fs:"transparent",lw:4.5,lc:"round",lj:"round"},this)).ply(Object.assign({closed:false,ls:"@nodfill",fs:"transparent",lw:2,lc:"round",lj:"round"},this))}});g2.prototype.bar=function(){return this.addCommand({c:"bar",a:arguments[0]})};g2.prototype.bar.prototype=g2.mix(g2.prototype.lin.prototype,{g2(){return g2().lin(Object.assign({ls:"@linkcolor",lw:6,lc:"round"},this))}});g2.prototype.bar2=function(){return this.addCommand({c:"bar2",a:arguments[0]})};g2.prototype.bar2.prototype=g2.mix(g2.prototype.lin.prototype,{g2(){const args=Object.assign({},this);return g2().lin({x1:args.x1,y1:args.y1,x2:args.x2,y2:args.y2,ls:"@nodcolor",lw:7,lc:"round"}).lin({x1:args.x1,y1:args.y1,x2:args.x2,y2:args.y2,ls:"@nodfill2",lw:4.5,lc:"round"}).lin({x1:args.x1,y1:args.y1,x2:args.x2,y2:args.y2,ls:"@nodfill",lw:2,lc:"round"})}});g2.prototype.pulley=function(){return this.addCommand({c:"pulley",a:arguments[0]})};g2.prototype.pulley.prototype=g2.mix(g2.prototype.cir.prototype,{g2(){const args=Object.assign({},this);return g2().beg({x:args.x,y:args.y,w:args.w}).cir({x:0,y:0,r:args.r,ls:"@nodcolor",fs:"#e6e6e6",lw:1}).cir({x:0,y:0,r:args.r-5,ls:"@nodcolor",fs:"#e6e6e6",lw:1}).cir({x:0,y:0,r:args.r-6,ls:"#8e8e8e",fs:"transparent",lw:2}).cir({x:0,y:0,r:args.r-8,ls:"#aeaeae",fs:"transparent",lw:2}).cir({x:0,y:0,r:args.r-10,ls:"#cecece",fs:"transparent",lw:2}).end()}});g2.prototype.pulley2=function(){return this.addCommand({c:"pulley2",a:arguments[0]})};g2.prototype.pulley2.prototype=g2.mix(g2.prototype.cir.prototype,{g2(){const args=Object.assign({},this);return g2().beg({x:args.x,y:args.y,w:args.w}).bar2({x1:0,y1:-args.r+4,x2:0,y2:args.r-4}).bar2({x1:-args.r+4,y1:0,x2:args.r-4,y2:0}).cir({x:0,y:0,r:args.r-2.5,ls:"#e6e6e6",fs:"transparent",lw:5}).cir({x:0,y:0,r:args.r,ls:"@nodcolor",fs:"transparent",lw:1}).cir({x:0,y:0,r:args.r-5,ls:"@nodcolor",fs:"transparent",lw:1}).end()}});g2.prototype.rope=function(){return this.addCommand({c:"rope",a:arguments[0]})};g2.prototype.rope.prototype=g2.mix(g2.prototype.lin.prototype,{g2(){const args=Object.assign({w:0},this);let x1="p1"in args?args.p1.x:"x1"in args?args.x1:"x"in args?args.x:0;let y1="p1"in args?args.p1.y:"y1"in args?args.y1:"y"in args?args.y:0;let x2="p2"in args?args.p2.x:"x2"in args?args.x2:"dx"in args?x1+args.dx:"r"in args?x1+args.r*Math.cos(args.w):x1+10;let y2="p2"in args?args.p2.y:"y2"in args?args.y2:"dy"in args?y1+args.dy:"r"in args?y1+args.r*Math.sin(args.w):y1;let Rmin=10;let R1=args.r1>Rmin?args.r1-2.5:args.r1<-Rmin?args.r1+2.5:0;let R2=args.r2>Rmin?args.r2-2.5:args.r2=pi.y||y>=pj.y)&&(y<=pi.y||y<=pj.y)&&(x<=pi.x||x<=pj.x)&&pi.y!==pj.y&&(pi.x===pj.x||x<=pj.x+(y-pj.y)*(pi.x-pj.x)/(pi.y-pj.y))){match++}}return match%2!=0}return g2().ply({pts:args.pts,closed:true,ls:"transparent",fs:"@linkfill"}).ins(g=>{for(const pts of startLoc){let dist=10*args.lw||10;const{x:x,y:y}=args.pointAt(pts);const t={x:x+Math.cos(args.w)*dist,y:y+Math.sin(args.w)*dist};if(isPntInPly(t,{pts:arr})){while(isPntInPly(t,{pts:arr})){dist++;t.x=x+Math.cos(args.w)*dist,t.y=y+Math.sin(args.w)*dist}g.vec({x1:x,y1:y,x2:t.x,y2:t.y,ls:args.ls||"darkred",lw:args.lw||1})}}})}});"use strict"; +/** + * g2.chart (c) 2015-18 Stefan Goessner + * @author Stefan Goessner + * @license MIT License + * @requires g2.core.js + * @requires g2.ext.js + * @typedef g2 + * @returns {object} chart + * @param {object} args - Chart arguments object or + * @property {float} x - x-position of lower left corner of chart rectangle. + * @property {float} y - y-position of lower left corner of chart rectangle. + * @property {float} [b=150] - width of chart rectangle. + * @property {float} [h=100] - height of chart rectangle. + * @property {string} [ls] - border color. + * @property {string} [fs] - fill color. + * @property {(string|object)} [title] - chart title. + * @property {string} [title.text] - chart title text string. + * @property {float} [title.offset=0] - chart title vertical offset. + * @property {object} [title.style] - chart title style. + * @property {string} [title.style.font=14px serif] - chart title font. + * @property {string} [title.style.thal=center] - chart title horizontal align. + * @property {string} [title.style.tval=bottom] - chart title vertical align. + * @property {array} [funcs=[]] - array of dataset `data` and/or function `fn` objects. + * @property {object} [funcs[item]] - dataset or function object. + * @property {array} [funcs[item].data] - data points as flat array `[x,y,..]`, array of point arrays `[[x,y],..]` or array of point objects `[{x,y},..]`. + * @property {function} [funcs[item].fn] - function `y = f(x)` recieving x-value returning y-value. + * @property {float} [funcs[item].dx] - x increment to apply to function `fn`. Ignored with data points. + * @property {boolean} [funcs[item].fill] - fill region between function graph and x-origin line. + * @property {boolean} [funcs[item].dots] - place circular dots at data points (Avoid with `fn`s). + * @property {boolean|object} [xaxis=false] - x-axis. + * @property {boolean|object} [xaxis.grid=false] - x-axis grid lines. + * @property {string} [xaxis.grid.ls] - x-axis grid line style (color). + * @property {string} [xaxis.grid.lw] - x-axis grid line width. + * @property {string} [xaxis.grid.ld] - x-axis grid line dash style. + * @property {boolean} [xaxis.line=true] - display x-axis base line. + * @property {boolean} [xaxis.origin=false] - display x-axis origin line. + * @property {boolean|object} [yaxis=false] - y-axis. + * @property {boolean|object} [yaxis.grid=false] - y-axis grid lines. + * @property {string} [yaxis.grid.ls] - y-axis grid line style color. + * @property {string} [yaxis.grid.lw] - y-axis grid line width. + * @property {string} [yaxis.grid.ld] - y-axis grid line dash style. + * @property {boolean} [yaxis.line=true] - display y-axis base line. + * @property {boolean} [yaxis.origin=false] - display y-axis origin line. + * @property {float} [xmin] - minimal x-axis value. If not given it is calculated from chart data values. + * @property {float} [xmax] - maximal x-axis value. If not given it is calculated from chart data values. + * @property {float} [ymin] - minimal y-axis value. If not given it is calculated from chart data values. + * @property {float} [ymax] - maximal y-axis value. If not given it is calculated from chart data values. + */g2.prototype.chart=function chart({x:x,y:y,b:b,h:h,style:style,title:title,funcs:funcs,xaxis:xaxis,xmin:xmin,xmax:xmax,yaxis:yaxis,ymin:ymin,ymax:ymax}){return this.addCommand({c:"chart",a:arguments[0]})};g2.prototype.chart.prototype={g2(){const g=g2(),funcs=this.get("funcs"),title=this.title&&this.get("title");if(!this.b)this.b=this.defaults.b;if(!this.h)this.h=this.defaults.h;if(funcs&&funcs.length){const tmp=[this.xmin===undefined,this.xmax===undefined,this.ymin===undefined,this.ymax===undefined];funcs.forEach(f=>this.initFunc(f,...tmp))}this.xAxis=this.autoAxis(this.get("xmin"),this.get("xmax"),0,this.b);this.yAxis=this.autoAxis(this.get("ymin"),this.get("ymax"),0,this.h);g.rec({x:this.x,y:this.y,b:this.b,h:this.h,fs:this.get("fs"),ls:this.get("ls")});g.beg(Object.assign({x:this.x,y:this.y,lw:1},this.defaults.style,this.style));if(title)g.txt(Object.assign({str:this.title&&this.title.text||this.title,x:this.get("b")/2,y:this.get("h")+this.get("title","offset"),w:0},this.defaults.title.style,this.title&&this.title.style||{}));if(this.xaxis)this.drawXAxis(g);if(this.yaxis)this.drawYAxis(g);g.end();if(funcs)funcs.forEach((fnc,i)=>{this.drawFunc(g,fnc,this.defaults.colors[i%this.defaults.colors.length])});return g},initFunc(fn,setXmin,setXmax,setYmin,setYmax){let itr;if(fn.data&&fn.data.length){itr=fn.itr=g2.pntItrOf(fn.data)}else if(fn.fn&&fn.dx){const xmin=+this.xmin||this.defaults.xmin;const xmax=+this.xmax||this.defaults.xmax;itr=fn.itr=(i=>{let x=xmin+i*fn.dx;return{x:x,y:fn.fn(x)}});itr.len=(xmax-xmin)/fn.dx+1}if(itr&&(setXmin||setXmax||setYmin||setYmax)){const xarr=[];const yarr=[];for(let i=0;ithis.xmax)this.xmax=xmax}if(setYmin){const ymin=Math.min(...yarr);if(!this.ymin||yminthis.ymax)this.ymax=ymax}if(fn.color&&typeof fn.color==="number")fn.color=this.defaults.colors[fn.color%this.defaults.colors.length]}},autoAxis(zmin,zmax,tmin,tmax){let base=2,exp=1,eps=Math.sqrt(Number.EPSILON),Dz=zmax-zmin||1,Dt=tmax-tmin||1,scl=Dz>eps?Dt/Dz:1,dz=base*Math.pow(10,exp),dt=Math.floor(scl*dz),N,dt01,i0,j0,jth,t0,res;while(dt<14||dt>35){if(dt<14){if(base==1)base=2;else if(base==2)base=5;else if(base==5){base=1;exp++}}else{if(base==1){base=5;exp--}else if(base==2)base=1;else if(base==5)base=2}dz=base*Math.pow(10,exp);dt=scl*dz}i0=(scl*Math.abs(zmin)+eps/2)%dt9?5:2;return{zmin:zmin,zmax:zmax,base:base,exp:exp,scl:scl,dt:dt,dz:dz,N:N,t0:t0,z0:z0,i0:i0,j0:j0,jth:jth,itr(i){return{t:this.t0+i*this.dt,z:parseFloat((this.z0+i*this.dz).toFixed(Math.abs(this.exp))),maj:(this.j0-this.i0+i)%this.jth===0}}}},drawXAxis(g){let tick,showgrid=this.xaxis&&this.xaxis.grid,gridstyle=showgrid&&Object.assign({},this.defaults.xaxis.grid,this.xaxis.grid),showaxis=this.xaxis||this.xAxis,axisstyle=showaxis&&Object.assign({},this.defaults.xaxis.style,this.defaults.xaxis.labels.style,this.xaxis&&this.xaxis.style||{}),showline=showaxis&&this.get("xaxis","line"),showlabels=this.xAxis&&showaxis&&this.get("xaxis","labels"),showticks=this.xAxis&&showaxis&&this.get("xaxis","ticks"),ticklen=showticks?this.get("xaxis","ticks","len"):0,showorigin=showaxis&&this.get("xaxis","origin"),title=this.xaxis&&(this.get("xaxis","title","text")||this.xaxis.title)||"";g.beg(axisstyle);for(let i=0;i=0)g.lin({x1:-this.xAxis.zmin*this.xAxis.scl,y1:0,x2:-this.xAxis.zmin*this.xAxis.scl,y2:this.h});if(title)g.txt(Object.assign({str:title.text||title,x:this.b/2,y:-(this.get("xaxis","title","offset")+(showticks&&this.get("xaxis","ticks","len")||0)+(showlabels&&this.get("xaxis","labels","offset")||0)+(showlabels&&parseFloat(this.get("xaxis","labels","style","font"))||0)),w:0},this.get("xaxis","title","style")));g.end()},drawYAxis(g){let tick,showgrid=this.yaxis&&this.yaxis.grid,gridstyle=showgrid&&Object.assign({},this.defaults.yaxis.grid,this.yaxis.grid),showaxis=this.yaxis||this.yAxis,axisstyle=showaxis&&Object.assign({},this.defaults.yaxis.style,this.defaults.yaxis.labels.style,this.yaxis&&this.yaxis.style||{}),showline=showaxis&&this.get("yaxis","line"),showlabels=this.yAxis&&showaxis&&this.get("yaxis","labels"),showticks=this.yAxis&&showaxis&&this.get("yaxis","ticks"),ticklen=showticks?this.get("yaxis","ticks","len"):0,showorigin=showaxis&&this.get("yaxis","origin"),title=this.yaxis&&(this.get("yaxis","title","text")||this.yaxis.title)||"";g.beg(axisstyle);for(let i=0;i=0)g.lin({x1:0,y1:-this.yAxis.zmin*this.yAxis.scl,x2:this.b,y2:-this.yAxis.zmin*this.yAxis.scl});if(title)g.txt(Object.assign({str:title.text||title,x:-(this.get("yaxis","title","offset")+(showticks&&this.get("yaxis","ticks","len")||0)+(showlabels&&this.get("yaxis","labels","offset")||0)+(showlabels&&parseFloat(this.get("yaxis","labels","style","font"))||0)),y:this.h/2,w:Math.PI/2},this.get("yaxis","title","style")));g.end()},drawFunc(g,fn,defaultcolor){let itr=fn.itr;if(itr){let fill=fn.fill||fn.style&&fn.style.fs&&fn.style.fs!=="transparent",color=fn.color=fn.color||fn.style&&fn.style.ls||defaultcolor,plydata=[],args=Object.assign({pts:plydata,closed:false,ls:color,fs:fill?g2.color.rgbaStr(color,.125):"transparent",lw:1},fn.style);if(fill)plydata.push(this.pntOf({x:itr(0).x,y:0}));for(let i=0,n=itr.len;iobject | Custom options object. It is simply copied into the 'g2' instance, but not used from the g2 kernel. | +| [opts] | object | Custom options object. | **Example** ```js @@ -18,36 +18,33 @@ const ctx = document.getElementById("c").getContext("2d"); g2() ``` * [g2](#g2) ⇒ [g2](#g2) - * _instance_ - * [.clr()](#g2+clr) ⇒ object - * [.view()](#g2+view) ⇒ object - * [.grid()](#g2+grid) ⇒ object - * [.cir()](#g2+cir) ⇒ object - * [.ell()](#g2+ell) ⇒ object - * [.arc()](#g2+arc) ⇒ object - * [.rec()](#g2+rec) ⇒ object - * [.lin()](#g2+lin) ⇒ object - * [.ply()](#g2+ply) ⇒ object - * [.txt()](#g2+txt) ⇒ object - * [.use()](#g2+use) ⇒ object - * [.img()](#g2+img) ⇒ object - * [.beg()](#g2+beg) ⇒ object - * [.end()](#g2+end) ⇒ object - * [.p()](#g2+p) ⇒ object - * [.z()](#g2+z) ⇒ object - * [.m()](#g2+m) ⇒ object - * [.l()](#g2+l) ⇒ object - * [.q()](#g2+q) ⇒ object - * [.c()](#g2+c) ⇒ object - * [.a()](#g2+a) ⇒ object - * [.stroke()](#g2+stroke) ⇒ object - * [.fill()](#g2+fill) ⇒ object - * [.drw()](#g2+drw) ⇒ object - * [.del()](#g2+del) ⇒ object - * [.ins()](#g2+ins) ⇒ object - * [.exe(ctx)](#g2+exe) ⇒ object - * _static_ - * [.mixin()](#g2.mixin) + * [.clr()](#g2+clr) ⇒ object + * [.view()](#g2+view) ⇒ object + * [.grid()](#g2+grid) ⇒ object + * [.cir()](#g2+cir) ⇒ object + * [.ell()](#g2+ell) ⇒ object + * [.arc()](#g2+arc) ⇒ object + * [.rec()](#g2+rec) ⇒ object + * [.lin()](#g2+lin) ⇒ object + * [.ply()](#g2+ply) ⇒ object + * [.txt()](#g2+txt) ⇒ object + * [.use()](#g2+use) ⇒ object + * [.img()](#g2+img) ⇒ object + * [.beg()](#g2+beg) ⇒ object + * [.end()](#g2+end) ⇒ object + * [.p()](#g2+p) ⇒ object + * [.z()](#g2+z) ⇒ object + * [.m()](#g2+m) ⇒ object + * [.l()](#g2+l) ⇒ object + * [.q()](#g2+q) ⇒ object + * [.c()](#g2+c) ⇒ object + * [.a()](#g2+a) ⇒ object + * [.stroke()](#g2+stroke) ⇒ object + * [.fill()](#g2+fill) ⇒ object + * [.drw()](#g2+drw) ⇒ object + * [.del()](#g2+del) ⇒ object + * [.ins()](#g2+ins) ⇒ object + * [.exe(ctx)](#g2+exe) ⇒ object @@ -481,7 +478,7 @@ Create line segment to point. **Example** ```js -g2().p() // Begin path. .m({x:0,y:50}) // Move to point. .l({x:300,y:0}) // Line segment to point. .l(x:400,y:100}) // ... .stroke() // Stroke path. +g2().p() // Begin path. .m({x:0,y:50}) // Move to point. .l({x:300,y:0}) // Line segment to point. .l({x:400,y:100}) // ... .stroke() // Stroke path. ``` @@ -557,7 +554,7 @@ Draw arc with angular range to target point. **Example** ```js -g2().p() // Begin path. .m({x:50,y:50}) // Move to point. .a({dw:2,x:300,y:100}) // Create cubic bezier curve. .stroke() // Stroke path. .exe(ctx); // Render to canvas context. +g2().p() // Begin path. .m({x:50,y:50}) // Move to point. .a({dw:2,x:300,y:100}) // Create arc segment. .stroke() // Stroke path. .exe(ctx); // Render to canvas context. ``` @@ -648,9 +645,3 @@ Execute g2 commands. It does so automatically and recursively with 'use'ed comma | --- | --- | --- | | ctx | object | Context. | - - -### g2.mixin() -Replacement for Object.assign, as it does not assign getters and setter properly ... See https://medium.com/@benastontweet/mixins-in-javascript-700ec81f5e5c - -**Kind**: static method of [g2](#g2) diff --git a/docs/api/g2.ext.md b/docs/api/g2.ext.md index 283eb03..4b2f09a 100644 --- a/docs/api/g2.ext.md +++ b/docs/api/g2.ext.md @@ -10,92 +10,338 @@ Additional methods for g2. * [g2](#g2) ⇒ [g2](#g2) * _instance_ + * [.nod](#g2+nod) + * [new g2.prototype.nod(args)](#new_g2+nod_new) + * [.dblnod](#g2+dblnod) + * [new g2.prototype.dblnod()](#new_g2+dblnod_new) + * [.pol](#g2+pol) + * [new g2.prototype.pol(args)](#new_g2+pol_new) + * [.gnd](#g2+gnd) + * [new g2.prototype.gnd(args)](#new_g2+gnd_new) + * [.origin](#g2+origin) + * [new g2.prototype.origin(args)](#new_g2+origin_new) + * [.hdl(args)](#g2+hdl) ⇒ object + * [.nodflt(args)](#g2+nodflt) ⇒ object + * [.vec(args)](#g2+vec) ⇒ object + * [.avec(args)](#g2+avec) ⇒ object + * [.dim(args)](#g2+dim) ⇒ object + * [.adim(args)](#g2+adim) ⇒ object * [.spline()](#g2+spline) ⇒ object - * [.label()](#g2+label) ⇒ object - * [.mark()](#g2+mark) ⇒ object * _static_ * [.symbol](#g2.symbol) : object + * [.flatten()](#g2.flatten) - + -### g2.spline() ⇒ object -Draw spline by points. Implementing a centripetal Catmull-Rom spline (thus avoiding cusps and self-intersections). Using iterator function for getting points from array by index. It must return current point object {x,y} or object {done:true}. Default iterator expects sequence of x/y-coordinates as a flat array [x,y,...], array of [[x,y],...] arrays or array of [{x,y},...] objects. +### g2.nod +**Kind**: instance class of [g2](#g2) +**Properties** + +| Name | Type | Description | +| --- | --- | --- | +| x | number | x-value center. | +| y | number | y-value center. | + + + +#### new g2.prototype.nod(args) +Node symbol. + + +| Param | Type | Description | +| --- | --- | --- | +| args | object | symbol arguments object. | + +**Example** +```js +g2().nod({x:10,y:10}) +``` + + +### g2.dblnod +**Kind**: instance class of [g2](#g2) +**Properties** + +| Name | Type | Description | +| --- | --- | --- | +| x | number | x-value center. | +| y | number | y-value center. | + + + +#### new g2.prototype.dblnod() +Double nod symbol + +**Returns**: object - g2 + +| Param | Type | Description | +| --- | --- | --- | +| | object | symbol arguments object. | + +**Example** +```js +g2().dblnod({x:10,y:10}) +``` + + +### g2.pol +**Kind**: instance class of [g2](#g2) +**Properties** + +| Name | Type | Description | +| --- | --- | --- | +| x | number | x-value center. | +| y | number | y-value center. | + + + +#### new g2.prototype.pol(args) +Pole symbol. + +**Returns**: object - g2 + +| Param | Type | Description | +| --- | --- | --- | +| args | object | symbol arguments object. | + +**Example** +```js +g2().pol({x:10,y:10}) +``` + + +### g2.gnd +**Kind**: instance class of [g2](#g2) +**Properties** + +| Name | Type | Description | +| --- | --- | --- | +| x | number | x-value center. | +| y | number | y-value center. | + + + +#### new g2.prototype.gnd(args) +Ground symbol. + + +| Param | Type | Description | +| --- | --- | --- | +| args | object | arguments object. | + +**Example** +```js +g2().gnd({x:10,y:10}) +``` + + +### g2.origin +**Kind**: instance class of [g2](#g2) +**Properties** + +| Name | Type | Description | +| --- | --- | --- | +| x | number | x-value center. | +| y | number | y-value center. | +| w | number | angle in radians. | + + + +#### new g2.prototype.origin(args) +Origin symbol + +**Returns**: object - g2 + +| Param | Type | Description | +| --- | --- | --- | +| args | object | symbol arguments object. | + +**Example** +```js +g2().view({cartesian:true}) .origin({x:10,y:10}) +``` + + +### g2.hdl(args) ⇒ object +Draw interactive handle. **Kind**: instance method of [g2](#g2) **Returns**: object - g2 -**See** -- https://pomax.github.io/bezierinfo -- https://de.wikipedia.org/wiki/Kubisch_Hermitescher_Spline +| Param | Type | Description | +| --- | --- | --- | +| args | object | handle object. | +**Properties** + +| Name | Type | Description | +| --- | --- | --- | +| x | number | x-value center. | +| y | number | y-value center. | + +**Example** +```js +g2().hdl({x:100,y:80}) // Draw handle. +``` + + +### g2.nodflt(args) ⇒ object +**Kind**: instance method of [g2](#g2) +**Returns**: object - g2 | Param | Type | Description | | --- | --- | --- | -| | object | spline arguments object. | +| args | object | symbol arguments object. | + +**Properties** + +| Name | Type | Description | +| --- | --- | --- | +| x | number | x-value center. | +| y | number | y-value center. | + +**Example** +```js +g2().view({cartesian:true}) .nodflt({x:10,y:10}) +``` + + +### g2.vec(args) ⇒ object +Draw vector arrow. + +**Kind**: instance method of [g2](#g2) +**Returns**: object - g2 + +| Param | Type | Description | +| --- | --- | --- | +| args | object | vector arguments object. | + +**Properties** + +| Name | Type | Description | +| --- | --- | --- | +| x1 | number | start x coordinate. | +| y1 | number | start y coordinate. | +| x2 | number | end x coordinate. | +| y2 | number | end y coordinate. | + +**Example** +```js +g2().vec({x1:50,y1:20,x2:250,y2:120}) +``` + + +### g2.avec(args) ⇒ object +Arc as Vector + +**Kind**: instance method of [g2](#g2) +**Returns**: object - g2 + +| Param | Type | Description | +| --- | --- | --- | +| args | object | angular dimension arguments. | **Properties** | Name | Type | Default | Description | | --- | --- | --- | --- | -| pts | Array.<object> \| Array.<Array.<number>> \| Array.<number> | | array of points. | -| [closed] | bool | false | closed spline. | +| x | number | | start x coordinate. | +| y | number | | start y coordinate. | +| r | number | | radius | +| [w] | number | 0 | start angle (in radian). | +| [dw] | number | Math.PI/2 | angular range in radian. In case of positive values it is running counterclockwise with right handed (cartesian) coordinate system. | **Example** ```js -g2().spline({pts:[100,50,50,150,150,150,100,50]}) +g2().avec({x:100,y:70,r:50,w:pi/3,dw:4*pi/3}) ``` - + -### g2.label() ⇒ object -Add label to certain elements. Please note that cartesian flag is necessary. +### g2.dim(args) ⇒ object +Linear Dimension **Kind**: instance method of [g2](#g2) **Returns**: object - g2 | Param | Type | Description | | --- | --- | --- | -| | object | label arguments object. | +| args | object | dimension arguments object. | **Properties** -| Name | Type | Description | +| Name | Type | Default | Description | +| --- | --- | --- | --- | +| x1 | number | | start x coordinate. | +| y1 | number | | start y coordinate. | +| x2 | number | | end x coordinate. | +| y2 | number | | end y coordinate. | +| off | number | | offset. | +| [inside] | boolean | true | draw dimension arrows between or outside of ticks. | + +**Example** +```js +g2().dim({x1:60,y1:40,x2:190,y2:120}) +``` + + +### g2.adim(args) ⇒ object +Angular dimension + +**Kind**: instance method of [g2](#g2) +**Returns**: object - g2 +**Depricated**: boolean [inside] - draw dimension arrows between ticks. + +| Param | Type | Description | | --- | --- | --- | -| str | string | label text | -| loc | number \| string | label location depending on referenced element.
'c': centered, wrt. rec, cir, arc
'beg','mid', 'end', wrt. lin
'n', 'ne', 'e', 'se', 's', 'sw', 'w', or 'nw': cardinal directions | -| off | number | offset distance [optional]. | +| args | object | angular dimension arguments. | + +**Properties** + +| Name | Type | Default | Description | +| --- | --- | --- | --- | +| x | number | | start x coordinate. | +| y | number | | start y coordinate. | +| r | number | | radius | +| [w] | number | 0 | start angle (in radian). | +| [dw] | number | Math.PI/2 | angular range in radian. In case of positive values it is running counterclockwise with right handed (cartesian) coordinate system. | +| [outside] | boolean | false | draw dimension arrows outside of ticks. | **Example** ```js -g2().view({cartesian:true}) .cir({x:10,y:10,r:5}) .label({str:'hello',loc:'s',off:10}) +g2().adim({x:100,y:70,r:50,w:pi/3,dw:4*pi/3}) ``` - + -### g2.mark() ⇒ object -Draw marker on line element. +### g2.spline() ⇒ object +Draw spline by points. Implementing a centripetal Catmull-Rom spline (thus avoiding cusps and self-intersections). Using iterator function for getting points from array by index. It must return current point object {x,y} or object {done:true}. Default iterator expects sequence of x/y-coordinates as a flat array [x,y,...], array of [[x,y],...] arrays or array of [{x,y},...] objects. **Kind**: instance method of [g2](#g2) **Returns**: object - g2 +**See** + +- https://pomax.github.io/bezierinfo +- https://de.wikipedia.org/wiki/Kubisch_Hermitescher_Spline + | Param | Type | Description | | --- | --- | --- | -| | object | Marker arguments object. | +| | object | spline arguments object. | **Properties** | Name | Type | Default | Description | | --- | --- | --- | --- | -| mrk | object \| string | | `g2` object or `name` of mark in `symbol` namespace. | -| loc | number \| string \| Array.<number> \| Array.<string> | | line location ['beg','end',0.1,0.9,'mid',...].
| -| [dir] | int | 0 | Direction:
-1 : negative tangent direction
0 : no orientation (rotation)
1 : positive tangent direction | +| pts | Array.<object> \| Array.<Array.<number>> \| Array.<number> | | array of points. | +| [closed] | bool | false | closed spline. | **Example** ```js -g2().lin({x1:10,y1:10,x2:100,y2:10}) .mark({mrk:"tick",loc:0.75,dir:1}) +g2().spline({pts:[100,50,50,150,150,150,100,50]}) ``` ### g2.symbol : object -Extensed style values. Not really meant to get overwritten. But if you actually want, proceed.
Theses styles can be referenced using the comfortable '@' syntax. +Extended style values. Not really meant to get overwritten. But if you actually want, proceed.
These styles can be referenced using the comfortable '@' syntax. **Kind**: static namespace of [g2](#g2) **Properties** @@ -119,3 +365,9 @@ Extensed style values. Not really meant to get overwritten. But if you actually | [symbol.labelOffset] | number | 5 | default label offset distance. | | [symbol.labelSignificantDigits] | number | 3 | default label's significant digits after numbering point. | + + +### g2.flatten() +Flatten object properties (evaluate getters) + +**Kind**: static method of [g2](#g2) diff --git a/docs/api/g2.mec.md b/docs/api/g2.mec.md index f4619ca..da8484b 100644 --- a/docs/api/g2.mec.md +++ b/docs/api/g2.mec.md @@ -9,141 +9,21 @@ Mechanical extensions. (Requires cartesian coordinates) **License**: MIT License * [g2](#g2) ⇒ [g2](#g2) - * _instance_ - * [.dim()](#g2+dim) ⇒ object - * [.adim()](#g2+adim) ⇒ object - * [.vec()](#g2+vec) ⇒ object - * [.avec()](#g2+avec) ⇒ object - * [.slider()](#g2+slider) ⇒ object - * [.spring()](#g2+spring) ⇒ object - * [.damper()](#g2+damper) ⇒ object - * [.link()](#g2+link) ⇒ object - * [.link2()](#g2+link2) ⇒ object - * [.beam()](#g2+beam) ⇒ object - * [.beam2()](#g2+beam2) ⇒ object - * [.bar()](#g2+bar) ⇒ object - * [.bar2()](#g2+bar2) ⇒ object - * [.pulley()](#g2+pulley) ⇒ object - * [.pulley2()](#g2+pulley2) ⇒ object - * [.rope()](#g2+rope) ⇒ object - * [.ground()](#g2+ground) ⇒ object - * [.load()](#g2+load) ⇒ object - * [.pol()](#g2+pol) ⇒ object - * [.gnd()](#g2+gnd) ⇒ object - * [.nod()](#g2+nod) ⇒ object - * [.dblnod()](#g2+dblnod) ⇒ object - * [.nodfix()](#g2+nodfix) ⇒ object - * [.nodflt()](#g2+nodflt) ⇒ object - * [.origin()](#g2+origin) ⇒ object - * _static_ - * [.State](#g2.State) : object - - - -### g2.dim() ⇒ object -Dimension + * [.slider()](#g2+slider) ⇒ object + * [.spring()](#g2+spring) ⇒ object + * [.damper()](#g2+damper) ⇒ object + * [.link()](#g2+link) ⇒ object + * [.link2()](#g2+link2) ⇒ object + * [.beam()](#g2+beam) ⇒ object + * [.beam2()](#g2+beam2) ⇒ object + * [.bar()](#g2+bar) ⇒ object + * [.bar2()](#g2+bar2) ⇒ object + * [.pulley()](#g2+pulley) ⇒ object + * [.pulley2()](#g2+pulley2) ⇒ object + * [.rope()](#g2+rope) ⇒ object + * [.ground()](#g2+ground) ⇒ object + * [.load()](#g2+load) ⇒ object -**Kind**: instance method of [g2](#g2) -**Returns**: object - g2 - -| Param | Type | Description | -| --- | --- | --- | -| | object | dimension arguments object. | - -**Properties** - -| Name | Type | Default | Description | -| --- | --- | --- | --- | -| x1 | number | | start x coordinate. | -| y1 | number | | start y coordinate. | -| x2 | number | | end x coordinate. | -| y2 | number | | end y coordinate. | -| off | number | | offset. | -| over | number | | overshoot of offset lines. | -| [inside] | boolean | true | draw dimension arrows between or outside of ticks. | - -**Example** -```js -g2().dim({x1:60,y1:40,x2:190,y2:120}) -``` - - -### g2.adim() ⇒ object -Angular dimension - -**Kind**: instance method of [g2](#g2) -**Returns**: object - g2 - -| Param | Type | Description | -| --- | --- | --- | -| | object | angular dimension arguments. | - -**Properties** - -| Name | Type | Default | Description | -| --- | --- | --- | --- | -| x | number | | start x coordinate. | -| y | number | | start y coordinate. | -| r | number | | radius | -| [w] | number | 0 | start angle (in radian). | -| [dw] | number | Math.PI/2 | angular range in radian. In case of positive values it is running counterclockwise with right handed (cartesian) coordinate system. | -| [inside] | boolean | true | draw dimension arrows between or outside of ticks. | - -**Example** -```js -g2().adim({x:100,y:70,r:50,w:pi/3,dw:4*pi/3}) -``` - - -### g2.vec() ⇒ object -Draw vector arrow. - -**Kind**: instance method of [g2](#g2) -**Returns**: object - g2 - -| Param | Type | Description | -| --- | --- | --- | -| | object | vector arguments object. | - -**Properties** - -| Name | Type | Description | -| --- | --- | --- | -| x1 | number | start x coordinate. | -| y1 | number | start y coordinate. | -| x2 | number | end x coordinate. | -| y2 | number | end y coordinate. | - -**Example** -```js -g2().vec({x1:50,y1:20,x2:250,y2:120}) -``` - - -### g2.avec() ⇒ object -Draw vector with an angle - -**Kind**: instance method of [g2](#g2) -**Returns**: object - g2 - -| Param | Type | Description | -| --- | --- | --- | -| | object | angle vector arguments object | - -**Properties** - -| Name | Type | Default | Description | -| --- | --- | --- | --- | -| x | number | | x-value center. | -| y | number | | y-value center. | -| r | number | | radius. | -| [w] | number | 0 | start angle (in radian). | -| [dw] | number | 2*pi | angular range in Radians. | - -**Example** -```js -g2().avec({x:300,y:400,r:390,w:-Math.PI/4,dw:-Math.PI/2}) .exe(ctx); -``` ### g2.slider() ⇒ object @@ -489,178 +369,3 @@ Polygonial line load. The first and last point define the base line onto which t | w | number | angle of vectors. | | spacing | number | spacing of the vectors drawn as a positive real number, interprete as
* spacing < 1: spacing = 1/m with a partition of m.
* spacing > 1: length of spacing. | - - -### g2.pol() ⇒ object -Creates a symbol at given coordinates. - -**Kind**: instance method of [g2](#g2) -**Returns**: object - g2 - -| Param | Type | Description | -| --- | --- | --- | -| | object | symbol arguments object. | - -**Properties** - -| Name | Type | Description | -| --- | --- | --- | -| x | number | x-value center. | -| y | number | y-value center. | - -**Example** -```js -g2().pol({x:10,y:10}) -``` - - -### g2.gnd() ⇒ object -**Kind**: instance method of [g2](#g2) -**Returns**: object - g2 - -| Param | Type | Description | -| --- | --- | --- | -| | object | symbol arguments object. | - -**Properties** - -| Name | Type | Description | -| --- | --- | --- | -| x | number | x-value center. | -| y | number | y-value center. | - -**Example** -```js -g2().gnd({x:10,y:10}) -``` - - -### g2.nod() ⇒ object -**Kind**: instance method of [g2](#g2) -**Returns**: object - g2 - -| Param | Type | Description | -| --- | --- | --- | -| | object | symbol arguments object. | - -**Properties** - -| Name | Type | Description | -| --- | --- | --- | -| x | number | x-value center. | -| y | number | y-value center. | - -**Example** -```js -g2().nod({x:10,y:10}) -``` - - -### g2.dblnod() ⇒ object -**Kind**: instance method of [g2](#g2) -**Returns**: object - g2 - -| Param | Type | Description | -| --- | --- | --- | -| | object | symbol arguments object. | - -**Properties** - -| Name | Type | Description | -| --- | --- | --- | -| x | number | x-value center. | -| y | number | y-value center. | - -**Example** -```js -g2().dblnod({x:10,y:10}) -``` - - -### g2.nodfix() ⇒ object -Since some symbols are not symmetrical, the cartesian mode is recommended. - -**Kind**: instance method of [g2](#g2) -**Returns**: object - g2 - -| Param | Type | Description | -| --- | --- | --- | -| | object | symbol arguments object. | - -**Properties** - -| Name | Type | Description | -| --- | --- | --- | -| x | number | x-value center. | -| y | number | y-value center. | - -**Example** -```js -g2().view({cartesian:true}) .nodfix({x:10,y:10}) -``` - - -### g2.nodflt() ⇒ object -**Kind**: instance method of [g2](#g2) -**Returns**: object - g2 - -| Param | Type | Description | -| --- | --- | --- | -| | object | symbol arguments object. | - -**Properties** - -| Name | Type | Description | -| --- | --- | --- | -| x | number | x-value center. | -| y | number | y-value center. | - -**Example** -```js -g2().view({cartesian:true}) .nodflt({x:10,y:10}) -``` - - -### g2.origin() ⇒ object -**Kind**: instance method of [g2](#g2) -**Returns**: object - g2 - -| Param | Type | Description | -| --- | --- | --- | -| | object | symbol arguments object. | - -**Properties** - -| Name | Type | Description | -| --- | --- | --- | -| x | number | x-value center. | -| y | number | y-value center. | - -**Example** -```js -g2().view({cartesian:true}) .origin({x:10,y:10}) -``` - - -### g2.State : object -Mechanical style values. Not really meant to get overwritten. But if you actually want, proceed.
Theses styles can be referenced using the comfortable '@' syntax. - -**Kind**: static namespace of [g2](#g2) -**Properties** - -| Name | Type | Default | Description | -| --- | --- | --- | --- | -| State | object | | `g2` state namespace. | -| [State.nodcolor] | string | "#333" | node color. | -| [State.nodfill] | string | "#dedede" | node fill color. | -| [State.nodfill2] | string | "#aeaeae" | alternate node fill color, somewhat darker. | -| [State.linkcolor] | string | "#666" | link color. | -| [State.linkfill] | string | "rgba(225,225,225,0.75)" | link fill color, semi-transparent. | -| [State.dimcolor] | string | "darkslategray" | dimension color. | -| [State.solid] | array | [] | solid line style. | -| [State.dash] | array | [15,10] | dashed line style. | -| [State.dot] | array | [4,4] | dotted line style. | -| [State.dashdot] | array | [25,6.5,2,6.5] | dashdotted line style. | -| [State.labelOffset] | number | 5 | default label offset distance. | -| [State.labelSignificantDigits] | number | 3 | default label's significant digits after numbering point. | - diff --git a/docs/img/adim.png b/docs/img/adim.png index c7fa45c..ac46535 100644 Binary files a/docs/img/adim.png and b/docs/img/adim.png differ diff --git a/docs/img/arc2.png b/docs/img/arc2.png new file mode 100644 index 0000000..a85488a Binary files /dev/null and b/docs/img/arc2.png differ diff --git a/docs/img/avec.png b/docs/img/avec.png index eeb710c..6c3a1d9 100644 Binary files a/docs/img/avec.png and b/docs/img/avec.png differ diff --git a/docs/img/beg-end.png b/docs/img/beg-end.png new file mode 100644 index 0000000..701f9a4 Binary files /dev/null and b/docs/img/beg-end.png differ diff --git a/docs/img/chart2.png b/docs/img/chart2.png new file mode 100644 index 0000000..5ffe4c7 Binary files /dev/null and b/docs/img/chart2.png differ diff --git a/docs/img/chart3.png b/docs/img/chart3.png new file mode 100644 index 0000000..b6e71e6 Binary files /dev/null and b/docs/img/chart3.png differ diff --git a/docs/img/chart4.png b/docs/img/chart4.png new file mode 100644 index 0000000..a9362d4 Binary files /dev/null and b/docs/img/chart4.png differ diff --git a/docs/img/cir.png b/docs/img/cir.png index 6103786..33b6cac 100644 Binary files a/docs/img/cir.png and b/docs/img/cir.png differ diff --git a/docs/img/cir2.png b/docs/img/cir2.png new file mode 100644 index 0000000..0736d85 Binary files /dev/null and b/docs/img/cir2.png differ diff --git a/docs/img/dim2.png b/docs/img/dim2.png new file mode 100644 index 0000000..e16a7ec Binary files /dev/null and b/docs/img/dim2.png differ diff --git a/docs/img/drw2.png b/docs/img/drw2.png new file mode 100644 index 0000000..8137729 Binary files /dev/null and b/docs/img/drw2.png differ diff --git a/docs/img/drw3.png b/docs/img/drw3.png new file mode 100644 index 0000000..a3a22d4 Binary files /dev/null and b/docs/img/drw3.png differ diff --git a/docs/img/g2.ext.js.png b/docs/img/g2.ext.js.png new file mode 100644 index 0000000..09b877a Binary files /dev/null and b/docs/img/g2.ext.js.png differ diff --git a/docs/img/g2.mec.js.png b/docs/img/g2.mec.js.png new file mode 100644 index 0000000..fde9754 Binary files /dev/null and b/docs/img/g2.mec.js.png differ diff --git a/docs/img/image2.png b/docs/img/image2.png new file mode 100644 index 0000000..5cebb11 Binary files /dev/null and b/docs/img/image2.png differ diff --git a/docs/img/img10.png b/docs/img/img10.png new file mode 100644 index 0000000..34fc0db Binary files /dev/null and b/docs/img/img10.png differ diff --git a/docs/img/img11.png b/docs/img/img11.png new file mode 100644 index 0000000..3a6d067 Binary files /dev/null and b/docs/img/img11.png differ diff --git a/docs/img/img12.png b/docs/img/img12.png new file mode 100644 index 0000000..7b5cae8 Binary files /dev/null and b/docs/img/img12.png differ diff --git a/docs/img/img13.png b/docs/img/img13.png new file mode 100644 index 0000000..5557b3e Binary files /dev/null and b/docs/img/img13.png differ diff --git a/docs/img/img2.png b/docs/img/img2.png new file mode 100644 index 0000000..5cebb11 Binary files /dev/null and b/docs/img/img2.png differ diff --git a/docs/img/img3.png b/docs/img/img3.png new file mode 100644 index 0000000..bd63007 Binary files /dev/null and b/docs/img/img3.png differ diff --git a/docs/img/img4.png b/docs/img/img4.png new file mode 100644 index 0000000..6802c78 Binary files /dev/null and b/docs/img/img4.png differ diff --git a/docs/img/img5.png b/docs/img/img5.png new file mode 100644 index 0000000..455e5dd Binary files /dev/null and b/docs/img/img5.png differ diff --git a/docs/img/img6.png b/docs/img/img6.png new file mode 100644 index 0000000..4975a7e Binary files /dev/null and b/docs/img/img6.png differ diff --git a/docs/img/img7.png b/docs/img/img7.png new file mode 100644 index 0000000..9766107 Binary files /dev/null and b/docs/img/img7.png differ diff --git a/docs/img/img8.png b/docs/img/img8.png new file mode 100644 index 0000000..c5f1d33 Binary files /dev/null and b/docs/img/img8.png differ diff --git a/docs/img/img9.png b/docs/img/img9.png new file mode 100644 index 0000000..0c0ca24 Binary files /dev/null and b/docs/img/img9.png differ diff --git a/docs/img/label2.png b/docs/img/label2.png new file mode 100644 index 0000000..f53e3b0 Binary files /dev/null and b/docs/img/label2.png differ diff --git a/docs/img/label3.png b/docs/img/label3.png new file mode 100644 index 0000000..137874b Binary files /dev/null and b/docs/img/label3.png differ diff --git a/docs/img/lin.png b/docs/img/lin.png index cac074f..44604be 100644 Binary files a/docs/img/lin.png and b/docs/img/lin.png differ diff --git a/docs/img/load2.png b/docs/img/load2.png new file mode 100644 index 0000000..b685485 Binary files /dev/null and b/docs/img/load2.png differ diff --git a/docs/img/mark.png b/docs/img/mark.png index 9f1af66..64c641e 100644 Binary files a/docs/img/mark.png and b/docs/img/mark.png differ diff --git a/docs/img/mark2.png b/docs/img/mark2.png new file mode 100644 index 0000000..9349a8f Binary files /dev/null and b/docs/img/mark2.png differ diff --git a/docs/img/mark3.png b/docs/img/mark3.png new file mode 100644 index 0000000..7b30022 Binary files /dev/null and b/docs/img/mark3.png differ diff --git a/docs/img/mec.png b/docs/img/mec.png index 3554f94..89816a4 100644 Binary files a/docs/img/mec.png and b/docs/img/mec.png differ diff --git a/docs/img/path.png b/docs/img/path.png new file mode 100644 index 0000000..a42b9ff Binary files /dev/null and b/docs/img/path.png differ diff --git a/docs/img/path2.png b/docs/img/path2.png new file mode 100644 index 0000000..281afcd Binary files /dev/null and b/docs/img/path2.png differ diff --git a/docs/img/path3.png b/docs/img/path3.png new file mode 100644 index 0000000..df1a85e Binary files /dev/null and b/docs/img/path3.png differ diff --git a/docs/img/path4.png b/docs/img/path4.png new file mode 100644 index 0000000..5b3a799 Binary files /dev/null and b/docs/img/path4.png differ diff --git a/docs/img/path5.png b/docs/img/path5.png new file mode 100644 index 0000000..071b9cd Binary files /dev/null and b/docs/img/path5.png differ diff --git a/docs/img/ply.png b/docs/img/ply.png index 6c85a72..ac6840b 100644 Binary files a/docs/img/ply.png and b/docs/img/ply.png differ diff --git a/docs/img/ply2.png b/docs/img/ply2.png new file mode 100644 index 0000000..2ece2fb Binary files /dev/null and b/docs/img/ply2.png differ diff --git a/docs/img/ply3.png b/docs/img/ply3.png new file mode 100644 index 0000000..2ece2fb Binary files /dev/null and b/docs/img/ply3.png differ diff --git a/docs/img/ply4.png b/docs/img/ply4.png new file mode 100644 index 0000000..2ece2fb Binary files /dev/null and b/docs/img/ply4.png differ diff --git a/docs/img/ply5.png b/docs/img/ply5.png new file mode 100644 index 0000000..ba40419 Binary files /dev/null and b/docs/img/ply5.png differ diff --git a/docs/img/rec.png b/docs/img/rec.png index 25c93d1..c5b6a92 100644 Binary files a/docs/img/rec.png and b/docs/img/rec.png differ diff --git a/docs/img/shadow.png b/docs/img/shadow.png new file mode 100644 index 0000000..c6c0708 Binary files /dev/null and b/docs/img/shadow.png differ diff --git a/docs/img/spline2.png b/docs/img/spline2.png new file mode 100644 index 0000000..cf56f44 Binary files /dev/null and b/docs/img/spline2.png differ diff --git a/docs/img/spline3.png b/docs/img/spline3.png new file mode 100644 index 0000000..efee615 Binary files /dev/null and b/docs/img/spline3.png differ diff --git a/docs/img/symbols.png b/docs/img/symbols.png index bf30669..164138d 100644 Binary files a/docs/img/symbols.png and b/docs/img/symbols.png differ diff --git a/docs/img/txt.png b/docs/img/txt.png index ac007bf..326dba5 100644 Binary files a/docs/img/txt.png and b/docs/img/txt.png differ diff --git a/docs/img/txt2.png b/docs/img/txt2.png new file mode 100644 index 0000000..aebd6f4 Binary files /dev/null and b/docs/img/txt2.png differ diff --git a/docs/img/txt3.png b/docs/img/txt3.png new file mode 100644 index 0000000..754ded9 Binary files /dev/null and b/docs/img/txt3.png differ diff --git a/docs/img/txt4.png b/docs/img/txt4.png new file mode 100644 index 0000000..f744b74 Binary files /dev/null and b/docs/img/txt4.png differ diff --git a/docs/img/txt5.png b/docs/img/txt5.png new file mode 100644 index 0000000..ed4b410 Binary files /dev/null and b/docs/img/txt5.png differ diff --git a/docs/img/use2.png b/docs/img/use2.png new file mode 100644 index 0000000..9c3f15a Binary files /dev/null and b/docs/img/use2.png differ diff --git a/docs/img/vec.png b/docs/img/vec.png index 24e4d67..0d8b950 100644 Binary files a/docs/img/vec.png and b/docs/img/vec.png differ diff --git a/docs/img/view.png b/docs/img/view.png index 2ebc098..c957a8d 100644 Binary files a/docs/img/view.png and b/docs/img/view.png differ diff --git a/docs/img/view2.png b/docs/img/view2.png new file mode 100644 index 0000000..7b3caf4 Binary files /dev/null and b/docs/img/view2.png differ diff --git a/docs/img/view3.png b/docs/img/view3.png new file mode 100644 index 0000000..c7bc2ed Binary files /dev/null and b/docs/img/view3.png differ diff --git a/docs/img/view4.png b/docs/img/view4.png new file mode 100644 index 0000000..ac3e3ce Binary files /dev/null and b/docs/img/view4.png differ diff --git a/docs/index.html b/docs/index.html index 0355ed7..fac2c44 100644 --- a/docs/index.html +++ b/docs/index.html @@ -33,7 +33,8 @@
  • Insert
  • Animation
  • Interactivity
  • -
  • Custom HTML Element
  • +
  • Custom HTML element
  • +
  • Custom chart element

    • @@ -70,7 +71,7 @@

      Main features #
    • Method chaining.
    • Support of cartesian coordinates.
    • Viewport pan and zoom transformations.
    • -
    • Low level path commands with short names adopted from SVG.
    • +
    • Low-level path commands with short names adopted from SVG.
    • Higher level element commands.
    • Maintaining a state stack for styling and transformations.
    • Easy way to build custom symbol libraries.
    • @@ -78,16 +79,16 @@

      Main features #
    • No dependency.

    Minimal Example #

    -
    <canvas id="c" width="200", height="100"></canvas> -<script src="g2.js"></script> +
    <canvas id="c" width="200", height="100"></canvas> +<script src="g2.js"></script> <script> - const ctx = document.getElementById("c").getContext("2d"); // define context + const ctx = document.getElementById("c").getContext("2d"); // define context g2().rec({x:40,y:30,b:120,h:40, // create g2 object, add rectangle - ls:"green",fs:"orange",lw:3}) // with style properties. + ls:"green",fs:"orange",lw:3}) // with style properties. .exe(ctx); // draw to canvas. </script>
    -

    first

    +

    ![first](/img/g2-first.png

    API Reference #

    See the API Reference for g2 for details.

    Also see the API Reference for g2.ext and the API Reference for g2.mec.

    @@ -103,7 +104,7 @@

    Cheat Sheet #

    GitCDN #

    Use the link https://cdn.jsdelivr.net/gh/goessner/g2/dist/g2.js for getting the latest commit as a raw file.

    In HTML use ...

    -
    <script src="https://cdn.jsdelivr.net/gh/goessner/g2/dist/g2.js"></script> +
    <script src="https://cdn.jsdelivr.net/gh/goessner/g2/dist/g2.js"></script>

    Tests #

    Tests are found on the Github Page

    diff --git a/docs/index.md b/docs/index.md index c3a4c12..f5b1e62 100644 --- a/docs/index.md +++ b/docs/index.md @@ -9,7 +9,7 @@ "tags": [] --- - [![License](http://img.shields.io/:license-mit-blue.svg)](https://github.com/goessner/g2/blob/master/LICENSE) +[![License](http://img.shields.io/:license-mit-blue.svg)](https://github.com/goessner/g2/blob/master/LICENSE) [![npm](https://img.shields.io/npm/v/g2d.svg)](https://npmjs.com/package/g2d) [![npm](https://img.shields.io/npm/dt/g2d.svg)](https://npmjs.com/package/g2d) [![no dependencies](https://img.shields.io/gemnasium/mathiasbynens/he.svg)](https://github.com/goessner/g2) @@ -26,7 +26,7 @@ So the API is minimal and easy to understand. The library is tiny, fast and rend * Method chaining. * Support of cartesian coordinates. * Viewport pan and zoom transformations. -* Low level path commands with short names adopted from SVG. +* Low-level path commands with short names adopted from SVG. * Higher level element commands. * Maintaining a state stack for styling and transformations. * Easy way to build custom symbol libraries. @@ -45,7 +45,7 @@ So the API is minimal and easy to understand. The library is tiny, fast and rend .exe(ctx); // draw to canvas. ``` -![first](/img/g2-first.png) +![first](/img/g2-first.png ## API Reference See the [API Reference for g2](api/g2.core.md) for details. diff --git a/docs/index/index.chart.js b/docs/index/index.chart.js index de3515a..d76271a 100644 --- a/docs/index/index.chart.js +++ b/docs/index/index.chart.js @@ -1,48 +1,43 @@ -new TestContainer("chart", - tests = [ - new Test('chart', - `g2().view({cartesian:true}) - .chart({x:40,y:40,b:190,h:90, +tests.push( + new Test('chart', +`g2().view({cartesian:true}) + .chart({x:40,y:40,b:190,h:90, funcs:[{data:[-2,4,2,-1,3,3,4,2],fill:true}], title:{text:"chart"}, xaxis:{title:"x-axis",grid:true,origin:true}, yaxis:{title:"y-axis"}, })`), - new Test('chart-2', - `g2().view({cartesian:true}) - .chart({x:40,y:40,b:190,h:90,fs:'black', - funcs:[ - {data:[-2,4,2,-1,3,3,4,2],style:{ls:'violet',lw:3}}, - {fn:Math.sin,dx:Math.PI/20,style:{ls:'green',lw:3}} - ], - title:{text:"chart"}, - xaxis:{title:"x-axis",origin:true}, - yaxis:{title:"y-axis",origin:true}, - })`), + new Test('chart2', + `g2().view({cartesian:true}) +.chart({x:40,y:40,b:190,h:90,fs:'black', +funcs:[ + {data:[-2,4,2,-1,3,3,4,2],style:{ls:'violet',lw:3}}, + {fn:Math.sin,dx:Math.PI/20,style:{ls:'green',lw:3}} +], +title:{text:"chart"}, +xaxis:{title:"x-axis",origin:true}, +yaxis:{title:"y-axis",origin:true}, +})`), - new Test('chart-3', - `g2().view({cartesian:true}) - .chart({x:40,y:40,b:190,h:90,fs:'black', - funcs:[ - {data:[-2,4,2,-1,3,3,4,2],style:{ls:'violet', - lw:3,sh:[5,5,5,'rgba(255,0,255,0.7)']}}, - {fn:Math.sin,dx:Math.PI/20,style:{ls:'green', - lw:3,sh:[5,5,5,'rgba(0,255,0,0.5)']}} - ], - title:{text:"chart"}, - xaxis:{title:"x-axis",origin:true}, - yaxis:{title:"y-axis",origin:true}, - })`), - new Test('chart-4', - `g2().view({cartesian:true}) - .chart({x:40,y:40,b:190,h:90, - funcs:[{data:[-2,4,2,-1,3,3,4,2],fill:true}], - title:{text:"chart",style:{fs:'blue'}}, - xaxis:{ - title:{text:"x-axis",style:{fs:'#0f0'}}, - grid:true,origin:true,labels:{style:{fs:'#f0f'}} - }, - yaxis:{title:"y-axis",style:{fs:'rgb(255,0,0)'}}, - })`), - ] -); + new Test('chart3', + `g2().view({cartesian:true}) +.chart({x:40,y:40,b:190,h:90,fs:'black', + funcs:[ + {data:[-2,4,2,-1,3,3,4,2],style:{ls:'violet',lw:3,sh:[5,5,5,'rgba(255,0,255,0.7)']}}, + {fn:Math.sin,dx:Math.PI/20,style:{ls:'green',lw:3,sh:[5,5,5,'rgba(0,255,0,0.5)']}} + ], + title:{text:"chart"}, + xaxis:{title:"x-axis",origin:true}, + yaxis:{title:"y-axis",origin:true}, +})`), + new Test('chart4', + `g2().view({cartesian:true}) +.chart({x:40,y:40,b:190,h:90, + funcs:[{data:[-2,4,2,-1,3,3,4,2],fill:true}], + title:{text:"chart",style:{fs:'blue'}}, + xaxis:{ + title:{text:"x-axis",style:{fs:'#0f0'}}, + grid:true,origin:true,labels:{style:{fs:'#f0f'}} + }, + yaxis:{title:"y-axis",style:{fs:'rgb(255,0,0)'}}, +})`)) diff --git a/docs/index/index.core.js b/docs/index/index.core.js deleted file mode 100644 index 703eb60..0000000 --- a/docs/index/index.core.js +++ /dev/null @@ -1,269 +0,0 @@ -new TestContainer( - "core", - tests = [ - new Test('clr', - `g2().grid().clr()`), - - new Test('view', - `g2().view({cartesian:true}).grid().origin()`), - - new Test('view-2', - `g2().view({x:25,y:30}).grid().origin()`), - - new Test('view-3', - `g2().view({scl:1.5,x:35,y:40}).grid().origin()`), - - new Test('view-4', - `g2().view({x:35,y:35}).grid().view({scl:1.5}).origin()`), - - new Test('grid', - `g2().grid()`), - - new Test('cir', - `g2().cir({x:40,y:40,r:20,fs:'yellow'}) - .cir({x:120,y:60,r:40,ls:'red'}) - .cir({x:200,y:80,r:30,ld:[1,0,1]})`), - - new Test('cir-2', - `g2().cir({x:40,y:40,r:20,fs:'yellow',sh:[5,5,5,'rgba(0,0,0,0.5)']}) - .cir({x:120,y:60,r:40,ls:'red',sh:[5,5,5,'rgba(0,0,0,0.7)']}) - .cir({x:200,y:80,r:30,ld:[1,0,1],sh:[5,5,5,'rgba(0,0,0,1)']})`), - - new Test('ell', - `g2().ell({x:100,y:50,rx:20,ry:40,rot:2,fs:'orange'}) - .ell({x:210,y:50,rx:10,ry:30,w:1,dw:5,ls:'blue'})`), - - new Test('arc', - `g2().arc({x:90,y:50,r:35,w:1/3*pi,dw:4/3*pi, - ls:'blue',lw:5,fs:'#ddd'}) - .arc({x:120,y:50,r:35,w:1/3*pi,dw:-2/3*pi, - ls:'blue',lw:5,fs:'#ddd'})`), - - new Test('arc-2', - `g2().arc({x:90,y:50,r:35,w:1/3*pi,dw:4/3*pi, - ls:'green',lw:5,fs:'#ddd',sh:[5,5,5,'rgba(0,0,0,0.5)']}) - .arc({x:120,y:50,r:35,w:1/3*pi,dw:-2/3*pi, - ls:'green',lw:5,fs:'#ddd',sh:[5,5,5,'rgba(0,0,0,0.5)']})`), - - new Test('rec', - `g2().rec({x:60,y:20,b:80,h:40,ls:'red',lw:3,fs:'#ddd'}) - .rec({x:120,y:70,b:80,h:40,ls:'blue',lw:3, - fs:'#ddd',sh:[5,5,5,'rgba(0,0,0,0.5)']})`), - - new Test('lin', - `g2().lin({x1:20,y1:30,x2:180,y2:80,ls:'green',lw:3}) - .lin({x1:30,y1:100,x2:200,y2:120,ls:'orange', - lw:3,sh:[5,5,5,'rgba(0,0,0,0.5)']})`), - - new Test('ply', - `g2().ply({pts:[20,10,60,80,120,30,180,90], - ls:'red',lw:3,fs:'#ddd'})`), - - new Test('ply-2', - `g2().ply({pts:[[20,10],[60,80],[120,30],[180,90]], - ls:'red',lw:3,fs:'#ddd',w:0.3})`), - - new Test('ply-3', - `g2().ply({y:30, - pts:[{x:20,y:10},{x:60,y:80},{x:120,y:30},{x:180,y:90}], - ls:'red',lw:3,fs:'#ddd',w:-0.3})`), - - new Test('ply-4', - `g2().ply({pts:[20,10,60,80,120,30,180,90],ls:'red',lw:3}) - .ply({pts:[20,50,60,110,120,60,180,120], - ls:'red',lw:3,sh:[5,5,5,'rgba(0,0,0,0.5)']})`), - - new Test('ply-5', - `g2().ply({pts:[20,10,60,80,120,30,180,90],closed:true, - ls:'red',lw:3,fs:'#ddd'})`), - - new Test('txt', - `g2().txt({str:'Hello',x:30,y:30,w:0, - fs:'red',font:'normal 30px serif'}) - .txt({str:'World!',x:120,y:30,w:0,fs:'blue',font:'normal 30px serif', - sh:[5,5,5,'rgba(0,0,0,0.5)']})`), - - new Test('txt-2', - `g2().txt({str:'Hello',x:100,y:50,w:0}) - .txt({str:'Hello',x:100,y:50,w:pi/2}) - .txt({str:'Hello',x:100,y:50,w:pi}) - .txt({str:'Hello',x:100,y:50,w:-pi/2})`), - - new Test('txt-3', - `g2().grid({color:'#ccc',size:25}) - .txt({str:'LL',x:100,y:25,w:0,tval:'bottom'}) - .txt({str:'ML',x:100,y:50,w:0,tval:'middle'}) - .txt({str:'UL',x:100,y:75,w:0,tval:'top'})`), - - new Test('txt-4', - `g2().grid({color:'#ccc',size:25}) - .beg({thal:'center'}) - .txt({str:'LL',x:100,y:25,w:0,tval:'bottom'}) - .txt({str:'ML',x:100,y:50,w:0,tval:'middle'}) - .txt({str:'UL',x:100,y:75,w:0,tval:'top'}) - .end()`), - - new Test('txt-5', - `g2().grid({color:'#ccc',size:25}) - .beg({thal:'right'}) - .txt({str:'LL',x:100,y:25,w:0,tval:'bottom'}) - .txt({str:'ML',x:100,y:50,w:0,tval:'middle'}) - .txt({str:'UL',x:100,y:75,w:0,tval:'top'}) - .end()`), - - new Test('use', - `g2().ins(() => { symbol = g2() - .rec({x:-25,y:-25,b:50,h:50,ls:'gray',lw:3,fs:'@fs2'}) - .cir({x:0,y:0,r:20}) - }) - .use({grp:symbol,x:65,y:50,fs:'red',fs2:'green'}) - .use({grp:symbol,x:135,y:50,fs:'blue',fs2:'yellow'})`), - - new Test('use-2', - `g2().ins(() => {smiley = g2() - .cir({x:0,y:0,r:5}) - .arc({x:0,y:0,r:3,w:0.5,dw:2}) - .cir({x:-2,y:-1,r:1,fs:'snow'}) - .cir({x:2,y:-1,r:1,fs:'snow'})}) - .use({grp:smiley,x:50,y:50,fs:'yellow',scl:4,lw:1}) - .use({grp:smiley,x:150,y:50,fs:'orange',scl:5,lw:1});`), - - new Test('img', - `g2().img({uri:'img/atom.png',b:200,h:100})`), - - new Test('img-2', - `// intended 'file Not Found' error. -g2().img({uri:'unknown.png',x:1,y:1,b:30,h:30})`), - - new Test('img-3', - `g2().img({uri:'img/imageTst.png'})`), - - new Test('img-4', - `g2().img({uri:'img/imageTst.png', x: 10, y: 10})`), - - new Test('img-5', - `g2().img({uri:'img/imageTst.png', x: 10, y: 10, b: 100, h: 50})`), - - new Test('img-6', - `g2().img({uri:'img/imageTst.png',x:10,y:10,b:100,h:50,w:0.5}) - .rec({x:10,y:10,b:100,h:50})`), - - new Test('img-7', - `g2().img({uri:'img/imageTst.png',y:60,xoff:-10,yoff:-50,b:100,h:50})`), - - new Test('img-8', - `g2().img({uri:'img/imageTst.png',y:60,x:10,yoff:-50,b:100,h:50,w:0.5}) - .rec({x:10,y:10,b:100,h:50})`), - - new Test('img-9', - `g2().img({uri:'img/imageTst.png',y:10,x:10,scl:0.5})`), - - new Test('img-10', - `g2().img({uri:'img/imageTst.png',x:10,y:10,b:200,h:100,sx:10,sy:10}) - .rec({x: 10, y: 10, b: 200, h: 100})`), - - new Test('img-11', - `g2().img({uri:'img/imageTst.png',x:10,y:10,b:200,h:100, - sx:10,sy:10,w:0.5}) - .beg({x: 10, y: 10, w: 0.5}) - .rec({b: 200, h: 100}) - .end()`), - - new Test('img-12', - `g2().img({uri:'img/imageTst.png',x:50,y:50,b:140,h:65,xoff:10,yoff:10}) - .rec({x: 50, y: 50, b: 150, h: 75})`), - - new Test('img-13', - `g2().img({uri:'img/imageTst.png',x:50,y:50,b:140,h:65, - xoff:10,yoff:10,w:0.5}) - .beg({x:50, y:50, w:0.5}) - .rec({b: 150, h: 75}) - .end()`), - - new Test('img-14', - `g2().img({uri:'img/imageTst.png',x:10,y:10,sb:100,sh:100,h:60})`), - - new Test('img-15', - `g2().img({uri:'img/imageTst.png',x:10,y:10,sb:100,sh:100,h:60,w:0.5})`), - - new Test('img-16', - `g2().view({cartesian:true}) - .img({uri:'img/imageTst.png'})`), - - new Test('img-17', - `g2().view({cartesian:true}) - .img({uri:'img/imageTst.png', y: 10, x: 20, scl: 0.5, xoff:50, w:0.4}) `), - - new Test('img-18', - `g2().view({cartesian:true}) - .img({uri:'img/imageTst.png',x:10,y:10,scl:.4}) - .img({uri:'img/imageTst.png',x:80,y:50,scl:.4}) - .img({uri:'img/imageTst.png',x:160,y:100,scl:.4})`), - - new Test('img-19', - `g2().view({cartesian:true}) - .img({uri:'img/imageTst.png',x:10,y:10,sy:50,scl:.4}) - .cir({x:10,y:10,r:5})`), - - new Test('beg', - `g2().beg({w:0.2,scl:2,ls:'#666',fs:'orange', - lw:3,lc:'round',lj:'round'}) - .rec({x:30,y:10,b:30,h:20}) - .cir({x:130,y:20,r:20,fs:'green'}) - .end()`), - - new Test('end', - `g2().beg({w:0.2,scl:2,ls:'#666',fs:'orange', - lw:3,lc:'round',lj:'round'}) - .rec({x:30,y:10,b:30,h:20}) - .end() - .cir({x:130,y:20,r:20,fs:'green'})`), - - new Test('stroke', - `g2().p() - .m({x:25,y:25}) - .q({x1:50,y1:0,x:75,y:25}) - .a({dw:-pi/2,x:75,y:75}) - .c({x1:50,y1:75,x2:50,y2:25,x:25,y:25}) - .z() - .stroke({ls:'#888',lw:8,lc:'round',lj:'round'})`), - - new Test('fill', - `g2().p() - .m({x:25,y:25}) - .q({x1:50,y1:0,x:75,y:25}) - .a({dw:-pi/2,x:75,y:75}) - .c({x1:50,y1:75,x2:50,y2:25,x:25,y:25}) - .z() - .fill({fs:'green'})`), - - new Test('drw', - `g2().p() - .m({x:25,y:25}) - .q({x1:50,y1:0,x:75,y:25}) - .a({dw:-pi/2,x:75,y:75}) - .c({x1:50,y1:75,x2:50,y2:25,x:25,y:25}) - .z() - .drw({ls:'#888',fs:'green',lw:8,lc:'round',lj:'round'})`), - - new Test('drw-2', - `g2().drw({d:'M100,10L123.5,82.4L61,37.6'+'L138,37.6L76.5,82.4Z'})`), - - new Test('drw-3', - `g2().drw({lw:4,ls:'#080',fs:'#0f0', - d:'M100,10L123.5,82.4L61,37.6'+'L138,37.6L76.5,82.4Z'})`), - - new Test('del', - `g2().rec({x:60,y:30,b:80,h:40}).del().cir({x:100,y:50,r:35})`), - - new Test('ins', -`const node = { fill:'lime', g2() { - return g2().cir({x:160,y:50,r:15,fs:this.fill,lw:4,sh:[8,8,8,"gray"]})} -}; -let color = 'red'; -g2().cir({x:40,y:50,r:15,fs:color,lw:4,sh:[8,8,8,"gray"]}) - .ins(()=>{color='green'}) - .cir({x:80,y:50,r:15,fs:color,lw:4,sh:[8,8,8,"gray"]}) - .ins((g)=>g.cir({x:120,y:50,r:15,fs:'orange',lw:4,sh:[8,8,8,"gray"]})) - .ins(node)`) - ]); diff --git a/docs/index/index.ext.js b/docs/index/index.ext.js deleted file mode 100644 index fb6896c..0000000 --- a/docs/index/index.ext.js +++ /dev/null @@ -1,66 +0,0 @@ -new TestContainer("ext", - tests = [ - - new Test('spline', - `g2().spline({pts:[50,40,150,60,150,60],closed:false,lw:4,ls:'#080'})`), - - new Test('spline-2', - `g2().spline({pts:[50,20,50,80,150,20],closed:false,lw:4,ls:'#080'})`), - - new Test('spline-3', - `g2().spline({pts:[50,30,150,20,150,80,50,70],closed:true, - lw:4,ls:'#080',fs:'orange'})`), - - new Test('label', - `g2().view({cartesian:true}) - .cir({x:75,y:75,r:50,ld:[6,3]}) - .label({str:"c",loc:"c"}) - .label({str:"n",loc:"n",off:10}) - .label({str:"ne",loc:"ne",off:10}) - .label({str:"e",loc:"e",off:10}) - .label({str:"se",loc:"se",off:10}) - .label({str:"s",loc:"s",off:10}) - .label({str:"sw",loc:"sw",off:10}) - .label({str:"w",loc:"w",off:10}) - .label({str:"nw",loc:"nw",off:10})`), - - new Test('label-2', - `g2().view({cartesian:true}) - .rec({x:50,y:30,b:100,h:75,ld:[6,3]}) - .label({str:"c",loc:"c",border:true}) - .label({str:"n",loc:"n",off:10,border:true}) - .label({str:"ne",loc:"ne",off:10,border:true}) - .label({str:"e",loc:"e",off:10,border:true}) - .label({str:"se",loc:"se",off:10,border:true}) - .label({str:"s",loc:"s",off:10,border:true}) - .label({str:"sw",loc:"sw",off:10,border:true}) - .label({str:"w",loc:"w",off:10,border:true}) - .label({str:"nw",loc:"nw",off:10,border:true})`), - - new Test('label-3', - `g2().view({cartesian:true}) - .pol({x:20,y:75}).label({str:'pol',loc:'s',off:25}) - .gnd({x:60,y:75}).label({str:'gnd',loc:'s',off:25}) - .nod({x:100,y:75}).label({str:'nod',loc:'s',off:25}) - .dblnod({x:140,y:75}).label({str:'dblnod',loc:'s',off:25}) - .nodfix({x:180,y:75}).label({str:'nodfix',loc:'s',off:25}) - .nodflt({x:220,y:75}).label({str:'nodflt',loc:'s',off:25}) - .origin({x:260,y:75}).label({str:'origin',loc:'s',off:25})`), - - new Test('mark', - `g2().view({cartesian:true}) - .cir({x:75,y:75,r:50,ld:[6,3]}) - .mark({mrk:"dot",loc:["c","n","ne","e","se","s","sw","w","nw"]})`), - - new Test('mark-2', - `g2().view({cartesian:true}) - .rec({x:50,y:30,b:100,h:75,ld:[6,3]}) - .mark({mrk:"dot",loc:["c","n","ne","e","se","s","sw","w","nw"]})`), - - new Test('mark-3', - `g2().view({cartesian:true}) - .ply({pts:[40,40,100,40,100,100,200,100,200,40]}) - .mark({mrk:"dot",loc:["beg",0.25,"mid",0.75,1]}) - .mark({mrk:"tick",loc:[0.1,0.3,0.5,0.7,0.9]})`), - - ]); \ No newline at end of file diff --git a/docs/index/index.js b/docs/index/index.js new file mode 100644 index 0000000..296314a --- /dev/null +++ b/docs/index/index.js @@ -0,0 +1,221 @@ +class Test { + constructor(name,src) { + this.name = name; + this.src = src; + } +} +const pi = Math.PI; +const tests = [ + +new Test('view', +`g2().view({cartesian:true}).grid().origin()`), + +new Test('view2', +`g2().view({x:25,y:30}).grid().origin()`), + +new Test('view3', +`g2().view({scl:1.5,x:35,y:40}).grid().origin()`), + +new Test('view4', +`g2().view({x:35,y:35}).grid().view({scl:1.5}).origin()`), + +new Test('del', +`g2().rec({x:60,y:30,b:80,h:40}).del().cir({x:100,y:50,r:35})`), + +new Test('lin', +`g2().lin({x1:20,y1:30,x2:180,y2:80,ls:'green',lw:3})`), + +new Test('rec', +`g2().rec({x:60,y:30,b:80,h:40,ls:'red',lw:3,fs:'#ddd'})`), + +new Test('cir', +`g2().cir({x:100,y:50,r:35})`), + +new Test('arc', +`g2().arc({x:90,y:50,r:35,w:1/3*pi,dw:4/3*pi,ls:'blue',lw:5,fs:'#ddd'}) +.arc({x:120,y:50,r:35,w:1/3*pi,dw:-2/3*pi,ls:'blue',lw:5,fs:'#ddd'})`), + +new Test('ply', +`g2().ply({pts:[20,10,60,80,120,30,180,90],ls:'red',lw:3})`), + +new Test('ply2', +`g2().ply({pts:[20,10,60,80,120,30,180,90],ls:'red',lw:3,fs:'#ddd'})`), + +new Test('ply3', +`g2().ply({pts:[[20,10],[60,80],[120,30],[180,90]],ls:'red',lw:3,fs:'#ddd'})`), + +new Test('ply4', +`g2().ply({pts:[{x:20,y:10},{x:60,y:80},{x:120,y:30},{x:180,y:90}],ls:'red',lw:3,fs:'#ddd'})`), + +new Test('ply5', +`g2().ply({pts:[20,10,60,80,120,30,180,90],closed:true,ls:'red',lw:3,fs:'#ddd'})`), + +new Test('path', +`g2() +.p() +.m({x:25,y:25}) +.q({x1:50,y1:0,x:75,y:25}) +.a({dw:-pi/2,x:75,y:75}) +.c({x1:50,y1:75,x2:50,y2:25,x:25,y:25}) +.z() +.stroke({ls:'#888',lw:8,lc:'round',lj:'round'})`), + +new Test('path2', +`g2() +.p() +.m({x:25,y:25}) +.q({x1:50,y1:0,x:75,y:25}) +.a({dw:-pi/2,x:75,y:75}) +.c({x1:50,y1:75,x2:50,y2:25,x:25,y:25}) +.z() +.fill({fs:'green'})`), + +new Test('path3', +`g2() +.p() +.m({x:25,y:25}) +.q({x1:50,y1:0,x:75,y:25}) +.a({dw:-pi/2,x:75,y:75}) +.c({x1:50,y1:75,x2:50,y2:25,x:25,y:25}) +.z() +.drw({ls:'#888',fs:'green',lw:8, +lc:'round',lj:'round'})`), + +new Test('path4', +`g2().drw({d:'M100,10L123.5,82.4L61,37.6'+'L138,37.6L76.5,82.4Z'})`), + +new Test('path5', +`g2().drw({lw:4,ls:'#080',fs:'#0f0',d:'M100,10L123.5,82.4L61,37.6'+'L138,37.6L76.5,82.4Z'})`), + +new Test('spline', +`g2().spline({pts:[50,40,150,60,150,60],closed:false,lw:4,ls:'#080'})`), + +new Test('spline2', +`g2().spline({pts:[50,20,50,80,150,20],closed:false,lw:4,ls:'#080'})`), + +new Test('spline3', +`g2().spline({pts:[50,30,150,20,150,80,50,70],closed:true,lw:4,ls:'#080',fs:'orange'})`), + +new Test('txt', +`g2().txt({str:'Hello',x:30,y:30,w:0,fs:'red',font:'normal 30px serif'})`), + +new Test('txt2', +`g2().txt({str:'Hello',x:100,y:50,w:0}) +.txt({str:'Hello',x:100,y:50,w:pi/2}) +.txt({str:'Hello',x:100,y:50,w:pi}) +.txt({str:'Hello',x:100,y:50,w:-pi/2})`), + +new Test('txt3', +`g2().grid({color:'#ccc',size:25}) +.txt({str:'LL',x:100,y:25,w:0,tval:'bottom'}) +.txt({str:'ML',x:100,y:50,w:0,tval:'middle'}) +.txt({str:'UL',x:100,y:75,w:0,tval:'top'})`), + +new Test('txt4', +`g2().grid({color:'#ccc',size:25}) +.beg({thal:'center'}) +.txt({str:'LL',x:100,y:25,w:0,tval:'bottom'}) +.txt({str:'ML',x:100,y:50,w:0,tval:'middle'}) +.txt({str:'UL',x:100,y:75,w:0,tval:'top'}) +.end()`), + +new Test('txt5', +`g2().grid({color:'#ccc',size:25}) +.beg({thal:'right'}) +.txt({str:'LL',x:100,y:25,w:0,tval:'bottom'}) +.txt({str:'ML',x:100,y:50,w:0,tval:'middle'}) +.txt({str:'UL',x:100,y:75,w:0,tval:'top'}) +.end()`), + +new Test('img', +`g2().img({uri:'../docs/img/atom.png',b:200,h:100})`), + +new Test('img2', +`// intended 'file Not Found' error. +g2().img({uri:'unknown.png',x:1,y:1,b:30,h:30})`), + +new Test('img3', +`g2().img({uri:'img/imageTst.png'})`), + +new Test('img4', +`g2().img({uri:'img/imageTst.png', x: 10, y: 10})`), + +new Test('img5', +`g2().img({uri:'img/imageTst.png', x: 10, y: 10, b: 100, h: 50})`), + +new Test('img6', +`g2().img({uri:'img/imageTst.png', x: 10, y: 10 , b: 100, h: 50, w: 0.5}) + .rec({x: 10, y: 10, b: 100, h: 50})`), + +new Test('img7', +`g2() + .img({uri:'img/imageTst.png', x: 10, yoff: -50, y: 60, b: 100, h: 50, w: 0.5}) + .rec({x: 10, y: 10, b: 100, h: 50})`), + +new Test('img8', +`g2().img({uri:'img/imageTst.png', y: 10, x: 10, scl: 0.5})`), + +new Test('img9', +`g2().img({uri:'img/imageTst.png', x: 10, y: 10, dx: 100, dy: 100, h: 60})`), + +new Test('img10', +`g2().view({cartesian:true}) + .img({uri:'img/imageTst.png'})`), + +new Test('img11', +`g2().view({cartesian:true}) + .img({uri:'img/imageTst.png', x: 10, y: 10, scl:.4}) + .img({uri:'img/imageTst.png', x: 80, y: 50, scl:.4}) + .img({uri:'img/imageTst.png', x: 160, y: 100, scl:.4})`), + +new Test('img12', +`g2().img({uri:'img/imageTst.png', x: 10, y: 10, b: 200, h: 100, sx: 10, sy: 10, w: 0.5}) + .beg({x: 10, y: 10, w: 0.5}) + .rec({b: 200, h: 100}) + .end()`), + +new Test('img13', +`g2().img({uri:'img/imageTst.png', x: 50, y: 50, b:140, h:65, xoff: 10, yoff: 10, w:0.5}) + .beg({x:50, y:50, w:0.5}) + .rec({b: 150, h: 75}) + .end()`), + +new Test('beg-end', +`g2().beg({x:70,y:30,w:0.2,scl:2,ls:'#666',fs:'orange',lw:3,lc:'round',lj:'round'}) +.rec({x:0,y:0,b:30,h:20}).end()`), + +new Test('use', +`g2() +.ins(() =>{ symbol = g2() +.rec({x:-25,y:-25,b:50,h:50,ls:'gray',lw:3,fs:'@fs2'}) +.cir({x:0,y:0,r:20})}) +.use({grp:symbol,x:65,y:50,fs:'red',fs2:'green'}) +.use({grp:symbol,x:135,y:50,fs:'blue',fs2:'yellow'})`), + +new Test('use2', +`g2().ins(() => {smiley = g2() +.cir({x:0,y:0,r:5}) +.arc({x:0,y:0,r:3,w:0.5,dw:2}) +.cir({x:-2,y:-1,r:1,fs:'snow'}) +.cir({x:2,y:-1,r:1,fs:'snow'})}) +.use({grp:smiley,x:50,y:50,fs:'yellow',scl:4,lw:1}) +.use({grp:smiley,x:150,y:50,fs:'orange',scl:5,lw:1});`), + +new Test('shadow', +`g2().beg({lw:3,ls:'#456',fs:'yellow',ld:[8, 4, 2, 4], sh:[5,5,5,'rgba(0,0,0,0.7)']}) +.rec({x:30,y:40,b:50,h:20}) +.cir({x:140,y:50,r:40})`), + +new Test('grid', +`g2().grid()`), + +new Test('mark', +`g2().view({cartesian:true}) + .ply({y:40,pts:[40,40,100,40,100,100,200,100,200,40], mark:4}) + .ply({y:-30,pts:[40,40,100,40,100,100,200,100,200,40], mark:4, closed:true})`), + +new Test('mark2', +`g2().cir({x:50,y:50,r:25,mark:{symbol:'sqr',loc:[0,0.5,0.7]}}) + .rec({x:150,y:30,b:40,h:60,mark:{symbol:g2.symbol.sqr,count:6}}) + .lin({x1:50,y1:130,x2:150,y2:130,mark:8})`) +]; \ No newline at end of file diff --git a/docs/index/index.mec.js b/docs/index/index.mec.js index cc70729..6bf4ef0 100644 --- a/docs/index/index.mec.js +++ b/docs/index/index.mec.js @@ -1,127 +1,133 @@ -new TestContainer("mec", - tests = [ - new Test('dim', - `g2().dim({x1:50,y1:75,x2:230,y2:75}) - .dim({x1:50,y1:125,x2:230,y2:125,inside:false})`), - - new Test('dim-2', - `A = {x1: 30,y1:50,x2:200,y2:120} -g2().lin({...A,ls:"red",lw:2}) - .dim({...A, off:-15,over:5})`), - - new Test('adim', - `g2().adim({x:100,y:70,r:50,w:pi/3,dw:4*pi/3}) - .adim({x:200,y:70,r:50,w:pi/3,dw:-4*pi/3,inside:false})`), - - new Test('vec', - `g2().vec({x1:50,y1:20,x2:250,y2:120})`), +tests.push( +new Test ('g2.mec.js', + `g2().txt({str:"the following tests are for g2.mec.js",x:50,y:50})`), - new Test('avec', - `g2().avec({x: 220, y: 50, r: 30, w: 1, dw: -Math.PI}) +new Test('symbols', +` +const label = (str) => ({str, loc:'s', off:15}); +g2().view({cartesian:true}) + .pol({x:20,y:75,label:label('pol')}) + .gnd({x:60,y:75,label:label('gnd')}) + .nod({x:100,y:75,label:label('nod')}) + .dblnod({x:140,y:75,label:label('dblnod')}) + .nodfix({x:180,y:75,label:label('nodfix')}) + .nodflt({x:220,y:75,label:label('nodflt')}) + .origin({x:260,y:75,label:label('origin')})`), + +new Test('vec', +`g2().vec({x1:50,y1:20,x2:250,y2:120})`), + +new Test('avec', +`g2().avec({x: 220, y: 50, r: 30, w: 1, dw: -Math.PI}) .avec({x: 180, y: 80, w: Math.PI/2, dw: Math.PI, r: 50}) .lin({x1: 180, x2: 180, y1: 20, y2: 140}) .avec({x: 65, y: 65, r: 60, w: 1}) - .avec({x: 80, y: 50, r: 30})`), + .avec({x: 80, y: 50, r: 70, dw: 1, w:2}) + .lin({x1:80,y1:50,x2:80+Math.cos(3)*75,y2:50+Math.sin(3)*75}) + .lin({x1:80,y1:50,x2:80+Math.cos(2)*75,y2:50+Math.sin(2)*75})`), + +new Test('dim', +`g2().dim({x1:20,y1:75,x2:200,y2:75}) + .dim({x1:20,y1:125,x2:200,y2:125,inside:false})`), + +new Test('dim2', +`A = {x1: 30, y1:50, x2: 200, y2: 50}; +B = {x1: 40, y1: 80, x2: 210, y2: 130}; +g2().lin({...A, ls: "red", lw: 2}) + .dim({...A, off: 15}) + .lin({...B, ls: "red", lw: 2}) + .dim({...B, off: 10})`), + +new Test('adim', +`g2().adim({x:100,y:70,r:50,w:pi/3,dw:4*pi/3}) + .adim({x:200,y:70,r:50,w:pi/3,dw:-4*pi/3,inside:false})`), - new Test('slider', - `g2().grid().slider({x:150,y:75,w:Math.PI/4,b:64,h:32})`), +new Test('slider', +`g2().grid().slider({x:150,y:75,w:Math.PI/4,b:64,h:32})`), - new Test('spring', - `g2().spring({x1:50,y1:50,x2:200,y2:25}) +new Test('spring', +`g2().spring({x1:50,y1:50,x2:200,y2:25}) .spring({x1:50,y1:100,x2:200,y2:125,h:32})`), - new Test('damper', - `g2().damper({x1:120,y1:60,x2:240,y2:70}) +new Test('damper', +`g2().damper({x1:120,y1:60,x2:240,y2:70}) .damper({x1:50,y1:100,x2:200,y2:125,h:32})`), - new Test('link', - `let A = {x:50,y:25},B = {x:150,y:25}, +new Test('link', +`let A = {x:50,y:25},B = {x:150,y:25}, C = {x:50,y:75}, D = {x:100,y:75}, E = {x:50,y:125}; g2().view({cartesian:true}) .link({pts:[A,B,E,A,D,C]})`), - new Test('link2', - `let A = {x:50,y:25},B = {x:150,y:25}, +new Test('link2', +`let A = {x:50,y:25},B = {x:150,y:25}, C = {x:50,y:75}, D = {x:100,y:75}, E = {x:50,y:125}; g2().view({cartesian:true}) .link2({pts:[A,B,E,A,D,C]}) - .nodfix({...A,scl:1.5}) - .nod({...B,scl:1.5}) - .nod({...C,scl:1.5}) - .nod({...D,scl:1.5}) - .nodflt({...E,w:-Math.PI/2,scl:1.5})`), - - new Test('beam', - `g2().view({cartesian:true}) + .nodfix({...A,scl:1.2}) + .nod({...B,scl:1.2}) + .nod({...C,scl:1.2}) + .nod({...D,scl:1.2}) + .nodflt({...E,w:-Math.PI/2,scl:1.2})`), + +new Test('beam', +`g2().view({cartesian:true}) .beam({pts:[[200,125],[50,125],[50,50],[200,50]]})`), - new Test('beam2', - `g2().view({cartesian:true}) +new Test('beam2', +`g2().view({cartesian:true}) .beam2({pts:[[200,125],[50,125],[50,50],[200,50]]})`), - new Test('bar', - `g2().bar({x1:50,y1:20,x2:250,y2:120})`), +new Test('bar', +`g2().bar({x1:50,y1:20,x2:250,y2:120})`), - new Test('bar2', - `g2().bar2({x1:50,y1:120,x2:250,y2:120})`), +new Test('bar2', +`g2().bar2({x1:50,y1:120,x2:250,y2:120})`), - new Test('pulley', - `g2().view({cartesian:true}) +new Test('pulley', +`g2().view({cartesian:true}) .pulley({x:150,y:80,r:50})`), - new Test('pulley2', - `g2().view({cartesian:true}) +new Test('pulley2', +`g2().view({cartesian:true}) .pulley2({x:150,y:80,r:50})`), - new Test('rope', - `let A = {x:50,y:30}, B = {x:200,y:75}; +new Test('rope', +`let A = {x:50,y:30}, B = {x:200,y:75}; g2().view({cartesian:true}) .pulley({...A,r:20}).pulley2({...B,r:40}) .rope({p1:A,r1:20,p2:B,r2:40})`), - new Test('ground', - `g2().ground({pts:[25,25,25,75,75,75,75,25,125,25], - pos:'left'})`), +new Test('ground', +`g2().ground({pts:[25,25,25,75,75,75,75,25,125,25],pos:'left'})`), - new Test('load', - `g2().view({cartesian:true}) +new Test('load', +`g2().view({cartesian:true}) .bar({x1:20,y1:50,x2:220,y2:50}) .load({pts:[20,50,30,140,250,90,220,50], lw:1,w:-Math.PI/3,ls:"blue"})`), - new Test('load-2', - `g2().view({cartesian:true}) +new Test('load2', +`g2().view({cartesian:true}) .load({pts:[10,10,100,10,100,130,200,130,200,60,10,60],spacing:10})`), - new Test('pol', - `g2().view({cartesian:true}) - .pol({x:20,y:75}) - .gnd({x:60,y:75}) - .nod({x:100,y:75}) - .dblnod({x:140,y:75}) - .nodfix({x:180,y:75}) - .nodflt({x:220,y:75}) - .origin({x:260,y:75})`), - - new Test('mec', - `let A = {x:50,y:25},B = {x:150,y:25}, +new Test('mec', +`let A = {x:50,y:25},B = {x:150,y:25}, C = {x:50,y:75}, D = {x:100,y:75}, E = {x:50,y:125}; g2().view({cartesian:true}) .link2({pts:[A,B,E,A,D,C]}) - .nodfix({...A,scl:1.5}) - .nod({...B,scl:1.5}) - .nod({...C,scl:1.5}) - .nod({...D,scl:1.5}) - .nodflt({...E,w:-Math.PI/2,scl:1.5}) - .ground({pts:[{x:E.x-23,y:E.y+20}, - {x:A.x-23,y:A.y-18},{x:D.x,y:A.y-18}]}) + .nodfix({...A,scl:1.2}) + .nod({...B,scl:1.2}) + .nod({...C,scl:1.2}) + .nod({...D,scl:1.2}) + .nodflt({...E,w:-Math.PI/2,scl:1.2}) + .ground({pts:[{x:E.x-23,y:E.y+20},{x:A.x-23,y:A.y-18},{x:D.x,y:A.y-18}]}) .vec({x1:D.x,y1:D.y,x2:D.x+50,y2:D.y,ls:'darkred',lw:2}) .vec({x1:B.x,y1:B.y,x2:B.x,y2:B.y-20,ls:'darkred',lw:0.5}) - .dim({x1:E.x,y1:E.y+15,x2:D.x,y2:E.y+15}).label({str:'b',loc:'s'}) - .dim({x1:D.x,y1:E.y+15,x2:B.x,y2:E.y+15}).label({str:'b',loc:'s'}) - .dim({x1:B.x+15,y1:E.y,x2:B.x+15,y2:D.y}).label({str:'b',loc:'w'}) - .dim({x1:B.x+15,y1:D.y,x2:B.x+15,y2:B.y}).label({str:'b',loc:'w'})`) - ]); + .dim({x1:E.x,y1:E.y+15,x2:D.x,y2:E.y+15,label:'b'}) + .dim({x1:D.x,y1:E.y+15,x2:B.x,y2:E.y+15,label:'b'}) + .dim({x1:B.x+15,y1:E.y,x2:B.x+15,y2:D.y,label:'b'}) + .dim({x1:B.x+15,y1:D.y,x2:B.x+15,y2:B.y,label:'b'})`)); diff --git a/docs/microjam.md/Animation.html b/docs/microjam.md/Animation.html index 7c3a7db..9cb917b 100644 --- a/docs/microjam.md/Animation.html +++ b/docs/microjam.md/Animation.html @@ -33,7 +33,8 @@
  • Insert
  • Animation
  • Interactivity
  • -
  • Custom HTML Element
  • +
  • Custom HTML element
  • +
  • Custom chart element

    • @@ -87,13 +88,13 @@

      Example #

      // draw pendulum ... g.del() // delete all commands in the queue ... .clr() // clear the renderers viewport ... - .ply({pts:[44,40,56,40,x0,y0],closed:true,fs:"#ddd"}) + .ply({pts:[44,40,56,40,x0,y0],closed:true,fs:"#ddd"}) .lin({x1:x0,y1:y0,x2:x,y2:y,lw:3}) // pendulum begins .. - .cir({x:x0,y:y0,r:3,fs:"#ddd"}) // .. and ends here. - .cir({x:x,y:y,r:8,fs:"#ddd"}) + .cir({x:x0,y:y0,r:3,fs:"#ddd"}) // .. and ends here. + .cir({x:x,y:y,r:8,fs:"#ddd"}) .exe(ctx); phi += dphi; // increment angle ... - requestAnimationFrame(render); // requedst browser to call 'render' again ... + requestAnimationFrame(render); // requedst browser to call 'render' again ... } render(); // initial render call.
    @@ -109,7 +110,7 @@

    Example - Enhanced let phi = 0; world = g2().clr() // clear and then define static world - .ply({pts:[44,40,56,40,x0,y0],closed:true,fs:"#ddd"}) + .ply({pts:[44,40,56,40,x0,y0],closed:true,fs:"#ddd"}) .use({grp:pendulum}) // reference pendulum here... .cir({x:x0,y:y0,r:3}) function render() { @@ -117,7 +118,7 @@

    Example - Enhanced // rebuild pendulum... .lin({x1:x0,y1:y0,x2:x,y2:y,lw:3}) // pendulum begins .. - .cir({x:x,y:y,r:8,fs:"#ddd"}) + .cir({x:x,y:y,r:8,fs:"#ddd"}) world.exe(ctx); phi += dphi; requestAnimationFrame(render); @@ -139,10 +140,10 @@

    Example - Enhanced Example - Timed Animation #

    Using window.requestAnimationFrame makes timed animation particularly easy. It passes a timestamp with every call to the callback function.

    -
    const ctx = document.getElementById("c").getContext("2d"), +
    const ctx = document.getElementById("c").getContext("2d"), pendulum = g2(), x0 = 50, y0 = 50, r = 40, omega = 1.5, world = g2().clr() - .ply({pts:[44,40,56,40,x0,y0],closed:true,fs:"#ddd"}) + .ply({pts:[44,40,56,40,x0,y0],closed:true,fs:"#ddd"}) .use({grp:pendulum}) .cir({x:x0,y:y0,r:3}); function render(time) { @@ -153,7 +154,7 @@

    Example - Timed Animation x1:x0,y1:y0,x2:x,y2:y,lw:3}) - .cir({x:x,y:y,r:8,fs:"#ddd"}) + .cir({x:x,y:y,r:8,fs:"#ddd"}) world.exe(ctx); requestAnimationFrame(render); } diff --git a/docs/microjam.md/Concepts.html b/docs/microjam.md/Concepts.html index 623634f..698d804 100644 --- a/docs/microjam.md/Concepts.html +++ b/docs/microjam.md/Concepts.html @@ -33,7 +33,8 @@
  • Insert
  • Animation
  • Interactivity
  • -
  • Custom HTML Element
  • +
  • Custom HTML element
  • +
  • Custom chart element

  • star1

    There are only two objects g2 and ctx involved. Both are nearly completely independent @@ -157,20 +158,20 @@

    Separating Model from View
    class Circle { - constructor(x,y,r) {this.x=x;this.y=y;this.r=r} - render(g) {g.cir(this.x,this.y,this.r)} + constructor(x,y,r) {this.x=x;this.y=y;this.r=r} + render(g) {g.cir(this.x,this.y,this.r)} } class Rect { - constructor(x,y,b,h) {this.x=x;this.y=y;this.b=b;this.h=h} - render(g) {g.rec(this.x,this.y,this.w,this.h)} + constructor(x,y,b,h) {this.x=x;this.y=y;this.b=b;this.h=h} + render(g) {g.rec(this.x,this.y,this.w,this.h)} } const model = [new Circle({x:40,y:60,r:20}),new Rect({x:45,y:25,b:40,h:40})], g = g2().grid(), - ctx1 = document.getElementById('c1').getContext('2d'), // view 1 - ctx2 = document.getElementById('c2').getContext('2d'); // view 2 + ctx1 = document.getElementById('c1').getContext('2d'), // view 1 + ctx2 = document.getElementById('c2').getContext('2d'); // view 2 for (let i of model) { // build command queue of ... - i.render(g); // ... model's shapes drawing commands. + i.render(g); // ... model's shapes drawing commands. } g.exe(ctx1); // render to view 1 g.exe(ctx2); // render to view 2 diff --git a/docs/microjam.md/Elements.html b/docs/microjam.md/Elements.html index a4450fe..8fb84b7 100644 --- a/docs/microjam.md/Elements.html +++ b/docs/microjam.md/Elements.html @@ -33,7 +33,8 @@
  • Insert
  • Animation
  • Interactivity
  • -
  • Custom HTML Element
  • +
  • Custom HTML element
  • +
  • Custom chart element




  • map

    @@ -104,12 +105,12 @@

    View Manipulation extra zoom window

    HTML Input Element #

    HTML input elements can be used to modify graphics related data.

    -
    <canvas id="c" width="300" height="200"></canvas> +
    <canvas id="c" width="300" height="200"></canvas> <br> -<input id="range" type="range" style="min-width:275px;vertical-align:middle;margin:0;padding:0" - min="0" max="360" value="0" step="1" /> -<output id="output" for="range" style="text-align:right;">0</output> -<script src='g2.core.js'></script> +<input id="range" type="range" style="min-width:275px;vertical-align:middle;margin:0;padding:0" + min="0" max="360" value="0" step="1" /> +<output id="output" for="range" style="text-align:right;">0</output> +<script src='g2.core.js'></script> <script> function position(phi) { const sp = Math.sin(phi), cp = Math.cos(phi), @@ -118,7 +119,7 @@

    HTML Input Element 0, yC = 2*a*sp; mec.del() // ... and build new mechanism geometry. - .ell({x:xA,y:yA,rx:a,ry:this.rx,rot:-phi,lw:2,ld:[12,4,2,4]}) + .ell({x:xA,y:yA,rx:a,ry:this.rx,rot:-phi,lw:2,ld:[12,4,2,4]}) .ply({pts:[0,0,xA,yA,xB,yB,xC,yC],lw:5}) .beg({...el}) .cir({x:xA,y:yA,r:6}) @@ -144,17 +145,17 @@

    HTML Input Element let dirty = true, phi = 0, xA, yA, xB, yB; - const ctx = document.getElementById("c").getContext("2d"), - range = document.getElementById("range"), - output = document.getElementById("output"), - el = {fs:"papayawhip",lw:2}, + const ctx = document.getElementById("c").getContext("2d"), + range = document.getElementById("range"), + output = document.getElementById("output"), + el = {fs:"papayawhip",lw:2}, xA0 = 150, yA0 = 150, a = 60, b = 80, e = 0; mec = g2(), // dynamic mechanism. world = g2().clr().cartesian().grid() // static world. - .cir({x:xA0, y:yA0, r:2*a, ls:"darkslategray", lw:2, ld:[12,4,2,4]}) + .cir({x:xA0, y:yA0, r:2*a, ls:"darkslategray", lw:2, ld:[12,4,2,4]}) .use({grp:mec, x:xA0, y:yA0}) .cir({x:xA0, y:yA0, r:6,...el}); - range.addEventListener("input",setPhi,false); + range.addEventListener("input",setPhi,false); render(); </script>

    @@ -165,12 +166,12 @@

    Using Layers #

    Here is a clearly written in depth article about HTML canvas layering.

    -
    <div style="position:relative; width:641px; height:481px;"> - <canvas id="bg" width="641" height="481" style="position:absolute;"></canvas> - <canvas id="fg" width="641" height="481" style="position:absolute;"></canvas> +
    <div style="position:relative; width:641px; height:481px;"> + <canvas id="bg" width="641" height="481" style="position:absolute;"></canvas> + <canvas id="fg" width="641" height="481" style="position:absolute;"></canvas> </div> <p>Select square by pointer and drag it.</p> -<script src='g2.core.js'></script> +<script src='g2.core.js'></script> <script> function onbuttondown(e) { // select square at pointer location. const x = e.clientX - Math.floor(viewport.left), @@ -181,7 +182,7 @@

    Using Layers #0 && y - rects[i].y <= sz) { bgDirty = fgDirty = true; // mark layers by dirty flags. selIdx = i; // memoize selected square. - ctxfg.canvas.addEventListener('mousemove',onmove,false); // ready to drag. + ctxfg.canvas.addEventListener('mousemove',onmove,false); // ready to drag. break; } } @@ -189,7 +190,7 @@

    Using Layers #function onbuttonup(e) { // deselect square if (selIdx != -1) { // is one selected bgDirty = fgDirty = true; // mark layers by dirty flags. - ctxfg.canvas.removeEventListener('mousemove',onmove,false); // finish dragging. + ctxfg.canvas.removeEventListener('mousemove',onmove,false); // finish dragging. selIdx = -1; // nothing selected. } } @@ -205,7 +206,7 @@

    Using Layers #for (let i=0; i<rects.length-1; i++) { if (i !== selIdx) { bg.rec({x:rects[i].x,y:rects[i].y,b:sz,h:sz, - ls:"darkslategray",lw:3,fs:"papayawhip"}); + ls:"darkslategray",lw:3,fs:"papayawhip"}); } } bg.exe(ctxbg); @@ -215,9 +216,9 @@

    Using Layers #if (selIdx != -1) { fg.rec({x:rects[selIdx].x,y:rects[selIdx].y,b:sz,h:sz, - lw:3,sh:[0,0,10,"black"]}) + lw:3,sh:[0,0,10,"black"]}) .rec({x:rects[selIdx].x,y:rects[selIdx].y,b:sz,h:sz, - lw:3,ls:"darkslategray",fs:"orange"}); + lw:3,ls:"darkslategray",fs:"orange"}); } fg.exe(ctxfg); fgDirty = false; @@ -227,8 +228,8 @@

    Using Layers #let bgDirty = true, fgDirty = true, selIdx = -1; - const ctxbg = document.getElementById("bg").getContext("2d"), - ctxfg = document.getElementById("fg").getContext("2d"), + const ctxbg = document.getElementById("bg").getContext("2d"), + ctxfg = document.getElementById("fg").getContext("2d"), viewport = ctxbg.canvas.getBoundingClientRect(), bg = g2(), // Background layer command queue. fg = g2(), // Foreground layer command queue for dragging. @@ -239,8 +240,8 @@

    Using Layers #y: Math.random()*(viewport.height-sz) }; } - ctxfg.canvas.addEventListener('mousedown', onbuttondown, false); - ctxfg.canvas.addEventListener('mouseup', onbuttonup, false); + ctxfg.canvas.addEventListener('mousedown', onbuttondown, false); + ctxfg.canvas.addEventListener('mouseup', onbuttonup, false); render(); // initial call to render loop. </script> diff --git a/docs/microjam.md/Paths.html b/docs/microjam.md/Paths.html index 7ace8de..31a21a4 100644 --- a/docs/microjam.md/Paths.html +++ b/docs/microjam.md/Paths.html @@ -33,7 +33,8 @@
  • Insert
  • Animation
  • Interactivity
  • -
  • Custom HTML Element
  • +
  • Custom HTML element
  • +
  • Custom chart element

    • @@ -156,8 +157,8 @@

      An optional second argument d of stroke, fill or drw can be used to hand over an SVG path data string. In that case previously defined path commands are ignored.

      -
      const star = "M100,10L123.5,82.4L61,37.6L138,37.6L76.5,82.4Z"; -g2().drw({d:star,lw:4,ls:"red",fs:"orange"}) +
      const star = "M100,10L123.5,82.4L61,37.6L138,37.6L76.5,82.4Z"; +g2().drw({d:star,lw:4,ls:"red",fs:"orange"}) .exe(ctx);

      star

      diff --git a/docs/microjam.md/Reuse.html b/docs/microjam.md/Reuse.html index 61856d4..7736c18 100644 --- a/docs/microjam.md/Reuse.html +++ b/docs/microjam.md/Reuse.html @@ -33,7 +33,8 @@
    • Insert
    • Animation
    • Interactivity
    • -
    • Custom HTML Element
    • +
    • Custom HTML element
    • +
    • Custom chart element

      @@ -89,12 +90,12 @@

      use #

      Example #

      const smiley = g2().cir({x:0,y:0,r:5}) // set smiley origin in its center ! .arc({x:0,y:0,r:3,w:0.8,dw:2}) - .beg({fs:"snow"}) // snow colored eyes. + .beg({fs:"snow"}) // snow colored eyes. .cir({x:-2,y:-1,r:1}) .cir({x:2,y:-1,r:1}) .end(); -g2().use({grp:smiley,x:50,y:50,scl:4,lw:0.75,fs:"yellow"}) - .use({grp:smiley,x:150,y:50,scl:5,lw:0.4,fs:"orange",ls:"green"}) +g2().use({grp:smiley,x:50,y:50,scl:4,lw:0.75,fs:"yellow"}) + .use({grp:smiley,x:150,y:50,scl:5,lw:0.4,fs:"orange",ls:"green"}) .exe(ctx);

      smiley example

      @@ -104,14 +105,14 @@

      Example #

      So if you want to reuse geometry more than once, consider using use!

      Example 2 #

      function rollingWheel() { - const r=40,dphi=11/20*Math.PI,style={fs:"transparent",lw:8}, + const r=40,dphi=11/20*Math.PI,style={fs:"transparent",lw:8}, wheel = g2().cir({x:0,y:0,r:r,...style}) .lin({x1:-0.8*r,y1:0,x2:0.8*r,y2:0,...style}) - .cir({x:0,y:0,r:4,fs:"snow",lw:2}), + .cir({x:0,y:0,r:4,fs:"snow",lw:2}), rolling = g2(); for (let alpha=0.3,phi=0; alpha<=1; alpha+=0.1,phi+=dphi) { - rolling.use({grp:wheel,x:r*phi,y:54,w:phi,ls:'rgba(0,0,0,'+alpha*alpha*alpha+')'}); + rolling.use({grp:wheel,x:r*phi,y:54,w:phi,ls:'rgba(0,0,0,'+alpha*alpha*alpha+')'}); } return rolling; } @@ -126,17 +127,17 @@

      Example 2 #

      But we can do even better than that. We slightly modify the previous example and hand over the main g2 object to the service function.

      Example 2 Enhanced #

      -
      <canvas id="c" width="550" height="100"></canvas> -<script src='g2.core.js'></script> +
      <canvas id="c" width="550" height="100"></canvas> +<script src='g2.core.js'></script> <script> - const ctx=document.getElementById("c").getContext("2d"); + const ctx=document.getElementById("c").getContext("2d"); function rollingWheel(g) { - let r=40,dphi=11/20*Math.PI,style={fs:"transparent",lw:8}, + let r=40,dphi=11/20*Math.PI,style={fs:"transparent",lw:8}, wheel = g2().cir({x:0,y:0,r:r,...style}) .lin({x1:-0.8*r,y1:0,x2:0.8*r,y2:0,...style}) - .cir({x:0,y:0,r:4,fs:"snow",lw:2}); + .cir({x:0,y:0,r:4,fs:"snow",lw:2}); for (let alpha=0.3,phi=0; alpha<=1;alpha+=0.1,phi+=dphi) - g.use({grp:wheel,x:r*phi,y:54,w:phi,ls:'rgba(0,0,0,'+alpha*alpha*alpha+')'}); + g.use({grp:wheel,x:r*phi,y:54,w:phi,ls:'rgba(0,0,0,'+alpha*alpha*alpha+')'}); return g; } @@ -157,21 +158,21 @@

      Example 2 with ins

      Example 3 #

      Use is exceptionally useful if the same element is used over and over again

      -
      <canvas id="c1" width=200, height=200 style="border:1px solid black"></canvas> -<canvas id="c2" width=200, height=200 style="border:1px solid black"></canvas> -<script src="./g2.core.js"></script> +
      <canvas id="c1" width=200, height=200 style="border:1px solid black"></canvas> +<canvas id="c2" width=200, height=200 style="border:1px solid black"></canvas> +<script src="./g2.core.js"></script> <script> const pi = Math.PI, - ctx1 = document.getElementById('c1').getContext('2d'), - ctx2 = document.getElementById('c2').getContext('2d'), - world = g2().rec({x:0,y:0,b:200,h:200,fs:"skyblue"}) - .ply({pts:[0,100,40,...],fs:"burlywood"}) - .rec({x:0,y:0,b:200,y:150,fs:"limegreen"}), - tree = g2().rec({x:-5,y:0,b:10,h:40,fs:"brown"}) - .beg({fs:"forestgreen"}) + ctx1 = document.getElementById('c1').getContext('2d'), + ctx2 = document.getElementById('c2').getContext('2d'), + world = g2().rec({x:0,y:0,b:200,h:200,fs:"skyblue"}) + .ply({pts:[0,100,40,...],fs:"burlywood"}) + .rec({x:0,y:0,b:200,y:150,fs:"limegreen"}), + tree = g2().rec({x:-5,y:0,b:10,h:40,fs:"brown"}) + .beg({fs:"forestgreen"}) .cir({x:15,y:-20,r:10}) ... - .cir({x:0,y:-20,r:20,ls:"transparent"}) + .cir({x:0,y:-20,r:20,ls:"transparent"}) .end(); g2().use({grp:world,x:0,y:0}) @@ -196,7 +197,7 @@

      Example 3 #

      Predefined command queues #

      Please note that predefined command queues can also be used without any manipulation of their styling, as shown in previous occasions.

      -
      const block = g2().rec({b:20,h:20,x:50,y:100,fs:"burlywood",ls:"saddlebrown"}); +
      const block = g2().rec({b:20,h:20,x:50,y:100,fs:"burlywood",ls:"saddlebrown"}); block.exe(ctx);
      diff --git a/docs/microjam.md/State-and-Style.html b/docs/microjam.md/State-and-Style.html index e35fe6d..efb21be 100644 --- a/docs/microjam.md/State-and-Style.html +++ b/docs/microjam.md/State-and-Style.html @@ -33,7 +33,8 @@
    • Insert
    • Animation
    • Interactivity
    • -
    • Custom HTML Element
    • +
    • Custom HTML element
    • +
    • Custom chart element

      @@ -138,13 +139,13 @@

      Example 1 #

      -
      <canvas id="c" width="100" height="100"></canvas> -<script src='g2.core.js'></script> +
      <canvas id="c" width="100" height="100"></canvas> +<script src='g2.core.js'></script> <script> - const ctx=document.getElementById("c").getContext("2d"), - style={ls:"red",lw:3,fs:"ddd"}; - g2().rec({x:20,y:10,b:60,h:30,ls:'maroon',fs:'khaki'}) - .cir({x:50,y:75,r:20,lw:2,ld:[6,3],fs:'orange'}) + const ctx=document.getElementById("c").getContext("2d"), + style={ls:"red",lw:3,fs:"ddd"}; + g2().rec({x:20,y:10,b:60,h:30,ls:'maroon',fs:'khaki'}) + .cir({x:50,y:75,r:20,lw:2,ld:[6,3],fs:'orange'}) .exe(ctx) </script>
      @@ -212,41 +213,41 @@

      Example 1 #

      Example 2 #

      -
      <canvas id="c" width="450" height="100"></canvas> -<script src='g2.core.js'></script> -<script> - const ctx=document.getElementById("c").getContext("2d"), - fs1={fs:"darkolivegreen",font:"24px serif"}, - fs2={fs:"steelblue",font:"30px serif"}, - fs3={...fs2,fs:"midnightblue"}; - g2().beg({fs:"gainsboro",ls:"firebrick",lw:3, - thal:"center",tval:"middle"}) - .rec({x:20,y:20,b:80,h:50}) - .txt({str:"one",x:60,y:45,...fs1}) - .beg({y:-10,fs:"darkseagreen",ls:"cadetblue",lw:5}) - .rec({x:120,y:20,b:80,h:50}) - .txt({str:"two",x:160,y:45,...fs2}) - .beg({ls:"@fs",fs:"cadetblue"}) - .rec({x:220,y:20,b:80,h:50}) - .txt({str:"three",x:260,y:45,...fs3}) - .end() - .end() - .rec({x:320,y:20,b:80,h:50}) - .txt({str:"four",x:360,y:45,...fs1}) - .end() +
      <canvas id="c" width="450" height="100"></canvas> +<script src='g2.core.js'></script> +<script> + const ctx=document.getElementById("c").getContext("2d"), + fs1={fs:"darkolivegreen",font:"24px serif"}, + fs2={fs:"steelblue",font:"30px serif"}, + fs3={...fs2,fs:"midnightblue"}; + g2().beg({fs:"gainsboro",ls:"firebrick",lw:3, + thal:"center",tval:"middle"}) + .rec({x:20,y:20,b:80,h:50}) + .txt({str:"one",x:60,y:45,...fs1}) + .beg({y:-10,fs:"darkseagreen",ls:"cadetblue",lw:5}) + .rec({x:120,y:20,b:80,h:50}) + .txt({str:"two",x:160,y:45,...fs2}) + .beg({ls:"@fs",fs:"cadetblue"}) + .rec({x:220,y:20,b:80,h:50}) + .txt({str:"three",x:260,y:45,...fs3}) + .end() + .end() + .rec({x:320,y:20,b:80,h:50}) + .txt({str:"four",x:360,y:45,...fs1}) + .end() .exe(ctx); </script>

      beg Element

      Example Yinyang #

      -
      <canvas id="c" width="100" height="100"></canvas> -<script src='g2.core.js'></script> +
      <canvas id="c" width="100" height="100"></canvas> +<script src='g2.core.js'></script> <script> - const ctx=document.getElementById("c").getContext("2d"), - pi = Math.PI, dark = "#444", light = "#eee"; + const ctx=document.getElementById("c").getContext("2d"), + pi = Math.PI, dark = "#444", light = "#eee"; g2().beg({x:50,y:50,scl:2,fs:light,ls:dark,lw:2}) .cir({x:0,y:0,r:20}) - .beg({fs:"@ls"}) + .beg({fs:"@ls"}) .p() .m({x:0,y:-20}) .a({dw:pi,x:0,y:0}) @@ -255,7 +256,7 @@

      Example Yinyang x:0,y:-10,r:3}) - .cir({x:0,y:10,r:3,ls:light,fs:"@ls"}) + .cir({x:0,y:10,r:3,ls:light,fs:"@ls"}) .end() .end() .exe(ctx); diff --git a/docs/microjam.md/View.html b/docs/microjam.md/View.html index 5716a69..2fccee2 100644 --- a/docs/microjam.md/View.html +++ b/docs/microjam.md/View.html @@ -33,7 +33,8 @@
    • Insert
    • Animation
    • Interactivity
    • -
    • Custom HTML Element
    • +
    • Custom HTML element
    • +
    • Custom chart element

      @@ -82,16 +83,16 @@

      This output is generated by the follwing code addressing different renderer contexts.

      world = g2().view(worldview) - .rec({x:0,y:0,b:200,h:200,fs:"skyblue"}) - .ply({pts:[0,100,40,...],fs:"burlywood"}) - .rec({x:0,y:0,b:200,y:150,fs:"limegreen"}), + .rec({x:0,y:0,b:200,h:200,fs:"skyblue"}) + .ply({pts:[0,100,40,...],fs:"burlywood"}) + .rec({x:0,y:0,b:200,y:150,fs:"limegreen"}), tree = g2().view(treeview) - .rec({x:-5,y:0,b:10,h:40,fs:"brown"}) - .beg({fs:"forestgreen"}) + .rec({x:-5,y:0,b:10,h:40,fs:"brown"}) + .beg({fs:"forestgreen"}) .cir({x:15,y:-20,r:10}) ... - .cir({x:0,y:-20,r:20,ls:"transparent"}) + .cir({x:0,y:-20,r:20,ls:"transparent"}) .end(); g2().use({grp:world,x:0,y:0}) diff --git a/docs/microjam.md/g2.chart.html b/docs/microjam.md/g2.chart.html index 31b3063..a999839 100644 --- a/docs/microjam.md/g2.chart.html +++ b/docs/microjam.md/g2.chart.html @@ -33,7 +33,8 @@
    • Insert
    • Animation
    • Interactivity
    • -
    • Custom HTML Element
    • +
    • Custom HTML element
    • +
    • Custom chart element


      @@ -58,21 +59,20 @@

      g2.ext.js #

      See the API Reference

      g2 extension for expanded functionality #

      -

      g2.ext adds so functions which can be invoked on existing g2 elements, which is used in extensions like label and mark, aswell as in other extensions like g2.mec.js.

      -

      get length of line in label #

      +

      g2.ext adds properties that can be used on standard g2 elements, which is used in extensions like label and mark, as well as in other extensions like g2.mec.js.

      +

      get the length of a line in label #

      g2().view({cartesian:true}) - .lin({x1:10,y1:10,x2:90,y2:90,ld:[6,3]}) - .label({str:"@len",off:20}) + .lin({x1:10,y1:10,x2:90,y2:90,ld:[6,3], label:`@len`})

      label

      -

      How label works is shown further down on this page.

      +

      How label works is shown further below on this page.

      Splines #

      -

      g2.ext.js introduces a new method which draws splines by points by +

      g2.ext.js introduces a new method that draws splines by points by implementing a centripetal Catmull-Rom spline (thus avoiding cusps and self-intersections).

      The spline command expects an object with a pts property analogous to the ply method from g2.js. This property can have different formats, for each of which there must be provided an individual -array iterator. g2 implements three standard iterators which need not be specified by last itr argument.

      +array iterator. g2 implements three standard iterators which need not be specified by the last itr argument.

      @@ -96,37 +96,43 @@

      Splines #

      Another property of string is closed which can have boolean values true and false, which are interpreted as closed flags (also analogous to ply).

      -
      g2().style({ls:"red",lw:3,fs:"#ddd"}) +
      g2().style({ls:"red",lw:3,fs:"#ddd"}) .spline([10,10,40,70,60,30,90,80]) .spline([[110,10],[140,70],[160,30],[190,80]],true) - .spline([{x:210,y:10},{x:240,y:70},{x:260,y:30},{x:290,y:80}],'split'); + .spline([{x:210,y:10},{x:240,y:70},{x:260,y:30},{x:290,y:80}],'split');

      splines

      -

      Other ressources on splines:

      +

      Other resources on splines:

      Labels #

      -

      If you want to add text to a geometric element, you can always use the basic txt element. However some -element types support the smarter label element, which comes with a more intuitive relative positioning with respect -to its nearest previous element. Reliable positioning requires always cartesian coordinates.

      +

      If you want to add text to a geometric element, you can always use the basic txt element. +However, some element types support the smarter label property, which comes with a more intuitive relative positioning concerning its nearest previous element. +Reliable positioning always requires cartesian coordinates.

      +
      +

      Please note that with g2 version 3 labels as element have been deprecated in favor of usage as property.

      +
      - + - + + + + +
      ElementProperty Meaning
      label({str,loc,off})label: {str,loc,off} Label element placing string str at location loc using offset off.
      label: strSame, but with loc set to 'se' and off to 1.
      -

      Point like and rectangular elements provide locations according cardinal directions. Linear elements provide parameterized numerical -or named locations.

      +

      Point-like and rectangular elements provide locations according to cardinal directions.Linear elements provide parameterized numerical or named locations.

      @@ -185,7 +191,10 @@

      Labels #

      With linear, polygonial and spline elements a positive value means right of and a negativ value means left of the line at the specified point. If there is no offset distance specified, the global g2.State.labelOffset's value is taken. Please note, that cardinal locations are not sensitive to transformations.

      Markers #

      -

      Markers can be placed onto the outline of the previous element.

      +

      Markers can be placed onto the outline of an element.

      +
      +

      Please note that with g2 version 3 marks as element have been deprecated in favor of usage as property.

      +
      @@ -195,14 +204,21 @@

      Markers #

      - - + + + + + +
      mark({mrk,loc})Marker element placing marker symbol mrk at locations loc.mark:{mark,loc||count}Marker element placing marker symbol mark at locations loc. If count is given instead of loc, the marks are distributed evenly on the respective element.
      mark:loc||countSame, but with mark set to "tick" as default.
      -

      Elements with a unique center and rectangular elements provide locations according cardinal directions. -Linear elements provide parameterized numerical or named locations. The spline element only support indexed locations. It does not support parameterized locations.

      -

      Locations can also be passed as arrays, which makes it convenient for multiple marks with same mrk to be just passed in one calling of the mark method e.g mark({mrk:"tick",loc:[0,0.25,"mid",0.9]})

      +

      Elements with a unique center and rectangular elements provide locations according to the cardinal directions. +Linear elements provide parameterized numerical or named locations. +The spline element only supports indexed locations. +It does not support parameterized locations.

      +

      Locations can also be passed as arrays, which makes it convenient for multiple marks with the same mark to be just passed in one calling of the mark method e.g mark({mark:"tick",loc:[0,0.25,"mid",0.9]})

      +

      Default symbols are "tick", "sqr" and "dot".

      @@ -256,6 +272,176 @@

      Markers #

      +
      spline markers
      +

      Symbols #

      +

      img.symbols.mec

      +

      These symbols expect cartesian mode to be true, notable when using nodfix, nodflt and origin.

      +

      Some useful symbols implemented into g2.ext.js are:

      + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
      ElementImageMeaning
      pol({x,y})pol-defCreates a pol symbol.
      gnd({x,y})gnd-defCreates a gnd symbol.
      nod({x,y})nod-defCreates a nod symbol.
      dblnod({x,y})dblnod-defCreates a dblnod symbol.
      nodfix({x,y})nodfix-defCreates a nodfix symbol.
      nodflt({x,y})nodflt-defCreates a nodflt symbol.
      origin({x,y})origin-defCreates a origin symbol.
      +

      Helpful elements #

      +

      g2.ext.js also provides some extra elements that are based on core elements:

      + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
      ElementImageMeaning
      vec({x1,y1,x2,y2})vec-defLinear vector element which starts at x1, y1 and points to x2, y2.
      avec({x,y,r,w,dw})avec-defAngular vector element with center at x, y, radius r and spans a vectro from w to w+dw.
      dim({x1,y1,x2,y2,inside})dim-defLinear dimension element which starts at x1, y1 and points to x2, y2.
      adim({x,y,r,w,dw,inside})adim-defArc dimension element by center coordinates x, y, radius r, start angle w and angular range dw.
      +

      Styles #

      +

      There are some predefined colors, line styles and other constants. +They can be overwritten if needed.

      + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
      NameTypeDefaultDescription
      Stateobjectg2 state namespace.
      State.nodcolorstring"#333"node color.
      State.nodfillstring"#dedede"node fill color.
      State.nodfill2string"#aeaeae"alternate node fill color, somewhat darker.
      State.linkcolorstring"#666"link color.
      State.linkfillstring"rgba(200,200,200,0.5)"link fill color, semi-transparent.
      State.solidarray[]solid line style.
      State.dasharray[15,10]dashed line style.
      State.dotarray[4,4]dotted line style.
      State.dashdotarray[25,6.5,2,6.5]dashdotted line style.
      State.labelOffsetnumber5default label offset distance.
      State.labelSignificantDigitsnumber3default label's significant digits after floating point.
      diff --git a/docs/microjam.md/g2.ext.md b/docs/microjam.md/g2.ext.md index 1ed5b74..8de5cda 100644 --- a/docs/microjam.md/g2.ext.md +++ b/docs/microjam.md/g2.ext.md @@ -16,29 +16,28 @@ ### g2 extension for expanded functionality ## -g2.ext adds so functions which can be invoked on existing g2 elements, which is used in extensions like `label` and `mark`, aswell as in other extensions like `g2.mec.js`. +g2.ext adds properties that can be used on standard g2 elements, which is used in extensions like `label` and `mark`, as well as in other extensions like `g2.mec.js`. -#### get length of line in label +#### get the length of a line in `label` ```javascript g2().view({cartesian:true}) - .lin({x1:10,y1:10,x2:90,y2:90,ld:[6,3]}) - .label({str:"@len",off:20}) + .lin({x1:10,y1:10,x2:90,y2:90,ld:[6,3], label:`@len`}) ``` ![label](../img/get-len.png) -How label works is shown further down on this page. +How `label` works is shown further below on this page. ### Splines -g2.ext.js introduces a new method which draws splines by points by +g2.ext.js introduces a new method that draws splines by points by implementing a centripetal Catmull-Rom spline (thus avoiding cusps and self-intersections). The `spline` command expects an object with a `pts` property analogous to the [ply](https://github.com/goessner/g2/wiki/Elements#ply-element) method from g2.js. This property can have different formats, for each of which there must be provided an individual -array iterator. _g2_ implements three standard iterators which need not be specified by last `itr` argument. +array iterator. _g2_ implements three standard iterators which need not be specified by the last `itr` argument. Property Format | comment -------- | ------- @@ -57,22 +56,24 @@ g2().style({ls:"red",lw:3,fs:"#ddd"}) ![splines](../img/splines.png) -Other ressources on splines: +Other resources on splines: * [cubic hermite spline](https://en.wikipedia.org/wiki/Cubic_Hermite_spline) * [bézier curves](https://pomax.github.io/bezierinfo/) ### Labels -If you want to add text to a geometric element, you can always use the basic `txt` element. However some -element types support the smarter `label` element, which comes with a more intuitive relative positioning with respect -to its nearest previous element. Reliable positioning requires always *cartesian* coordinates. +If you want to add text to a geometric element, you can always use the basic `txt` element. +However, some element types support the smarter `label` property, which comes with a more intuitive relative positioning concerning its nearest previous element. +Reliable positioning always requires *cartesian* coordinates. -Element | Meaning +> Please note that with `g2` version 3 labels as element have been deprecated in favor of usage as property. + +Property | Meaning :--------: | :--------: -`label({str,loc,off})` | Label element placing string `str` at location `loc` using offset `off`. +`label: {str,loc,off}` | Label element placing string `str` at location `loc` using offset `off`. +`label: str` | Same, but with `loc` set to `'se'` and off to `1`. -Point like and rectangular elements provide locations according cardinal directions. Linear elements provide parameterized numerical -or named locations. +Point-like and rectangular elements provide locations according to cardinal directions.Linear elements provide parameterized numerical or named locations. | Type | Elements | default | locations | offset | img | | :-----: | :------: | :-----: | :-----: | :-----: | :-----: | @@ -85,20 +86,25 @@ or named locations. A numerical offset always means *outside* of closed shapes with regard to the specified point. With linear, polygonial and spline elements a positive value means *right of* and a negativ value means *left of* the line at the specified point. If there is no offset distance specified, the global `g2.State.labelOffset`'s value is taken. Please note, that cardinal locations are not sensitive to transformations. - - ### Markers -Markers can be placed onto the outline of the previous element. +Markers can be placed onto the outline of an element. + +> Please note that with `g2` version 3 marks as element have been deprecated in favor of usage as property. Element | Meaning :--------: | :--------: -`mark({mrk,loc})` | Marker element placing marker symbol `mrk` at locations `loc`. +`mark:{mark,loc||count}` | Marker element placing marker symbol `mark` at locations `loc`. If `count` is given instead of `loc`, the marks are distributed evenly on the respective element. +`mark:loc||count` | Same, but with mark set to `"tick"` as default. + +Elements with a unique center and rectangular elements provide locations according to the cardinal directions. +Linear elements provide parameterized numerical or named locations. +The `spline` element only supports indexed locations. +It does not support parameterized locations. -Elements with a unique center and rectangular elements provide locations according cardinal directions. -Linear elements provide parameterized numerical or named locations. The `spline` element only support indexed locations. It does not support parameterized locations. +Locations can also be passed as arrays, which makes it convenient for multiple marks with the same `mark` to be just passed in one calling of the mark method e.g `mark({mark:"tick",loc:[0,0.25,"mid",0.9]})` -Locations can also be passed as arrays, which makes it convenient for multiple marks with same `mrk` to be just passed in one calling of the mark method e.g `mark({mrk:"tick",loc:[0,0.25,"mid",0.9]})` +Default symbols are `"tick"`, `"sqr"` and `"dot"`. | Type | Elements | default | locations |dir | img | | :-----: | :------: | :-----: | :-----: | :-----: | :-----: | @@ -108,3 +114,50 @@ Locations can also be passed as arrays, which makes it convenient for multiple m | Polygonial elements | `ply, ground`
      `link, link2` | `0.5` | `beg, end, mid, all`
      `#index`
      normalized numerical parameter |`-1,0,1` | ![polygonial markers](../img/poly-markers.png) | Spline element | `spline` | `beg` | `beg, end, mid, all`
      `#index` |`-1,0,1` | ![spline markers](../img/spline-markers.png) +### Symbols ### +![img.symbols.mec](../img/symbols.png) + +These symbols expect `cartesian` mode to be true, notable when using `nodfix`, `nodflt` and `origin`. + +Some useful symbols implemented into `g2.ext.js` are: + +Element | Image | Meaning +-------- | -------- | ------ +`pol({x,y})` | ![pol-def](../img/pol-2.png) | Creates a pol symbol. +`gnd({x,y})` | ![gnd-def](../img/gnd.png) | Creates a gnd symbol. +`nod({x,y})` | ![nod-def](../img/nod.png) | Creates a nod symbol. +`dblnod({x,y})` | ![dblnod-def](../img/dblnod.png) | Creates a dblnod symbol. +`nodfix({x,y})` | ![nodfix-def](../img/nodfix.png) | Creates a nodfix symbol. +`nodflt({x,y})` | ![nodflt-def](../img/nodflt.png) | Creates a nodflt symbol. +`origin({x,y})` | ![origin-def](../img/origin_sym.png) | Creates a origin symbol. + +### Helpful elements + +`g2.ext.js` also provides some extra elements that are based on `core` elements: + +Element | Image | Meaning +-------- | -------- | ------ +`vec({x1,y1,x2,y2})` | ![vec-def](../img/vec-def.png) | Linear vector element which starts at `x1, y1` and points to `x2, y2`. +`avec({x,y,r,w,dw})` | ![avec-def](../img/avec-def.png) | Angular vector element with center at `x, y`, radius `r` and spans a vectro from `w` to `w+dw`. +`dim({x1,y1,x2,y2,inside})` | ![dim-def](../img/dim-def.png) | Linear dimension element which starts at `x1, y1` and points to `x2, y2`. +`adim({x,y,r,w,dw,inside})` | ![adim-def](../img/adim-def.png) | Arc dimension element by center coordinates `x`, `y`, radius `r`, start angle `w` and angular range `dw`. + +### Styles + +There are some predefined colors, line styles and other constants. +They can be overwritten if needed. + +| Name | Type | Default | Description | +| --- | --- | --- | --- | +| State | object | | `g2` state namespace. | +| State.nodcolor | string | "#333" | node color. | +| State.nodfill | string | "#dedede" | node fill color. | +| State.nodfill2 | string | "#aeaeae" | alternate node fill color, somewhat darker. | +| State.linkcolor | string | "#666" | link color. | +| State.linkfill | string | "rgba(200,200,200,0.5)" | link fill color, semi-transparent. | +| State.solid | array | [] | solid line style. | +| State.dash | array | [15,10] | dashed line style. | +| State.dot | array | [4,4] | dotted line style. | +| State.dashdot | array | [25,6.5,2,6.5] | dashdotted line style. | +| State.labelOffset | number | 5 | default label offset distance. | +| State.labelSignificantDigits | number | 3 | default label's significant digits after floating point. | diff --git a/docs/microjam.md/g2.html.html b/docs/microjam.md/g2.html.html index 66cf81d..01c9ba1 100644 --- a/docs/microjam.md/g2.html.html +++ b/docs/microjam.md/g2.html.html @@ -33,7 +33,8 @@
    • Insert
    • Animation
    • Interactivity
    • -
    • Custom HTML Element
    • +
    • Custom HTML element
    • +
    • Custom chart element

    diff --git a/docs/microjam.md/g2.mec.md b/docs/microjam.md/g2.mec.md index 1fe473b..81b6ee7 100644 --- a/docs/microjam.md/g2.mec.md +++ b/docs/microjam.md/g2.mec.md @@ -9,7 +9,6 @@ "tags": [] --- - [![License](https://img.shields.io/badge/license-MIT-blue.svg)](https://github.com/goessner/g2/blob/master/LICENSE) [![npm](https://img.shields.io/npm/v/g2d-mec.svg)](https://www.npmjs.com/package/g2d-mec) [![npm](https://img.shields.io/npm/dt/g2d-mec.svg)](https://www.npmjs.com/package/g2d-mec) @@ -22,35 +21,12 @@ Some of these elements use functions of `g2.ext`, so it is advisable to prepend `g2.ext` before using `g2.mec`. -### Mechanical Symbols ### -![img.symbols.mec](../img/symbols.png) - - -Since `g2.mec` is an extension for mechanical purposes, these symbols expect `g2` to be in `cartesian` mode. -This is notable with `nodfix`, `nodflt` and `origin`, since they will be upside down if the `cartesian` mode is not set. - -`g2.mec.js` provides some useful mechanical symbols for the [`g2.core.js`](https://github.com/goessner/g2.core) graphics library. - -Element | Image | Meaning --------- | -------- | ------ -`pol({x,y})` | ![pol-def](../img/pol-2.png) | Creates a pol symbol. -`gnd({x,y})` | ![gnd-def](../img/gnd.png) | Creates a gnd symbol. -`nod({x,y})` | ![nod-def](../img/nod.png) | Creates a nod symbol. -`dblnod({x,y})` | ![dblnod-def](../img/dblnod.png) | Creates a dblnod symbol. -`nodfix({x,y})` | ![nodfix-def](../img/nodfix.png) | Creates a nodfix symbol. -`nodflt({x,y})` | ![nodflt-def](../img/nodflt.png) | Creates a nodflt symbol. -`origin({x,y})` | ![origin-def](../img/origin_sym.png) | Creates a origin symbol. - ### Mechanical Elements ### `g2.mec.js` also provides some mechanical elements. Element | Image | Meaning -------- | -------- | ------ -`vec({x1,y1,x2,y2})` | ![vec-def](../img/vec-def.png) | Linear vector element which starts at `x1, y1` and points to `x2, y2`. -`avec({x,y,r,w,dw})` | ![avec-def](../img/avec-def.png) | Angular vector element with center at `x, y`, radius `r` and spans a vectro from `w` to `w+dw`. -`dim({x1,y1,x2,y2,inside})` | ![dim-def](../img/dim-def.png) | Linear dimension element which starts at `x1, y1` and points to `x2, y2`. -`adim({x,y,r,w,dw,inside})` | ![adim-def](../img/adim-def.png) | Arc dimension element by center coordinates `x`, `y`, radius `r`, start angle `w` and angular range `dw`. `slider({x,y,w,b,h})` | ![slider-def](../img/slider-def.png) | Slider element by center coordinates `x, y` and rotation angle `w` with breadth `b` and height `w`. `spring({x1,y1,x2,y2})` | ![spring-def](../img/spring-def.png) | Symbolical Spring element which starts at `x1, y1` and points to `x2, y2`. `damper({x1,y1,x2,y2})` | ![damper-def](../img/damper-def.png) | Symbolical Damper element which starts at `x1, y1` and points to `x2, y2`. @@ -66,28 +42,6 @@ Element | Image | Meaning `pulley2({x,y,r})` | ![pulley2-def](../img/pulley2-def.png) | Pulley2 element by position `x, y` including rotation angle `w` and radius `r`. `rope({x1,y1,r1,x2,y2,r2})` | ![rope-def](../img/rope-def.png) | Rope element tangential to two circles with given center points `p1`[`{x, y}`] and `p2`[`{x, y}`] and radii `r1` and `r2`. You can switch between the four possible tangents by the sign of the radii. - -### Mechanical Styles - -There are some predefined colors, line styles and other constants. -You can overwrite them, if you are not comfortable with it. - -| Name | Type | Default | Description | -| --- | --- | --- | --- | -| State | object | | `g2` state namespace. | -| State.nodcolor | string | "#333" | node color. | -| State.nodfill | string | "#dedede" | node fill color. | -| State.nodfill2 | string | "#aeaeae" | alternate node fill color, somewhat darker. | -| State.linkcolor | string | "#666" | link color. | -| State.linkfill | string | "rgba(200,200,200,0.5)" | link fill color, semi-transparent. | -| State.solid | array | [] | solid line style. | -| State.dash | array | [15,10] | dashed line style. | -| State.dot | array | [4,4] | dotted line style. | -| State.dashdot | array | [25,6.5,2,6.5] | dashdotted line style. | -| State.labelOffset | number | 5 | default label offset distance. | -| State.labelSignificantDigits | number | 3 | default label's significant digits after floating point. | - - ### Truss Example You can see a simple truss below, composed from mechanical symbols and elements. diff --git a/docs/navigation.md b/docs/navigation.md index faf9568..1255419 100644 --- a/docs/navigation.md +++ b/docs/navigation.md @@ -8,7 +8,8 @@ - [Insert]({base}/microjam.md/Insert.html) - [Animation]({base}/microjam.md/Animation.html) - [Interactivity]({base}/microjam.md/Interactivity.html) - - [Custom HTML Element]({base}/microjam.md/g2.html.html) + - [Custom HTML element]({base}/microjam.md/g2.html.html) + - [Custom chart element]({base}/microjam.md/g2.chart.html.html) --- diff --git a/docs/pages.json b/docs/pages.json deleted file mode 100644 index ebf5357..0000000 --- a/docs/pages.json +++ /dev/null @@ -1 +0,0 @@ -[{"layout":"page","title":"g2-Main page","date":"2020-06-01","description":"microjam for g2","tags":[],"category":[],"math":false,"uri":"c:\\Users\\me\\devel\\HiWi\\g2\\docs\\index.md","reluri":"index","base":".","header":"Main Page","uses":[{"uri":"navigation.md"}],"permalink":"#"},{"layout":"page","title":"g2-Animation","date":"2020-06-01","description":"g2-Animation","tags":[],"category":[],"math":false,"uri":"c:\\Users\\me\\devel\\HiWi\\g2\\docs\\microjam.md\\Animation.md","reluri":"microjam.md/Animation","base":"..","header":"Animation","permalink":"#","uses":[{"uri":"navigation.md"}]},{"layout":"page","title":"g2-Concepts","date":"2020-06-01","description":"g2-Concepts","tags":[],"category":[],"math":false,"uri":"c:\\Users\\me\\devel\\HiWi\\g2\\docs\\microjam.md\\Concepts.md","reluri":"microjam.md/Concepts","base":"..","header":"Concepts","permalink":"#","uses":[{"uri":"navigation.md"}]},{"layout":"page","title":"Elements","date":"2020-06-01","description":"Elements","tags":[],"category":[],"math":false,"uri":"c:\\Users\\me\\devel\\HiWi\\g2\\docs\\microjam.md\\Elements.md","reluri":"microjam.md/Elements","base":"..","header":"g2-Elements","permalink":"#","uses":[{"uri":"navigation.md"}]},{"layout":"page","title":"g2.chart","date":"2020-06-01","description":"g2.chart","tags":[],"category":[],"math":false,"uri":"c:\\Users\\me\\devel\\HiWi\\g2\\docs\\microjam.md\\g2.chart.md","reluri":"microjam.md/g2.chart","base":"..","header":"Extension: g2.chart","permalink":"#","uses":[{"uri":"navigation.md"}]},{"layout":"page","title":"g2.ext","date":"2020-06-01","description":"g2.ext","tags":[],"category":[],"math":false,"uri":"c:\\Users\\me\\devel\\HiWi\\g2\\docs\\microjam.md\\g2.ext.md","reluri":"microjam.md/g2.ext","base":"..","header":"Extension: g2.ext","permalink":"#","uses":[{"uri":"navigation.md"}]},{"layout":"page","title":"g2-Element","date":"2020-06-22","description":"g2-Element","tags":[],"category":[],"math":false,"uri":"c:\\Users\\me\\devel\\HiWi\\g2\\docs\\microjam.md\\g2.html.md","reluri":"microjam.md/g2.html","base":"..","header":"Custom HTML Element","permalink":"#","uses":[{"uri":"navigation.md"}]},{"layout":"page","title":"g2.mec","date":"2020-06-01","description":"g2.mec","tags":[],"category":[],"math":false,"uri":"c:\\Users\\me\\devel\\HiWi\\g2\\docs\\microjam.md\\g2.mec.md","reluri":"microjam.md/g2.mec","base":"..","header":"Extension: g2.mec","permalink":"#","uses":[{"uri":"navigation.md"}]},{"layout":"page","title":"g2-Getting started","date":"2020-06-01","description":"g2-Getting started","tags":[],"category":[],"math":false,"uri":"c:\\Users\\me\\devel\\HiWi\\g2\\docs\\microjam.md\\Getting-Started.md","reluri":"microjam.md/Getting-Started","base":"..","header":"Getting started","permalink":"#","uses":[{"uri":"navigation.md"}]},{"layout":"page","title":"g2-Insert","date":"2020-06-01","description":"g2-Insert","tags":[],"category":[],"math":false,"uri":"c:\\Users\\me\\devel\\HiWi\\g2\\docs\\microjam.md\\Insert.md","reluri":"microjam.md/Insert","base":"..","header":"Insert","permalink":"#","uses":[{"uri":"navigation.md"}]},{"layout":"page","title":"g2-Interactivity","date":"2020-06-01","description":"g2-Interactivity","tags":[],"category":[],"math":false,"uri":"c:\\Users\\me\\devel\\HiWi\\g2\\docs\\microjam.md\\Interactivity.md","reluri":"microjam.md/Interactivity","base":"..","header":"Interactivity","permalink":"#","uses":[{"uri":"navigation.md"}]},{"layout":"page","title":"g2-Paths","date":"2020-06-01","description":"g2-Paths","tags":[],"category":[],"math":false,"uri":"c:\\Users\\me\\devel\\HiWi\\g2\\docs\\microjam.md\\Paths.md","reluri":"microjam.md/Paths","base":"..","header":"Paths","permalink":"#","uses":[{"uri":"navigation.md"}]},{"layout":"page","title":"g2-Reuse","date":"2020-06-01","description":"g2-Reuse","tags":[],"category":[],"math":false,"uri":"c:\\Users\\me\\devel\\HiWi\\g2\\docs\\microjam.md\\Reuse.md","reluri":"microjam.md/Reuse","base":"..","header":"Reuse","permalink":"#","uses":[{"uri":"navigation.md"}]},{"layout":"page","title":"g2-State and Style","date":"2020-06-01","description":"g2-State and Style","tags":[],"category":[],"math":false,"uri":"c:\\Users\\me\\devel\\HiWi\\g2\\docs\\microjam.md\\State-and-Style.md","reluri":"microjam.md/State-and-Style","base":"..","header":"State and Style","permalink":"#","uses":[{"uri":"navigation.md"}]},{"layout":"page","title":"g2-View","date":"2020-06-01","description":"g2-View","tags":[],"category":[],"math":false,"uri":"c:\\Users\\me\\devel\\HiWi\\g2\\docs\\microjam.md\\View.md","reluri":"microjam.md/View","base":"..","header":"View","permalink":"#","uses":[{"uri":"navigation.md"}]}] \ No newline at end of file diff --git a/docs/tests.html b/docs/tests.html index c30194d..e4e167a 100644 --- a/docs/tests.html +++ b/docs/tests.html @@ -22,127 +22,107 @@ } + - -
    + + + + + + + + + + + +

    g2 Tests

    g2.js, g2.ext.js, g2.mec.js are referenced. canvas and image should look alike.
    -

    Error found at:

    +

    Error at number:

    + + + + + + + + + + + + +
    #namesrccanvasimageerror
    - - - - - - \ No newline at end of file + diff --git a/package.json b/package.json index e965fee..b5415f9 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "g2d", - "version": "2.5.0", + "version": "3.0.0", "description": "g2 - a tiny 2d graphics command library", "keywords": [ "vector", @@ -12,11 +12,14 @@ "javascript" ], "main": "g2.js", + "microjam": {}, "scripts": { - "build": "npm run minify && npm run jsdoc", - "minify": "uglifyjs ./src/g2.core.js ./src/g2.lib.js ./src/g2.ext.js ./src/g2.mec.js ./src/g2.chart.js -o ./src/g2.js --comments", - "concat": "type .\\src\\g2.core.js .\\src\\g2.lib.js .\\src\\g2.ext.js .\\src\\g2.mec.js .\\src\\g2.chart.js >.\\src\\g2.full.js", + "build": "npm run minify && npm run jsdoc && npm run 7z", + "minify": "uglifyjs ./src/g2.core.js ./src/g2.lib.js ./src/g2.ext.js ./src/g2.mec.js ./src/g2.chart.js -o ./dist/g2.js --comments", + "concat": "concat ./src/g2.core.js ./src/g2.lib.js ./src/g2.ext.js ./src/g2.mec.js ./src/g2.chart.js >./dist/g2.full.js", "jsdoc": "jsdoc2md ./src/g2.core.js > ./docs/api/g2.core.md && jsdoc2md ./src/g2.mec.js > ./docs/api/g2.mec.md && jsdoc2md ./src/g2.ext.js > ./docs/api/g2.ext.md && jsdoc2md ./src/g2.chart.js > ./docs/api/g2.chart.md", + "g2.html": "concat ./src/g2.core.js ./src/g2.io.js ./src/g2.lib.js ./src/g2.ext.js ./src/g2.mec.js ./src/g2.selector.js ./bin/canvasInteractor.js ./src/g2.element.js >./dist/g2.html.js", + "g2.chart.html": "concat ./src/g2.core.js ./src/g2.io.js ./src/g2.lib.js ./src/g2.ext.js ./src/g2.chart.js ./src/g2.selector.js ./bin/canvasInteractor.js ./src/g2.chart.element.js > ./dist/g2.chart.html.js", "7z": "7z -tgzip a ./g2.min.js.gz ./g2.min.js" }, "author": "Stefan Goessner ", @@ -28,7 +31,7 @@ "microjam": {}, "devDependencies": { "concat": "^1.0.0", - "jsdoc-to-markdown": "^1.3.2", + "jsdoc-to-markdown": "^5.0.3", "uglify-es": "^3.3.9" } } diff --git a/sample/charttst.html b/sample/charttst.html new file mode 100644 index 0000000..9ece5fd --- /dev/null +++ b/sample/charttst.html @@ -0,0 +1,57 @@ + + + + + + + \ No newline at end of file diff --git a/sample/g2-circle.html b/sample/g2-circle.html new file mode 100644 index 0000000..8bf96cb --- /dev/null +++ b/sample/g2-circle.html @@ -0,0 +1,45 @@ + + +g2 ellipse + + + + + + + + + + + + \ No newline at end of file diff --git a/sample/g2-core.html b/sample/g2-core.html new file mode 100644 index 0000000..c72229e --- /dev/null +++ b/sample/g2-core.html @@ -0,0 +1,17 @@ + + +g2 core + + + + + + + + \ No newline at end of file diff --git a/sample/g2-drag.html b/sample/g2-drag.html new file mode 100644 index 0000000..582c265 --- /dev/null +++ b/sample/g2-drag.html @@ -0,0 +1,32 @@ + + +g2 simple drag + + + + + + + + + + + + \ No newline at end of file diff --git a/sample/g2-handles.html b/sample/g2-handles.html new file mode 100644 index 0000000..3104215 --- /dev/null +++ b/sample/g2-handles.html @@ -0,0 +1,38 @@ + + +g2 handles + + + + + + + + + + + + \ No newline at end of file diff --git a/sample/g2-interactive-labels.html b/sample/g2-interactive-labels.html new file mode 100644 index 0000000..de2c854 --- /dev/null +++ b/sample/g2-interactive-labels.html @@ -0,0 +1,46 @@ + + +g2 labels + + + + + + + + + + + + \ No newline at end of file diff --git a/sample/g2-labels.html b/sample/g2-labels.html new file mode 100644 index 0000000..e71e5fa --- /dev/null +++ b/sample/g2-labels.html @@ -0,0 +1,33 @@ + + +g2 labels + + + + + + + + + + \ No newline at end of file diff --git a/sample/g2-nolabel.html b/sample/g2-nolabel.html new file mode 100644 index 0000000..62bcb51 --- /dev/null +++ b/sample/g2-nolabel.html @@ -0,0 +1,16 @@ + + +g2 labels + + + + + + + + \ No newline at end of file diff --git a/sample/g2-pan.html b/sample/g2-pan.html new file mode 100644 index 0000000..f7184b4 --- /dev/null +++ b/sample/g2-pan.html @@ -0,0 +1,23 @@ + + +g2 simple pan + + + + + + + + + \ No newline at end of file diff --git a/sample/g2-usebug.html b/sample/g2-usebug.html new file mode 100644 index 0000000..8810499 --- /dev/null +++ b/sample/g2-usebug.html @@ -0,0 +1,21 @@ + + +g2 use bug + + + + + + + + + + \ No newline at end of file diff --git a/sample/g2.element.test.html b/sample/g2.element.test.html new file mode 100644 index 0000000..0dc10a0 --- /dev/null +++ b/sample/g2.element.test.html @@ -0,0 +1,18 @@ + + +g2.chart.html - test + + + + +[{ + "data": [-2,6,0,-2,2,1,4,1,6,5], "fill":true, "dots":true +}] + + + + + + + + diff --git a/sample/g2.element.test2.html b/sample/g2.element.test2.html new file mode 100644 index 0000000..2536ff8 --- /dev/null +++ b/sample/g2.element.test2.html @@ -0,0 +1,32 @@ + + +g2.html - test + + + + +{ "main":[ + {"c":"cir","a":{"x":250,"y":100,"r":30,"fs":"orange"}}, + {"c":"cir","a":{"x":250,"y":100,"r":15,"fs":"snow"}} + ] +} + +
    + +{ "main":[ + {"c":"nodfix","a":{}}, + {"c":"nod","a":{"y":50}}, + {"c":"nod","a":{"x":150,"y":150}}, + {"c":"nodfix","a":{"x":200}} + ] +} + + + + + + diff --git a/sample/g2.element.test3.html b/sample/g2.element.test3.html new file mode 100644 index 0000000..a37abce --- /dev/null +++ b/sample/g2.element.test3.html @@ -0,0 +1,20 @@ + + +g2.html - bug + + + + + { "main":[ + {"c":"lin","a":{"x1":"@A.x","y1":"@A.y","x2":"@B.x","y2":"@B.y","lw":3,"ls":"@linkcolor"}}, + {"c":"cir","a":{"id":"A","x":150,"y":100,"r":5,"fs":"orange"}}, + {"c":"cir","a":{"id":"B","x":-50,"y":100,"r":5,"fs":"orange"}} + ] +} + + + + + diff --git a/sample/g2.element.test4.html b/sample/g2.element.test4.html new file mode 100644 index 0000000..964ed51 --- /dev/null +++ b/sample/g2.element.test4.html @@ -0,0 +1,31 @@ + + +g2.html - constraining + + + + + + + + diff --git a/sample/g2chart1.html b/sample/g2chart1.html new file mode 100644 index 0000000..d43257a --- /dev/null +++ b/sample/g2chart1.html @@ -0,0 +1,16 @@ + + + + + + Document + + + + [{ + "data": [-2,6,0,-2,2,1,4,1,6,5], "fill":true, "dots":true + }] + + + + \ No newline at end of file diff --git a/sample/g2chart2.html b/sample/g2chart2.html new file mode 100644 index 0000000..5c1457a --- /dev/null +++ b/sample/g2chart2.html @@ -0,0 +1,18 @@ + + + + + + Document + + + + [ + {"fn":Math.sin,"dx":0.104,"fill":true}, + {"fn":Math.tan,"dx":0.0349,"fill":true} + ] + + + + + \ No newline at end of file diff --git a/sample/g2imgtst.html b/sample/g2imgtst.html index efeeaa2..ad03eff 100644 --- a/sample/g2imgtst.html +++ b/sample/g2imgtst.html @@ -6,19 +6,16 @@ - - + + + - - + + \ No newline at end of file diff --git a/sample/svgpath.html b/sample/svgpath.html index 858c9c1..3de6f2e 100644 --- a/sample/svgpath.html +++ b/sample/svgpath.html @@ -8,8 +8,8 @@ - - + + diff --git a/src/g2.chart.element.js b/src/g2.chart.element.js new file mode 100644 index 0000000..18a0bc5 --- /dev/null +++ b/src/g2.chart.element.js @@ -0,0 +1,111 @@ +"use strict"; + +class G2ChartElement extends HTMLElement { + static get observedAttributes() { + return [ + 'width', + 'height', + 'xmin', + 'xmax', + 'ymin', + 'ymax', + 'title', + ]; + } + + constructor() { + super(); + this._root = this.attachShadow({ mode: 'open' }); + } + + get width() { return +this.getAttribute('width') || 301; } + set width(q) { q && this.setAttribute('width', q) } + get height() { return +this.getAttribute('height') || 201; } + set height(q) { q && this.setAttribute('height', q) } + get xmin() { return +this.getAttribute('xmin') || undefined; } + set xmin(q) { return q && +this.setAttribute('xmin', q) } + get xmax() { return +this.getAttribute('xmax') || undefined; } + set xmax(q) { return q && +this.setAttribute('xmax', q) } + get ymin() { return +this.getAttribute('ymin') || undefined; } + set ymin(q) { return q && +this.setAttribute('ymin', q) } + get ymax() { return +this.getAttribute('ymax') || undefined; } + set ymax(q) { return q && +this.setAttribute('ymax', q) } + get title() { return this.getAttribute('title') || ''; } + set title(q) { return q && this.setAttribute('title', q) } + + connectedCallback() { + this._root.innerHTML = G2ChartElement.template({ + width: this.width, height: this.height + }); + + this._ctx = this._root.getElementById('cnv').getContext('2d'); + + const t = 35; + this._chart = { + x: t, + y: t, + xmin: this.xmin, + xmax: this.xmax, + ymin: this.ymin, + ymax: this.ymax, + title: this.title, + b: this.width - t * 2, + h: this.height - t * 2, + xaxis: () => this.xaxis || {}, + yaxis: () => this.yaxis || {}, + title: () => this.title || "", + funcs: () => this.funcs + }; + + this._g = g2().del().clr().view({ cartesian: true }).chart(this._chart); + + try { + // If not true, the element should be referenced by another module. + if (this.innerHTML !== '') { + // Remove all functions (declared by "fn": fn) and fetch them before parsing. + // Then bring them back in using Function: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/eval#Never_use_eval + + // Find all functions declared by "fn:" like here: https://goessner.github.io/g2/g2.chart.html#example-multiple-functions + const funcRegEx = /(("|')fn("|'):)([^(,|})]+)/g; + const funcs = JSON.parse(this.innerHTML.replace(funcRegEx, '"fn":"PLACEHOLDER"').trim()); + let itr = 0; + for (const a of this.innerHTML.matchAll(funcRegEx)) { + funcs[itr].fn = (() => Function('"use strict"; return (' + a[4] + ')')())(); + itr++; + } + this.funcs = funcs; + } + } + catch (e) { + console.warn(e); + this._g.txt({ str: e, y: 5 }); + } + finally { + this._g.nod({ + x: () => this.nod && this.nod().x, + y: () => this.nod && this.nod().y, + scl: () => this.nod && this.nod().scl || 0 + }); + this.render(); + } + } + + render() { + this._g.exe(this._ctx); + } + + setFuncs(funcs) { + this.funcs = funcs; + return this; + } + + disconnectedCallback() { + // TODO + } + + static template({ width, height }) { + return `` + } +} +customElements.define('g2-chart', G2ChartElement); diff --git a/src/g2.core.js b/src/g2.core.js index 705003c..217d68d 100644 --- a/src/g2.core.js +++ b/src/g2.core.js @@ -2,12 +2,12 @@ "use strict" /** - * g2.core (c) 2013-19 Stefan Goessner + * g2.core (c) 2013-20 Stefan Goessner * @author Stefan Goessner * @license MIT License * @link https://github.com/goessner/g2 * @typedef {g2} - * @param {object} [opts] Custom options object. It is simply copied into the 'g2' instance, but not used from the g2 kernel. + * @param {object} [opts] Custom options object. * @description Create a 2D graphics command queue object. Call without using 'new'. * @returns {g2} * @example @@ -21,7 +21,7 @@ function g2(opts) { let o = Object.create(g2.prototype); o.commands = []; - if (opts) Object.assign(o,opts); + if (opts) Object.assign(o, opts); return o; } @@ -31,7 +31,7 @@ g2.prototype = { * @method * @returns {object} g2 */ - clr() { return this.addCommand({c:'clr'}); }, + clr() { return this.addCommand({ c: 'clr' }); }, /** * Set the view by placing origin coordinates and scaling factor in device units @@ -44,7 +44,7 @@ g2.prototype = { * @property {number} [y=0] - y-origin in device units. * @property {boolean} [cartesian=false] - set cartesian flag. */ - view({scl,x,y,cartesian}) { return this.addCommand({c:'view',a:arguments[0]}); }, + view({ scl, x, y, cartesian }) { return this.addCommand({ c: 'view', a: arguments[0] }); }, /** * Draw grid. @@ -54,7 +54,7 @@ g2.prototype = { * @property {string} [color=#ccc] - change color. * @property {number} [size=20] - change space between lines. */ - grid({color,size}={}) { return this.addCommand({c:'grid',a:arguments[0]}); }, + grid({ color, size } = {}) { return this.addCommand({ c: 'grid', a: arguments[0] }); }, /** * Draw circle by center and radius. @@ -73,7 +73,7 @@ g2.prototype = { * @example * g2().cir({x:100,y:80,r:20}) // Draw circle. */ - cir({x,y,r,w}) { return this.addCommand({c:'cir',a:arguments[0]}); }, + cir({ x, y, r, w }) { return this.addCommand({ c: 'cir', a: arguments[0] }); }, /** * Draw ellipse by center and radius for x and y. @@ -96,7 +96,7 @@ g2.prototype = { * @example * g2().ell({x:100,y:80,rx:20,ry:30,w:0,dw:2*Math.PI/4,rot:1}) // Draw circle. */ - ell({x,y,rx,ry,w,dw,rot}) { return this.addCommand({c:'ell',a:arguments[0]}); }, + ell({ x, y, rx, ry, w, dw, rot }) { return this.addCommand({ c: 'ell', a: arguments[0] }); }, /** * Draw arc by center point, radius, start angle and angular range. @@ -119,7 +119,7 @@ g2.prototype = { * g2().arc({x:300,y:400,r:390,w:-Math.PI/4,dw:-Math.PI/2}) * .exe(ctx); */ - arc({x,y,r,w,dw}) { return this.addCommand({c:'arc',a:arguments[0]}); }, + arc({ x, y, r, w, dw }) { return this.addCommand({ c: 'arc', a: arguments[0] }); }, /** * Draw rectangle by anchor point and dimensions. @@ -141,7 +141,7 @@ g2.prototype = { * @example * g2().rec({x:100,y:80,b:40,h:30}) // Draw rectangle. */ - rec({x,y,b,h}) { return this.addCommand({c:'rec',a:arguments[0]}); }, + rec({ x, y, b, h }) { return this.addCommand({ c: 'rec', a: arguments[0] }); }, /** * Draw line by start point and end point. @@ -161,7 +161,7 @@ g2.prototype = { * @example * g2().lin({x1:10,x2:10,y1:190,y2:10}) // Draw line. */ - lin({x1,y1,x2,y2}) { return this.addCommand({c:'lin',a:arguments[0]}); }, + lin({ x1, y1, x2, y2 }) { return this.addCommand({ c: 'lin', a: arguments[0] }); }, /** * Draw polygon by points. @@ -193,9 +193,9 @@ g2.prototype = { * .ply({pts:[{x:160,y:70},{x:180,y:80},{x:140,y:90}]}), * .exe(ctx); */ - ply({pts,format,closed,x,y,w}) { + ply({ pts, format, closed, x, y, w }) { arguments[0]._itr = format && g2.pntIterator[format](pts) || g2.pntItrOf(pts); - return this.addCommand({c:'ply',a:arguments[0]}); + return this.addCommand({ c: 'ply', a: arguments[0] }); }, /** @@ -219,7 +219,7 @@ g2.prototype = { * [Font]{@link https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/font} * [styling]{@link https://html.spec.whatwg.org/multipage/canvas.html#dom-context-2d-font} */ - txt({str,x,y,w}) { return this.addCommand({c:'txt',a:arguments[0]}); }, + txt({ str, x, y, w }) { return this.addCommand({ c: 'txt', a: arguments[0] }); }, /** * Reference g2 graphics commands from another g2 object or a predefined g2.symbol. @@ -255,11 +255,12 @@ g2.prototype = { * g2.symbol.cross = g2().lin({x1:5,y1:5,x2:-5,y2:-5}).lin({x1:5,y1:-5,x2:-5,y2:5}); // Define symbol. * g2().use({grp:"cross",x:100,y:100}) // Draw cross at position 100,100. */ - use({grp,x,y,w,scl}) { - if (typeof grp === "string") // must be a member name of the 'g2.symbol' namespace - arguments[0].grp = grp = g2.symbol[grp]; - if (grp && grp !== this) // avoid self reference .. - this.addCommand({c:'use',a:arguments[0]}); + use({ grp, x, y, w, scl }) { + if (grp && grp !== this) { // avoid self reference .. + if (typeof grp === "string") // must be a member name of the 'g2.symbol' namespace + arguments[0].grp = g2.symbol[(grp in g2.symbol) ? grp : 'unknown']; + this.addCommand({ c: 'use', a: arguments[0] }); + } return this; }, @@ -283,7 +284,7 @@ g2.prototype = { * @property {number} [w = 0] - rotation angle (about upper left, in radians). * @property {number} [scl = 1] - image scaling. */ - img({uri,x,y,b,h,sx,sy,sb,sh,xoff,yoff,w,scl}) { return this.addCommand({c:'img',a:arguments[0]}); }, + img({ uri, x, y, b, h, sx, sy, sb, sh, xoff, yoff, w, scl }) { return this.addCommand({ c: 'img', a: arguments[0] }); }, /** * Begin subcommands. Current state is saved. @@ -313,7 +314,7 @@ g2.prototype = { * [Font]{@link https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/font} * [styling]{@link https://html.spec.whatwg.org/multipage/canvas.html#dom-context-2d-font} */ - beg({x,y,w,scl,matrix}={}) { return this.addCommand({c:'beg',a:arguments[0]}); }, + beg({ x, y, w, scl, matrix } = {}) { return this.addCommand({ c: 'beg', a: arguments[0] }); }, /** * End subcommands. Previous state is restored. @@ -323,12 +324,12 @@ g2.prototype = { */ end() { // ignore 'end' commands without matching 'beg' let myBeg = 1, - findMyBeg = (cmd) => { - if (cmd.c === 'beg') myBeg--; + findMyBeg = (cmd) => { // care about nested beg...end blocks ... + if (cmd.c === 'beg') myBeg--; else if (cmd.c === 'end') myBeg++; return myBeg === 0; } - return g2.getCmdIdx(this.commands,findMyBeg) !== false ? this.addCommand({c:'end'}) : this; + return g2.cmdIdxBy(this.commands, findMyBeg) !== false ? this.addCommand({ c: 'end' }) : this; }, /** @@ -336,14 +337,14 @@ g2.prototype = { * @method * @returns {object} g2 */ - p() { return this.addCommand({c:'p'}); }, + p() { return this.addCommand({ c: 'p' }); }, /** * Close current path by straight line. * @method * @returns {object} g2 */ - z() { return this.addCommand({c:'z'}); }, + z() { return this.addCommand({ c: 'z' }); }, /** * Move to point. @@ -353,7 +354,7 @@ g2.prototype = { * @property {number} x - move to x coordinate * @property {number} y - move to y coordinate */ - m({x,y}) { return this.addCommand({c:'m',a:arguments[0]}); }, + m({ x, y }) { return this.addCommand({ c: 'm', a: arguments[0] }); }, /** * Create line segment to point. @@ -366,10 +367,10 @@ g2.prototype = { * g2().p() // Begin path. * .m({x:0,y:50}) // Move to point. * .l({x:300,y:0}) // Line segment to point. - * .l(x:400,y:100}) // ... + * .l({x:400,y:100}) // ... * .stroke() // Stroke path. */ - l({x,y}) { return this.addCommand({c:'l',a:arguments[0]}); }, + l({ x, y }) { return this.addCommand({ c: 'l', a: arguments[0] }); }, /** * Create quadratic bezier curve segment to point. @@ -386,7 +387,7 @@ g2.prototype = { * .q({x1:200,y1:200,x:400,y:0}) // Quadratic bezier curve segment. * .stroke() // Stroke path. */ - q({x1,y1,x,y}) { return this.addCommand({c:'q',a:arguments[0]});}, + q({ x1, y1, x, y }) { return this.addCommand({ c: 'q', a: arguments[0] }); }, /** * Create cubic bezier curve to point. @@ -406,7 +407,7 @@ g2.prototype = { * .stroke() // Stroke path. * .exe(ctx); // Render to canvas context. */ - c({x1,y1,x2,y2,x,y}) { return this.addCommand({c:'c',a:arguments[0]}); }, + c({ x1, y1, x2, y2, x, y }) { return this.addCommand({ c: 'c', a: arguments[0] }); }, /** * Draw arc with angular range to target point. @@ -419,15 +420,15 @@ g2.prototype = { * @example * g2().p() // Begin path. * .m({x:50,y:50}) // Move to point. - * .a({dw:2,x:300,y:100}) // Create cubic bezier curve. + * .a({dw:2,x:300,y:100}) // Create arc segment. * .stroke() // Stroke path. * .exe(ctx); // Render to canvas context. */ - a({dw,x,y}) { - let prvcmd = this.commands[this.commands.length-1]; - g2.cpyProp(prvcmd.a,'x',arguments[0],'_xp'); - g2.cpyProp(prvcmd.a,'y',arguments[0],'_yp'); - return this.addCommand({c:'a',a:arguments[0]}); + a({ dw, x, y }) { + let prvcmd = this.commands[this.commands.length - 1]; + g2.cpyProp(prvcmd.a, 'x', arguments[0], '_xp'); + g2.cpyProp(prvcmd.a, 'y', arguments[0], '_yp'); + return this.addCommand({ c: 'a', a: arguments[0] }); }, /** @@ -437,7 +438,7 @@ g2.prototype = { * @param {object} - stroke arguments object. * @property {string} [d = undefined] - SVG path definition string. Current path is ignored then. */ - stroke({d}={}) { return this.addCommand({c:'stroke',a:arguments[0]}); }, + stroke({ d } = {}) { return this.addCommand({ c: 'stroke', a: arguments[0] }); }, /** * Fill the current path or path object. @@ -446,7 +447,7 @@ g2.prototype = { * @param {object} - fill arguments object. * @property {string} [d = undefined] - SVG path definition string. Current path is ignored then. */ - fill({d}={}) { return this.addCommand({c:'fill',a:arguments[0]}); }, + fill({ d } = {}) { return this.addCommand({ c: 'fill', a: arguments[0] }); }, /** * Shortcut for stroke and fill the current path or path object. @@ -456,7 +457,7 @@ g2.prototype = { * @param {object} - drw arguments object. * @property {string} [d = undefined] - SVG path definition string. Current path is ignored then. */ - drw({d,lsh}={}) { return this.addCommand({c:'drw',a:arguments[0]}); }, + drw({ d, lsh } = {}) { return this.addCommand({ c: 'drw', a: arguments[0] }); }, /** * Delete all commands beginning from `idx` to end of command queue. @@ -484,9 +485,9 @@ g2.prototype = { * .ins(node) // draw node. * .exe(ctx) // render to canvas context. */ - ins(fn) { - return typeof fn === 'function' ? (fn(this) || this) - : typeof fn === 'object' ? ( this.commands.push({c:'ins',a:fn}), this ) // no 'addCommand' .. ! + ins(arg) { + return typeof arg === 'function' ? (arg(this) || this) // no further processing by handler ... + : typeof arg === 'object' ? (this.commands.push({ a: arg }), this) // no explicit command name .. ! : this; }, /** @@ -502,14 +503,32 @@ g2.prototype = { return this; }, // helpers ... - addCommand({c,a}) { - if (a && Object.getPrototypeOf(a) === Object.prototype) { // modify only pure argument objects 'a' - for (const key in a) - if (!Object.getOwnPropertyDescriptor(a,key).get // if 'key' is no getter ... - && key[0] !== '_' // and no private property ... - && typeof a[key] === 'function') { // and a function - Object.defineProperty(a, key, { get:a[key], enumerable:true, configurable:true, writabel:false }); + addCommand({ c, a }) { + if (a && Object.getPrototypeOf(a) === Object.prototype) { // modify only pure argument objects 'a' .. ! + for (const key in a) { + if (!Object.getOwnPropertyDescriptor(a, key).get // if 'key' is no getter ... + && key[0] !== '_' // and no private property ... + && typeof a[key] === 'function') { // and a function ... make it a getter + Object.defineProperty(a, key, { get: a[key], enumerable: true, configurable: true, writabel: false }); + } + if (typeof a[key] === 'string' && a[key][0] === '@') { // referring values by neighbor id's + const refidIdx = a[key].indexOf('.'); + const refid = refidIdx > 0 ? a[key].substr(1, refidIdx - 1) : ''; + const refkey = refid ? a[key].substr(refidIdx + 1) : ''; + const refcmd = refid ? () => this.commands.find((cmd) => cmd.a && cmd.a.id === refid) : undefined; + + if (refcmd) + Object.defineProperty(a, key, { + get: function () { + const rc = refcmd(); + return rc && (refkey in rc.a) ? rc.a[refkey] : 0; + }, + enumerable: true, + configurable: true, + writabel: false + }); } + } if (g2.prototype[c].prototype) Object.setPrototypeOf(a, g2.prototype[c].prototype); } this.commands.push(arguments[0]); @@ -518,81 +537,84 @@ g2.prototype = { }; // statics -g2.defaultStyle = {fs:'transparent',ls:'#000',lw:1,lc:"butt",lj:"miter",ld:[],ml:10,sh:[0,0],lsh:false,font:'14px serif',thal:'start',tval:'alphabetic'}; -g2.symbol = {}; -g2.handler = function(ctx) { +g2.defaultStyle = { fs: 'transparent', ls: '#000', lw: 1, lc: "butt", lj: "miter", ld: [], ml: 10, sh: [0, 0], lsh: false, font: '14px serif', thal: 'start', tval: 'alphabetic' }; +g2.symbol = { + unknown: g2().cir({ r: 12, fs: 'orange' }).txt({ str: '?', thal: 'center', tval: 'middle', font: 'bold 20pt serif' }) +}; +g2.handler = function (ctx) { let hdl; for (let h of g2.handler.factory) - if ((hdl = h(ctx)) !== false) - return hdl; + if ((hdl = h(ctx)) !== false) + return hdl; return false; } g2.handler.factory = []; // predefined polyline/spline point iterators g2.pntIterator = { - "x,y": function(pts) { - function pitr(i) { return {x:pts[2*i],y:pts[2*i+1]}; }; - Object.defineProperty(pitr, 'len', { get:() => pts.length/2, enumerable:true, configurable:true, writabel:false }); - return pitr; - }, - "[x,y]": function(pts) { - function pitr(i) { return pts[i] ? {x:pts[i][0],y:pts[i][1]} : undefined; }; - Object.defineProperty(pitr, 'len', { get:() => pts.length, enumerable:true, configurable:true, writabel:false }); - return pitr; - }, - "{x,y}": function(pts) { - function pitr(i) { return pts[i]; }; - Object.defineProperty(pitr, 'len', { get:() => pts.length, enumerable:true, configurable:true, writabel:false }); - return pitr; - } + "x,y": function (pts) { + function pitr(i) { return { x: pts[2 * i], y: pts[2 * i + 1] }; }; + Object.defineProperty(pitr, 'len', { get: () => pts.length / 2, enumerable: true, configurable: true, writabel: false }); + return pitr; + }, + "[x,y]": function (pts) { + function pitr(i) { return pts[i] ? { x: pts[i][0], y: pts[i][1] } : undefined; }; + Object.defineProperty(pitr, 'len', { get: () => pts.length, enumerable: true, configurable: true, writabel: false }); + return pitr; + }, + "{x,y}": function (pts) { + function pitr(i) { return pts[i]; }; + Object.defineProperty(pitr, 'len', { get: () => pts.length, enumerable: true, configurable: true, writabel: false }); + return pitr; + } }; -g2.pntItrOf = function(pts) { - return !(pts && pts.length) ? undefined - : typeof pts[0] === "number" ? g2.pntIterator["x,y"](pts) - : Array.isArray(pts[0]) && pts[0].length >= 2 ? g2.pntIterator["[x,y]"](pts) - : typeof pts[0] === "object" && "x" in pts[0] && "y" in pts[0] ? g2.pntIterator["{x,y}"](pts) - : undefined; +g2.pntItrOf = function (pts) { + return !(pts && pts.length) ? undefined + : typeof pts[0] === "number" ? g2.pntIterator["x,y"](pts) + : Array.isArray(pts[0]) && pts[0].length >= 2 ? g2.pntIterator["[x,y]"](pts) + : typeof pts[0] === "object" && "x" in pts[0] && "y" in pts[0] ? g2.pntIterator["{x,y}"](pts) + : undefined; }; /** * Get index of command resolving 'callbk' to 'true' starting from end of the queue walking back.
    * Similar to 'Array.prototype.findIndex', only working reverse. * @private */ -g2.getCmdIdx = function(cmds,callbk) { - for (let i = cmds.length-1; i >= 0; i--) - if (callbk(cmds[i],i,cmds)) +g2.cmdIdxBy = function (cmds, callbk) { + for (let i = cmds.length - 1; i >= 0; i--) + if (callbk(cmds[i], i, cmds)) return i; return false; // command with index '0' signals 'failing' ... }; /** * Replacement for Object.assign, as it does not assign getters and setter properly ... + * See https://github.com/tc39/proposal-object-getownpropertydescriptors * See https://medium.com/@benastontweet/mixins-in-javascript-700ec81f5e5c + * Shallow copy of prototypes (think interfaces) + * @private */ -g2.mixin = function mixin(obj, ...protos) { - protos.forEach(p => { - Object.keys(p).forEach(k => { - Object.defineProperty(obj, k, Object.getOwnPropertyDescriptor(p, k)); - }) - }) - return obj; +g2.mix = function mix(...protos) { + let mixture = {}; + for (const p of protos) + mixture = Object.defineProperties(mixture, Object.getOwnPropertyDescriptors(p)); + return mixture; } /** - * Copy properties, even as getters .. a useful fraction of the above .. + * Copy properties, even as getters .. a useful part of the above .. * @private */ -g2.cpyProp = function(from,fromKey,to,toKey) { Object.defineProperty(to, toKey, Object.getOwnPropertyDescriptor(from, fromKey)); } +g2.cpyProp = function (from, fromKey, to, toKey) { Object.defineProperty(to, toKey, Object.getOwnPropertyDescriptor(from, fromKey)); } // Html canvas handler -g2.canvasHdl = function(ctx) { +g2.canvasHdl = function (ctx) { if (this instanceof g2.canvasHdl) { if (ctx instanceof CanvasRenderingContext2D) { this.ctx = ctx; this.cur = g2.defaultStyle; this.stack = [this.cur]; - this.matrix = [[1,0,0,1,0.5,0.5]]; + this.matrix = [[1, 0, 0, 1, 0.5, 0.5]]; this.gridBase = 2; this.gridExp = 1; return this; @@ -600,120 +622,138 @@ g2.canvasHdl = function(ctx) { else return null; } - return g2.canvasHdl.apply(Object.create(g2.canvasHdl.prototype),arguments); + return g2.canvasHdl.apply(Object.create(g2.canvasHdl.prototype), arguments); }; g2.handler.factory.push((ctx) => ctx instanceof g2.canvasHdl ? ctx - : ctx instanceof CanvasRenderingContext2D ? g2.canvasHdl(ctx) : false); + : ctx instanceof CanvasRenderingContext2D ? g2.canvasHdl(ctx) : false); g2.canvasHdl.prototype = { - init(grp,style) { + init(grp, style) { this.stack.length = 1; this.matrix.length = 1; - this.initStyle(style ? Object.assign({},this.cur,style) : this.cur); + this.initStyle(style ? Object.assign({}, this.cur, style) : this.cur); return true; }, async exe(commands) { for (let cmd of commands) { - if (cmd.c && this[cmd.c]) { + // cmd.a is an object offering a `g2` method, so call it and execute its returned commands array. + if (cmd.a && cmd.a.g2) { + const cmds = cmd.a.g2().commands; + // If false, ext was not applied to this cmd. But the command still renders + if (cmds) { + this.exe(cmds); + continue; + } + } + // cmd.a is a `g2` object, so directly execute its commands array. + else if (cmd.a && cmd.a.commands) { + this.exe(cmd.a.commands); + continue; + } + if (cmd.c && this[cmd.c]) { // explicit command name .. ! const rx = this[cmd.c](cmd.a); if (rx && rx instanceof Promise) { await rx; } - } else if (cmd.a && 'g2' in cmd.a) - this.exe(cmd.a.g2().commands); + } } }, - view({x=0,y=0,scl=1,cartesian=false}) { - this.pushTrf(cartesian ? [scl,0,0,-scl,x,this.ctx.canvas.height-1-y] - : [scl,0,0,scl,x,y] ); + view({ x = 0, y = 0, scl = 1, cartesian = false }) { + this.pushTrf(cartesian ? [scl, 0, 0, -scl, x, this.ctx.canvas.height - 1 - y] + : [scl, 0, 0, scl, x, y]); }, - grid({color='#ccc',size}={}) { + grid({ color = '#ccc', size } = {}) { let ctx = this.ctx, b = ctx.canvas.width, h = ctx.canvas.height, - {x,y,scl} = this.uniTrf, + { x, y, scl } = this.uniTrf, sz = size || this.gridSize(scl), - xoff = x%sz, yoff = y%sz; + xoff = x % sz, yoff = y % sz; ctx.save(); - ctx.setTransform(1,0,0,1,0,0); + ctx.setTransform(1, 0, 0, 1, 0, 0); ctx.strokeStyle = color; ctx.lineWidth = 1; ctx.beginPath(); - for (let x=xoff,nx=b+1; x Number.EPSILON && Math.abs(r) > Number.EPSILON) { this.ctx.beginPath(); - this.ctx.arc(x,y,Math.abs(r),w,w+dw,dw<0); + this.ctx.arc(x, y, Math.abs(r), w, w + dw, dw < 0); this.drw(arguments[0]); } else if (Math.abs(dw) < Number.EPSILON && Math.abs(r) > Number.EPSILON) { const cw = Math.cos(w), sw = Math.sin(w); this.ctx.beginPath(); - this.ctx.moveTo(x-r*cw,y-r*sw); - this.ctx.lineTo(x+r*cw,y+r*sw); + this.ctx.moveTo(x - r * cw, y - r * sw); + this.ctx.lineTo(x + r * cw, y + r * sw); } - // else // nothing to draw with r === 0 + // else // nothing to draw with r === 0 }, - ell({x=0,y=0,rx,ry,w=0,dw=2*Math.PI,rot=0}) { + ell({ rx, ry, w = 0, dw = 2 * Math.PI, rot = 0 }) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; this.ctx.beginPath(); - this.ctx.ellipse(x,y,Math.abs(rx),Math.abs(ry),rot,w,w+dw,dw<0); + this.ctx.ellipse(x, y, Math.abs(rx), Math.abs(ry), rot, w, w + dw, dw < 0); this.drw(arguments[0]); }, - rec({x=0,y=0,b,h}) { - let tmp = this.setStyle(arguments[0]); - this.ctx.fillRect(x,y,b,h); - this.ctx.strokeRect(x,y,b,h); + rec({ b, h }) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + const tmp = this.setStyle(arguments[0]); + this.ctx.fillRect(x, y, b, h); + this.ctx.strokeRect(x, y, b, h); this.resetStyle(tmp); }, - lin({x1=0,y1=0,x2,y2}) { - let ctx = this.ctx; - ctx.beginPath(); - ctx.moveTo(x1,y1); - ctx.lineTo(x2,y2); - this.stroke(arguments[0]); + lin(args) { + this.ctx.beginPath(); + this.ctx.moveTo(args.p1 && args.p1.x || args.x1 || 0, args.p1 && args.p1.y || args.y1 || 0); + this.ctx.lineTo(args.p2 && args.p2.x || args.x2 || 0, args.p2 && args.p2.y || args.y2 || 0); + this.stroke(args); }, - ply: function({pts,closed,x=0,y=0,w=0,_itr}) { + ply({ pts, closed, w = 0, _itr }) { if (_itr && _itr.len) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; let p, i, len = _itr.len, istrf = !!(x || y || w), cw, sw; - if (istrf) this.setTrf([cw=(w?Math.cos(w):1),sw=(w?Math.sin(w):0),-sw,cw,x||0,y||0]); + if (istrf) this.setTrf([cw = (w ? Math.cos(w) : 1), sw = (w ? Math.sin(w) : 0), -sw, cw, x, y]); this.ctx.beginPath(); - this.ctx.moveTo((p=_itr(0)).x,p.y); - for (i=1; i < len; i++) - this.ctx.lineTo((p=_itr(i)).x,p.y); + this.ctx.moveTo((p = _itr(0)).x, p.y); + for (i = 1; i < len; i++) + this.ctx.lineTo((p = _itr(i)).x, p.y); if (closed) // closed then .. this.ctx.closePath(); this.drw(arguments[0]); if (istrf) this.resetTrf(); - return i-1; // number of points .. + return i - 1; // number of points .. } return 0; }, - txt({str,x=0,y=0,w=0,unsizable}) { - let tmp = this.setStyle(arguments[0]), + txt({ str, w = 0/*,unsizable*/ }) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; + const tmp = this.setStyle(arguments[0]), sw = w ? Math.sin(w) : 0, cw = w ? Math.cos(w) : 1, - trf = this.isCartesian ? [cw,sw,sw,-cw,x,y] - : [cw,sw,-sw,cw,x,y]; - this.setTrf(unsizable ? this.concatTrf(this.unscaleTrf({x,y}),trf) : trf); + trf = this.isCartesian ? [cw, sw, sw, -cw, x, y] + : [cw, sw, -sw, cw, x, y]; + this.setTrf(trf); // this.setTrf(unsizable ? this.concatTrf(this.unscaleTrf({x,y}),trf) : trf); if (this.ctx.fillStyle === 'rgba(0, 0, 0, 0)') { this.ctx.fillStyle = this.ctx.strokeStyle; tmp.fs = 'transparent'; } - this.ctx.fillText(str,0,0); + this.ctx.fillText(str, 0, 0); this.resetTrf(); this.resetStyle(tmp); }, @@ -734,8 +774,8 @@ g2.canvasHdl.prototype = { resolve(img); img = undefined; }; - img.addEventListener('error', error, {once:true}); - img.addEventListener('load', load, {once:true}); + img.addEventListener('error', error, { once: true }); + img.addEventListener('load', load, { once: true }); }); try { @@ -763,38 +803,39 @@ g2.canvasHdl.prototype = { } return img; }, - async img({uri,x=0,y=0,b,h,sx=0,sy=0,sb,sh,xoff=0,yoff=0,w=0,scl=1}) { + async img({ uri, x = 0, y = 0, b, h, sx = 0, sy = 0, sb, sh, xoff = 0, yoff = 0, w = 0, scl = 1 }) { const img_ = await this.loadImage(uri); this.ctx.save(); const cart = this.isCartesian ? -1 : 1; sb = sb || img_.width; b = b || img_.width; sh = (sh || img_.height); - h = (h || img_.height)*cart; - yoff*=cart; - w*=cart; - y = this.isCartesian ? -(y/scl)+sy : y/scl; - const [cw,sw] = [Math.cos(w), Math.sin(w)]; - this.ctx.scale(scl, scl*cart); - this.ctx.transform(cw, sw, -sw, cw,x/scl,y); - this.ctx.drawImage(img_,sx,sy,sb,sh,xoff,yoff,b,h); + h = (h || img_.height) * cart; + yoff *= cart; + w *= cart; + y = this.isCartesian ? -(y / scl) + sy : y / scl; + const [cw, sw] = [Math.cos(w), Math.sin(w)]; + this.ctx.scale(scl, scl * cart); + this.ctx.transform(cw, sw, -sw, cw, x / scl, y); + this.ctx.drawImage(img_, sx, sy, sb, sh, xoff, yoff, b, h); this.ctx.restore(); }, - use({grp}) { + use({ grp }) { this.beg(arguments[0]); this.exe(grp.commands); this.end(); }, - beg({x=0,y=0,w=0,scl=1,matrix,unsizable}={}) { + beg({ w = 0, scl = 1, matrix/*,unsizable*/ } = {}) { + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; let trf = matrix; if (!trf) { let ssw, scw; - ssw = w ? Math.sin(w)*scl : 0; - scw = w ? Math.cos(w)*scl : scl; - trf = [scw,ssw,-ssw,scw,x,y]; + ssw = w ? Math.sin(w) * scl : 0; + scw = w ? Math.cos(w) * scl : scl; + trf = [scw, ssw, -ssw, scw, x, y]; } this.pushStyle(arguments[0]); - this.pushTrf(unsizable ? this.concatTrf(this.unscaleTrf({x,y}),trf) : trf); + this.pushTrf(trf); // this.pushTrf(unsizable ? this.concatTrf(this.unscaleTrf({x,y}),trf) : trf); }, end() { this.popTrf(); @@ -802,20 +843,21 @@ g2.canvasHdl.prototype = { }, p() { this.ctx.beginPath(); }, z() { this.ctx.closePath(); }, - m({x,y}) { this.ctx.moveTo(x,y); }, - l({x,y}) { this.ctx.lineTo(x,y); }, - q({x,y,x1,y1}) { this.ctx.quadraticCurveTo(x1,y1,x,y); }, - c({x,y,x1,y1,x2,y2}) { this.ctx.bezierCurveTo(x1,y1,x2,y2,x,y); }, - a({x,y,dw,k,phi,_xp,_yp}) { // todo: fix elliptical arc bug ... + m({ x, y }) { this.ctx.moveTo(x, y); }, + l({ x, y }) { this.ctx.lineTo(x, y); }, + q({ x, y, x1, y1 }) { this.ctx.quadraticCurveTo(x1, y1, x, y); }, + c({ x, y, x1, y1, x2, y2 }) { this.ctx.bezierCurveTo(x1, y1, x2, y2, x, y); }, + a({ dw, k, phi, _xp, _yp }) { // todo: fix elliptical arc bug ... + const { x = 0, y = 0 } = arguments[0].p !== undefined ? arguments[0].p : arguments[0]; if (k === undefined) k = 1; // ratio r1/r2 if (Math.abs(dw) > Number.EPSILON) { if (k === 1) { // circular arc ... - let x12 = x-_xp, y12 = y-_yp; - let tdw_2 = Math.tan(dw/2), - rx = (x12 - y12/tdw_2)/2, ry = (y12 + x12/tdw_2)/2, - R = Math.hypot(rx,ry), - w = Math.atan2(-ry,-rx); - this.ctx.ellipse(_xp+rx,_yp+ry,R,R,0,w,w+dw,this.cartesian?dw>0:dw<0); + let x12 = x - _xp, y12 = y - _yp; + let tdw_2 = Math.tan(dw / 2), + rx = (x12 - y12 / tdw_2) / 2, ry = (y12 + x12 / tdw_2) / 2, + R = Math.hypot(rx, ry), + w = Math.atan2(-ry, -rx); + this.ctx.ellipse(_xp + rx, _yp + ry, R, R, 0, w, w + dw, this.cartesian ? dw > 0 : dw < 0); } else { // elliptical arc .. still buggy .. ! if (phi === undefined) phi = 0; @@ -823,46 +865,46 @@ g2.canvasHdl.prototype = { y1 = dw > 0 ? _yp : y, x2 = dw > 0 ? x : _xp, y2 = dw > 0 ? y : _yp; - let x12 = x2-x1, y12 = y2-y1, + let x12 = x2 - x1, y12 = y2 - y1, _dw = (dw < 0) ? dw : -dw; // if (dw < 0) dw = -dw; // test for bugs .. ! let cp = phi ? Math.cos(phi) : 1, sp = phi ? Math.sin(phi) : 0, - dx = -x12*cp - y12*sp, dy = -x12*sp - y12*cp, - sdw_2 = Math.sin(_dw/2), - R = Math.sqrt((dx*dx + dy*dy/(k*k))/(4*sdw_2*sdw_2)), - w = Math.atan2(k*dx,dy) - _dw/2, - x0 = x1 - R*Math.cos(w), - y0 = y1 - R*k*Math.sin(w); - this.ctx.ellipse(x0,y0,R, R*k,phi,w,w+dw,this.cartesian?dw>0:dw<0); + dx = -x12 * cp - y12 * sp, dy = -x12 * sp - y12 * cp, + sdw_2 = Math.sin(_dw / 2), + R = Math.sqrt((dx * dx + dy * dy / (k * k)) / (4 * sdw_2 * sdw_2)), + w = Math.atan2(k * dx, dy) - _dw / 2, + x0 = x1 - R * Math.cos(w), + y0 = y1 - R * k * Math.sin(w); + this.ctx.ellipse(x0, y0, R, R * k, phi, w, w + dw, this.cartesian ? dw > 0 : dw < 0); } } else - this.ctx.lineTo(x,y); + this.ctx.lineTo(x, y); }, - stroke({d}={}) { + stroke({ d } = {}) { let tmp = this.setStyle(arguments[0]); d ? this.ctx.stroke(new Path2D(d)) : this.ctx.stroke(); // SVG path syntax this.resetStyle(tmp); }, - fill({d}={}) { + fill({ d } = {}) { let tmp = this.setStyle(arguments[0]); d ? this.ctx.fill(new Path2D(d)) : this.ctx.fill(); // SVG path syntax this.resetStyle(tmp); }, - drw({d,lsh}={}) { + drw({ d, lsh } = {}) { let ctx = this.ctx, tmp = this.setStyle(arguments[0]), p = d && new Path2D(d); // SVG path syntax d ? ctx.fill(p) : ctx.fill(); if (ctx.shadowColor !== 'rgba(0, 0, 0, 0)' && ctx.fillStyle !== 'rgba(0, 0, 0, 0)' && !lsh) { - let shc = ctx.shadowColor; // usually avoid stroke shadow when filling ... - ctx.shadowColor = 'rgba(0, 0, 0, 0)'; - d ? ctx.stroke(p) : ctx.stroke(); - ctx.shadowColor = shc; + let shc = ctx.shadowColor; // usually avoid stroke shadow when filling ... + ctx.shadowColor = 'rgba(0, 0, 0, 0)'; + d ? ctx.stroke(p) : ctx.stroke(); + ctx.shadowColor = shc; } else - d ? ctx.stroke(p) : ctx.stroke(); + d ? ctx.stroke(p) : ctx.stroke(); this.resetStyle(tmp); }, @@ -875,32 +917,34 @@ g2.canvasHdl.prototype = { lc: (ctx) => ctx.lineCap, lj: (ctx) => ctx.lineJoin, ld: (ctx) => ctx.getLineDash(), + ldoff: (ctx) => ctx.lineDashOffset, ml: (ctx) => ctx.miterLimit, - sh: (ctx) => [ctx.shadowOffsetX||0,ctx.shadowOffsetY||0, - ctx.shadowBlur||0,ctx.shadowColor||'black'], + sh: (ctx) => [ctx.shadowOffsetX || 0, ctx.shadowOffsetY || 0, + ctx.shadowBlur || 0, ctx.shadowColor || 'black'], font: (ctx) => ctx.font, thal: (ctx) => ctx.textAlign, tval: (ctx) => ctx.textBaseline, }, set: { - fs: (ctx,q) => { ctx.fillStyle=q; }, - ls: (ctx,q) => { ctx.strokeStyle=q; }, - lw: (ctx,q) => { ctx.lineWidth=q; }, - lc: (ctx,q) => { ctx.lineCap=q; }, - lj: (ctx,q) => { ctx.lineJoin=q; }, - ld: (ctx,q) => { ctx.setLineDash(q); }, - ml: (ctx,q) => { ctx.miterLimit=q; }, - sh: (ctx,q) => { + fs: (ctx, q) => { ctx.fillStyle = q; }, + ls: (ctx, q) => { ctx.strokeStyle = q; }, + lw: (ctx, q) => { ctx.lineWidth = q; }, + lc: (ctx, q) => { ctx.lineCap = q; }, + lj: (ctx, q) => { ctx.lineJoin = q; }, + ld: (ctx, q) => { ctx.setLineDash(q); }, + ldoff: (ctx, q) => { ctx.lineDashOffset = q; }, + ml: (ctx, q) => { ctx.miterLimit = q; }, + sh: (ctx, q) => { if (q) { - ctx.shadowOffsetX = q[0]||0; - ctx.shadowOffsetY = q[1]||0; - ctx.shadowBlur = q[2]||0; - ctx.shadowColor = q[3]||'black'; + ctx.shadowOffsetX = q[0] || 0; + ctx.shadowOffsetY = q[1] || 0; + ctx.shadowBlur = q[2] || 0; + ctx.shadowColor = q[3] || 'black'; } }, - font: (ctx,q) => { ctx.font=q; }, - thal: (ctx,q) => { ctx.textAlign=q; }, - tval: (ctx,q) => { ctx.textBaseline=q; } + font: (ctx, q) => { ctx.font = q; }, + thal: (ctx, q) => { ctx.textAlign = q; }, + tval: (ctx, q) => { ctx.textBaseline = q; } }, initStyle(style) { for (const key in style) @@ -915,7 +959,7 @@ g2.canvasHdl.prototype = { let ref = style[key].substr(1); style[key] = g2.symbol[ref] || this.get[ref] && this.get[ref](this.ctx); } - if ((q=this.get[key](this.ctx)) !== style[key]) { + if ((q = this.get[key](this.ctx)) !== style[key]) { prv[key] = q; this.set[key](this.ctx, style[key]); } @@ -936,68 +980,68 @@ g2.canvasHdl.prototype = { style[key] = g2.symbol[ref] || this.get[ref] && this.get[ref](this.ctx); } if (this.cur[key] !== style[key]) - this.set[key](this.ctx, (cur[key]=style[key])); + this.set[key](this.ctx, (cur[key] = style[key])); } - this.stack.push(this.cur = Object.assign({},this.cur,cur)); + this.stack.push(this.cur = Object.assign({}, this.cur, cur)); }, popStyle() { let cur = this.stack.pop(); - this.cur = this.stack[this.stack.length-1]; + this.cur = this.stack[this.stack.length - 1]; for (const key in this.cur) if (this.get[key] && this.cur[key] !== cur[key]) - this.set[key](this.ctx, this.cur[key]); + this.set[key](this.ctx, this.cur[key]); }, - concatTrf(q,t) { + concatTrf(q, t) { return [ - q[0]*t[0]+q[2]*t[1], - q[1]*t[0]+q[3]*t[1], - q[0]*t[2]+q[2]*t[3], - q[1]*t[2]+q[3]*t[3], - q[0]*t[4]+q[2]*t[5]+q[4], - q[1]*t[4]+q[3]*t[5]+q[5] + q[0] * t[0] + q[2] * t[1], + q[1] * t[0] + q[3] * t[1], + q[0] * t[2] + q[2] * t[3], + q[1] * t[2] + q[3] * t[3], + q[0] * t[4] + q[2] * t[5] + q[4], + q[1] * t[4] + q[3] * t[5] + q[5] ]; }, initTrf() { this.ctx.setTransform(...this.matrix[0]); }, setTrf(t) { - this.ctx.setTransform(...this.concatTrf(this.matrix[this.matrix.length-1],t)); + this.ctx.setTransform(...this.concatTrf(this.matrix[this.matrix.length - 1], t)); }, resetTrf() { - this.ctx.setTransform(...this.matrix[this.matrix.length-1]); + this.ctx.setTransform(...this.matrix[this.matrix.length - 1]); }, pushTrf(t) { - let q_t = this.concatTrf(this.matrix[this.matrix.length-1],t); + let q_t = this.concatTrf(this.matrix[this.matrix.length - 1], t); this.matrix.push(q_t); this.ctx.setTransform(...q_t); }, popTrf() { this.matrix.pop(); - this.ctx.setTransform(...this.matrix[this.matrix.length-1]); + this.ctx.setTransform(...this.matrix[this.matrix.length - 1]); }, get isCartesian() { // det of mat2x2 < 0 ! - let m = this.matrix[this.matrix.length-1]; - return m[0]*m[3] - m[1]*m[2] < 0; + let m = this.matrix[this.matrix.length - 1]; + return m[0] * m[3] - m[1] * m[2] < 0; }, get uniTrf() { - let m = this.matrix[this.matrix.length-1]; - return {x:m[4],y:m[5],scl:Math.hypot(m[0],m[1]),cartesian:m[0]*m[3] - m[1]*m[2] < 0}; + let m = this.matrix[this.matrix.length - 1]; + return { x: m[4], y: m[5], scl: Math.hypot(m[0], m[1]), cartesian: m[0] * m[3] - m[1] * m[2] < 0 }; }, - unscaleTrf({x,y}) { // remove scaling effect (make unzoomable with respect to (x,y)) - let m = this.matrix[this.matrix.length-1], - invscl = 1/Math.hypot(m[0],m[1]); - return [invscl,0,0,invscl,(1-invscl)*x,(1-invscl)*y]; + unscaleTrf({ x, y }) { // remove scaling effect (make unzoomable with respect to (x,y)) + let m = this.matrix[this.matrix.length - 1], + invscl = 1 / Math.hypot(m[0], m[1]); + return [invscl, 0, 0, invscl, (1 - invscl) * x, (1 - invscl) * y]; }, gridSize(scl) { let base = this.gridBase, exp = this.gridExp, sz; - while ((sz = scl*base*Math.pow(10,exp)) < 14 || sz > 35) { + while ((sz = scl * base * Math.pow(10, exp)) < 14 || sz > 35) { if (sz < 14) { - if (base == 1) base = 2; + if (base == 1) base = 2; else if (base == 2) base = 5; else if (base == 5) { base = 1; exp++; } } else { - if (base == 1) { base = 5; exp--; } + if (base == 1) { base = 5; exp--; } else if (base == 2) base = 1; else if (base == 5) base = 2; } @@ -1008,18 +1052,5 @@ g2.canvasHdl.prototype = { } } -// utils - -g2.zoomView = function({scl,x,y}) { return { scl, x:(1-scl)*x, y:(1-scl)*y } } -// fn argument must be a function with (optional) timestamp 't' as single argument -// returning true to continue or false to stop RAF. -g2.render = function render(fn) { - function animate(t) { - if (fn(t)) - requestAnimationFrame(animate); - } - animate(performance.now()); -} - // use it with node.js ... ? if (typeof module !== 'undefined') module.exports = g2; diff --git a/src/g2.d.ts b/src/g2.d.ts deleted file mode 100644 index c63c383..0000000 --- a/src/g2.d.ts +++ /dev/null @@ -1,228 +0,0 @@ -// // Type definitions for g2 2.0 -// // Project: https://github.com/goessner/g2 -// // Definitions by: Stefan Goessner - -/** - * create a g2object - * https://github.com/goessner/g2/wiki - */ -declare function g2():g2; - -interface coordinate { - x: number; - y: number; -} -interface radii { - rx: number; - ry: number; -} -interface magnitude { - b: number; - h: number; -} -interface position1 { - x1: number; - y1: number; -} -interface position2 { - x2: number; - y2: number; -} -interface position_n { - xn: number; - yn: number; -} -interface delta_coordinate { - dx: number; - dy: number; -} -interface off_coordinate { - xoff: number; - yoff: number; -} -interface radius { r: number;} -interface scale { scl: number;} -interface cartesian { cartesian: boolean;} -interface color { color: string;} -interface size { size: number;} -interface delta_angle { dw: number;} -interface angle { w: number;} -interface rotation { rot: number;} -interface closed { closed: boolean;} -interface points { pts: position_n[];} -interface text { str: string;} -interface g2object { grp: g2;} -interface uri { uri: string;} -interface matrix { matrix: number[]} -interface svg { - /* - * Draw a SVG Path. - */ - d: string;} - -interface view extends coordinate, scale, cartesian {} -interface grid extends color, size {} -interface cir extends coordinate, radius {} -interface ell extends coordinate, radii, angle, delta_angle, rotation {} -interface arc extends coordinate, radius, angle, delta_angle {} -interface rec extends coordinate, magnitude {} -interface lin extends position1, position2 {} -interface ply extends coordinate, closed, angle, points {} -interface txt extends text, coordinate, angle {} -interface use extends g2object, coordinate, scale, angle {} -interface img extends uri, coordinate, angle, magnitude, delta_coordinate, off_coordinate {} -interface beg extends coordinate, angle, scale, matrix {} -interface m extends coordinate {} -interface l extends coordinate {} -interface q extends position1, position2 {} -interface c extends coordinate, position1, position2 {} -interface a extends coordinate, angle {} -interface stroke extends svg {} -interface fill extends svg {} -interface drw extends svg {} - -interface g2 { - - /** - * clear viewport - * https://github.com/goessner/g2/wiki/animation - */ - clr:()=>g2; - - /** - * change view - * https://github.com/goessner/g2/wiki/view - */ - view:(obj:view)=>g2; - - /** - * create grid - * https://github.com/goessner/g2/wiki/ - */ - grid:(obj:grid)=>g2; - - /** - * create a circle - * https://github.com/goessner/g2/wiki/elements - */ - cir:(obj:cir)=>g2; - - /** - * create a ellipse - * https://github.com/goessner/g2/wiki/elements - */ - ell:(obj:ell)=>g2; - - /** - * create an arc - * https://github.com/goessner/g2/wiki/elements - */ - arc:(obj:arc)=>g2; - - /** - * create a rectangle - */ - rec:(obj:rec)=>g2; - - /** - * create a line - * https://github.com/goessner/g2/wiki/elements - */ - lin:(obj:lin)=>g2; - - /** - * create a polygon - * https://github.com/goessner/g2/wiki/elements - */ - ply:(obj:ply)=>g2; - - /** - * write a text - * https://github.com/goessner/g2/wiki/elements - */ - txt:(obj:txt)=>g2; - - /** - * reuse a g2-object - * https://github.com/goessner/g2/wiki/reuse - */ - use:(obj:use)=>g2; - - /** - * import an image - * https://github.com/goessner/g2/wiki/elements - */ - img:(obj:img)=>g2; - - /** - * create a "view-matrix" - * https://github.com/goessner/g2/wiki - */ - beg:(obj:beg)=>g2; - - /** - * pops the "top-matrix" - * https://github.com/goessner/g2/wiki - */ - end:()=>g2; - - /** - * begin new path - * https://github.com/goessner/g2/wiki/paths - */ - p:()=>g2; - - /** - * close path - * https://github.com/goessner/g2/wiki/paths - */ - z:()=>g2; - - /** - * move to position - * https://github.com/goessner/g2/wiki/paths - */ - m:(obj:m)=>g2; - - /** - * create line segment - * https://github.com/goessner/g2/wiki/paths - */ - l:(obj:l)=>g2; - - /** - * quadratic curve to - * https://github.com/goessner/g2/wiki/paths - */ - q:(obj:q)=>g2; - - /** - * bezier-curve to - * https://github.com/goessner/g2/wiki/paths - */ - c:(obj:c)=>g2; - - /** - * arc to - * https://github.com/goessner/g2/wiki/paths - */ - a:(obj:a)=>g2; - - /** - * stroke the previously defined path - * https://github.com/goessner/g2/wiki/paths - */ - stroke:(obj:stroke)=>g2; - - /** - * fill the previously defined path - * https://github.com/goessner/g2/wiki/paths - */ - fill:(obj:fill)=>g2; - - /** - * fill and stroke the previously defined path in that order - * https://github.com/goessner/g2/wiki/paths - */ - drw:(obj:drw)=>g2; -} \ No newline at end of file diff --git a/src/g2.editor.js b/src/g2.editor.js deleted file mode 100644 index bd2edcc..0000000 --- a/src/g2.editor.js +++ /dev/null @@ -1,138 +0,0 @@ -/** - * g2.editor.js (c) 2018 Stefan Goessner - * @file editor interfaces for `g2` elements. - * @author Stefan Goessner - * @license MIT License - */ -"use strict" - -/** - * Extensions. - * (Requires cartesian coordinate system) - * @namespace - */ -var g2 = g2 || { prototype:{} }; // for jsdoc only ... - -// extend prototypes for argument objects -g2.editor = function() { - if (this instanceof g2.editor) { - this.draggable = false; - this.handles = g2(); - return this; - } - return g2.editor.apply(Object.create(g2.editor.prototype),arguments); -}; -g2.handler.factory.push((ctx) => ctx instanceof g2.editor ? ctx : false); - -g2.NONE = 0x0; g2.OVER = 0x1; g2.DRAG = 0x2; g2.EDIT = 0x4; -g2.editor.state = ['NONE','OVER','DRAG','OVER+DRAG','EDIT','OVER+EDIT']; - -g2.editor.prototype = { - init(grp) { - return true; - }, - get selection() { - return this.handles.commands.length && this.handles || false; - }, - get dragInfo() { return this.draggable && '_info' in this.draggable && this.draggable._info() }, - g2() { return this.handles; }, - exe(commands) { - if (this.evt) { // selection state can change only with events... ! - if (this.selection) { - if (this.evt.type === 'click') { // something selected .. - for (let cmd of commands) - if (cmd.a && cmd.a.state) - cmd.a.state = g2.NONE; - this.handles.del(); - } - else { - for (let cmd of this.handles.commands) // treat handles interactivity ! - if (cmd.c === 'handle') - this.hit(cmd.a); - } - } - else { - for (let elm=false, i=commands.length; i && !elm; i--) // stop after first hit .. starting from list end ! - elm = this.hit(commands[i-1].a); - } - this.evt = false; - } - }, - on({type,x,y,dx,dy,btn}) { - this.evt = {type,x,y,dx,dy,btn,eps:2}; - return this; - }, - hit(elm) { - const {type,dx,dy} = this.evt; - if (!elm) { // commands without arguments object .. ! - return false; - } - else if (elm.state & g2.EDIT) { // in EDIT mode - if (type === 'click') // leave EDIT mode .. - elm.state ^= g2.EDIT; - } - else if (!(elm.state & g2.OVER)) { // not in OVER mode - if (type === 'pointer' && this.elementHit(elm,this.evt)) // enter OVER mode .. - elm.state |= g2.OVER; - } - else if (elm.state & g2.DRAG) { // in DRAG mode - if (type === 'drag' && elm.drag) // drag element .. - elm.drag({dx,dy}); - else if (type === 'buttonup') // leave DRAG mode .. - this.elementDragEnd(elm); - else if (type === 'click') - this.elementEdit(elm); // enter EDIT mode .. - } - else if (elm.state & g2.OVER) { // in OVER mode - if (type === 'pointer' && !this.elementHit(elm,this.evt)) // leave OVER mode .. - elm.state ^= g2.OVER; - else if (type === 'buttondown') // enter DRAG mode .. - this.elementDragBeg(elm); - } - else - console.log('what state .. '+g2.editor.state[elm.state]); - - this.curState = g2.editor.state[elm && elm.state || 0]; - - return elm.state && elm; - }, - elementHit(elm,{x,y,eps}) { - return elm.isSolid ? elm.hitInner && elm.hitInner({x,y,eps}) - : elm.hitContour && elm.hitContour({x,y,eps}); - }, - elementDragBeg(elm) { - elm.state |= g2.DRAG; - this.draggable = elm; - }, - elementDragEnd(elm) { - elm.state ^= g2.DRAG; - this.draggable = false; - }, - elementEdit(elm) { - if ('handles' in elm) { - elm.state = elm.state ^ g2.OVER | g2.EDIT; - elm.handles(this.handles); - } - }, -}; - -// implement zoom agnostic handle ... -g2.prototype.handle = function handle({x,y,_update,info}) { return this.addCommand({c:'handle',a:arguments[0]}); }; -g2.prototype.handle.prototype = { - isSolid: true, - sz: 4, - hitInner({x,y,eps}) { return Math.abs(this.x-x) < this.sz+eps && Math.abs(this.y-y) < this.sz+eps; }, - drag({dx,dy}) { this.x += dx; this.y += dy; this._update({x:this.x,y:this.y,dx,dy}); }, - get fs() { return this.state & g2.DRAG ? 'beige' : 'lightgoldenrodyellow' }, - get sh() { return this.state ? [0,0,5,"gray"] : false }, - _info() { return `x:${this.x}
    y:${this.y}` } -} -g2.canvasHdl.prototype.handle = function({x,y,sz,fs,sh}) { - let m = this.matrix[this.matrix.length-1], - scl = Math.hypot(m[0],m[1]), - z = sz/scl, - tmp = this.setStyle({ls:'#444',lw:1/scl,fs,sh}); - this.ctx.fillRect(x-z+0.5,y-z+0.5,2*z,2*z); - this.ctx.strokeRect(x-z+0.5,y-z+0.5,2*z,2*z); - this.resetStyle(tmp); -} diff --git a/src/g2.element.js b/src/g2.element.js new file mode 100644 index 0000000..4ad0d26 --- /dev/null +++ b/src/g2.element.js @@ -0,0 +1,164 @@ +/** + * g2.element.js (c) 2019-20 Stefan Goessner + * @license MIT License + */ +"use strict"; + +class G2Element extends HTMLElement { + static get observedAttributes() { + return ['width', 'height','cartesian','grid', 'x0', 'y0', 'darkmode', 'interactive','background']; + } + + constructor() { + super(); + this._root = this.attachShadow({ mode:'open' }); + } + + get width() { return +this.getAttribute('width') || 301; } + set width(q) { if (q) this.setAttribute('width',q); } + get height() { return +this.getAttribute('height') || 201; } + set height(q) { if (q) this.setAttribute('height',q); } + get x0() { return (+this.getAttribute('x0')) || 0; } + set x0(q) { if (q) this.setAttribute('x0',q); } + get y0() { return (+this.getAttribute('y0')) || 0; } + set y0(q) { if (q) this.setAttribute('y0',q); } + get cartesian() { return this.hasAttribute('cartesian'); } + set cartesian(q) { q ? this.setAttribute('cartesian','') : this.removeAttribute('cartesian'); } + get grid() { return this.hasAttribute('grid') || false; } + set grid(q) { q ? this.setAttribute('grid','') : this.removeAttribute('grid'); } + get darkmode() { return this.hasAttribute('darkmode') || false; } + get interactive() { return this.hasAttribute('interactive') || false; } + get background() { return this.getAttribute('background') || false; } + get g() { return this._g; } + get canvas() { return this._ctx && this._ctx.canvas || false } + get readyState() { return !!this._g; } + + update() { + if (!this.interactive && this._g && this._ctx) + this._g.exe(this._ctx); + } + + init() { + const state = {x:this.x0,y:this.y0,cartesian:this.cartesian}; + // add shadow dom + this._root.innerHTML = G2Element.template({width:this.width,height:this.height,darkmode:this.darkmode}); + // cache elements of shadow dom + this._logview = this._root.getElementById('logview'); + // set up canvas context + this._ctx = this._root.getElementById('cnv').getContext('2d'); + // set up canvas background + if (this.background && G2Element[this.background]) + this._ctx.canvas.style.backgroundImage = 'url('+G2Element[this.background]+')'; + // set up canvas interactor + if (this.interactive) { + this._interactor = canvasInteractor.create(this._ctx, state); + this._selector = g2.selector(this._interactor.evt); + this._interactor.on('tick', e => this.ontick(e)) + .on('pan', e => this.onpan(e)) + .on('drag', e => this.ondrag(e)) + .on('wheel', e => this.onwheel(e)); + } + this._g = g2().clr().view(this.interactive && this._interactor.view || state); + if (this.grid) this._g.grid({color:this.darkmode?'#999':'#ccc'}); + this.initContent(this.innerHTML.trim(), e => this.log(e)); + if (this.interactive) + this._interactor.startTimer(); + else // no tick timer .. ! + this._g.exe(this._ctx); + this.dispatchEvent(new CustomEvent('init')); + } + initContent(content, onErr) { + if (!content) return; + try { + content = JSON.parse(content); // is valid JSON string ? + content = g2.io.parseGrp(content, 'main', onErr); // is valid g2 json string + if (content && content.commands) { // content is a valid g2 object. + this._g.commands.push(...content.commands); // inject commands of `main` group ... + } // ... into `_g` with a little brute force. + } + catch(e) { + const w = this.width, h = this.height, x0 = this.x0, y0 = this.y0, dx = w/20, dy = h/20, + x = q => q*w-x0, y = q => q*h-y0; + + content = g2({id:'main'}).stroke({"d":`M${x(0.05)},${y(0.05)} ${x(0.45)},${y(0.45)}M${x(0.55)},${y(0.55)} ${x(0.95)},${y(0.95)}M${x(0.05)},${y(0.95)} ${x(0.45)},${y(0.55)}M${x(0.55)},${y(0.45)} ${x(0.95)},${y(0.05)}`,lw:10,ls:"red",lc:"round"}); + this._g.ins(content); + onErr(e.message); + } + return !!content; + } + deinit() { + delete this._selector; + if (this.interactive) { + delete this._interactor.deinit(); + delete this._interactor; + delete this._selector; + } + delete this._g; + // delete cached data + delete this._ctx; + } + log(str) { + this._logview.innerHTML += str; + } + + ontick(e) { + this.dispatchEvent(new CustomEvent('tick')); + if (this.interactive) + this._g.exe(this._selector); + this._g.exe(this._ctx); + } + onpan(e) { + this._interactor.view.x = this.x0 += e.dx; + this._interactor.view.y = this.y0 += e.dy; + } + ondrag(e) { // only modify selected geometry here .. do not redraw .. ! + if (this._selector.selection && this._selector.selection.drag) { + this._selector.selection.drag({x:e.xusr,y:e.yusr,dx:e.dxusr,dy:e.dyusr,mode:'drag'}); + this.dispatchEvent(new CustomEvent('drag')); + } + } + onwheel(e) { + this._interactor.view.x = e.x + e.dscl*(this._interactor.view.x - e.x); + this._interactor.view.y = e.y + e.dscl*(this._interactor.view.y - e.y); + this._interactor.view.scl *= e.dscl; + } + + on(hdl,fn) { } + + // standard lifecycle callbacks + // https://developer.mozilla.org/en-US/docs/Web/Web_Components/Using_custom_elements + connectedCallback() { + this.init(); + } + disconnectedCallback() { + this.deinit(); + } + attributeChangedCallback(name, oldval, val) { + if (this._root && this._root.getElementById('cnv')) { + if (name === 'width') { // todo: preserve minimum width + this._root.getElementById('cnv').setAttribute('width',val); + this._root.querySelector('.status').style.width = val+'px'; + } + if (name === 'height') // todo: preserve minimum height + this._root.getElementById('cnv').setAttribute('height',val); + } + } + + static template({width,height,darkmode}) { +return ` + +
    +
    +
    
    +
    +` + } + + static get paper() { return " "; } +} +customElements.define('g-2', G2Element); diff --git a/src/g2.ext.js b/src/g2.ext.js index 3137a1e..8a0fcad 100644 --- a/src/g2.ext.js +++ b/src/g2.ext.js @@ -2,7 +2,7 @@ "use strict" /** - * g2.ext (c) 2015-18 Stefan Goessner + * g2.ext (c) 2015-20 Stefan Goessner * @author Stefan Goessner * @license MIT License * @requires g2.core.js @@ -10,163 +10,635 @@ * @description Additional methods for g2. * @returns {g2} */ -var g2 = g2 || { prototype:{} }; // for jsdoc only ... + +var g2 = g2 || { prototype: {} }; // for jsdoc only ... // constants for element selection / editing g2.NONE = 0x0; g2.OVER = 0x1; g2.DRAG = 0x2; g2.EDIT = 0x4; -// prototypes for extending argument objects -g2.prototype.lin.prototype = { +/** + * Extended style values. + * Not really meant to get overwritten. But if you actually want, proceed.
    + * These styles can be referenced using the comfortable '@' syntax. + * @namespace + * @property {object} symbol `g2` symbol namespace. + * @property {object} [symbol.tick] Predefined symbol: a little tick + * @property {object} [symbol.dot] Predefined symbol: a little dot + * @property {object} [symbol.sqr] Predefined symbol: a little square + * @property {string} [symbol.nodcolor=#333] node color. + * @property {string} [symbol.nodfill=#dedede] node fill color. + * @property {string} [symbol.nodfill2=#aeaeae] alternate node fill color, somewhat darker. + * @property {string} [symbol.linkcolor=#666] link color. + * @property {string} [symbol.linkfill=rgba(225,225,225,0.75)] link fill color, semi-transparent. + * @property {string} [symbol.dimcolor=darkslategray] dimension color. + * @property {array} [symbol.solid=[]] solid line style. + * @property {array} [symbol.dash=[15,10]] dashed line style. + * @property {array} [symbol.dot=[4,4]] dotted line style. + * @property {array} [symbol.dashdot=[25,6.5,2,6.5]] dashdotted line style. + * @property {number} [symbol.labelOffset=5] default label offset distance. + * @property {number} [symbol.labelSignificantDigits=3] default label's significant digits after numbering point. + */ +g2.symbol = g2.symbol || {}; +g2.symbol.tick = g2().p().m({ x: 0, y: -2 }).l({ x: 0, y: 2 }).stroke({ lc: "round", lwnosc: true }); +g2.symbol.dot = g2().cir({ x: 0, y: 0, r: 2, ls: "transparent" }); +g2.symbol.sqr = g2().rec({ x: -1.5, y: -1.5, b: 3, h: 3, ls: "transparent" }); + +g2.symbol.nodcolor = "#333"; +g2.symbol.nodfill = "#dedede"; +g2.symbol.nodfill2 = "#aeaeae"; +g2.symbol.linkcolor = "#666"; +g2.symbol.linkfill = "rgba(225,225,225,0.75)"; +g2.symbol.dimcolor = "darkslategray"; +g2.symbol.solid = []; +g2.symbol.dash = [15, 10]; +g2.symbol.dot = [4, 4]; +g2.symbol.dashdot = [25, 6.5, 2, 6.5]; +g2.symbol.labelSignificantDigits = 3; // 0.1234 => 0.123, 0.01234 => 0.0123, 1.234 => 1.23, 12.34 => 12.3, 123.4 => 123, 1234 => 1234 + +/** +* Flatten object properties (evaluate getters) +*/ +g2.flatten = function (obj) { + const args = Object.create(null); // important ! + for (let p in obj) + if (typeof obj[p] !== 'function') + args[p] = obj[p]; + return args; +} +/* +g2.strip = function(obj,prop) { + const clone = Object.create(Object.getPrototypeOf(obj),Object.getOwnPropertyDescriptors(obj)); + Object.defineProperty(clone, prop, { get:undefined, enumerable:true, configurable:true, writabel:false }); + return clone; +} +*/ +g2.pointIfc = { + // p vector notation ! ... helps to avoid object destruction + get p() { return { x: this.x, y: this.y }; }, // visible if 'p' is *not* explicite given. + get x() { return Object.getOwnPropertyDescriptor(this, 'p') ? this.p.x : 0; }, + get y() { return Object.getOwnPropertyDescriptor(this, 'p') ? this.p.y : 0; }, + set x(q) { if (Object.getOwnPropertyDescriptor(this, 'p')) this.p.x = q; }, + set y(q) { if (Object.getOwnPropertyDescriptor(this, 'p')) this.p.y = q; }, +} + +g2.labelIfc = { + getLabelOffset() { const off = this.label.off !== undefined ? +this.label.off : 1; return off + Math.sign(off) * (this.lw || 2) / 2; }, + getLabelString() { + let s = typeof this.label === 'object' ? this.label.str : typeof this.label === 'string' ? this.label : '?'; + if (s && s[0] === "@" && this[s.substr(1)]) { + s = s.substr(1); + let val = this[s]; + val = Number.isInteger(val) ? val + : Number(val).toFixed(Math.max(g2.symbol.labelSignificantDigits - Math.log10(val), 0)); + + s = `${val}${s === 'angle' ? "°" : ""}`; + } + return s; + }, + drawLabel(g) { + const lbl = this.label; + const font = lbl.font || g2.defaultStyle.font; + const h = parseInt(font); // font height (px assumed !) + const str = this.getLabelString(); + const rx = (str.length || 1) * 0.75 * h / 2, ry = 1.25 * h / 2; // ellipse semi-axes length + const pos = this.pointAt(lbl.loc || this.lbloc || 'se'); + const off = this.getLabelOffset(); + const p = { + x: pos.x + pos.nx * (off + Math.sign(off) * rx), + y: pos.y + pos.ny * (off + Math.sign(off) * ry) + }; + + if (lbl.border) g.ell({ x: p.x, y: p.y, rx, ry, ls: lbl.fs || 'black', fs: lbl.fs2 || '#ffc' }); + g.txt({ + str, x: p.x, y: p.y, + thal: "center", tval: "middle", + fs: lbl.fs || 'black', font: lbl.font + }); + return g; + } +} + +g2.markIfc = { + markAt(loc) { + const p = this.pointAt(loc); + const w = Math.atan2(p.ny, p.nx) + Math.PI / 2; + return { + grp: this.getMarkSymbol(), x: p.x, y: p.y, w: w, scl: this.lw || 1, + ls: this.ls || '#000', fs: this.fs || this.ls || '#000' + } + }, + getMarkSymbol() { + // Use tick as default + const mrk = this.mark + if (typeof mrk === 'number' || !mrk) return g2.symbol.tick; + if (typeof mrk.symbol === 'object') return mrk.symbol; + if (typeof mrk.symbol === 'string') return g2.symbol[mrk.symbol] + }, + // loop is for elements that close, e.g. rec or cir => loc at 0 === loc at 1 + drawMark(g, closed = false) { + let loc; + if (Array.isArray(this.mark)) { + loc = this.mark; + } + else { + const count = typeof this.mark === 'object' ? this.mark.count : this.mark; + loc = count ? + Array.from(Array(count)).map((_, i) => i / (count - !closed)) : + this.mark.loc; + } + for (let l of loc) { + g.use(this.markAt(l)); + } + return g; + } +} + +g2.prototype.cir.prototype = g2.mix(g2.pointIfc, g2.labelIfc, g2.markIfc, { + w: 0, // default start angle (used for dash-dot orgin and editing) + lbloc: 'c', + get isSolid() { return this.fs && this.fs !== 'transparent' }, + get len() { return 2 * Math.PI * this.r; }, + get lsh() { return this.state & g2.OVER; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false }, + get g2() { // dynamically switch existence of method via getter ... cool ! + const e = g2(); // hand object stripped from `g2` + this.label && e.ins((g) => this.drawLabel(g)); + this.mark && e.ins((g) => this.drawMark(g, true)); + return () => g2().cir(g2.flatten(this)).ins(e); // avoiding infinite recursion ! + }, + pointAt(loc) { + const Q = Math.SQRT2 / 2; + const LOC = { c: [0, 0], e: [1, 0], ne: [Q, Q], n: [0, 1], nw: [-Q, Q], w: [-1, 0], sw: [-Q, -Q], s: [0, -1], se: [Q, -Q] }; + const q = (loc + 0 === loc) ? [Math.cos(loc * 2 * Math.PI), Math.sin(loc * 2 * Math.PI)] + : (LOC[loc || "c"] || [0, 0]); + return { + x: this.x + q[0] * this.r, + y: this.y + q[1] * this.r, + nx: q[0], + ny: q[1] + }; + }, + hit({ x, y, eps }) { + return this.isSolid ? g2.isPntInCir({ x, y }, this, eps) + : g2.isPntOnCir({ x, y }, this, eps); + }, + drag({ dx, dy }) { this.x += dx; this.y += dy }, +}); + +g2.prototype.lin.prototype = g2.mix(g2.labelIfc, g2.markIfc, { + // p1 vector notation ! + get p1() { return { x1: this.x1, y1: this.y1 }; }, // relevant if 'p1' is *not* explicite given. + get x1() { return Object.getOwnPropertyDescriptor(this, 'p1') ? this.p1.x : 0; }, + get y1() { return Object.getOwnPropertyDescriptor(this, 'p1') ? this.p1.y : 0; }, + set x1(q) { if (Object.getOwnPropertyDescriptor(this, 'p1')) this.p1.x = q; }, + set y1(q) { if (Object.getOwnPropertyDescriptor(this, 'p1')) this.p1.y = q; }, + // p2 vector notation ! + get p2() { return { x2: this.x2, y2: this.y2 }; }, // relevant if 'p2' is *not* explicite given. + get x2() { return Object.getOwnPropertyDescriptor(this, 'p2') ? this.p2.x : 0; }, + get y2() { return Object.getOwnPropertyDescriptor(this, 'p2') ? this.p2.y : 0; }, + set x2(q) { if (Object.getOwnPropertyDescriptor(this, 'p2')) this.p2.x = q; }, + set y2(q) { if (Object.getOwnPropertyDescriptor(this, 'p2')) this.p2.y = q; }, + isSolid: false, get len() { return Math.hypot(this.x2 - this.x1, this.y2 - this.y1); }, - get sh() { return this.state & g2.OVER ? [0,0,5,"black"] : false }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false }, + get g2() { // dynamically switch existence of method via getter ... ! + const e = g2(); + this.label && e.ins(e => this.drawLabel(e)); + this.mark && e.ins(e => this.drawMark(e)); + return () => g2().lin(g2.flatten(this)).ins(e); + }, + pointAt(loc) { - let t = loc==="beg" ? 0 - : loc==="end" ? 1 - : (loc+0 === loc) ? loc // numerical arg .. - : 0.5, // 'mid' .. + let t = loc === "beg" ? 0 + : loc === "end" ? 1 + : (loc + 0 === loc) ? loc // numerical arg .. + : 0.5, // 'mid' .. dx = this.x2 - this.x1, dy = this.y2 - this.y1, - len = Math.hypot(dx,dy); + len = Math.hypot(dx, dy); return { - x: this.x1 + dx*t, - y: this.y1 + dy*t, - dx: len ? dx/len : 1, - dy: len ? dy/len : 0 - }; + x: this.x1 + dx * t, + y: this.y1 + dy * t, + nx: len ? dy / len : 0, + ny: len ? -dx / len : -1 + }; + }, + hit({ x, y, eps }) { + return g2.isPntOnLin({ x, y }, { x: this.x1, y: this.y1 }, { x: this.x2, y: this.y2 }, eps); }, - hitContour({x,y,eps}) { return g2.isPntOnLin({x,y},{x:this.x1,y:this.y1},{x:this.x2,y:this.y2},eps) }, - drag({dx,dy}) { + drag({ dx, dy }) { this.x1 += dx; this.x2 += dx; this.y1 += dy; this.y2 += dy; - }, - handles(grp) { - grp.handle({x:this.x1,y:this.y1,_update:({dx,dy})=>{this.x1+=dx;this.y1+=dy}}) - .handle({x:this.x2,y:this.y2,_update:({dx,dy})=>{this.x2+=dx;this.y2+=dy}}); } -}; +}); -g2.prototype.rec.prototype = { - _dir: { c:[0,0],e:[1,0],ne:[1,1],n:[0,1],nw:[-1,1], - w:[-1,0],sw:[-1,-1],s:[0,-1],se:[1,-1] }, - get len() { return 2*(this.b+this.h); }, +g2.prototype.rec.prototype = g2.mix(g2.pointIfc, g2.labelIfc, g2.markIfc, { + get len() { return 2 * (this.b + this.h); }, get isSolid() { return this.fs && this.fs !== 'transparent' }, - get len() { return 2*Math.PI*this.r; }, get lsh() { return this.state & g2.OVER; }, - get sh() { return this.state & g2.OVER ? [0,0,5,"black"] : false; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false; }, + get g2() { // dynamically switch existence of method via getter ... ! + const e = g2(); + this.label && e.ins(e => this.drawLabel(e)); + this.mark && e.ins(e => this.drawMark(e, true)); + return () => g2().rec(g2.flatten(this)).ins(e); + }, + lbloc: 'c', pointAt(loc) { - const q = this._dir[loc || "c"] || this._dir['c'], nx = q[0], ny = q[1]; + const locAt = (loc) => { + const o = { c: [0, 0], e: [1, 0], ne: [0.95, 0.95], n: [0, 1], nw: [-0.95, 0.95], w: [-1, 0], sw: [-0.95, -0.95], s: [0, -1], se: [0.95, -0.95] }; + + if (o[loc]) return o[loc]; + + const w = 2 * Math.PI * loc + pi / 4; + if (loc <= 0.25) return [1 / Math.tan(w), 1]; + if (loc <= 0.50) return [-1, -Math.tan(w)]; + if (loc <= 0.75) return [- 1 / Math.tan(w), -1]; + if (loc <= 1.00) return [1, Math.tan(w)]; + } + const q = locAt(loc); return { - x: this.x + (1 + nx)*this.b/2, - y: this.y + (1 + ny)*this.h/2, - dx: -ny, - dy: nx + x: this.x + (1 + q[0]) * this.b / 2, + y: this.y + (1 + q[1]) * this.h / 2, + nx: 1 - Math.abs(q[0]) < 0.01 ? q[0] : 0, + ny: 1 - Math.abs(q[1]) < 0.01 ? q[1] : 0 }; }, - hitContour({x,y,eps}) { return g2.isPntOnBox({x,y},{x:this.x+this.b/2,y:this.y+this.h/2,b:this.b/2,h:this.h/2},eps) }, - hitInner({x,y,eps}) {return g2.isPntInBox({x,y},{x:this.x+this.b/2,y:this.y+this.h/2,b:this.b/2,h:this.h/2},eps) }, - drag({dx,dy}) { this.x += dx; this.y += dy } -}; + hit({ x, y, eps }) { + return this.isSolid ? g2.isPntInBox({ x, y }, { x: this.x + this.b / 2, y: this.y + this.h / 2, b: this.b / 2, h: this.h / 2 }, eps) + : g2.isPntOnBox({ x, y }, { x: this.x + this.b / 2, y: this.y + this.h / 2, b: this.b / 2, h: this.h / 2 }, eps); + }, + drag({ dx, dy }) { this.x += dx; this.y += dy } +}); -g2.prototype.cir.prototype = { - w: 0, // default start angle (used for dash-dot orgin and editing) - _dir: { c:[0,0],e:[1,0],ne:[Math.SQRT2/2,Math.SQRT2/2],n:[0,1],nw:[-Math.SQRT2/2,Math.SQRT2/2], - w:[-1,0],sw:[-Math.SQRT2/2,-Math.SQRT2/2],s:[0,-1],se:[Math.SQRT2/2,-Math.SQRT2/2] }, - get isSolid() { return this.fs && this.fs !== 'transparent' }, - get len() { return 2*Math.PI*this.r; }, - get lsh() { return this.state & g2.OVER; }, - get sh() { return this.state & g2.OVER ? [0,0,5,"black"] : false }, +g2.prototype.arc.prototype = g2.mix(g2.pointIfc, g2.labelIfc, g2.markIfc, { + get len() { return Math.abs(this.r * this.dw); }, + isSolid: false, + get angle() { return this.dw / Math.PI * 180; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false }, + get g2() { // dynamically switch existence of method via getter ... ! + const e = g2(); + this.label && e.ins(e => this.drawLabel(e)); + this.mark && e.ins(e => this.drawMark(e)); + return () => g2().arc(g2.flatten(this)).ins(e); + }, + lbloc: 'mid', pointAt(loc) { - let q = (loc+0 === loc) ? [Math.cos(loc*2*Math.PI),Math.sin(loc*2*Math.PI)] - : (this._dir[loc || "c"] || [0,0]), - nx = q[0], ny = q[1]; + let t = loc === "beg" ? 0 + : loc === "end" ? 1 + : loc === "mid" ? 0.5 + : loc + 0 === loc ? loc + : 0.5, + ang = (this.w || 0) + t * (this.dw || Math.PI * 2), cang = Math.cos(ang), sang = Math.sin(ang), r = loc === "c" ? 0 : this.r; return { - x: this.x + nx*this.r, - y: this.y + ny*this.r, - dx: -ny, - dy: nx }; - }, - hitContour({x,y,eps}) { return g2.isPntOnCir({x,y},this,eps) }, - hitInner({x,y,eps}) {return g2.isPntInCir({x,y},this,eps) }, - drag({dx,dy}) { this.x += dx; this.y += dy }, - handles(grp) { - const p0 = { - x:this.x, y:this.y, - _update:({dx,dy})=>{this.x+=dx;this.y+=dy;p1.x+=dx;p1.y+=dy;} - }; - const p1 = { - x:this.x+this.r*Math.cos(this.w||0), - y:this.y+this.r*Math.sin(this.w||0), - _info:()=>`r:${this.r.toFixed(1)}
    w:${(this.w/Math.PI*180).toFixed(1)}°`, - _update:({x,y}) => { - this.r = Math.hypot(y-this.y,x-this.x); - this.w = Math.atan2(y-this.y,x-this.x); - } + x: this.x + r * cang, + y: this.y + r * sang, + nx: cang, + ny: sang }; - grp.lin({x1:()=>this.x,y1:()=>this.y,x2:()=>p1.x,y2:()=>p1.y,ld:[4,3],ls:'#666'}) - .handle(p0) - .handle(p1); + }, + hit({ x, y, eps }) { return g2.isPntOnArc({ x, y }, this, eps) }, + drag({ dx, dy }) { this.x += dx; this.y += dy; }, +}); + +/** +* Draw interactive handle. +* @method +* @returns {object} g2 +* @param {object} - handle object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().hdl({x:100,y:80}) // Draw handle. +*/ +g2.prototype.hdl = function (args) { return this.addCommand({ c: 'hdl', a: args }); } +g2.prototype.hdl.prototype = g2.mix(g2.prototype.cir.prototype, { + r: 5, + isSolid: true, + draggable: true, + lbloc: 'se', + get lsh() { return this.state & g2.OVER; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false }, + g2() { + const { x, y, r, b = 4, shape = 'cir', ls = 'black', fs = '#ccc', sh } = this; + return shape === 'cir' ? g2().cir({ x, y, r, ls, fs, sh }).ins((g) => this.label && this.drawLabel(g)) + : g2().rec({ x: x - b, y: y - b, b: 2 * b, h: 2 * b, ls, fs, sh }).ins((g) => this.label && this.drawLabel(g)); } -}; +}); + +/** +* Node symbol. +* @constructor +* @param {object} - symbol arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().nod({x:10,y:10}) +*/ + +g2.prototype.nod = function (args = {}) { return this.addCommand({ c: 'nod', a: args }); } +g2.prototype.nod.prototype = g2.mix(g2.prototype.cir.prototype, { + r: 5, + ls: '@nodcolor', + fs: g2.symbol.nodfill, + isSolid: true, + lbloc: 'se', + g2() { // in contrast to `g2.prototype.cir.prototype`, `g2()` is called always ! + return g2() + .cir({ ...g2.flatten(this), r: this.r * (this.scl !== undefined ? this.scl : 1) }) + .ins((g) => this.label && this.drawLabel(g)) + } +}); + +/** + * Double nod symbol + * @constructor + * @returns {object} g2 + * @param {object} - symbol arguments object. + * @property {number} x - x-value center. + * @property {number} y - y-value center. + * @example + * g2().dblnod({x:10,y:10}) +*/ +g2.prototype.dblnod = function ({ x = 0, y = 0 }) { return this.addCommand({ c: 'dblnod', a: arguments[0] }); } +g2.prototype.dblnod.prototype = g2.mix(g2.prototype.cir.prototype, { + get r() { return 6; }, + get isSolid() { return true; }, + g2() { + return g2() + .beg({ x: this.x, y: this.y }) + .cir({ r: 6, ls: '@nodcolor', fs: '@nodfill', sh: this.sh }) + .cir({ r: 3, ls: '@nodcolor', fs: '@nodfill2' }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) + +/** +* Pole symbol. +* @constructor +* @returns {object} g2 +* @param {object} - symbol arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().pol({x:10,y:10}) +*/ +g2.prototype.pol = function (args = {}) { return this.addCommand({ c: 'pol', a: args }); } +g2.prototype.pol.prototype = g2.mix(g2.prototype.nod.prototype, { + g2() { + return g2() + .beg(g2.flatten(this)) + .cir({ r: 6, fs: g2.symbol.nodfill }) + .cir({ r: 2.5, fs: '@ls', ls: 'transparent' }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) -g2.prototype.arc.prototype = { - get len() { return Math.abs(this.r*this.dw); }, - get angle() { return this.dw/Math.PI*180; }, +/** +* Ground symbol. +* @constructor +* @param {object} - arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().gnd({x:10,y:10}) +*/ +g2.prototype.gnd = function (args = {}) { return this.addCommand({ c: 'gnd', a: args }); } +g2.prototype.gnd.prototype = g2.mix(g2.prototype.nod.prototype, { + g2() { + return g2() + .beg(g2.flatten(this)) + .cir({ x: 0, y: 0, r: 6 }) + .p() + .m({ x: 0, y: 6 }) + .a({ dw: Math.PI / 2, x: -6, y: 0 }) + .l({ x: 6, y: 0 }) + .a({ dw: -Math.PI / 2, x: 0, y: -6 }) + .z() + .fill({ fs: g2.symbol.nodcolor }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) + +g2.prototype.nodfix = function (args = {}) { return this.addCommand({ c: 'nodfix', a: args }); } +g2.prototype.nodfix.prototype = g2.mix(g2.prototype.nod.prototype, { + g2() { + return g2() + .beg(g2.flatten(this)) + .p() + .m({ x: -8, y: -12 }) + .l({ x: 0, y: 0 }) + .l({ x: 8, y: -12 }) + .drw({ fs: g2.symbol.nodfill2 }) + .cir({ x: 0, y: 0, r: this.r }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) +/** +* @method +* @returns {object} g2 +* @param {object} - symbol arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @example +* g2().view({cartesian:true}) + * .nodflt({x:10,y:10}) +*/ +g2.prototype.nodflt = function (args = {}) { return this.addCommand({ c: 'nodflt', a: args }); } +g2.prototype.nodflt.prototype = g2.mix(g2.prototype.nod.prototype, { + g2() { + return g2() + .beg(g2.flatten(this)) + .p() + .m({ x: -8, y: -12 }) + .l({ x: 0, y: 0 }) + .l({ x: 8, y: -12 }) + .drw({ ls: g2.symbol.nodcolor, fs: g2.symbol.nodfill2 }) + .cir({ x: 0, y: 0, r: this.r, ls: g2.symbol.nodcolor, fs: g2.symbol.nodfill }) + .lin({ x1: -9, y1: -19, x2: 9, y2: -19, ls: g2.symbol.nodfill2, lw: 5 }) + .lin({ x1: -9, y1: -15.5, x2: 9, y2: -15.5, ls: g2.symbol.nodcolor, lw: 2 }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}) + +/** +* Draw vector arrow. +* @method +* @returns {object} g2 +* @param {object} - vector arguments object. +* @property {number} x1 - start x coordinate. +* @property {number} y1 - start y coordinate. +* @property {number} x2 - end x coordinate. +* @property {number} y2 - end y coordinate. +* @example +* g2().vec({x1:50,y1:20,x2:250,y2:120}) +*/ +g2.prototype.vec = function vec(args) { return this.addCommand({ c: 'vec', a: args }); } +g2.prototype.vec.prototype = g2.mix(g2.prototype.lin.prototype, { + g2() { + const { x1, y1, x2, y2, lw = 1, ls = '#000', ld = [], fs = ls || '#000', lc = 'round', lj = 'round', } = this; + const dx = x2 - x1, dy = y2 - y1, r = Math.hypot(dx, dy); + const b = 3 * (1 + lw) > r ? r / 3 : (1 + lw); + const arrowHead = () => g2().p().m({ x: 0, y: 0 }).l({ x: -5 * b, y: b }).a({ dw: -Math.PI / 3, x: -5 * b, y: -b }).z().drw({ ls, fs, lc, lj }); + return g2() + .beg({ x: x1, y: y1, w: Math.atan2(dy, dx), lc, lj }) + .p().m({ x: 0, y: 0 }) + .l({ x: r - 3 * b, y: 0 }) + .stroke({ ls, lw, ld }) + .use({ grp: arrowHead, x: r, y: 0 }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}); + +/** +* Arc as Vector +* @method +* @returns {object} g2 +* @param {object} - angular dimension arguments. +* @property {number} x - start x coordinate. +* @property {number} y - start y coordinate. +* @property {number} r - radius +* @property {number} [w=0] - start angle (in radian). +* @property {number} [dw=Math.PI/2] - angular range in radian. In case of positive values it is running counterclockwise with + * right handed (cartesian) coordinate system. +* @example +* g2().avec({x:100,y:70,r:50,w:pi/3,dw:4*pi/3}) +*/ +g2.prototype.avec = function avec(args) { return this.addCommand({ c: 'avec', a: args }); } +g2.prototype.avec.prototype = g2.mix(g2.prototype.arc.prototype, { + g2() { + const { x, y, r, w, dw = 0, lw = 1, lc = 'round', lj = 'round', ls, fs = ls || "#000", label } = this; + const b = 3 * (1 + lw) > r ? r / 3 : (1 + lw), bw = 5 * b / r; + const arrowHead = () => g2().p().m({ x: 0, y: 2 * b }).l({ x: 0, y: -2 * b }).m({ x: 0, y: 0 }).l({ x: -5 * b, y: b }) + .a({ dw: -Math.PI / 3, x: -5 * b, y: -b }).z().drw({ ls, fs }); + + return g2() + .beg({ x, y, w, ls, lw, lc, lj }) + .arc({ r, w: 0, dw }) + .use({ + grp: arrowHead, x: r * Math.cos(dw), y: r * Math.sin(dw), + w: (dw >= 0 ? dw + Math.PI / 2 - bw / 2 : dw - Math.PI / 2 + bw / 2) + }) + .end() + .ins((g) => label && this.drawLabel(g)); + } +}); + +/** +* Linear Dimension +* @method +* @returns {object} g2 +* @param {object} - dimension arguments object. +* @property {number} x1 - start x coordinate. +* @property {number} y1 - start y coordinate. +* @property {number} x2 - end x coordinate. +* @property {number} y2 - end y coordinate. +* @property {number} off - offset. +* @property {boolean} [inside=true] - draw dimension arrows between or outside of ticks. +* @example +* g2().dim({x1:60,y1:40,x2:190,y2:120}) +*/ +g2.prototype.dim = function dim(args) { return this.addCommand({ c: 'dim', a: args }); } +g2.prototype.dim.prototype = g2.mix(g2.prototype.lin.prototype, { pointAt(loc) { - let t = loc==="beg" ? 0 - : loc==="end" ? 1 - : loc==="mid" ? 0.5 - : loc+0 === loc ? loc - : 0.5, - ang = this.w+t*this.dw, cang = Math.cos(ang), sang = Math.sin(ang), r = loc === "c" ? 0 : this.r; - return { - x: this.x + r*cang, - y: this.y + r*sang, - dx: -sang, - dy: cang - }; + const pnt = g2.prototype.lin.prototype.pointAt.call(this, loc); + if (this.off) { + pnt.x += this.off * pnt.nx; + pnt.y += this.off * pnt.ny; + } + return pnt; }, - isSolid: false, - get sh() { return this.state & g2.OVER ? [0,0,5,"black"] : false }, - hitContour({x,y,eps}) { return g2.isPntOnArc({x,y},this,eps) }, - drag({dx,dy}) { this.x += dx; this.y += dy; }, - handles(grp) { - const p0 = { - x:this.x, y:this.y, - _update:({dx,dy})=>{this.x+=dx;this.y+=dy;p1.x+=dx;p1.y+=dy;p2.x+=dx;p2.y+=dy;} - }, - p1 = { - x:this.x+this.r*Math.cos(this.w), - y:this.y+this.r*Math.sin(this.w), - _info:()=>`r:${this.r.toFixed(1)}
    w:${(this.w/Math.PI*180).toFixed(1)}°`, - _update:({x,y})=>{ - this.r = Math.hypot(y-this.y,x-this.x); - this.w = Math.atan2(y-this.y,x-this.x); - p2.x = this.x+this.r*Math.cos(this.w+this.dw); - p2.y = this.y+this.r*Math.sin(this.w+this.dw); } - }, - dw = this.dw, - p2 = { - x:this.x+this.r*Math.cos(this.w+this.dw), - y:this.y+this.r*Math.sin(this.w+this.dw), - _info:()=>`dw:${(this.dw/Math.PI*180).toFixed(1)}°`, - _update:({x,y})=>{ // bug with negative 'this.w' ... - let lam = g2.toArc(g2.toPi2(Math.atan2(y-this.y,x-this.x)),g2.toPi2(this.w),dw); - this.dw = lam*dw; - } - }; - if (this.w === undefined) this.w = 0; - grp.lin({x1:()=>this.x,y1:()=>this.y,x2:()=>p1.x,y2:()=>p1.y,ld:[4,3],ls:'#666'}) - .lin({x1:()=>this.x,y1:()=>this.y,x2:()=>p2.x,y2:()=>p2.y,ld:[4,3],ls:'#666'}) - .handle(p0) - .handle(p1) - .handle(p2); + g2() { + const { x1, y1, x2, y2, lw = 1, lc = 'round', lj = 'round', off = 0, inside = true, ls, fs = ls || "#000", label } = this; + const dx = x2 - x1, dy = y2 - y1, r = Math.hypot(dx, dy); + const b = 3 * (1 + lw) > r ? r / 3 : (1 + lw); + const arrowHead = () => g2().p().m({ x: 0, y: 2 * b }).l({ x: 0, y: -2 * b }).m({ x: 0, y: 0 }).l({ x: -5 * b, y: b }) + .a({ dw: -Math.PI / 3, x: -5 * b, y: -b }).z().drw({ ls, fs }); + return g2() + .beg({ x: x1 + off / r * dy, y: y1 - off / r * dx, w: Math.atan2(dy, dx), ls, fs, lw, lc, lj }) + .lin({ x1: (inside ? 4 * b : 0), y1: 0, x2: (inside ? r - 4 * b : r), y2: 0 }) + .use({ grp: arrowHead, x: r, y: 0, w: (inside ? 0 : Math.PI) }) + .use({ grp: arrowHead, x: 0, y: 0, w: (inside ? Math.PI : 0) }) + .lin({ x1: 0, y1: off, x2: 0, y2: 0 }) + .lin({ x1: r, y1: off, x2: r, y2: 0 }) + .end() + .ins((g) => label && this.drawLabel(g)); } -}; +}); + +/** +* Angular dimension +* @method +* @returns {object} g2 +* @param {object} - angular dimension arguments. +* @property {number} x - start x coordinate. +* @property {number} y - start y coordinate. +* @property {number} r - radius +* @property {number} [w=0] - start angle (in radian). +* @property {number} [dw=Math.PI/2] - angular range in radian. In case of positive values it is running counterclockwise with + * right handed (cartesian) coordinate system. +* @property {boolean} [outside=false] - draw dimension arrows outside of ticks. +* @depricated {boolean} [inside] - draw dimension arrows between ticks. +* @example +* g2().adim({x:100,y:70,r:50,w:pi/3,dw:4*pi/3}) +*/ +g2.prototype.adim = function adim(args) { return this.addCommand({ c: 'adim', a: args }); } +g2.prototype.adim.prototype = g2.mix(g2.prototype.arc.prototype, { + g2() { + const { x, y, r, w, dw, lw = 1, lc = 'round', lj = 'round', ls, fs = ls || "#000", label } = this; + const b = 3 * (1 + lw) > r ? r / 3 : (1 + lw), bw = 5 * b / r; + const arrowHead = () => g2().p().m({ x: 0, y: 2 * b }).l({ x: 0, y: -2 * b }).m({ x: 0, y: 0 }).l({ x: -5 * b, y: b }) + .a({ dw: -Math.PI / 3, x: -5 * b, y: -b }).z().drw({ ls, fs }); + + const outside = (this.inside !== undefined && this.outside === undefined) ? !this.inside : !!this.outside; // still support depricated property ! + + return g2() + .beg({ x, y, w, ls, lw, lc, lj }) + .arc({ r, w: 0, dw }) + .use({ grp: arrowHead, x: r, y: 0, w: (!outside && dw > 0 || outside && dw < 0 ? -Math.PI / 2 + bw / 2 : Math.PI / 2 - bw / 2) }) + .use({ grp: arrowHead, x: r * Math.cos(dw), y: r * Math.sin(dw), w: (!outside && dw > 0 || outside && dw < 0 ? dw + Math.PI / 2 - bw / 2 : dw - Math.PI / 2 + bw / 2) }) + .end() + .ins((g) => label && this.drawLabel(g)); + } +}); -g2.prototype.ply.prototype = { +/** +* Origin symbol +* @constructor +* @returns {object} g2 +* @param {object} - symbol arguments object. +* @property {number} x - x-value center. +* @property {number} y - y-value center. +* @property {number} w - angle in radians. +* @example +* g2().view({cartesian:true}) + * .origin({x:10,y:10}) +*/ +g2.prototype.origin = function (args = {}) { return this.addCommand({ c: 'origin', a: args }); } +g2.prototype.origin.prototype = g2.mix(g2.prototype.nod.prototype, { + lbloc: 'sw', + g2() { + const { x, y, w, ls = '#000', lw = 1 } = this; + return g2() + .beg({ x, y, w, ls }) + .vec({ x1: 0, y1: 0, x2: 40, y2: 0, lw, fs: '#ccc' }) + .vec({ x1: 0, y1: 0, x2: 0, y2: 40, lw, fs: '#ccc' }) + .cir({ x: 0, y: 0, r: lw + 1, fs: '#ccc' }) + .end() + .ins((g) => this.label && this.drawLabel(g)); + } +}); + +g2.prototype.ply.prototype = g2.mix(g2.labelIfc, g2.markIfc, { get isSolid() { return this.closed && this.fs && this.fs !== 'transparent'; }, - get sh() { return this.state & g2.OVER ? [0,0,5,"black"] : false; }, + get sh() { return this.state & g2.OVER ? [0, 0, 5, "black"] : false; }, // get len() { // let len_itr = 0; // let last_pt = {x:0,y:0}; @@ -177,29 +649,27 @@ g2.prototype.ply.prototype = { // return len_itr; // }, pointAt(loc) { - const t = loc==="beg" ? 0 - : loc==="end" ? 1 - : (loc+0 === loc) ? loc // numerical arg .. - : 0.5, // 'mid' .. + const t = loc === "beg" ? 0 + : loc === "end" ? 1 + : (loc + 0 === loc) ? loc // numerical arg .. + : 0.5, // 'mid' .. pitr = g2.pntItrOf(this.pts), pts = [], len = []; + for (let itr = 0; itr < pitr.len; itr++) { - const next = pitr(itr+1) ? pitr(itr+1) : pitr(0); - if ((itr === pitr.len-1 && this.closed) || itr < pitr.len-1) { - pts.push(pitr(itr)); - len.push(Math.hypot( - next.x-pitr(itr).x, - next.y-pitr(itr).y) - ); - } + const next = pitr((itr + 1) % pitr.len); + pts.push(pitr(itr)); + len.push(Math.hypot( + next.x - pitr(itr).x, + next.y - pitr(itr).y)); } - const {t2, x, y, dx, dy} = (() => { - const target = t * len.reduce((a,b) => a+b); - let tmp = 0; - for(let itr = 0; itr < pts.length; itr++) { + this.closed || len.pop(); + const { t2, x, y, dx, dy } = (() => { + const target = t * len.reduce((a, b) => a + b); + for (let itr = 0, tmp = 0; itr < pts.length; itr++) { tmp += len[itr]; - const next = pitr(itr+1).x ? pitr(itr+1) : pitr(0); + const next = pitr(itr + 1).x ? pitr(itr + 1) : pitr(0); if (tmp >= target) { return { t2: 1 - (tmp - target) / len[itr], @@ -211,60 +681,74 @@ g2.prototype.ply.prototype = { } } })(); - const len2 = Math.hypot(dx,dy); + const len2 = Math.hypot(dx, dy); return { - x: x + dx*t2, - y: y + dy*t2, - dx: len2 ? dx/len2 : 1, - dy: len2 ? dy/len2 : 0 + x: (this.x || 0) + x + dx * t2, + y: (this.y || 0) + y + dy * t2, + nx: len2 ? dy / len2 : 1, + ny: len2 ? dx / len2 : 0, }; }, - x: 0, y: 0, - hitContour({x,y,eps}) { let p={x:x-this.x,y:y-this.y}; return g2.isPntOnPly(p,this,eps) }, // translational only .. at current .. ! - hitInner({x,y,eps}) { let p={x:x-this.x,y:y-this.y}; return g2.isPntInPly(p,this,eps) }, // translational only .. at current .. ! - drag({dx,dy}) { this.x += dx; this.y += dy; }, - handles(grp) { - let p, slf=this; - for (let n = this._itr.len, i=0; i this.drawLabel(e)); + this.mark && e.ins(e => this.drawMark(e, this.closed)); + return () => g2().ply(g2.flatten(this)).ins(e); } -} +}); g2.prototype.use.prototype = { - _dir: g2.prototype.cir.prototype._dir, - r: 5, - pointAt: g2.prototype.cir.prototype.pointAt -}; + // p vector notation ! + get p() { return { x: this.x, y: this.y }; }, // relevant if 'p' is *not* explicite given. + get x() { return Object.getOwnPropertyDescriptor(this, 'p') ? this.p.x : 0; }, + get y() { return Object.getOwnPropertyDescriptor(this, 'p') ? this.p.y : 0; }, + set x(q) { if (Object.getOwnPropertyDescriptor(this, 'p')) this.p.x = q; }, + set y(q) { if (Object.getOwnPropertyDescriptor(this, 'p')) this.p.y = q; }, + isSolid: false, + /* + hit(at) { + for (const cmd of this.grp.commands) { + if (cmd.a.hit && cmd.a.hit(at)) + return true; + } + return false; + }, + + pointAt: g2.prototype.cir.prototype.pointAt, + */ +}; // complex macros / add prototypes to argument objects /** - * Draw spline by points. - * Implementing a centripetal Catmull-Rom spline (thus avoiding cusps and self-intersections). - * Using iterator function for getting points from array by index. - * It must return current point object {x,y} or object {done:true}. - * Default iterator expects sequence of x/y-coordinates as a flat array [x,y,...], - * array of [[x,y],...] arrays or array of [{x,y},...] objects. - * @see https://pomax.github.io/bezierinfo - * @see https://de.wikipedia.org/wiki/Kubisch_Hermitescher_Spline - * @method - * @returns {object} g2 - * @param {object} - spline arguments object. - * @property {object[] | number[][] | number[]} pts - array of points. - * @property {bool} [closed=false] - closed spline. - * @example - * g2().spline({pts:[100,50,50,150,150,150,100,50]}) - */ -g2.prototype.spline = function spline({pts,closed,x,y,w}) { +* Draw spline by points. +* Implementing a centripetal Catmull-Rom spline (thus avoiding cusps and self-intersections). +* Using iterator function for getting points from array by index. +* It must return current point object {x,y} or object {done:true}. +* Default iterator expects sequence of x/y-coordinates as a flat array [x,y,...], +* array of [[x,y],...] arrays or array of [{x,y},...] objects. +* @see https://pomax.github.io/bezierinfo +* @see https://de.wikipedia.org/wiki/Kubisch_Hermitescher_Spline +* @method +* @returns {object} g2 +* @param {object} - spline arguments object. +* @property {object[] | number[][] | number[]} pts - array of points. +* @property {bool} [closed=false] - closed spline. +* @example +* g2().spline({pts:[100,50,50,150,150,150,100,50]}) +*/ +g2.prototype.spline = function spline({ pts, closed, x, y, w }) { arguments[0]._itr = g2.pntItrOf(pts); - return this.addCommand({c:'spline',a:arguments[0]}); + return this.addCommand({ c: 'spline', a: arguments[0] }); } -g2.prototype.spline.prototype = g2.mixin({},g2.prototype.ply.prototype,{ - g2: function() { - let {pts,closed,x,y,w,ls,lw,fs,sh} = this, itr = this._itr, gbez; +g2.prototype.spline.prototype = g2.mix(g2.prototype.ply.prototype, { + g2: function () { + let { pts, closed, x, y, w, ls, lw, fs, sh } = this, itr = this._itr, gbez; if (itr) { let b = [], i, n = itr.len, p1, p2, p3, p4, d1, d2, d3, @@ -272,195 +756,45 @@ g2.prototype.spline.prototype = g2.mixin({},g2.prototype.ply.prototype,{ den2, den3, istrf = x || y || w; gbez = g2(); - if (istrf) gbez.beg({x,y,w}); + if (istrf) gbez.beg({ x, y, w }); gbez.p().m(itr(0)); - for (let i=0; i < (closed ? n : n-1); i++) { + for (let i = 0; i < (closed ? n : n - 1); i++) { if (i === 0) { - p1 = closed ? itr(n-1) : {x:2*itr(0).x-itr(1).x, y:2*itr(0).y-itr(1).y}; + p1 = closed ? itr(n - 1) : { x: 2 * itr(0).x - itr(1).x, y: 2 * itr(0).y - itr(1).y }; p2 = itr(0); p3 = itr(1); - p4 = n === 2 ? (closed ? itr(0) : {x:2*itr(1).x-itr(0).x, y:2*itr(1).y-itr(0).y}) : itr(2); - d1 = Math.max(Math.hypot(p2.x-p1.x,p2.y-p1.y),Number.EPSILON); // don't allow .. - d2 = Math.max(Math.hypot(p3.x-p2.x,p3.y-p2.y),Number.EPSILON); // zero point distances .. + p4 = n === 2 ? (closed ? itr(0) : { x: 2 * itr(1).x - itr(0).x, y: 2 * itr(1).y - itr(0).y }) : itr(2); + d1 = Math.max(Math.hypot(p2.x - p1.x, p2.y - p1.y), Number.EPSILON); // don't allow .. + d2 = Math.max(Math.hypot(p3.x - p2.x, p3.y - p2.y), Number.EPSILON); // zero point distances .. } else { p1 = p2; p2 = p3; p3 = p4; - p4 = (i === n-2) ? (closed ? itr(0) : {x:2*itr(n-1).x-itr(n-2).x, y:2*itr(n-1).y-itr(n-2).y}) - : (i === n-1) ? itr(1) - : itr(i+2); + p4 = (i === n - 2) ? (closed ? itr(0) : { x: 2 * itr(n - 1).x - itr(n - 2).x, y: 2 * itr(n - 1).y - itr(n - 2).y }) + : (i === n - 1) ? itr(1) + : itr(i + 2); d1 = d2; d2 = d3; } - d3 = Math.max(Math.hypot(p4.x-p3.x,p4.y-p3.y),Number.EPSILON); - d1d2 = Math.sqrt(d1*d2), d2d3 = Math.sqrt(d2*d3), - scl2 = 2*d1 + 3*d1d2 + d2, - scl3 = 2*d3 + 3*d2d3 + d2, - den2 = 3*(d1 + d1d2), - den3 = 3*(d3 + d2d3); + d3 = Math.max(Math.hypot(p4.x - p3.x, p4.y - p3.y), Number.EPSILON); + d1d2 = Math.sqrt(d1 * d2), d2d3 = Math.sqrt(d2 * d3), + scl2 = 2 * d1 + 3 * d1d2 + d2, + scl3 = 2 * d3 + 3 * d2d3 + d2, + den2 = 3 * (d1 + d1d2), + den3 = 3 * (d3 + d2d3); gbez.c({ x: p3.x, y: p3.y, - x1: (-d2*p1.x + scl2*p2.x + d1*p3.x)/den2, - y1: (-d2*p1.y + scl2*p2.y + d1*p3.y)/den2, - x2: (-d2*p4.x + scl3*p3.x + d3*p2.x)/den3, - y2: (-d2*p4.y + scl3*p3.y + d3*p2.y)/den3 + x1: (-d2 * p1.x + scl2 * p2.x + d1 * p3.x) / den2, + y1: (-d2 * p1.y + scl2 * p2.y + d1 * p3.y) / den2, + x2: (-d2 * p4.x + scl3 * p3.x + d3 * p2.x) / den3, + y2: (-d2 * p4.y + scl3 * p3.y + d3 * p2.y) / den3 }); } - gbez.c(closed ? {x:itr(0).x,y:itr(0).y} : {x:itr(n-1).x,y:itr(n-1).y}) + gbez.c(closed ? { x: itr(0).x, y: itr(0).y } : { x: itr(n - 1).x, y: itr(n - 1).y }) if (closed) gbez.z(); - gbez.drw({ls,lw,fs,sh}); + gbez.drw({ ls, lw, fs, sh }); if (istrf) gbez.end(); } return gbez; } -}) - -/** - * Add label to certain elements. - * Please note that cartesian flag is necessary. - * @method - * @returns {object} g2 - * @param {object} - label arguments object. - * @property {string} str - label text - * @property {number | string} loc - label location depending on referenced element.
    - * 'c': centered, wrt. rec, cir, arc
    - * 'beg','mid', 'end', wrt. lin
    - * 'n', 'ne', 'e', 'se', 's', 'sw', 'w', or 'nw': cardinal directions - * @property {number} off - offset distance [optional]. - * @example - * g2().view({cartesian:true}) - * .cir({x:10,y:10,r:5}) - * .label({str:'hello',loc:'s',off:10}) - */ -g2.prototype.label = function label({str,loc,off,fs,font,fs2}) { - let idx = g2.getCmdIdx(this.commands, (cmd) => { return cmd.a && 'pointAt' in cmd.a}); // find reference index of previous element adding label to ... - if (idx !== undefined) { - arguments[0]['_refelem'] = this.commands[idx]; - this.addCommand({c:'label', a: arguments[0]}); - } - return this; -} -g2.prototype.label.prototype = { - g2() { - let label = g2(); - if (this._refelem) { - let {str,loc,off,fs,font,border,fs2} = this, - p = this._refelem.a.pointAt(loc), // 'loc'ation in coordinates .. - tanlen = p.dx*p.dx || p.dy*p.dy; // tangent length .. (0 || 1) .. ! - let h = parseInt(font||g2.defaultStyle.font), // char height - diag, phi, n; // n .. str length - - if (str[0] === "@" && (s=this._refelem.a[str.substr(1)]) !== undefined) // expect 's' as string convertable to a number ... - str = "" + (Number.isInteger(+s) ? +s : Number(s).toFixed(Math.max(g2.symbol.labelSignificantDigits-Math.log10(s),0))) // use at least 3 significant digits after decimal point. - + (str.substr(1) === "angle" ? "°" : ""); - n = str.length; - if (tanlen > Number.EPSILON) { - diag = Math.hypot(p.dx,n*p.dy); - off = off === undefined ? 1 : off; - p.x += tanlen*p.dy*( off + n*n*0.8*h/2/diag*Math.sign(off)); - p.y += tanlen*p.dx*(-off - h/2/diag*Math.sign(off)); - } - fs = fs||'black'; - if (border) - label.ell({x:p.x,y:p.y,rx:n*0.8*h/2+2,ry:h/2+2,ls:fs||'black',fs:fs2||'#ffc'}); - // .rec({x:p.x-n*0.8*h/2/Math.SQRT2,y:p.y-h/2/Math.SQRT2,b:n*0.8*h/Math.SQRT2,h:h/Math.SQRT2}) - label.txt({ - str, x:p.x,y:p.y, - thal: "center",tval: "middle", - fs: fs||'black',font - }); - } - return label; - } -} - -/** - * Draw marker on line element. - * @method - * @returns {object} g2 - * @param {object} - Marker arguments object. - * @property {object | string} mrk - `g2` object or `name` of mark in `symbol` namespace. - * @property {number | string | number[] | string[]} loc - line location ['beg','end',0.1,0.9,'mid',...].
    - * - * @property {int} [dir=0] - Direction:
    - * -1 : negative tangent direction
    - * 0 : no orientation (rotation)
    - * 1 : positive tangent direction - * @example - * g2().lin({x1:10,y1:10,x2:100,y2:10}) - * .mark({mrk:"tick",loc:0.75,dir:1}) - * - */ -g2.prototype.mark = function mark({mrk,loc,dir,fs,ls}) { - let idx = g2.getCmdIdx(this.commands, (cmd) => { return cmd.a && 'pointAt' in cmd.a}); // find reference index of previous element adding mark to ... - if (idx !== undefined) { - arguments[0]['_refelem'] = this.commands[idx]; - this.addCommand({c:'mark', a: arguments[0]}); - } - return this; -} -g2.prototype.mark.prototype = { - markAt(elem,loc,mrk,dir,ls,fs) { - const p = elem.pointAt(loc), - w = dir < 0 ? Math.atan2(-p.dy,-p.dx) - :(dir > 0 || dir === undefined) ? Math.atan2( p.dy, p.dx) - : 0; - return { - grp:mrk,x:p.x,y:p.y,w:w,scl:elem.lw || 1, - ls:ls || elem.ls || 'black', - fs:fs || ls || elem.ls || 'black' - } - }, - g2() { - let {mrk,loc,dir,fs,ls} = this, - elem = this._refelem.a, - marks = g2(); - if (Array.isArray(loc)) - for (let l of loc) - marks.use(this.markAt(elem,l,mrk,dir,ls,fs)); - else - marks.use(this.markAt(elem,loc,mrk,dir,ls,fs)); - return marks; - } -} - -/** - * Extensed style values. - * Not really meant to get overwritten. But if you actually want, proceed.
    - * Theses styles can be referenced using the comfortable '@' syntax. - * @namespace - * @property {object} symbol `g2` symbol namespace. - * @property {object} [symbol.tick] Predefined symbol: a little tick - * @property {object} [symbol.dot] Predefined symbol: a little dot - * @property {object} [symbol.sqr] Predefined symbol: a little square - * @property {string} [symbol.nodcolor=#333] node color. - * @property {string} [symbol.nodfill=#dedede] node fill color. - * @property {string} [symbol.nodfill2=#aeaeae] alternate node fill color, somewhat darker. - * @property {string} [symbol.linkcolor=#666] link color. - * @property {string} [symbol.linkfill=rgba(225,225,225,0.75)] link fill color, semi-transparent. - * @property {string} [symbol.dimcolor=darkslategray] dimension color. - * @property {array} [symbol.solid=[]] solid line style. - * @property {array} [symbol.dash=[15,10]] dashed line style. - * @property {array} [symbol.dot=[4,4]] dotted line style. - * @property {array} [symbol.dashdot=[25,6.5,2,6.5]] dashdotted line style. - * @property {number} [symbol.labelOffset=5] default label offset distance. - * @property {number} [symbol.labelSignificantDigits=3] default label's significant digits after numbering point. - */ -g2.symbol = g2.symbol || {}; -g2.symbol.tick = g2().p().m({x:0,y:-2}).l({x:0,y:2}).stroke({lc:"round",lwnosc:true}); -g2.symbol.dot = g2().cir({x:0,y:0,r:2,ls:"transparent"}); -g2.symbol.sqr = g2().rec({x:-1.5,y:-1.5,b:3,h:3,ls:"transparent"}); - -g2.symbol.nodcolor = "#333"; -g2.symbol.nodfill = "#dedede"; -g2.symbol.nodfill2 = "#aeaeae"; -g2.symbol.linkcolor = "#666"; -g2.symbol.linkfill = "rgba(225,225,225,0.75)"; -g2.symbol.dimcolor = "darkslategray"; -g2.symbol.solid = []; -g2.symbol.dash = [15,10]; -g2.symbol.dot = [4,4]; -g2.symbol.dashdot = [25,6.5,2,6.5]; -g2.symbol.labelSignificantDigits = 3; // 0.1234 => 0.123, 0.01234 => 0.0123, 1.234 => 1.23, 12.34 => 12.3, 123.4 => 123, 1234 => 1234 - -// Helper methods .. not chainable. +}); \ No newline at end of file diff --git a/src/g2.full.js b/src/g2.full.js index 3fc488a..b5ad6d3 100644 --- a/src/g2.full.js +++ b/src/g2.full.js @@ -256,10 +256,11 @@ g2.prototype = { * g2().use({grp:"cross",x:100,y:100}) // Draw cross at position 100,100. */ use({grp,x,y,w,scl}) { - if (typeof grp === "string") // must be a member name of the 'g2.symbol' namespace - arguments[0].grp = grp = g2.symbol[grp]; - if (grp && grp !== this) // avoid self reference .. + if (grp && grp !== this) { // avoid self reference .. + if (typeof grp === "string") // must be a member name of the 'g2.symbol' namespace + arguments[0].grp = g2.symbol[(grp in g2.symbol) ? grp : 'unknown']; this.addCommand({c:'use',a:arguments[0]}); + } return this; }, @@ -323,12 +324,12 @@ g2.prototype = { */ end() { // ignore 'end' commands without matching 'beg' let myBeg = 1, - findMyBeg = (cmd) => { + findMyBeg = (cmd) => { // care about nested beg...end blocks ... if (cmd.c === 'beg') myBeg--; else if (cmd.c === 'end') myBeg++; return myBeg === 0; } - return g2.getCmdIdx(this.commands,findMyBeg) !== false ? this.addCommand({c:'end'}) : this; + return g2.cmdIdxBy(this.commands,findMyBeg) !== false ? this.addCommand({c:'end'}) : this; }, /** @@ -484,10 +485,10 @@ g2.prototype = { * .ins(node) // draw node. * .exe(ctx) // render to canvas context. */ - ins(fn) { - return typeof fn === 'function' ? (fn(this) || this) - : typeof fn === 'object' ? ( this.commands.push({c:'ins',a:fn}), this ) // no 'addCommand' .. ! - : this; + ins(arg) { + return typeof arg === 'function' ? (arg(this) || this) // no further processing by handler ... + : typeof arg === 'object' ? ( this.commands.push({a:arg}), this ) // no explicit command name .. ! + : this; }, /** * Execute g2 commands. It does so automatically and recursively with 'use'ed commands. @@ -503,13 +504,31 @@ g2.prototype = { }, // helpers ... addCommand({c,a}) { - if (a && Object.getPrototypeOf(a) === Object.prototype) { // modify only pure argument objects 'a' - for (const key in a) + if (a && Object.getPrototypeOf(a) === Object.prototype) { // modify only pure argument objects 'a' .. ! + for (const key in a) { if (!Object.getOwnPropertyDescriptor(a,key).get // if 'key' is no getter ... && key[0] !== '_' // and no private property ... - && typeof a[key] === 'function') { // and a function + && typeof a[key] === 'function') { // and a function ... make it a getter Object.defineProperty(a, key, { get:a[key], enumerable:true, configurable:true, writabel:false }); } + if (typeof a[key] === 'string' && a[key][0] === '@') { // referring values by neighbor id's + const refidIdx = a[key].indexOf('.'); + const refid = refidIdx > 0 ? a[key].substr(1,refidIdx-1) : ''; + const refkey = refid ? a[key].substr(refidIdx+1) : ''; + const refcmd = refid ? () => this.commands.find((cmd) => cmd.a && cmd.a.id === refid) : undefined; + + if (refcmd) + Object.defineProperty(a, key, { + get: function() { + const rc = refcmd(); + return rc && (refkey in rc.a) ? rc.a[refkey] : 0; + }, + enumerable: true, + configurable: true, + writabel: false + }); + } + } if (g2.prototype[c].prototype) Object.setPrototypeOf(a, g2.prototype[c].prototype); } this.commands.push(arguments[0]); @@ -519,7 +538,9 @@ g2.prototype = { // statics g2.defaultStyle = {fs:'transparent',ls:'#000',lw:1,lc:"butt",lj:"miter",ld:[],ml:10,sh:[0,0],lsh:false,font:'14px serif',thal:'start',tval:'alphabetic'}; -g2.symbol = {}; +g2.symbol = { + unknown: g2().cir({r:12}).txt({str:'?',thal:'center',tval:'middle',font:'bold 20pt serif'}) +}; g2.handler = function(ctx) { let hdl; for (let h of g2.handler.factory) @@ -559,7 +580,7 @@ g2.pntItrOf = function(pts) { * Similar to 'Array.prototype.findIndex', only working reverse. * @private */ -g2.getCmdIdx = function(cmds,callbk) { +g2.cmdIdxBy = function(cmds,callbk) { for (let i = cmds.length-1; i >= 0; i--) if (callbk(cmds[i],i,cmds)) return i; @@ -614,13 +635,17 @@ g2.canvasHdl.prototype = { }, async exe(commands) { for (let cmd of commands) { - if (cmd.c && this[cmd.c]) { + if (cmd.c && this[cmd.c]) { // explicit command name .. ! const rx = this[cmd.c](cmd.a); if (rx && rx instanceof Promise) { await rx; } - } else if (cmd.a && 'g2' in cmd.a) - this.exe(cmd.a.g2().commands); + } else if (cmd.a) { // should be from 'ins' command + if (cmd.a.commands) // cmd.a is a `g2` object, so directly execute its commands array. + this.exe(cmd.a.commands); + if (cmd.a.g2) // cmd.a is an object offering a `g2` method, so call it and execute its returned commands array. + this.exe(cmd.a.g2().commands); + } } }, view({x=0,y=0,scl=1,cartesian=false}) { @@ -875,6 +900,7 @@ g2.canvasHdl.prototype = { lc: (ctx) => ctx.lineCap, lj: (ctx) => ctx.lineJoin, ld: (ctx) => ctx.getLineDash(), + ldoff: (ctx) => ctx.lineDashOffset, ml: (ctx) => ctx.miterLimit, sh: (ctx) => [ctx.shadowOffsetX||0,ctx.shadowOffsetY||0, ctx.shadowBlur||0,ctx.shadowColor||'black'], @@ -889,6 +915,7 @@ g2.canvasHdl.prototype = { lc: (ctx,q) => { ctx.lineCap=q; }, lj: (ctx,q) => { ctx.lineJoin=q; }, ld: (ctx,q) => { ctx.setLineDash(q); }, + ldoff: (ctx,q) => { ctx.lineDashOffset=q; }, ml: (ctx,q) => { ctx.miterLimit=q; }, sh: (ctx,q) => { if (q) { @@ -1231,7 +1258,7 @@ g2.prototype.lin.prototype = { dy: len ? dy/len : 0 }; }, - hitContour({x,y,eps}) { return g2.isPntOnLin({x,y},{x:this.x1,y:this.y1},{x:this.x2,y:this.y2},eps) }, + hit({x,y,eps}) { return g2.isPntOnLin({x,y},{x:this.x1,y:this.y1},{x:this.x2,y:this.y2},eps) }, drag({dx,dy}) { this.x1 += dx; this.x2 += dx; this.y1 += dy; this.y2 += dy; @@ -1259,8 +1286,10 @@ g2.prototype.rec.prototype = { dy: nx }; }, - hitContour({x,y,eps}) { return g2.isPntOnBox({x,y},{x:this.x+this.b/2,y:this.y+this.h/2,b:this.b/2,h:this.h/2},eps) }, - hitInner({x,y,eps}) {return g2.isPntInBox({x,y},{x:this.x+this.b/2,y:this.y+this.h/2,b:this.b/2,h:this.h/2},eps) }, + hit({x,y,eps}) { + return this.isSolid ? g2.isPntInBox({x,y},{x:this.x+this.b/2,y:this.y+this.h/2,b:this.b/2,h:this.h/2},eps) + : g2.isPntOnBox({x,y},{x:this.x+this.b/2,y:this.y+this.h/2,b:this.b/2,h:this.h/2},eps); + }, drag({dx,dy}) { this.x += dx; this.y += dy } }; @@ -1282,8 +1311,10 @@ g2.prototype.cir.prototype = { dx: -ny, dy: nx }; }, - hitContour({x,y,eps}) { return g2.isPntOnCir({x,y},this,eps) }, - hitInner({x,y,eps}) {return g2.isPntInCir({x,y},this,eps) }, + hit({x,y,eps}) { + return this.isSolid ? g2.isPntInCir({x,y},this,eps) + : g2.isPntOnCir({x,y},this,eps); + }, drag({dx,dy}) { this.x += dx; this.y += dy }, handles(grp) { const p0 = { @@ -1324,7 +1355,7 @@ g2.prototype.arc.prototype = { }, isSolid: false, get sh() { return this.state & g2.OVER ? [0,0,5,"black"] : false }, - hitContour({x,y,eps}) { return g2.isPntOnArc({x,y},this,eps) }, + hit({x,y,eps}) { return g2.isPntOnArc({x,y},this,eps) }, drag({dx,dy}) { this.x += dx; this.y += dy; }, handles(grp) { const p0 = { @@ -1415,9 +1446,11 @@ g2.prototype.ply.prototype = { dy: len2 ? dy/len2 : 0 }; }, - x: 0, y: 0, - hitContour({x,y,eps}) { let p={x:x-this.x,y:y-this.y}; return g2.isPntOnPly(p,this,eps) }, // translational only .. at current .. ! - hitInner({x,y,eps}) { let p={x:x-this.x,y:y-this.y}; return g2.isPntInPly(p,this,eps) }, // translational only .. at current .. ! +// x: 0, y: 0, + hit({x,y,eps}) { + return this.isSolid ? g2.isPntInPly({x:x-this.x,y:y-this.y},this,eps) // translational transformation only .. at current .. ! + : g2.isPntOnPly({x:x-this.x,y:y-this.y},this,eps); + }, drag({dx,dy}) { this.x += dx; this.y += dy; }, handles(grp) { let p, slf=this; @@ -1429,10 +1462,16 @@ g2.prototype.ply.prototype = { } } +// use is currently not transformed g2.prototype.use.prototype = { - _dir: g2.prototype.cir.prototype._dir, - r: 5, - pointAt: g2.prototype.cir.prototype.pointAt + get isSolid() { return false }, + hit(at) { + for (const cmd of this.grp.commands) { + if (cmd.a.hit && cmd.a.hit(at)) + return true; + } + return false; + } }; // complex macros / add prototypes to argument objects @@ -1529,7 +1568,7 @@ g2.prototype.spline.prototype = g2.mixin({},g2.prototype.ply.prototype,{ * .label({str:'hello',loc:'s',off:10}) */ g2.prototype.label = function label({str,loc,off,fs,font,fs2}) { - let idx = g2.getCmdIdx(this.commands, (cmd) => { return cmd.a && 'pointAt' in cmd.a}); // find reference index of previous element adding label to ... + let idx = g2.cmdIdxBy(this.commands, (cmd) => { return cmd.a && 'pointAt' in cmd.a}); // find reference index of previous element adding label to ... if (idx !== undefined) { arguments[0]['_refelem'] = this.commands[idx]; this.addCommand({c:'label', a: arguments[0]}); @@ -1588,7 +1627,7 @@ g2.prototype.label.prototype = { * */ g2.prototype.mark = function mark({mrk,loc,dir,fs,ls}) { - let idx = g2.getCmdIdx(this.commands, (cmd) => { return cmd.a && 'pointAt' in cmd.a}); // find reference index of previous element adding mark to ... + let idx = g2.cmdIdxBy(this.commands, (cmd) => { return cmd.a && 'pointAt' in cmd.a}); // find reference index of previous element adding mark to ... if (idx !== undefined) { arguments[0]['_refelem'] = this.commands[idx]; this.addCommand({c:'mark', a: arguments[0]}); @@ -1621,7 +1660,7 @@ g2.prototype.mark.prototype = { } /** - * Extensed style values. + * Extended style values. * Not really meant to get overwritten. But if you actually want, proceed.
    * Theses styles can be referenced using the comfortable '@' syntax. * @namespace @@ -1659,7 +1698,18 @@ g2.symbol.dot = [4,4]; g2.symbol.dashdot = [25,6.5,2,6.5]; g2.symbol.labelSignificantDigits = 3; // 0.1234 => 0.123, 0.01234 => 0.0123, 1.234 => 1.23, 12.34 => 12.3, 123.4 => 123, 1234 => 1234 -// Helper methods .. not chainable. +g2.symbol.image = { graphite: new Image() }; +g2.symbol.image.graphite.src = g2GraphiteData(); +g2.symbol.graphite = g2Graphite(); + +function g2Graphite() { + const ctx = document.createElement("canvas").getContext("2d"); + return ctx.createPattern(g2.symbol.image.graphite, 'repeat'); +} + +function g2GraphiteData() { return ""; } + + "use strict" @@ -2374,14 +2424,12 @@ g2.prototype.pol.prototype = g2.mixin({}, g2.prototype.use.prototype, { * @example * g2().nod({x:10,y:10}) */ -g2.prototype.nod = function () { return this.addCommand({c:'nod',a:arguments[0]||{}}); } -g2.prototype.nod.prototype = g2.mixin({}, g2.prototype.use.prototype, { +g2.prototype.nod = function({x=0,y=0}) { return this.addCommand({c:'nod',a:arguments[0]}); } +g2.prototype.nod.prototype = g2.mixin({}, g2.prototype.cir.prototype, { + get r() { return 5; }, + get isSolid() { return true; }, g2() { - const args = Object.assign({x:0,y:0,scl:1,w:0},this); - return g2() - .beg({x:args.x,y:args.y,scl:args.scl,w:args.w}) - .cir({r:4,ls:'@nodcolor',fs:'@nodfill',lwnosc:true}) - .end(); + return g2().cir({x:this.x,y:this.y,r:this.r,ls:'@nodcolor',fs:'@nodfill',sh:this.sh}) } }) @@ -2394,20 +2442,21 @@ g2.prototype.nod.prototype = g2.mixin({}, g2.prototype.use.prototype, { * @example * g2().dblnod({x:10,y:10}) */ -g2.prototype.dblnod = function () { return this.addCommand({c:'dblnod',a:arguments[0]||{}}); } -g2.prototype.dblnod.prototype = g2.mixin({}, g2.prototype.use.prototype, { +g2.prototype.dblnod = function({x=0,y=0}) { return this.addCommand({c:'dblnod',a:arguments[0]}); } +g2.prototype.dblnod.prototype = g2.mixin({}, g2.prototype.cir.prototype, { + get r() { return 6; }, + get isSolid() { return true; }, g2() { - const args = Object.assign({x:0,y:0,scl:1,w:0},this); return g2() - .beg({x:args.x,y:args.y,scl:args.scl,w:args.w}) - .cir({r:6,ls:'@nodcolor',fs:'@nodfill'}) - .cir({r:3,ls:'@nodcolor',fs:'@nodfill2',lwnosc:true}) + .beg({x:this.x,y:this.y}) + .cir({r:6,ls:'@nodcolor',fs:'@nodfill',sh:this.sh}) + .cir({r:3,ls:'@nodcolor',fs:'@nodfill2'}) .end(); } }) /** - * Since some symbols are not symmetrical, the cartesian mode is recommended. + * Since some symbols are not symmetrical, cartesian mode is recommended. * @method * @returns {object} g2 * @param {object} - symbol arguments object. @@ -2417,18 +2466,19 @@ g2.prototype.dblnod.prototype = g2.mixin({}, g2.prototype.use.prototype, { * g2().view({cartesian:true}) * .nodfix({x:10,y:10}) */ -g2.prototype.nodfix = function () { return this.addCommand({c:'nodfix',a:arguments[0]||{}}); } -g2.prototype.nodfix.prototype = g2.mixin({}, g2.prototype.use.prototype, { +g2.prototype.nodfix = function ({x=0,y=0}) { return this.addCommand({c:'nodfix',a:arguments[0]||{x:0,y:0}}); } +g2.prototype.nodfix.prototype = g2.mixin({}, g2.prototype.cir.prototype, { + get r() { return 5; }, + get isSolid() { return true; }, g2() { - const args = Object.assign({x:0,y:0,scl:1,w:0},this); return g2() - .beg({x:args.x,y:args.y,scl:args.scl,w:args.w}) + .beg({x:this.x,y:this.y,scl:this.scl,w:this.w,sh:this.sh}) .p() .m({x:-8,y:-12}) .l({x:0,y:0}) .l({x:8,y:-12}) .drw({ls:'@nodcolor',fs:'@nodfill2'}) - .cir({x:0,y:0,r:4,ls:'@nodcolor',fs:'@nodfill'}) + .cir({x:0,y:0,r:this.r,ls:'@nodcolor',fs:'@nodfill'}) .end(); } }) @@ -2443,18 +2493,19 @@ g2.prototype.nodfix.prototype = g2.mixin({}, g2.prototype.use.prototype, { * g2().view({cartesian:true}) * .nodflt({x:10,y:10}) */ -g2.prototype.nodflt = function () { return this.addCommand({c:'nodflt',a:arguments[0]||{}}); } -g2.prototype.nodflt.prototype = g2.mixin({}, g2.prototype.use.prototype, { +g2.prototype.nodflt = function ({x=0,y=0}) { return this.addCommand({c:'nodflt',a:arguments[0]||{x,y}}); } +g2.prototype.nodflt.prototype = g2.mixin({}, g2.prototype.cir.prototype, { + get r() { return 5; }, + get isSolid() { return true; }, g2() { - const args = Object.assign({x:0,y:0,scl:1,w:0},this); return g2() - .beg({x:args.x,y:args.y,scl:args.scl,w:args.w}) + .beg({x:this.x,y:this.y,w:this.w,sh:this.sh}) .p() .m({x:-8,y:-12}) .l({x:0,y:0}) .l({x:8,y:-12}) .drw({ls:'@nodcolor',fs:'@nodfill2'}) - .cir({x:0,y:0,r:4,ls:'@nodcolor',fs:'@nodfill'}) + .cir({x:0,y:0,r:this.r,ls:'@nodcolor',fs:'@nodfill'}) .lin({x1:-9,y1:-19,x2:9,y2:-19,ls:'@nodfill2',lw:5,lwnosc:false}) .lin({x1:-9,y1:-15.5,x2:9,y2:-15.5,ls:'@nodcolor',lw:2,lwnosc:false}) .end(); @@ -2484,38 +2535,6 @@ g2.prototype.origin.prototype = g2.mixin({}, g2.prototype.use.prototype, { } }) -/** - * Mechanical style values. - * Not really meant to get overwritten. But if you actually want, proceed.
    - * Theses styles can be referenced using the comfortable '@' syntax. - * @namespace - * @property {object} State `g2` state namespace. - * @property {string} [State.nodcolor=#333] node color. - * @property {string} [State.nodfill=#dedede] node fill color. - * @property {string} [State.nodfill2=#aeaeae] alternate node fill color, somewhat darker. - * @property {string} [State.linkcolor=#666] link color. - * @property {string} [State.linkfill=rgba(225,225,225,0.75)] link fill color, semi-transparent. - * @property {string} [State.dimcolor=darkslategray] dimension color. - * @property {array} [State.solid=[]] solid line style. - * @property {array} [State.dash=[15,10]] dashed line style. - * @property {array} [State.dot=[4,4]] dotted line style. - * @property {array} [State.dashdot=[25,6.5,2,6.5]] dashdotted line style. - * @property {number} [State.labelOffset=5] default label offset distance. - * @property {number} [State.labelSignificantDigits=3] default label's significant digits after numbering point. - */ -g2.State = g2.State || {}; -g2.State.nodcolor = '#333'; -g2.State.nodfill = '#dedede'; -g2.State.nodfill2 = '#aeaeae'; -g2.State.linkcolor = '#666'; -g2.State.linkfill = 'rgba(225,225,225,0.75)'; -g2.State.dimcolor = 'darkslategray'; -g2.State.solid = []; -g2.State.dash = [15,10]; -g2.State.dot = [4,4]; -g2.State.dashdot = [25,6.5,2,6.5]; -g2.State.labelOffset = 5; -g2.State.labelSignificantDigits = 3; // 0.1234 => 0.123, 0.01234 => 0.0123, 1.234 => 1.23, 12.34 => 12.3, 123.4 => 123, 1234 => 1234 "use strict" /** diff --git a/src/g2.io.js b/src/g2.io.js index 70139f9..2ae4218 100644 --- a/src/g2.io.js +++ b/src/g2.io.js @@ -1,41 +1,41 @@ /** - * g2.io (c) 2013-17 Stefan Goessner + * g2.io (c) 2017-18 Stefan Goessner * @license MIT License - * @link https://github.com/goessner/g2 */ -"use strict" +"use strict"; g2.io = function() { - if (this instanceof g2.io) { - this.model = null; - this.grpidx = 0; - return this; - } - return g2.io.apply(Object.create(g2.io.prototype)); + if (this instanceof g2.io) { + this.model = null; + this.grpidx = 0; + return this; + } + return g2.io.apply(Object.create(g2.io.prototype)); }; g2.handler.factory.push((ctx) => ctx instanceof g2.io ? ctx : false); -g2.io.parse = function(str) { - let model = JSON.parse(str); - return g2.io.parseGrp(model,'main'); -} -g2.io.parseGrp = function(model, id) { +g2.io.parseGrp = function(model, id, onErr) { let g; + onErr = onErr || console.error; if (id in model) { g = g2({id}); for (let cmd of model[id]) { - if (cmd.c === 'use') + if (cmd.c === 'use') { cmd.a.grp = g2.io.parseGrp(model, cmd.a.grp); - else if (this[cmd.c]) - cmd.a ? this[cmd.c](cmd.a) : this[cmd.c](); - else // invalid g2 command ! - console.error(`io: Unable to handle command '${cmd.c}'`); + g[cmd.c](cmd.a); + } + else if (g[cmd.c]) + cmd.a ? g[cmd.c](cmd.a) : g[cmd.c](); + else // invalid g2 command ! + onErr(`g2.io: Unable to handle command '${cmd.c}'`) } return g; } else if (id in g2.symbol) return g2.symbol[id]; - return null; + else + onErr(`g2.io: Unable to find group with id '${id}'!`); + return false; } g2.io.prototype = { @@ -43,7 +43,7 @@ g2.io.prototype = { this.model = {'main':[]}; this.curgrp = this.model.main; this.grpidx = 0; - return true; + return true; }, exe: function(commands) { for (let cmd of commands) { diff --git a/src/g2.js b/src/g2.js deleted file mode 100644 index 4099730..0000000 --- a/src/g2.js +++ /dev/null @@ -1,91 +0,0 @@ -"use strict"; -/** - * g2.core (c) 2013-19 Stefan Goessner - * @author Stefan Goessner - * @license MIT License - * @link https://github.com/goessner/g2 - * @typedef {g2} - * @param {object} [opts] Custom options object. It is simply copied into the 'g2' instance, but not used from the g2 kernel. - * @description Create a 2D graphics command queue object. Call without using 'new'. - * @returns {g2} - * @example - * const ctx = document.getElementById("c").getContext("2d"); - * g2() // Create 'g2' instance. - * .lin({x1:50,y1:50,x2:100,y2:100}) // Append ... - * .lin({x1:100,y1:100,x2:200,y2:50}) // ... commands. - * .exe(ctx); // Execute commands addressing canvas context. - */function g2(opts){let o=Object.create(g2.prototype);o.commands=[];if(opts)Object.assign(o,opts);return o}g2.prototype={clr(){return this.addCommand({c:"clr"})},view({scl:scl,x:x,y:y,cartesian:cartesian}){return this.addCommand({c:"view",a:arguments[0]})},grid({color:color,size:size}={}){return this.addCommand({c:"grid",a:arguments[0]})},cir({x:x,y:y,r:r,w:w}){return this.addCommand({c:"cir",a:arguments[0]})},ell({x:x,y:y,rx:rx,ry:ry,w:w,dw:dw,rot:rot}){return this.addCommand({c:"ell",a:arguments[0]})},arc({x:x,y:y,r:r,w:w,dw:dw}){return this.addCommand({c:"arc",a:arguments[0]})},rec({x:x,y:y,b:b,h:h}){return this.addCommand({c:"rec",a:arguments[0]})},lin({x1:x1,y1:y1,x2:x2,y2:y2}){return this.addCommand({c:"lin",a:arguments[0]})},ply({pts:pts,format:format,closed:closed,x:x,y:y,w:w}){arguments[0]._itr=format&&g2.pntIterator[format](pts)||g2.pntItrOf(pts);return this.addCommand({c:"ply",a:arguments[0]})},txt({str:str,x:x,y:y,w:w}){return this.addCommand({c:"txt",a:arguments[0]})},use({grp:grp,x:x,y:y,w:w,scl:scl}){if(typeof grp==="string")arguments[0].grp=grp=g2.symbol[grp];if(grp&&grp!==this)this.addCommand({c:"use",a:arguments[0]});return this},img({uri:uri,x:x,y:y,b:b,h:h,sx:sx,sy:sy,sb:sb,sh:sh,xoff:xoff,yoff:yoff,w:w,scl:scl}){return this.addCommand({c:"img",a:arguments[0]})},beg({x:x,y:y,w:w,scl:scl,matrix:matrix}={}){return this.addCommand({c:"beg",a:arguments[0]})},end(){let myBeg=1,findMyBeg=cmd=>{if(cmd.c==="beg")myBeg--;else if(cmd.c==="end")myBeg++;return myBeg===0};return g2.getCmdIdx(this.commands,findMyBeg)!==false?this.addCommand({c:"end"}):this},p(){return this.addCommand({c:"p"})},z(){return this.addCommand({c:"z"})},m({x:x,y:y}){return this.addCommand({c:"m",a:arguments[0]})},l({x:x,y:y}){return this.addCommand({c:"l",a:arguments[0]})},q({x1:x1,y1:y1,x:x,y:y}){return this.addCommand({c:"q",a:arguments[0]})},c({x1:x1,y1:y1,x2:x2,y2:y2,x:x,y:y}){return this.addCommand({c:"c",a:arguments[0]})},a({dw:dw,x:x,y:y}){let prvcmd=this.commands[this.commands.length-1];g2.cpyProp(prvcmd.a,"x",arguments[0],"_xp");g2.cpyProp(prvcmd.a,"y",arguments[0],"_yp");return this.addCommand({c:"a",a:arguments[0]})},stroke({d:d}={}){return this.addCommand({c:"stroke",a:arguments[0]})},fill({d:d}={}){return this.addCommand({c:"fill",a:arguments[0]})},drw({d:d,lsh:lsh}={}){return this.addCommand({c:"drw",a:arguments[0]})},del(idx){this.commands.length=idx||0;return this},ins(fn){return typeof fn==="function"?fn(this)||this:typeof fn==="object"?(this.commands.push({c:"ins",a:fn}),this):this},exe(ctx){let handler=g2.handler(ctx);if(handler&&handler.init(this))handler.exe(this.commands);return this},addCommand({c:c,a:a}){if(a&&Object.getPrototypeOf(a)===Object.prototype){for(const key in a)if(!Object.getOwnPropertyDescriptor(a,key).get&&key[0]!=="_"&&typeof a[key]==="function"){Object.defineProperty(a,key,{get:a[key],enumerable:true,configurable:true,writabel:false})}if(g2.prototype[c].prototype)Object.setPrototypeOf(a,g2.prototype[c].prototype)}this.commands.push(arguments[0]);return this}};g2.defaultStyle={fs:"transparent",ls:"#000",lw:1,lc:"butt",lj:"miter",ld:[],ml:10,sh:[0,0],lsh:false,font:"14px serif",thal:"start",tval:"alphabetic"};g2.symbol={};g2.handler=function(ctx){let hdl;for(let h of g2.handler.factory)if((hdl=h(ctx))!==false)return hdl;return false};g2.handler.factory=[];g2.pntIterator={"x,y":function(pts){function pitr(i){return{x:pts[2*i],y:pts[2*i+1]}}Object.defineProperty(pitr,"len",{get:()=>pts.length/2,enumerable:true,configurable:true,writabel:false});return pitr},"[x,y]":function(pts){function pitr(i){return pts[i]?{x:pts[i][0],y:pts[i][1]}:undefined}Object.defineProperty(pitr,"len",{get:()=>pts.length,enumerable:true,configurable:true,writabel:false});return pitr},"{x,y}":function(pts){function pitr(i){return pts[i]}Object.defineProperty(pitr,"len",{get:()=>pts.length,enumerable:true,configurable:true,writabel:false});return pitr}};g2.pntItrOf=function(pts){return!(pts&&pts.length)?undefined:typeof pts[0]==="number"?g2.pntIterator["x,y"](pts):Array.isArray(pts[0])&&pts[0].length>=2?g2.pntIterator["[x,y]"](pts):typeof pts[0]==="object"&&"x"in pts[0]&&"y"in pts[0]?g2.pntIterator["{x,y}"](pts):undefined};g2.getCmdIdx=function(cmds,callbk){for(let i=cmds.length-1;i>=0;i--)if(callbk(cmds[i],i,cmds))return i;return false};g2.mixin=function mixin(obj,...protos){protos.forEach(p=>{Object.keys(p).forEach(k=>{Object.defineProperty(obj,k,Object.getOwnPropertyDescriptor(p,k))})});return obj};g2.cpyProp=function(from,fromKey,to,toKey){Object.defineProperty(to,toKey,Object.getOwnPropertyDescriptor(from,fromKey))};g2.canvasHdl=function(ctx){if(this instanceof g2.canvasHdl){if(ctx instanceof CanvasRenderingContext2D){this.ctx=ctx;this.cur=g2.defaultStyle;this.stack=[this.cur];this.matrix=[[1,0,0,1,.5,.5]];this.gridBase=2;this.gridExp=1;return this}else return null}return g2.canvasHdl.apply(Object.create(g2.canvasHdl.prototype),arguments)};g2.handler.factory.push(ctx=>ctx instanceof g2.canvasHdl?ctx:ctx instanceof CanvasRenderingContext2D?g2.canvasHdl(ctx):false);g2.canvasHdl.prototype={init(grp,style){this.stack.length=1;this.matrix.length=1;this.initStyle(style?Object.assign({},this.cur,style):this.cur);return true},async exe(commands){for(let cmd of commands){if(cmd.c&&this[cmd.c]){const rx=this[cmd.c](cmd.a);if(rx&&rx instanceof Promise){await rx}}else if(cmd.a&&"g2"in cmd.a)this.exe(cmd.a.g2().commands)}},view({x:x=0,y:y=0,scl:scl=1,cartesian:cartesian=false}){this.pushTrf(cartesian?[scl,0,0,-scl,x,this.ctx.canvas.height-1-y]:[scl,0,0,scl,x,y])},grid({color:color="#ccc",size:size}={}){let ctx=this.ctx,b=ctx.canvas.width,h=ctx.canvas.height,{x:x,y:y,scl:scl}=this.uniTrf,sz=size||this.gridSize(scl),xoff=x%sz,yoff=y%sz;ctx.save();ctx.setTransform(1,0,0,1,0,0);ctx.strokeStyle=color;ctx.lineWidth=1;ctx.beginPath();for(let x=xoff,nx=b+1;xNumber.EPSILON&&Math.abs(r)>Number.EPSILON){this.ctx.beginPath();this.ctx.arc(x,y,Math.abs(r),w,w+dw,dw<0);this.drw(arguments[0])}else if(Math.abs(dw)Number.EPSILON){const cw=Math.cos(w),sw=Math.sin(w);this.ctx.beginPath();this.ctx.moveTo(x-r*cw,y-r*sw);this.ctx.lineTo(x+r*cw,y+r*sw)}},ell({x:x=0,y:y=0,rx:rx,ry:ry,w:w=0,dw:dw=2*Math.PI,rot:rot=0}){this.ctx.beginPath();this.ctx.ellipse(x,y,Math.abs(rx),Math.abs(ry),rot,w,w+dw,dw<0);this.drw(arguments[0])},rec({x:x=0,y:y=0,b:b,h:h}){let tmp=this.setStyle(arguments[0]);this.ctx.fillRect(x,y,b,h);this.ctx.strokeRect(x,y,b,h);this.resetStyle(tmp)},lin({x1:x1=0,y1:y1=0,x2:x2,y2:y2}){let ctx=this.ctx;ctx.beginPath();ctx.moveTo(x1,y1);ctx.lineTo(x2,y2);this.stroke(arguments[0])},ply:function({pts:pts,closed:closed,x:x=0,y:y=0,w:w=0,_itr:_itr}){if(_itr&&_itr.len){let p,i,len=_itr.len,istrf=!!(x||y||w),cw,sw;if(istrf)this.setTrf([cw=w?Math.cos(w):1,sw=w?Math.sin(w):0,-sw,cw,x||0,y||0]);this.ctx.beginPath();this.ctx.moveTo((p=_itr(0)).x,p.y);for(i=1;i{const pimg=new Promise((resolve,reject)=>{let img=new Image;img.src=xuri;function error(err){img.removeEventListener("load",load);img=undefined;reject(err)}function load(){img.removeEventListener("error",error);resolve(img);img=undefined}img.addEventListener("error",error,{once:true});img.addEventListener("load",load,{once:true})});try{return await pimg}catch(err){if(xuri===this.errorImageStr){throw err}else{return await download(this.errorImageStr)}}};let img=this.images[uri];if(img!==undefined){return img instanceof Promise?await img:img}img=download(uri);this.images[uri]=img;try{img=await img}finally{this.images[uri]=img}return img},async img({uri:uri,x:x=0,y:y=0,b:b,h:h,sx:sx=0,sy:sy=0,sb:sb,sh:sh,xoff:xoff=0,yoff:yoff=0,w:w=0,scl:scl=1}){const img_=await this.loadImage(uri);this.ctx.save();const cart=this.isCartesian?-1:1;sb=sb||img_.width;b=b||img_.width;sh=sh||img_.height;h=(h||img_.height)*cart;yoff*=cart;w*=cart;y=this.isCartesian?-(y/scl)+sy:y/scl;const[cw,sw]=[Math.cos(w),Math.sin(w)];this.ctx.scale(scl,scl*cart);this.ctx.transform(cw,sw,-sw,cw,x/scl,y);this.ctx.drawImage(img_,sx,sy,sb,sh,xoff,yoff,b,h);this.ctx.restore()},use({grp:grp}){this.beg(arguments[0]);this.exe(grp.commands);this.end()},beg({x:x=0,y:y=0,w:w=0,scl:scl=1,matrix:matrix,unsizable:unsizable}={}){let trf=matrix;if(!trf){let ssw,scw;ssw=w?Math.sin(w)*scl:0;scw=w?Math.cos(w)*scl:scl;trf=[scw,ssw,-ssw,scw,x,y]}this.pushStyle(arguments[0]);this.pushTrf(unsizable?this.concatTrf(this.unscaleTrf({x:x,y:y}),trf):trf)},end(){this.popTrf();this.popStyle()},p(){this.ctx.beginPath()},z(){this.ctx.closePath()},m({x:x,y:y}){this.ctx.moveTo(x,y)},l({x:x,y:y}){this.ctx.lineTo(x,y)},q({x:x,y:y,x1:x1,y1:y1}){this.ctx.quadraticCurveTo(x1,y1,x,y)},c({x:x,y:y,x1:x1,y1:y1,x2:x2,y2:y2}){this.ctx.bezierCurveTo(x1,y1,x2,y2,x,y)},a({x:x,y:y,dw:dw,k:k,phi:phi,_xp:_xp,_yp:_yp}){if(k===undefined)k=1;if(Math.abs(dw)>Number.EPSILON){if(k===1){let x12=x-_xp,y12=y-_yp;let tdw_2=Math.tan(dw/2),rx=(x12-y12/tdw_2)/2,ry=(y12+x12/tdw_2)/2,R=Math.hypot(rx,ry),w=Math.atan2(-ry,-rx);this.ctx.ellipse(_xp+rx,_yp+ry,R,R,0,w,w+dw,this.cartesian?dw>0:dw<0)}else{if(phi===undefined)phi=0;let x1=dw>0?_xp:x,y1=dw>0?_yp:y,x2=dw>0?x:_xp,y2=dw>0?y:_yp;let x12=x2-x1,y12=y2-y1,_dw=dw<0?dw:-dw;let cp=phi?Math.cos(phi):1,sp=phi?Math.sin(phi):0,dx=-x12*cp-y12*sp,dy=-x12*sp-y12*cp,sdw_2=Math.sin(_dw/2),R=Math.sqrt((dx*dx+dy*dy/(k*k))/(4*sdw_2*sdw_2)),w=Math.atan2(k*dx,dy)-_dw/2,x0=x1-R*Math.cos(w),y0=y1-R*k*Math.sin(w);this.ctx.ellipse(x0,y0,R,R*k,phi,w,w+dw,this.cartesian?dw>0:dw<0)}}else this.ctx.lineTo(x,y)},stroke({d:d}={}){let tmp=this.setStyle(arguments[0]);d?this.ctx.stroke(new Path2D(d)):this.ctx.stroke();this.resetStyle(tmp)},fill({d:d}={}){let tmp=this.setStyle(arguments[0]);d?this.ctx.fill(new Path2D(d)):this.ctx.fill();this.resetStyle(tmp)},drw({d:d,lsh:lsh}={}){let ctx=this.ctx,tmp=this.setStyle(arguments[0]),p=d&&new Path2D(d);d?ctx.fill(p):ctx.fill();if(ctx.shadowColor!=="rgba(0, 0, 0, 0)"&&ctx.fillStyle!=="rgba(0, 0, 0, 0)"&&!lsh){let shc=ctx.shadowColor;ctx.shadowColor="rgba(0, 0, 0, 0)";d?ctx.stroke(p):ctx.stroke();ctx.shadowColor=shc}else d?ctx.stroke(p):ctx.stroke();this.resetStyle(tmp)},get:{fs:ctx=>ctx.fillStyle,ls:ctx=>ctx.strokeStyle,lw:ctx=>ctx.lineWidth,lc:ctx=>ctx.lineCap,lj:ctx=>ctx.lineJoin,ld:ctx=>ctx.getLineDash(),ml:ctx=>ctx.miterLimit,sh:ctx=>[ctx.shadowOffsetX||0,ctx.shadowOffsetY||0,ctx.shadowBlur||0,ctx.shadowColor||"black"],font:ctx=>ctx.font,thal:ctx=>ctx.textAlign,tval:ctx=>ctx.textBaseline},set:{fs:(ctx,q)=>{ctx.fillStyle=q},ls:(ctx,q)=>{ctx.strokeStyle=q},lw:(ctx,q)=>{ctx.lineWidth=q},lc:(ctx,q)=>{ctx.lineCap=q},lj:(ctx,q)=>{ctx.lineJoin=q},ld:(ctx,q)=>{ctx.setLineDash(q)},ml:(ctx,q)=>{ctx.miterLimit=q},sh:(ctx,q)=>{if(q){ctx.shadowOffsetX=q[0]||0;ctx.shadowOffsetY=q[1]||0;ctx.shadowBlur=q[2]||0;ctx.shadowColor=q[3]||"black"}},font:(ctx,q)=>{ctx.font=q},thal:(ctx,q)=>{ctx.textAlign=q},tval:(ctx,q)=>{ctx.textBaseline=q}},initStyle(style){for(const key in style)if(this.get[key]&&this.get[key](this.ctx)!==style[key])this.set[key](this.ctx,style[key])},setStyle(style){let q,prv={};for(const key in style){if(this.get[key]){if(typeof style[key]==="string"&&style[key][0]==="@"){let ref=style[key].substr(1);style[key]=g2.symbol[ref]||this.get[ref]&&this.get[ref](this.ctx)}if((q=this.get[key](this.ctx))!==style[key]){prv[key]=q;this.set[key](this.ctx,style[key])}}}return prv},resetStyle(style){for(const key in style)this.set[key](this.ctx,style[key])},pushStyle(style){let cur={};for(const key in style)if(this.get[key]){if(typeof style[key]==="string"&&style[key][0]==="@"){let ref=style[key].substr(1);style[key]=g2.symbol[ref]||this.get[ref]&&this.get[ref](this.ctx)}if(this.cur[key]!==style[key])this.set[key](this.ctx,cur[key]=style[key])}this.stack.push(this.cur=Object.assign({},this.cur,cur))},popStyle(){let cur=this.stack.pop();this.cur=this.stack[this.stack.length-1];for(const key in this.cur)if(this.get[key]&&this.cur[key]!==cur[key])this.set[key](this.ctx,this.cur[key])},concatTrf(q,t){return[q[0]*t[0]+q[2]*t[1],q[1]*t[0]+q[3]*t[1],q[0]*t[2]+q[2]*t[3],q[1]*t[2]+q[3]*t[3],q[0]*t[4]+q[2]*t[5]+q[4],q[1]*t[4]+q[3]*t[5]+q[5]]},initTrf(){this.ctx.setTransform(...this.matrix[0])},setTrf(t){this.ctx.setTransform(...this.concatTrf(this.matrix[this.matrix.length-1],t))},resetTrf(){this.ctx.setTransform(...this.matrix[this.matrix.length-1])},pushTrf(t){let q_t=this.concatTrf(this.matrix[this.matrix.length-1],t);this.matrix.push(q_t);this.ctx.setTransform(...q_t)},popTrf(){this.matrix.pop();this.ctx.setTransform(...this.matrix[this.matrix.length-1])},get isCartesian(){let m=this.matrix[this.matrix.length-1];return m[0]*m[3]-m[1]*m[2]<0},get uniTrf(){let m=this.matrix[this.matrix.length-1];return{x:m[4],y:m[5],scl:Math.hypot(m[0],m[1]),cartesian:m[0]*m[3]-m[1]*m[2]<0}},unscaleTrf({x:x,y:y}){let m=this.matrix[this.matrix.length-1],invscl=1/Math.hypot(m[0],m[1]);return[invscl,0,0,invscl,(1-invscl)*x,(1-invscl)*y]},gridSize(scl){let base=this.gridBase,exp=this.gridExp,sz;while((sz=scl*base*Math.pow(10,exp))<14||sz>35){if(sz<14){if(base==1)base=2;else if(base==2)base=5;else if(base==5){base=1;exp++}}else{if(base==1){base=5;exp--}else if(base==2)base=1;else if(base==5)base=2}}this.gridBase=base;this.gridExp=exp;return sz}};g2.zoomView=function({scl:scl,x:x,y:y}){return{scl:scl,x:(1-scl)*x,y:(1-scl)*y}};g2.render=function render(fn){function animate(t){if(fn(t))requestAnimationFrame(animate)}animate(performance.now())};if(typeof module!=="undefined")module.exports=g2; -/** - * g2.lib (c) 2013-17 Stefan Goessner - * geometric constants and higher functions - * @license MIT License - * @link https://github.com/goessner/g2 - */ -"use strict";var g2=g2||{};g2=Object.assign(g2,{EPS:Number.EPSILON,PI:Math.PI,PI2:2*Math.PI,SQRT2:Math.SQRT2,SQRT2_2:Math.SQRT2/2,toPi2(w){return(w%g2.PI2+g2.PI2)%g2.PI2},toPi(w){return(w=(w%g2.PI2+g2.PI2)%g2.PI2)>g2.PI?w-g2.PI2:w},toArc:function(w,w0,dw){if(dw>g2.EPS||dw<-g2.EPS){if(w0>w&&w0+dw>g2.PI2)w0-=g2.PI2;else if(w0eps&&Math.abs(dist-r)=0&&mu<=1},isPntOnPly({x:x,y:y},{pts:pts,closed:closed},eps=Number.EPSILON){for(var i=0,n=pts.length;i<(closed?n:n-1);i++)if(g2.isPntOnLin({x:x,y:y},pts[i],pts[(i+1)%n],eps))return true;return false},isPntOnBox({x:xp,y:yp},{x:x,y:y,b:b,h:h},eps=Number.EPSILON){var dx=x.p-x,dy=yp-y;return dx>=b-eps&&dx<=b+eps&&dy<=h+eps&&dy>=-h-eps||dx>=-b-eps&&dx<=b+eps&&dy<=h+eps&&dy>=h-eps||dx>=-b-eps&&dx<=-b+eps&&dy<=h+eps&&dy>=-h-eps||dx>=-b-eps&&dx<=b+eps&&dy<=-h+eps&&dy>=-h-eps},isPntInCir({x:xp,y:yp},{x:x,y:y,r:r}){return(x-xp)**2+(y-yp)**2pi.y||y>pj.y)&&(y<=pi.y||y<=pj.y)&&(x<=pi.x||x<=pj.x)&&pi.y!==pj.y&&(pi.x===pj.x||x<=pj.x+(y-pj.y)*(pi.x-pj.x)/(pi.y-pj.y)))match++;return match%2!=0},isPntInBox({x:xp,y:yp},{x:x,y:y,b:b,h:h}){var dx=xp-x,dy=yp-y;return dx>=-b&&dx<=b&&dy>=-h&&dy<=h},arc3pts(x1,y1,x2,y2,x3,y3){const dx1=x2-x1,dy1=y2-y1;const dx2=x3-x2,dy2=y3-y2;const den=dx1*dy2-dy1*dx2;const lam=Math.abs(den)>Number.EPSILON?.5*((dx1+dx2)*dx2+(dy1+dy2)*dy2)/den:0;const x0=lam?x1+.5*dx1-lam*dy1:x1+.5*(dx1+dx2);const y0=lam?y1+.5*dy1+lam*dx1:y1+.5*(dy1+dy2);const dx01=x1-x0,dy01=y1-y0;const dx03=x3-x0,dy03=y3-y0;const dw=lam?Math.atan2(dx01*dy03-dy01*dx03,dx01*dx03+dy01*dy03):0;const r=dw?Math.hypot(dy01,dx01):.5*Math.hypot(dy1+dy2,dx1+dx2);return{x:x0,y:y0,r:r,w:Math.atan2(dy01,dx01),dw:dw}}});"use strict"; -/** - * g2.ext (c) 2015-18 Stefan Goessner - * @author Stefan Goessner - * @license MIT License - * @requires g2.core.js - * @typedef {g2} - * @description Additional methods for g2. - * @returns {g2} - */var g2=g2||{prototype:{}};g2.NONE=0;g2.OVER=1;g2.DRAG=2;g2.EDIT=4;g2.prototype.lin.prototype={isSolid:false,get len(){return Math.hypot(this.x2-this.x1,this.y2-this.y1)},get sh(){return this.state&g2.OVER?[0,0,5,"black"]:false},pointAt(loc){let t=loc==="beg"?0:loc==="end"?1:loc+0===loc?loc:.5,dx=this.x2-this.x1,dy=this.y2-this.y1,len=Math.hypot(dx,dy);return{x:this.x1+dx*t,y:this.y1+dy*t,dx:len?dx/len:1,dy:len?dy/len:0}},hitContour({x:x,y:y,eps:eps}){return g2.isPntOnLin({x:x,y:y},{x:this.x1,y:this.y1},{x:this.x2,y:this.y2},eps)},drag({dx:dx,dy:dy}){this.x1+=dx;this.x2+=dx;this.y1+=dy;this.y2+=dy},handles(grp){grp.handle({x:this.x1,y:this.y1,_update:({dx:dx,dy:dy})=>{this.x1+=dx;this.y1+=dy}}).handle({x:this.x2,y:this.y2,_update:({dx:dx,dy:dy})=>{this.x2+=dx;this.y2+=dy}})}};g2.prototype.rec.prototype={_dir:{c:[0,0],e:[1,0],ne:[1,1],n:[0,1],nw:[-1,1],w:[-1,0],sw:[-1,-1],s:[0,-1],se:[1,-1]},get len(){return 2*(this.b+this.h)},get isSolid(){return this.fs&&this.fs!=="transparent"},get len(){return 2*Math.PI*this.r},get lsh(){return this.state&g2.OVER},get sh(){return this.state&g2.OVER?[0,0,5,"black"]:false},pointAt(loc){const q=this._dir[loc||"c"]||this._dir["c"],nx=q[0],ny=q[1];return{x:this.x+(1+nx)*this.b/2,y:this.y+(1+ny)*this.h/2,dx:-ny,dy:nx}},hitContour({x:x,y:y,eps:eps}){return g2.isPntOnBox({x:x,y:y},{x:this.x+this.b/2,y:this.y+this.h/2,b:this.b/2,h:this.h/2},eps)},hitInner({x:x,y:y,eps:eps}){return g2.isPntInBox({x:x,y:y},{x:this.x+this.b/2,y:this.y+this.h/2,b:this.b/2,h:this.h/2},eps)},drag({dx:dx,dy:dy}){this.x+=dx;this.y+=dy}};g2.prototype.cir.prototype={w:0,_dir:{c:[0,0],e:[1,0],ne:[Math.SQRT2/2,Math.SQRT2/2],n:[0,1],nw:[-Math.SQRT2/2,Math.SQRT2/2],w:[-1,0],sw:[-Math.SQRT2/2,-Math.SQRT2/2],s:[0,-1],se:[Math.SQRT2/2,-Math.SQRT2/2]},get isSolid(){return this.fs&&this.fs!=="transparent"},get len(){return 2*Math.PI*this.r},get lsh(){return this.state&g2.OVER},get sh(){return this.state&g2.OVER?[0,0,5,"black"]:false},pointAt(loc){let q=loc+0===loc?[Math.cos(loc*2*Math.PI),Math.sin(loc*2*Math.PI)]:this._dir[loc||"c"]||[0,0],nx=q[0],ny=q[1];return{x:this.x+nx*this.r,y:this.y+ny*this.r,dx:-ny,dy:nx}},hitContour({x:x,y:y,eps:eps}){return g2.isPntOnCir({x:x,y:y},this,eps)},hitInner({x:x,y:y,eps:eps}){return g2.isPntInCir({x:x,y:y},this,eps)},drag({dx:dx,dy:dy}){this.x+=dx;this.y+=dy},handles(grp){const p0={x:this.x,y:this.y,_update:({dx:dx,dy:dy})=>{this.x+=dx;this.y+=dy;p1.x+=dx;p1.y+=dy}};const p1={x:this.x+this.r*Math.cos(this.w||0),y:this.y+this.r*Math.sin(this.w||0),_info:()=>`r:${this.r.toFixed(1)}
    w:${(this.w/Math.PI*180).toFixed(1)}°`,_update:({x:x,y:y})=>{this.r=Math.hypot(y-this.y,x-this.x);this.w=Math.atan2(y-this.y,x-this.x)}};grp.lin({x1:()=>this.x,y1:()=>this.y,x2:()=>p1.x,y2:()=>p1.y,ld:[4,3],ls:"#666"}).handle(p0).handle(p1)}};g2.prototype.arc.prototype={get len(){return Math.abs(this.r*this.dw)},get angle(){return this.dw/Math.PI*180},pointAt(loc){let t=loc==="beg"?0:loc==="end"?1:loc==="mid"?.5:loc+0===loc?loc:.5,ang=this.w+t*this.dw,cang=Math.cos(ang),sang=Math.sin(ang),r=loc==="c"?0:this.r;return{x:this.x+r*cang,y:this.y+r*sang,dx:-sang,dy:cang}},isSolid:false,get sh(){return this.state&g2.OVER?[0,0,5,"black"]:false},hitContour({x:x,y:y,eps:eps}){return g2.isPntOnArc({x:x,y:y},this,eps)},drag({dx:dx,dy:dy}){this.x+=dx;this.y+=dy},handles(grp){const p0={x:this.x,y:this.y,_update:({dx:dx,dy:dy})=>{this.x+=dx;this.y+=dy;p1.x+=dx;p1.y+=dy;p2.x+=dx;p2.y+=dy}},p1={x:this.x+this.r*Math.cos(this.w),y:this.y+this.r*Math.sin(this.w),_info:()=>`r:${this.r.toFixed(1)}
    w:${(this.w/Math.PI*180).toFixed(1)}°`,_update:({x:x,y:y})=>{this.r=Math.hypot(y-this.y,x-this.x);this.w=Math.atan2(y-this.y,x-this.x);p2.x=this.x+this.r*Math.cos(this.w+this.dw);p2.y=this.y+this.r*Math.sin(this.w+this.dw)}},dw=this.dw,p2={x:this.x+this.r*Math.cos(this.w+this.dw),y:this.y+this.r*Math.sin(this.w+this.dw),_info:()=>`dw:${(this.dw/Math.PI*180).toFixed(1)}°`,_update:({x:x,y:y})=>{let lam=g2.toArc(g2.toPi2(Math.atan2(y-this.y,x-this.x)),g2.toPi2(this.w),dw);this.dw=lam*dw}};if(this.w===undefined)this.w=0;grp.lin({x1:()=>this.x,y1:()=>this.y,x2:()=>p1.x,y2:()=>p1.y,ld:[4,3],ls:"#666"}).lin({x1:()=>this.x,y1:()=>this.y,x2:()=>p2.x,y2:()=>p2.y,ld:[4,3],ls:"#666"}).handle(p0).handle(p1).handle(p2)}};g2.prototype.ply.prototype={get isSolid(){return this.closed&&this.fs&&this.fs!=="transparent"},get sh(){return this.state&g2.OVER?[0,0,5,"black"]:false},pointAt(loc){const t=loc==="beg"?0:loc==="end"?1:loc+0===loc?loc:.5,pitr=g2.pntItrOf(this.pts),pts=[],len=[];for(let itr=0;itr{const target=t*len.reduce((a,b)=>a+b);let tmp=0;for(let itr=0;itr=target){return{t2:1-(tmp-target)/len[itr],x:pts[itr].x,y:pts[itr].y,dx:next.x-pts[itr].x,dy:next.y-pts[itr].y}}}})();const len2=Math.hypot(dx,dy);return{x:x+dx*t2,y:y+dy*t2,dx:len2?dx/len2:1,dy:len2?dy/len2:0}},x:0,y:0,hitContour({x:x,y:y,eps:eps}){let p={x:x-this.x,y:y-this.y};return g2.isPntOnPly(p,this,eps)},hitInner({x:x,y:y,eps:eps}){let p={x:x-this.x,y:y-this.y};return g2.isPntInPly(p,this,eps)},drag({dx:dx,dy:dy}){this.x+=dx;this.y+=dy},handles(grp){let p,slf=this;for(let n=this._itr.len,i=0;i{return cmd.a&&"pointAt"in cmd.a});if(idx!==undefined){arguments[0]["_refelem"]=this.commands[idx];this.addCommand({c:"label",a:arguments[0]})}return this};g2.prototype.label.prototype={g2(){let label=g2();if(this._refelem){let{str:str,loc:loc,off:off,fs:fs,font:font,border:border,fs2:fs2}=this,p=this._refelem.a.pointAt(loc),tanlen=p.dx*p.dx||p.dy*p.dy;let h=parseInt(font||g2.defaultStyle.font),diag,phi,n;if(str[0]==="@"&&(s=this._refelem.a[str.substr(1)])!==undefined)str=""+(Number.isInteger(+s)?+s:Number(s).toFixed(Math.max(g2.symbol.labelSignificantDigits-Math.log10(s),0)))+(str.substr(1)==="angle"?"°":"");n=str.length;if(tanlen>Number.EPSILON){diag=Math.hypot(p.dx,n*p.dy);off=off===undefined?1:off;p.x+=tanlen*p.dy*(off+n*n*.8*h/2/diag*Math.sign(off));p.y+=tanlen*p.dx*(-off-h/2/diag*Math.sign(off))}fs=fs||"black";if(border)label.ell({x:p.x,y:p.y,rx:n*.8*h/2+2,ry:h/2+2,ls:fs||"black",fs:fs2||"#ffc"});label.txt({str:str,x:p.x,y:p.y,thal:"center",tval:"middle",fs:fs||"black",font:font})}return label}};g2.prototype.mark=function mark({mrk:mrk,loc:loc,dir:dir,fs:fs,ls:ls}){let idx=g2.getCmdIdx(this.commands,cmd=>{return cmd.a&&"pointAt"in cmd.a});if(idx!==undefined){arguments[0]["_refelem"]=this.commands[idx];this.addCommand({c:"mark",a:arguments[0]})}return this};g2.prototype.mark.prototype={markAt(elem,loc,mrk,dir,ls,fs){const p=elem.pointAt(loc),w=dir<0?Math.atan2(-p.dy,-p.dx):dir>0||dir===undefined?Math.atan2(p.dy,p.dx):0;return{grp:mrk,x:p.x,y:p.y,w:w,scl:elem.lw||1,ls:ls||elem.ls||"black",fs:fs||ls||elem.ls||"black"}},g2(){let{mrk:mrk,loc:loc,dir:dir,fs:fs,ls:ls}=this,elem=this._refelem.a,marks=g2();if(Array.isArray(loc))for(let l of loc)marks.use(this.markAt(elem,l,mrk,dir,ls,fs));else marks.use(this.markAt(elem,loc,mrk,dir,ls,fs));return marks}};g2.symbol=g2.symbol||{};g2.symbol.tick=g2().p().m({x:0,y:-2}).l({x:0,y:2}).stroke({lc:"round",lwnosc:true});g2.symbol.dot=g2().cir({x:0,y:0,r:2,ls:"transparent"});g2.symbol.sqr=g2().rec({x:-1.5,y:-1.5,b:3,h:3,ls:"transparent"});g2.symbol.nodcolor="#333";g2.symbol.nodfill="#dedede";g2.symbol.nodfill2="#aeaeae";g2.symbol.linkcolor="#666";g2.symbol.linkfill="rgba(225,225,225,0.75)";g2.symbol.dimcolor="darkslategray";g2.symbol.solid=[];g2.symbol.dash=[15,10];g2.symbol.dot=[4,4];g2.symbol.dashdot=[25,6.5,2,6.5];g2.symbol.labelSignificantDigits=3;"use strict"; -/** - * g2.mec (c) 2013-18 Stefan Goessner - * @author Stefan Goessner - * @license MIT License - * @requires g2.core.js - * @requires g2.ext.js - * @typedef {g2} - * @description Mechanical extensions. (Requires cartesian coordinates) - * @returns {g2} - */var g2=g2||{prototype:{}};g2.prototype.skip=function skip(tag){if(this.cmds.length)this.cmds[this.cmds.length-1].skip=tag;return this};g2.prototype.dim=function dim({}){return this.addCommand({c:"dim",a:arguments[0]})};g2.prototype.dim.prototype=g2.mixin({},g2.prototype.lin.prototype,{g2(){const args=Object.assign({lw:1,w:0,lc:"round",lj:"round",off:0,over:0,inside:true,fs:"#000"},this);const dx=args.x2-args.x1,dy=args.y2-args.y1,len=Math.hypot(dx,dy);args.fixed=args.fixed||len/2;const over=args.off>0?Math.abs(args.over):-Math.abs(args.over);const w=Math.atan2(dy,dx);return g2().beg({x:args.x1-args.off*Math.sin(w),y:args.y1+args.off*Math.cos(w),w:w}).vec({x1:args.inside?1:-25,y1:0,x2:0,y2:0,fixed:args.fixed,fs:args.fs,ls:args.ls,lw:args.lw}).vec({x1:args.inside?0:len+25,y1:0,x2:args.inside?len:len,y2:0,fixed:args.fixed,fs:args.fs,ls:args.ls,lw:args.lw}).ins(g=>{if(!args.inside){g.lin({x1:0,y1:0,x2:len,y2:0,fs:args.fs,ls:args.ls,lw:args.lw})}}).end().ins(g=>{if(!!args.off)g.lin({x1:args.x1,y1:args.y1,x2:args.x1-(over+args.off)*Math.sin(w),y2:args.y1+(over+args.off)*Math.cos(w),lw:args.lw/2,lw:args.lw/2,ls:args.ls,fs:args.fs}).lin({x1:args.x1+Math.cos(w)*len,y1:args.y1+Math.sin(w)*len,x2:args.x1+Math.cos(w)*len-(over+args.off)*Math.sin(w),y2:args.y1+Math.sin(w)*len+(over+args.off)*Math.cos(w),lw:args.lw/2,ls:args.ls,fs:args.fs})})}});g2.prototype.adim=function adim({}){return this.addCommand({c:"adim",a:arguments[0]})};g2.prototype.adim.prototype=g2.mixin({},g2.prototype.arc.prototype,{g2(){const args=Object.assign({lw:1,w:0,lc:"round",lj:"round",inside:true,fs:"#000"},this);return g2().beg({x:args.x,y:args.y,w:args.w}).arc({x:0,y:0,r:args.r,w:0,dw:args.dw,ls:args.ls,lw:args.lw}).vec({x1:args.inside?args.r-.15:args.r-3.708,y1:args.inside?1:24.723,x2:args.r,y2:0,fs:args.fs,ls:args.ls,lw:args.lw,fixed:30}).lin({x1:args.r-3.5,y1:0,x2:args.r+3.5,y2:0,fs:args.fs,ls:args.ls,lw:args.lw}).end().beg({x:args.x,y:args.y,w:args.w+args.dw}).vec({x1:args.inside?args.r-.15:args.r-3.708,y1:args.inside?-1:-24.723,x2:args.r,y2:0,fs:args.fs,ls:args.ls,lw:args.lw,fixed:30}).lin({x1:args.r-3.5,y1:0,x2:args.r+3.5,y2:0,fs:args.fs,ls:args.ls,lw:args.lw}).end()}});g2.prototype.vec=function vec({}){return this.addCommand({c:"vec",a:arguments[0]})};g2.prototype.vec.prototype=g2.mixin({},g2.prototype.lin.prototype,{g2(){const args=Object.assign({ls:"#000",fs:"@ls",lc:"round",lj:"round",lw:1,fixed:undefined},this);const dx=args.x2-args.x1,dy=args.y2-args.y1,r=Math.hypot(dx,dy);let z=args.head||2+args.lw;const z2=(args.fixed||r)/10;z=z>z2?z2:z;return g2().beg(Object.assign({},args,{x:args.x1,y:args.y1,w:Math.atan2(dy,dx)})).p().m({x:0,y:0}).l({x:r,y:0}).stroke({ls:args.ls}).p().m({x:r,y:0}).l({x:r-5*z,y:z}).a({dw:-Math.PI/3,x:r-5*z,y:-z}).z().drw({ls:args.ls,fs:args.fs}).end()}});g2.prototype.avec=function vec({}){return this.addCommand({c:"avec",a:arguments[0]})};g2.prototype.avec.prototype=g2.mixin({},g2.prototype.arc.prototype,{g2(){const args=Object.assign({ls:"#000",fs:"@ls",lc:"round",lj:"round",fixed:30,lw:1,dw:2*Math.PI},this);const z=args.fixed/2;return g2().beg({x:args.x,y:args.y,w:args.w+args.dw}).arc({x:0,y:0,r:args.r,w:-args.dw,dw:args.dw,ls:args.ls,lw:args.lw}).vec({x1:args.r*Math.sqrt(1-(z/args.r)**2),x2:args.r,y1:-1*Math.sign(args.dw)*z,y2:0,fixed:args.fixed}).end()}});g2.prototype.slider=function(){return this.addCommand({c:"slider",a:arguments[0]})};g2.prototype.slider.prototype=g2.mixin({},g2.prototype.rec.prototype,{g2(){const args=Object.assign({b:32,h:16,fs:"@linkfill"},this);return g2().beg({x:args.x,y:args.y,w:args.w,fs:args.fs}).rec({x:-args.b/2,y:-args.h/2,b:args.b,h:args.h}).end()}});g2.prototype.spring=function(){return this.addCommand({c:"spring",a:arguments[0]})};g2.prototype.spring.prototype=g2.mixin({},g2.prototype.lin.prototype,{g2(){const args=Object.assign({h:16},this);const len=Math.hypot(args.x2-args.x1,args.y2-args.y1);const xm=(args.x2+args.x1)/2;const ym=(args.y2+args.y1)/2;const h=args.h;const ux=(args.x2-args.x1)/len;const uy=(args.y2-args.y1)/len;return g2().p().m({x:args.x1,y:args.y1}).l({x:xm-ux*h/2,y:ym-uy*h/2}).l({x:xm+(-ux/6+uy/2)*h,y:ym+(-uy/6-ux/2)*h}).l({x:xm+(ux/6-uy/2)*h,y:ym+(uy/6+ux/2)*h}).l({x:xm+ux*h/2,y:ym+uy*h/2}).l({x:args.x2,y:args.y2}).stroke(Object.assign({},{ls:"@nodcolor"},this,{fs:"transparent",lc:"round",lj:"round"}))}});g2.prototype.damper=function(){return this.addCommand({c:"damper",a:arguments[0]})};g2.prototype.damper.prototype=g2.mixin({},g2.prototype.lin.prototype,{g2(){const args=Object.assign({h:16},this);const len=Math.hypot(args.x2-args.x1,args.y2-args.y1);const xm=(args.x2+args.x1)/2;const ym=(args.y2+args.y1)/2;const h=args.h;const ux=(args.x2-args.x1)/len;const uy=(args.y2-args.y1)/len;return g2().p().m({x:args.x1,y:args.y1}).l({x:xm-ux*h/2,y:ym-uy*h/2}).m({x:xm+(ux-uy)*h/2,y:ym+(uy+ux)*h/2}).l({x:xm+(-ux-uy)*h/2,y:ym+(-uy+ux)*h/2}).l({x:xm+(-ux+uy)*h/2,y:ym+(-uy-ux)*h/2}).l({x:xm+(ux+uy)*h/2,y:ym+(uy-ux)*h/2}).m({x:xm,y:ym}).l({x:args.x2,y:args.y2}).stroke(Object.assign({},{ls:"@nodcolor"},this,{fs:"transparent",lc:"round",lj:"round"}))}});g2.prototype.link=function(){return this.addCommand({c:"link",a:arguments[0]})};g2.prototype.link.prototype=g2.mixin({},g2.prototype.ply.prototype,{g2(){const args=Object.assign({ls:"@linkcolor",fs:"transparent"},this);return g2().ply(Object.assign({},this,{closed:true,ls:args.ls,fs:args.fs,lw:7,lc:"round",lj:"round"}))}});g2.prototype.link2=function(){return this.addCommand({c:"link2",a:arguments[0]})};g2.prototype.link2.prototype=g2.mixin({},g2.prototype.ply.prototype,{g2(){return g2().ply(Object.assign({closed:true,ls:"@nodcolor",fs:"transparent",lw:7,lc:"round",lj:"round"},this)).ply(Object.assign({closed:true,ls:"@nodfill2",fs:"transparent",lw:4.5,lc:"round",lj:"round"},this)).ply(Object.assign({closed:true,ls:"@nodfill",fs:"transparent",lw:2,lc:"round",lj:"round"},this))}});g2.prototype.beam=function(){return this.addCommand({c:"beam",a:arguments[0]})};g2.prototype.beam.prototype=g2.mixin({},g2.prototype.ply.prototype,{g2(){return g2().ply(Object.assign({closed:false,ls:"@linkcolor",fs:"transparent",lw:7,lc:"round",lj:"round"},this))}});g2.prototype.beam2=function(){return this.addCommand({c:"beam2",a:arguments[0]})};g2.prototype.beam2.prototype=g2.mixin({},g2.prototype.ply.prototype,{g2(){return g2().ply(Object.assign({closed:false,ls:"@nodcolor",fs:"transparent",lw:7,lc:"round",lj:"round"},this)).ply(Object.assign({closed:false,ls:"@nodfill2",fs:"transparent",lw:4.5,lc:"round",lj:"round"},this)).ply(Object.assign({closed:false,ls:"@nodfill",fs:"transparent",lw:2,lc:"round",lj:"round"},this))}});g2.prototype.bar=function(){return this.addCommand({c:"bar",a:arguments[0]})};g2.prototype.bar.prototype=g2.mixin({},g2.prototype.lin.prototype,{g2(){return g2().lin(Object.assign({ls:"@linkcolor",lw:6,lc:"round"},this))}});g2.prototype.bar2=function(){return this.addCommand({c:"bar2",a:arguments[0]})};g2.prototype.bar2.prototype=g2.mixin({},g2.prototype.lin.prototype,{g2(){const args=Object.assign({},this);return g2().lin({x1:args.x1,y1:args.y1,x2:args.x2,y2:args.y2,ls:"@nodcolor",lw:7,lc:"round"}).lin({x1:args.x1,y1:args.y1,x2:args.x2,y2:args.y2,ls:"@nodfill2",lw:4.5,lc:"round"}).lin({x1:args.x1,y1:args.y1,x2:args.x2,y2:args.y2,ls:"@nodfill",lw:2,lc:"round"})}});g2.prototype.pulley=function(){return this.addCommand({c:"pulley",a:arguments[0]})};g2.prototype.pulley.prototype=g2.mixin({},g2.prototype.cir.prototype,{g2(){const args=Object.assign({},this);return g2().beg({x:args.x,y:args.y,w:args.w}).cir({x:0,y:0,r:args.r,ls:"@nodcolor",fs:"#e6e6e6",lw:1}).cir({x:0,y:0,r:args.r-5,ls:"@nodcolor",fs:"#e6e6e6",lw:1}).cir({x:0,y:0,r:args.r-6,ls:"#8e8e8e",fs:"transparent",lw:2}).cir({x:0,y:0,r:args.r-8,ls:"#aeaeae",fs:"transparent",lw:2}).cir({x:0,y:0,r:args.r-10,ls:"#cecece",fs:"transparent",lw:2}).end()}});g2.prototype.pulley2=function(){return this.addCommand({c:"pulley2",a:arguments[0]})};g2.prototype.pulley2.prototype=g2.mixin({},g2.prototype.cir.prototype,{g2(){const args=Object.assign({},this);return g2().beg({x:args.x,y:args.y,w:args.w}).bar2({x1:0,y1:-args.r+4,x2:0,y2:args.r-4}).bar2({x1:-args.r+4,y1:0,x2:args.r-4,y2:0}).cir({x:0,y:0,r:args.r-2.5,ls:"#e6e6e6",fs:"transparent",lw:5}).cir({x:0,y:0,r:args.r,ls:"@nodcolor",fs:"transparent",lw:1}).cir({x:0,y:0,r:args.r-5,ls:"@nodcolor",fs:"transparent",lw:1}).end()}});g2.prototype.rope=function(){return this.addCommand({c:"rope",a:arguments[0]})};g2.prototype.rope.prototype=g2.mixin({},g2.prototype.lin.prototype,{g2(){const args=Object.assign({w:0},this);let x1="p1"in args?args.p1.x:"x1"in args?args.x1:"x"in args?args.x:0;let y1="p1"in args?args.p1.y:"y1"in args?args.y1:"y"in args?args.y:0;let x2="p2"in args?args.p2.x:"x2"in args?args.x2:"dx"in args?x1+args.dx:"r"in args?x1+args.r*Math.cos(args.w):x1+10;let y2="p2"in args?args.p2.y:"y2"in args?args.y2:"dy"in args?y1+args.dy:"r"in args?y1+args.r*Math.sin(args.w):y1;let Rmin=10;let R1=args.r1>Rmin?args.r1-2.5:args.r1<-Rmin?args.r1+2.5:0;let R2=args.r2>Rmin?args.r2-2.5:args.r2=pi.y||y>=pj.y)&&(y<=pi.y||y<=pj.y)&&(x<=pi.x||x<=pj.x)&&pi.y!==pj.y&&(pi.x===pj.x||x<=pj.x+(y-pj.y)*(pi.x-pj.x)/(pi.y-pj.y))){match++}}return match%2!=0}return g2().ply({pts:args.pts,closed:true,ls:"transparent",fs:"@linkfill"}).ins(g=>{for(const pts of startLoc){let dist=10*args.lw||10;const{x:x,y:y}=args.pointAt(pts),t={x:x+Math.cos(args.w)*dist,y:y+Math.sin(args.w)*dist};if(isPntInPly(t,{pts:arr})){while(isPntInPly(t,{pts:arr})){dist++;t.x=x+Math.cos(args.w)*dist,t.y=y+Math.sin(args.w)*dist}g.vec({x1:x,y1:y,x2:t.x,y2:t.y,ls:args.ls||"darkred",lw:args.lw||1})}}})}});g2.prototype.pol=function(){return this.addCommand({c:"pol",a:arguments[0]||{}})};g2.prototype.pol.prototype=g2.mixin({},g2.prototype.use.prototype,{g2(){const args=Object.assign({x:0,y:0,scl:1,w:0},this);return g2().beg({x:args.x,y:args.y,scl:args.scl,w:args.w}).cir({r:6,fs:"@nodfill"}).cir({r:2.5,fs:"@ls",ls:"transparent"}).end()}});g2.prototype.gnd=function(){return this.addCommand({c:"gnd",a:arguments[0]||{}})};g2.prototype.gnd.prototype=g2.mixin({},g2.prototype.use.prototype,{g2(){const args=Object.assign({x:0,y:0,scl:1,w:0},this);return g2().beg({x:args.x,y:args.y,scl:args.scl,w:args.w}).cir({x:0,y:0,r:6,ls:"@nodcolor",fs:"@nodfill",lwnosc:true}).p().m({x:0,y:6}).a({dw:Math.PI/2,x:-6,y:0}).l({x:6,y:0}).a({dw:-Math.PI/2,x:0,y:-6}).z().fill({fs:"@nodcolor"}).end()}});g2.prototype.nod=function(){return this.addCommand({c:"nod",a:arguments[0]||{}})};g2.prototype.nod.prototype=g2.mixin({},g2.prototype.use.prototype,{g2(){const args=Object.assign({x:0,y:0,scl:1,w:0},this);return g2().beg({x:args.x,y:args.y,scl:args.scl,w:args.w}).cir({r:4,ls:"@nodcolor",fs:"@nodfill",lwnosc:true}).end()}});g2.prototype.dblnod=function(){return this.addCommand({c:"dblnod",a:arguments[0]||{}})};g2.prototype.dblnod.prototype=g2.mixin({},g2.prototype.use.prototype,{g2(){const args=Object.assign({x:0,y:0,scl:1,w:0},this);return g2().beg({x:args.x,y:args.y,scl:args.scl,w:args.w}).cir({r:6,ls:"@nodcolor",fs:"@nodfill"}).cir({r:3,ls:"@nodcolor",fs:"@nodfill2",lwnosc:true}).end()}});g2.prototype.nodfix=function(){return this.addCommand({c:"nodfix",a:arguments[0]||{}})};g2.prototype.nodfix.prototype=g2.mixin({},g2.prototype.use.prototype,{g2(){const args=Object.assign({x:0,y:0,scl:1,w:0},this);return g2().beg({x:args.x,y:args.y,scl:args.scl,w:args.w}).p().m({x:-8,y:-12}).l({x:0,y:0}).l({x:8,y:-12}).drw({ls:"@nodcolor",fs:"@nodfill2"}).cir({x:0,y:0,r:4,ls:"@nodcolor",fs:"@nodfill"}).end()}});g2.prototype.nodflt=function(){return this.addCommand({c:"nodflt",a:arguments[0]||{}})};g2.prototype.nodflt.prototype=g2.mixin({},g2.prototype.use.prototype,{g2(){const args=Object.assign({x:0,y:0,scl:1,w:0},this);return g2().beg({x:args.x,y:args.y,scl:args.scl,w:args.w}).p().m({x:-8,y:-12}).l({x:0,y:0}).l({x:8,y:-12}).drw({ls:"@nodcolor",fs:"@nodfill2"}).cir({x:0,y:0,r:4,ls:"@nodcolor",fs:"@nodfill"}).lin({x1:-9,y1:-19,x2:9,y2:-19,ls:"@nodfill2",lw:5,lwnosc:false}).lin({x1:-9,y1:-15.5,x2:9,y2:-15.5,ls:"@nodcolor",lw:2,lwnosc:false}).end()}});g2.prototype.origin=function(){return this.addCommand({c:"origin",a:arguments[0]||{}})};g2.prototype.origin.prototype=g2.mixin({},g2.prototype.use.prototype,{g2(){const args=Object.assign({x:0,y:0,scl:1,w:0,z:3.5},this);return g2().beg({x:args.x,y:args.y,scl:args.scl,w:args.w,lc:"round",lj:"round",fs:"#ccc"}).vec({x1:0,y1:0,x2:10*args.z,y2:0,lw:.8,fs:"#ccc"}).vec({x1:0,y1:0,x2:0,y2:10*args.z,lw:.8,fs:"#ccc"}).cir({x:0,y:0,r:2.5,fs:"#ccc"}).end()}});g2.State=g2.State||{};g2.State.nodcolor="#333";g2.State.nodfill="#dedede";g2.State.nodfill2="#aeaeae";g2.State.linkcolor="#666";g2.State.linkfill="rgba(225,225,225,0.75)";g2.State.dimcolor="darkslategray";g2.State.solid=[];g2.State.dash=[15,10];g2.State.dot=[4,4];g2.State.dashdot=[25,6.5,2,6.5];g2.State.labelOffset=5;g2.State.labelSignificantDigits=3;"use strict"; -/** - * g2.chart (c) 2015-18 Stefan Goessner - * @author Stefan Goessner - * @license MIT License - * @requires g2.core.js - * @requires g2.ext.js - * @typedef g2 - * @returns {object} chart - * @param {object} args - Chart arguments object or - * @property {float} x - x-position of lower left corner of chart rectangle. - * @property {float} y - y-position of lower left corner of chart rectangle. - * @property {float} [b=150] - width of chart rectangle. - * @property {float} [h=100] - height of chart rectangle. - * @property {string} [ls] - border color. - * @property {string} [fs] - fill color. - * @property {(string|object)} [title] - chart title. - * @property {string} [title.text] - chart title text string. - * @property {float} [title.offset=0] - chart title vertical offset. - * @property {object} [title.style] - chart title style. - * @property {string} [title.style.font=14px serif] - chart title font. - * @property {string} [title.style.thal=center] - chart title horizontal align. - * @property {string} [title.style.tval=bottom] - chart title vertical align. - * @property {array} [funcs=[]] - array of dataset `data` and/or function `fn` objects. - * @property {object} [funcs[item]] - dataset or function object. - * @property {array} [funcs[item].data] - data points as flat array `[x,y,..]`, array of point arrays `[[x,y],..]` or array of point objects `[{x,y},..]`. - * @property {function} [funcs[item].fn] - function `y = f(x)` recieving x-value returning y-value. - * @property {float} [funcs[item].dx] - x increment to apply to function `fn`. Ignored with data points. - * @property {boolean} [funcs[item].fill] - fill region between function graph and x-origin line. - * @property {boolean} [funcs[item].dots] - place circular dots at data points (Avoid with `fn`s). - * @property {boolean|object} [xaxis=false] - x-axis. - * @property {boolean|object} [xaxis.grid=false] - x-axis grid lines. - * @property {string} [xaxis.grid.ls] - x-axis grid line style (color). - * @property {string} [xaxis.grid.lw] - x-axis grid line width. - * @property {string} [xaxis.grid.ld] - x-axis grid line dash style. - * @property {boolean} [xaxis.line=true] - display x-axis base line. - * @property {boolean} [xaxis.origin=false] - display x-axis origin line. - * @property {boolean|object} [yaxis=false] - y-axis. - * @property {boolean|object} [yaxis.grid=false] - y-axis grid lines. - * @property {string} [yaxis.grid.ls] - y-axis grid line style color. - * @property {string} [yaxis.grid.lw] - y-axis grid line width. - * @property {string} [yaxis.grid.ld] - y-axis grid line dash style. - * @property {boolean} [yaxis.line=true] - display y-axis base line. - * @property {boolean} [yaxis.origin=false] - display y-axis origin line. - * @property {float} [xmin] - minimal x-axis value. If not given it is calculated from chart data values. - * @property {float} [xmax] - maximal x-axis value. If not given it is calculated from chart data values. - * @property {float} [ymin] - minimal y-axis value. If not given it is calculated from chart data values. - * @property {float} [ymax] - maximal y-axis value. If not given it is calculated from chart data values. - */g2.prototype.chart=function chart({x:x,y:y,b:b,h:h,style:style,title:title,funcs:funcs,xaxis:xaxis,xmin:xmin,xmax:xmax,yaxis:yaxis,ymin:ymin,ymax:ymax}){return this.addCommand({c:"chart",a:arguments[0]})};g2.prototype.chart.prototype={g2(){const g=g2(),funcs=this.get("funcs"),title=this.title&&this.get("title");if(!this.b)this.b=this.defaults.b;if(!this.h)this.h=this.defaults.h;if(funcs&&funcs.length){const tmp=[this.xmin===undefined,this.xmax===undefined,this.ymin===undefined,this.ymax===undefined];funcs.forEach(f=>this.initFunc(f,...tmp))}this.xAxis=this.autoAxis(this.get("xmin"),this.get("xmax"),0,this.b);this.yAxis=this.autoAxis(this.get("ymin"),this.get("ymax"),0,this.h);g.rec({x:this.x,y:this.y,b:this.b,h:this.h,fs:this.get("fs"),ls:this.get("ls")});g.beg(Object.assign({x:this.x,y:this.y,lw:1},this.defaults.style,this.style));if(title)g.txt(Object.assign({str:this.title&&this.title.text||this.title,x:this.get("b")/2,y:this.get("h")+this.get("title","offset"),w:0},this.defaults.title.style,this.title&&this.title.style||{}));if(this.xaxis)this.drawXAxis(g);if(this.yaxis)this.drawYAxis(g);g.end();if(funcs)funcs.forEach((fnc,i)=>{this.drawFunc(g,fnc,this.defaults.colors[i%this.defaults.colors.length])});return g},initFunc(fn,setXmin,setXmax,setYmin,setYmax){let itr;if(fn.data&&fn.data.length){itr=fn.itr=g2.pntItrOf(fn.data)}else if(fn.fn&&fn.dx){const xmin=+this.xmin||this.defaults.xmin;const xmax=+this.xmax||this.defaults.xmax;itr=fn.itr=(i=>{let x=xmin+i*fn.dx;return{x:x,y:fn.fn(x)}});itr.len=(xmax-xmin)/fn.dx+1}if(itr&&(setXmin||setXmax||setYmin||setYmax)){const xarr=[];const yarr=[];for(let i=0;ithis.xmax)this.xmax=xmax}if(setYmin){const ymin=Math.min(...yarr);if(!this.ymin||yminthis.ymax)this.ymax=ymax}if(fn.color&&typeof fn.color==="number")fn.color=this.defaults.colors[fn.color%this.defaults.colors.length]}},autoAxis(zmin,zmax,tmin,tmax){let base=2,exp=1,eps=Math.sqrt(Number.EPSILON),Dz=zmax-zmin||1,Dt=tmax-tmin||1,scl=Dz>eps?Dt/Dz:1,dz=base*Math.pow(10,exp),dt=Math.floor(scl*dz),N,dt01,i0,j0,jth,t0,res;while(dt<14||dt>35){if(dt<14){if(base==1)base=2;else if(base==2)base=5;else if(base==5){base=1;exp++}}else{if(base==1){base=5;exp--}else if(base==2)base=1;else if(base==5)base=2}dz=base*Math.pow(10,exp);dt=scl*dz}i0=(scl*Math.abs(zmin)+eps/2)%dt9?5:2;return{zmin:zmin,zmax:zmax,base:base,exp:exp,scl:scl,dt:dt,dz:dz,N:N,t0:t0,z0:z0,i0:i0,j0:j0,jth:jth,itr(i){return{t:this.t0+i*this.dt,z:parseFloat((this.z0+i*this.dz).toFixed(Math.abs(this.exp))),maj:(this.j0-this.i0+i)%this.jth===0}}}},drawXAxis(g){let tick,showgrid=this.xaxis&&this.xaxis.grid,gridstyle=showgrid&&Object.assign({},this.defaults.xaxis.grid,this.xaxis.grid),showaxis=this.xaxis||this.xAxis,axisstyle=showaxis&&Object.assign({},this.defaults.xaxis.style,this.defaults.xaxis.labels.style,this.xaxis&&this.xaxis.style||{}),showline=showaxis&&this.get("xaxis","line"),showlabels=this.xAxis&&showaxis&&this.get("xaxis","labels"),showticks=this.xAxis&&showaxis&&this.get("xaxis","ticks"),ticklen=showticks?this.get("xaxis","ticks","len"):0,showorigin=showaxis&&this.get("xaxis","origin"),title=this.xaxis&&(this.get("xaxis","title","text")||this.xaxis.title)||"";g.beg(axisstyle);for(let i=0;i=0)g.lin({x1:-this.xAxis.zmin*this.xAxis.scl,y1:0,x2:-this.xAxis.zmin*this.xAxis.scl,y2:this.h});if(title)g.txt(Object.assign({str:title.text||title,x:this.b/2,y:-(this.get("xaxis","title","offset")+(showticks&&this.get("xaxis","ticks","len")||0)+(showlabels&&this.get("xaxis","labels","offset")||0)+(showlabels&&parseFloat(this.get("xaxis","labels","style","font"))||0)),w:0},this.get("xaxis","title","style")));g.end()},drawYAxis(g){let tick,showgrid=this.yaxis&&this.yaxis.grid,gridstyle=showgrid&&Object.assign({},this.defaults.yaxis.grid,this.yaxis.grid),showaxis=this.yaxis||this.yAxis,axisstyle=showaxis&&Object.assign({},this.defaults.yaxis.style,this.defaults.yaxis.labels.style,this.yaxis&&this.yaxis.style||{}),showline=showaxis&&this.get("yaxis","line"),showlabels=this.yAxis&&showaxis&&this.get("yaxis","labels"),showticks=this.yAxis&&showaxis&&this.get("yaxis","ticks"),ticklen=showticks?this.get("yaxis","ticks","len"):0,showorigin=showaxis&&this.get("yaxis","origin"),title=this.yaxis&&(this.get("yaxis","title","text")||this.yaxis.title)||"";g.beg(axisstyle);for(let i=0;i=0)g.lin({x1:0,y1:-this.yAxis.zmin*this.yAxis.scl,x2:this.b,y2:-this.yAxis.zmin*this.yAxis.scl});if(title)g.txt(Object.assign({str:title.text||title,x:-(this.get("yaxis","title","offset")+(showticks&&this.get("yaxis","ticks","len")||0)+(showlabels&&this.get("yaxis","labels","offset")||0)+(showlabels&&parseFloat(this.get("yaxis","labels","style","font"))||0)),y:this.h/2,w:Math.PI/2},this.get("yaxis","title","style")));g.end()},drawFunc(g,fn,defaultcolor){let itr=fn.itr;if(itr){let fill=fn.fill||fn.style&&fn.style.fs&&fn.style.fs!=="transparent",color=fn.color=fn.color||fn.style&&fn.style.ls||defaultcolor,plydata=[],args=Object.assign({pts:plydata,closed:false,ls:color,fs:fill?g2.color.rgbaStr(color,.125):"transparent",lw:1},fn.style);if(fill)plydata.push(this.pntOf({x:itr(0).x,y:0}));for(let i=0,n=itr.len;i 0 ? Math.abs(args.over) : -Math.abs(args.over); - const w = Math.atan2(dy,dx); - return g2() - .beg({x:args.x1 - args.off*Math.sin(w),y:args.y1 + args.off*Math.cos(w),w:w}) - .vec({ - x1:args.inside ? 1 : -25, - y1:0,x2:0,y2:0, - fixed:args.fixed, - fs:args.fs,ls:args.ls,lw:args.lw - }) - .vec({ - x1:args.inside ? 0 : len + 25,y1:0, - x2:args.inside ? len : len,y2:0, - fixed:args.fixed, - fs:args.fs,ls:args.ls,lw:args.lw - }) - .ins(g => {if(!args.inside) - {g.lin({x1:0,y1:0,x2:len,y2:0,fs:args.fs,ls:args.ls,lw:args.lw})} - }) - .end() - .ins(g => {if(!!args.off) - g.lin({ - x1:args.x1,y1:args.y1, - x2:args.x1 - (over + args.off)*Math.sin(w), - y2:args.y1 + (over + args.off)*Math.cos(w), - lw:args.lw/2,lw:args.lw/2,ls:args.ls,fs:args.fs}) - .lin({ - x1:args.x1+Math.cos(w)*len,y1:args.y1+Math.sin(w)*len, - x2:args.x1+Math.cos(w)*len-(over + args.off)*Math.sin(w), - y2:args.y1+Math.sin(w)*len+(over + args.off)*Math.cos(w), - lw:args.lw/2,ls:args.ls,fs:args.fs - }) - }); - } -}); - -/** - * Angular dimension - * @method - * @returns {object} g2 - * @param {object} - angular dimension arguments. - * @property {number} x - start x coordinate. - * @property {number} y - start y coordinate. - * @property {number} r - radius - * @property {number} [w=0] - start angle (in radian). - * @property {number} [dw=Math.PI/2] - angular range in radian. In case of positive values it is running counterclockwise with - * right handed (cartesian) coordinate system. - * @property {boolean} [inside=true] - draw dimension arrows between or outside of ticks. - * @example - * g2().adim({x:100,y:70,r:50,w:pi/3,dw:4*pi/3}) - */ -g2.prototype.adim = function adim({}) { return this.addCommand({c:'adim',a:arguments[0]}); } -g2.prototype.adim.prototype = g2.mixin({}, g2.prototype.arc.prototype, { - g2() { - const args = Object.assign({lw:1,w:0,lc:'round',lj:'round',inside:true,fs:"#000"}, this); - return g2() - .beg({x:args.x,y:args.y,w:args.w}) - .arc({x:0,y:0,r:args.r,w:0,dw:args.dw,ls:args.ls,lw:args.lw}) - .vec({ - x1:args.inside ? args.r-.15:args.r-3.708, - y1:args.inside?1:24.723,x2:args.r,y2:0,fs:args.fs,ls:args.ls,lw:args.lw,fixed:30}) - .lin({x1:args.r-3.5,y1:0,x2:args.r+3.5,y2:0,fs:args.fs,ls:args.ls,lw:args.lw}) - .end() - .beg({x:args.x,y:args.y,w:args.w+args.dw}) - .vec({ - x1:args.inside ? args.r-.15:args.r-3.708, - y1:args.inside?-1:-24.723,x2:args.r,y2:0,fs:args.fs,ls:args.ls,lw:args.lw,fixed:30}) - .lin({x1:args.r-3.5,y1:0,x2:args.r+3.5,y2:0,fs:args.fs,ls:args.ls,lw:args.lw}) - .end(); - } -}); - -/** - * Draw vector arrow. - * @method - * @returns {object} g2 - * @param {object} - vector arguments object. - * @property {number} x1 - start x coordinate. - * @property {number} y1 - start y coordinate. - * @property {number} x2 - end x coordinate. - * @property {number} y2 - end y coordinate. - * @example - * g2().vec({x1:50,y1:20,x2:250,y2:120}) - */ -g2.prototype.vec = function vec({}) { return this.addCommand({c:'vec',a:arguments[0]}); } -g2.prototype.vec.prototype = g2.mixin({},g2.prototype.lin.prototype,{ - g2() { - const args = Object.assign({ls:"#000",fs:"@ls",lc:'round',lj:'round',lw:1,fixed:undefined}, this); - const dx = args.x2-args.x1, dy = args.y2-args.y1, r = Math.hypot(dx,dy); - let z = args.head || 2+(args.lw); - const z2 = (args.fixed || r) / 10; - z = z > z2 ? z2 : z; - return g2() - .beg(Object.assign({}, args, {x:args.x1,y:args.y1,w:Math.atan2(dy,dx)})) - .p().m({x:0,y:0}) - .l({x:r,y:0}) - .stroke({ls:args.ls}) - .p().m({x:r,y:0}) - .l({x:r-5*z,y:z}) - .a({dw:-Math.PI/3,x:r-5*z,y:-z}) - .z() - .drw({ls:args.ls,fs:args.fs}) - .end(); - } -}) -/** - * Draw vector with an angle - * @method - * @returns {object} g2 - * @param {object} - angle vector arguments object - * @property {number} x - x-value center. - * @property {number} y - y-value center. - * @property {number} r - radius. - * @property {number} [w=0] - start angle (in radian). - * @property {number} [dw=2*pi] - angular range in Radians. - * @example - * g2().avec({x:300,y:400,r:390,w:-Math.PI/4,dw:-Math.PI/2}) - * .exe(ctx); - */ -g2.prototype.avec = function vec({}) { return this.addCommand({c: 'avec',a:arguments[0]}); } -g2.prototype.avec.prototype = g2.mixin({}, g2.prototype.arc.prototype, { - g2() { - const args = Object.assign({ls:"#000", fs:"@ls", lc:'round', lj:'round', fixed: 30, lw:1, dw: 2*Math.PI}, this); - const z = args.fixed / 2; - return g2() - .beg({x:args.x, y:args.y,w:args.w+args.dw}) - .arc({x:0, y:0, r:args.r, w:-args.dw, dw:args.dw, ls:args.ls, lw:args.lw}) - .vec({x1:args.r*Math.sqrt(1-(z/args.r)**2), x2: args.r, - y1:-1*Math.sign(args.dw)*z, y2:0, - fixed: args.fixed}) - .end() - } -}) +var g2 = g2 || { prototype:{} }; // for jsdoc only ... /** * Draw slider. @@ -193,7 +28,7 @@ g2.prototype.avec.prototype = g2.mixin({}, g2.prototype.arc.prototype, { * g2().slider({x:150,y:75,w:Math.PI/4,b:64,h:32}) */ g2.prototype.slider = function () { return this.addCommand({c:'slider',a:arguments[0]}); } -g2.prototype.slider.prototype = g2.mixin({},g2.prototype.rec.prototype,{ +g2.prototype.slider.prototype = g2.mix(g2.prototype.rec.prototype,{ g2() { const args = Object.assign({b:32,h:16,fs:'@linkfill'}, this); return g2() @@ -217,7 +52,7 @@ g2.prototype.slider.prototype = g2.mixin({},g2.prototype.rec.prototype,{ * g2().spring({x1:50,y1:100,x2:200,y2:75}) */ g2.prototype.spring = function () { return this.addCommand({c:'spring',a:arguments[0]}); } -g2.prototype.spring.prototype = g2.mixin({}, g2.prototype.lin.prototype,{ +g2.prototype.spring.prototype = g2.mix(g2.prototype.lin.prototype,{ g2() { const args = Object.assign({h:16}, this); const len = Math.hypot(args.x2-args.x1, args.y2-args.y1); @@ -251,7 +86,7 @@ g2.prototype.spring.prototype = g2.mixin({}, g2.prototype.lin.prototype,{ * * g2().damper({x1:60,y1:120,x2:200,y2:75}) */ g2.prototype.damper = function () { return this.addCommand({c:'damper',a:arguments[0]}); } -g2.prototype.damper.prototype = g2.mixin({}, g2.prototype.lin.prototype,{ +g2.prototype.damper.prototype = g2.mix(g2.prototype.lin.prototype,{ g2() { const args = Object.assign({h:16}, this); const len = Math.hypot(args.x2-args.x1, args.y2-args.y1); @@ -292,7 +127,7 @@ g2.prototype.damper.prototype = g2.mixin({}, g2.prototype.lin.prototype,{ * .link({pts:[A,B,E,A,D,C]}) */ g2.prototype.link = function () { return this.addCommand({c:'link',a:arguments[0]}); } -g2.prototype.link.prototype = g2.mixin({}, g2.prototype.ply.prototype,{ +g2.prototype.link.prototype = g2.mix(g2.prototype.ply.prototype,{ g2() { const args = Object.assign({ls:'@linkcolor',fs:'transparent'}, this); return g2().ply(Object.assign({}, this, {closed:true,ls:args.ls,fs:args.fs,lw:7,lc:'round',lj:'round'})); @@ -317,7 +152,7 @@ g2.prototype.link.prototype = g2.mixin({}, g2.prototype.ply.prototype,{ * .link({pts:[A,B,E,A,D,C]}) */ g2.prototype.link2 = function () { return this.addCommand({c:'link2',a:arguments[0]}); } -g2.prototype.link2.prototype = g2.mixin({}, g2.prototype.ply.prototype,{ +g2.prototype.link2.prototype = g2.mix(g2.prototype.ply.prototype,{ g2() { return g2() .ply(Object.assign({closed:true,ls:'@nodcolor',fs:'transparent',lw:7,lc:'round',lj:'round'},this)) @@ -340,7 +175,7 @@ g2.prototype.link2.prototype = g2.mixin({}, g2.prototype.ply.prototype,{ * .beam({pts:[[200,125][50,125][50,50][200,50]]}) */ g2.prototype.beam = function () { return this.addCommand({c:'beam',a:arguments[0]}); } -g2.prototype.beam.prototype = g2.mixin({}, g2.prototype.ply.prototype,{ +g2.prototype.beam.prototype = g2.mix(g2.prototype.ply.prototype,{ g2() { return g2().ply(Object.assign({closed:false,ls:'@linkcolor',fs:'transparent',lw:7,lc:'round',lj:'round'},this)); } @@ -360,7 +195,7 @@ g2.prototype.beam.prototype = g2.mixin({}, g2.prototype.ply.prototype,{ * .beam2({pts:[[200,125][50,125][50,50][200,50]]}) */ g2.prototype.beam2 = function () { return this.addCommand({c:'beam2',a:arguments[0]}); } -g2.prototype.beam2.prototype = g2.mixin({}, g2.prototype.ply.prototype,{ +g2.prototype.beam2.prototype = g2.mix(g2.prototype.ply.prototype,{ g2() { return g2() .ply(Object.assign({closed:false,ls:'@nodcolor',fs:'transparent',lw:7,lc:'round',lj:'round'},this)) @@ -382,7 +217,7 @@ g2.prototype.beam2.prototype = g2.mixin({}, g2.prototype.ply.prototype,{ * g2().bar({x1:50,y1:20,x2:250,y2:120}) */ g2.prototype.bar = function () { return this.addCommand({c:'bar',a:arguments[0]}); } -g2.prototype.bar.prototype = g2.mixin({}, g2.prototype.lin.prototype,{ +g2.prototype.bar.prototype = g2.mix(g2.prototype.lin.prototype,{ g2() { return g2().lin(Object.assign({ls:'@linkcolor',lw:6,lc:'round'},this)); } @@ -401,7 +236,7 @@ g2.prototype.bar.prototype = g2.mixin({}, g2.prototype.lin.prototype,{ * g2().bar2({x1:50,y1:20,x2:250,y2:120}) */ g2.prototype.bar2 = function () { return this.addCommand({c:'bar2',a:arguments[0]}); } -g2.prototype.bar2.prototype = g2.mixin({}, g2.prototype.lin.prototype,{ +g2.prototype.bar2.prototype = g2.mix(g2.prototype.lin.prototype,{ g2() { const args = Object.assign({}, this); return g2() @@ -424,7 +259,7 @@ g2.prototype.bar2.prototype = g2.mixin({}, g2.prototype.lin.prototype,{ * g2().pulley({x:100,y:75,r:50}) */ g2.prototype.pulley = function () { return this.addCommand({c:'pulley',a:arguments[0]}); } -g2.prototype.pulley.prototype = g2.mixin({}, g2.prototype.cir.prototype,{ +g2.prototype.pulley.prototype = g2.mix(g2.prototype.cir.prototype,{ g2() { const args = Object.assign({}, this); return g2() @@ -451,7 +286,7 @@ g2.prototype.pulley.prototype = g2.mixin({}, g2.prototype.cir.prototype,{ * g2().pulley2({x:50,y:30,r:25}) */ g2.prototype.pulley2 = function () { return this.addCommand({c:'pulley2',a:arguments[0]}); } -g2.prototype.pulley2.prototype = g2.mixin({}, g2.prototype.cir.prototype,{ +g2.prototype.pulley2.prototype = g2.mix(g2.prototype.cir.prototype,{ g2() { const args = Object.assign({}, this); return g2() @@ -481,7 +316,7 @@ g2.prototype.pulley2.prototype = g2.mixin({}, g2.prototype.cir.prototype,{ * .rope({p1:A,r1:20,p2:B,r2:40}) */ g2.prototype.rope = function () { return this.addCommand({c:'rope',a:arguments[0]}); } -g2.prototype.rope.prototype = g2.mixin({}, g2.prototype.lin.prototype,{ +g2.prototype.rope.prototype = g2.mix(g2.prototype.lin.prototype,{ g2() { const args = Object.assign({w:0}, this); let x1 = 'p1' in args ? args.p1.x @@ -535,7 +370,7 @@ g2.prototype.rope.prototype = g2.mixin({}, g2.prototype.lin.prototype,{ * g2().ground({pts:[25,25,25,75,75,75,75,25,125,25],pos:'left'}) */ g2.prototype.ground = function () { return this.addCommand({c:'ground',a:arguments[0]}); } -g2.prototype.ground.prototype = g2.mixin({}, g2.prototype.ply.prototype,{ +g2.prototype.ground.prototype = g2.mix(g2.prototype.ply.prototype,{ g2() { const args = Object.assign({h:4}, this); // , {closed: this.closed || false}); const itr = g2.pntItrOf(args.pts); @@ -593,7 +428,7 @@ g2.prototype.ground.prototype = g2.mixin({}, g2.prototype.ply.prototype,{ * * spacing > 1: length of spacing. */ g2.prototype.load = function () { return this.addCommand({c:'load',a:arguments[0]}); } -g2.prototype.load.prototype = g2.mixin({}, g2.prototype.ply.prototype,{ +g2.prototype.load.prototype = g2.mix(g2.prototype.ply.prototype,{ g2() { const args = Object.assign({ pointAt: this.pointAt, spacing: 20, w: -Math.PI/2 }, this); const pitr = g2.pntItrOf(args.pts), startLoc = [], arr = []; @@ -631,8 +466,8 @@ g2.prototype.load.prototype = g2.mixin({}, g2.prototype.ply.prototype,{ .ins(g => { for (const pts of startLoc) { let dist = (10*args.lw||10); // minimum distance a vector has to be - const {x,y} = args.pointAt(pts), - t = { + const {x,y} = args.pointAt(pts); + const t = { x:x+Math.cos(args.w)*dist, y:y+Math.sin(args.w)*dist }; @@ -653,204 +488,3 @@ g2.prototype.load.prototype = g2.mixin({}, g2.prototype.ply.prototype,{ }); } }); - -/** - * Creates a symbol at given coordinates. - * @method - * @returns {object} g2 - * @param {object} - symbol arguments object. - * @property {number} x - x-value center. - * @property {number} y - y-value center. - * @example - * g2().pol({x:10,y:10}) - */ -g2.prototype.pol = function () { return this.addCommand({c:'pol',a:arguments[0]||{}}); } -g2.prototype.pol.prototype = g2.mixin({}, g2.prototype.use.prototype, { - g2() { - const args = Object.assign({x:0,y:0,scl:1,w:0},this); - return g2() - .beg({x:args.x,y:args.y,scl:args.scl,w:args.w}) - .cir({r:6,fs:'@nodfill'}) - .cir({r:2.5,fs:'@ls',ls:'transparent'}) - .end(); - } -}) - -/** - * @method - * @returns {object} g2 - * @param {object} - symbol arguments object. - * @property {number} x - x-value center. - * @property {number} y - y-value center. - * @example - * g2().gnd({x:10,y:10}) -*/ - g2.prototype.gnd = function () { return this.addCommand({c:'gnd',a:arguments[0]||{}}); } - g2.prototype.gnd.prototype = g2.mixin({}, g2.prototype.use.prototype, { - g2() { - const args = Object.assign({x:0,y:0,scl:1,w:0},this); - return g2() - .beg({x:args.x,y:args.y,scl:args.scl,w:args.w}) - .cir({x:0,y:0,r:6,ls:'@nodcolor',fs:'@nodfill',lwnosc:true}) - .p() - .m({x:0,y:6}) - .a({dw:Math.PI/2,x:-6,y:0}) - .l({x:6,y:0}) - .a({dw:-Math.PI/2,x:0,y:-6}) - .z() - .fill({fs:'@nodcolor'}) - .end(); - } -}) - -/** - * @method - * @returns {object} g2 - * @param {object} - symbol arguments object. - * @property {number} x - x-value center. - * @property {number} y - y-value center. - * @example - * g2().nod({x:10,y:10}) -*/ -g2.prototype.nod = function () { return this.addCommand({c:'nod',a:arguments[0]||{}}); } -g2.prototype.nod.prototype = g2.mixin({}, g2.prototype.use.prototype, { - g2() { - const args = Object.assign({x:0,y:0,scl:1,w:0},this); - return g2() - .beg({x:args.x,y:args.y,scl:args.scl,w:args.w}) - .cir({r:4,ls:'@nodcolor',fs:'@nodfill',lwnosc:true}) - .end(); - } -}) - -/** - * @method - * @returns {object} g2 - * @param {object} - symbol arguments object. - * @property {number} x - x-value center. - * @property {number} y - y-value center. - * @example - * g2().dblnod({x:10,y:10}) -*/ -g2.prototype.dblnod = function () { return this.addCommand({c:'dblnod',a:arguments[0]||{}}); } -g2.prototype.dblnod.prototype = g2.mixin({}, g2.prototype.use.prototype, { - g2() { - const args = Object.assign({x:0,y:0,scl:1,w:0},this); - return g2() - .beg({x:args.x,y:args.y,scl:args.scl,w:args.w}) - .cir({r:6,ls:'@nodcolor',fs:'@nodfill'}) - .cir({r:3,ls:'@nodcolor',fs:'@nodfill2',lwnosc:true}) - .end(); - } -}) - -/** - * Since some symbols are not symmetrical, the cartesian mode is recommended. - * @method - * @returns {object} g2 - * @param {object} - symbol arguments object. - * @property {number} x - x-value center. - * @property {number} y - y-value center. - * @example - * g2().view({cartesian:true}) - * .nodfix({x:10,y:10}) -*/ -g2.prototype.nodfix = function () { return this.addCommand({c:'nodfix',a:arguments[0]||{}}); } -g2.prototype.nodfix.prototype = g2.mixin({}, g2.prototype.use.prototype, { - g2() { - const args = Object.assign({x:0,y:0,scl:1,w:0},this); - return g2() - .beg({x:args.x,y:args.y,scl:args.scl,w:args.w}) - .p() - .m({x:-8,y:-12}) - .l({x:0,y:0}) - .l({x:8,y:-12}) - .drw({ls:'@nodcolor',fs:'@nodfill2'}) - .cir({x:0,y:0,r:4,ls:'@nodcolor',fs:'@nodfill'}) - .end(); - } -}) - -/** - * @method - * @returns {object} g2 - * @param {object} - symbol arguments object. - * @property {number} x - x-value center. - * @property {number} y - y-value center. - * @example - * g2().view({cartesian:true}) - * .nodflt({x:10,y:10}) -*/ -g2.prototype.nodflt = function () { return this.addCommand({c:'nodflt',a:arguments[0]||{}}); } -g2.prototype.nodflt.prototype = g2.mixin({}, g2.prototype.use.prototype, { - g2() { - const args = Object.assign({x:0,y:0,scl:1,w:0},this); - return g2() - .beg({x:args.x,y:args.y,scl:args.scl,w:args.w}) - .p() - .m({x:-8,y:-12}) - .l({x:0,y:0}) - .l({x:8,y:-12}) - .drw({ls:'@nodcolor',fs:'@nodfill2'}) - .cir({x:0,y:0,r:4,ls:'@nodcolor',fs:'@nodfill'}) - .lin({x1:-9,y1:-19,x2:9,y2:-19,ls:'@nodfill2',lw:5,lwnosc:false}) - .lin({x1:-9,y1:-15.5,x2:9,y2:-15.5,ls:'@nodcolor',lw:2,lwnosc:false}) - .end(); - } -}) - -/** - * @method - * @returns {object} g2 - * @param {object} - symbol arguments object. - * @property {number} x - x-value center. - * @property {number} y - y-value center. - * @example - * g2().view({cartesian:true}) - * .origin({x:10,y:10}) -*/ -g2.prototype.origin = function () { return this.addCommand({c:'origin',a:arguments[0]||{}}); } -g2.prototype.origin.prototype = g2.mixin({}, g2.prototype.use.prototype, { - g2() { - const args = Object.assign({x:0,y:0,scl:1,w:0,z:3.5},this); - return g2() - .beg({x:args.x,y:args.y,scl:args.scl,w:args.w,lc:'round',lj:'round',fs:'#ccc'}) - .vec({x1:0,y1:0,x2:10*args.z,y2:0,lw:0.8,fs:'#ccc'}) - .vec({x1:0,y1:0,x2:0,y2:10*args.z,lw:0.8,fs:'#ccc'}) - .cir({x:0,y:0,r:2.5,fs:'#ccc'}) - .end(); - } -}) - -/** - * Mechanical style values. - * Not really meant to get overwritten. But if you actually want, proceed.
    - * Theses styles can be referenced using the comfortable '@' syntax. - * @namespace - * @property {object} State `g2` state namespace. - * @property {string} [State.nodcolor=#333] node color. - * @property {string} [State.nodfill=#dedede] node fill color. - * @property {string} [State.nodfill2=#aeaeae] alternate node fill color, somewhat darker. - * @property {string} [State.linkcolor=#666] link color. - * @property {string} [State.linkfill=rgba(225,225,225,0.75)] link fill color, semi-transparent. - * @property {string} [State.dimcolor=darkslategray] dimension color. - * @property {array} [State.solid=[]] solid line style. - * @property {array} [State.dash=[15,10]] dashed line style. - * @property {array} [State.dot=[4,4]] dotted line style. - * @property {array} [State.dashdot=[25,6.5,2,6.5]] dashdotted line style. - * @property {number} [State.labelOffset=5] default label offset distance. - * @property {number} [State.labelSignificantDigits=3] default label's significant digits after numbering point. - */ -g2.State = g2.State || {}; -g2.State.nodcolor = '#333'; -g2.State.nodfill = '#dedede'; -g2.State.nodfill2 = '#aeaeae'; -g2.State.linkcolor = '#666'; -g2.State.linkfill = 'rgba(225,225,225,0.75)'; -g2.State.dimcolor = 'darkslategray'; -g2.State.solid = []; -g2.State.dash = [15,10]; -g2.State.dot = [4,4]; -g2.State.dashdot = [25,6.5,2,6.5]; -g2.State.labelOffset = 5; -g2.State.labelSignificantDigits = 3; // 0.1234 => 0.123, 0.01234 => 0.0123, 1.234 => 1.23, 12.34 => 12.3, 123.4 => 123, 1234 => 1234 diff --git a/src/g2.selector.js b/src/g2.selector.js new file mode 100644 index 0000000..a276861 --- /dev/null +++ b/src/g2.selector.js @@ -0,0 +1,79 @@ +/** + * g2.selector.js (c) 2018 Stefan Goessner + * @file selector for `g2` elements. + * @author Stefan Goessner + * @license MIT License + */ +/* jshint -W014 */ + +/** + * Extensions. + * (Requires cartesian coordinate system) + * @namespace + */ +var g2 = g2 || { prototype:{} }; // for jsdoc only ... + +// extend prototypes for argument objects +g2.selector = function(evt) { + if (this instanceof g2.selector) { + this.selection = false; + this.evt = evt; // sharing evt object with canvasInteractor as owner ... important ! + return this; + } + return g2.selector.apply(Object.create(g2.selector.prototype), arguments); +}; +g2.handler.factory.push((ctx) => ctx instanceof g2.selector ? ctx : false); + +// g2.selector.state = ['NONE','OVER','DRAG','OVER+DRAG','EDIT','OVER+EDIT']; + +g2.selector.prototype = { + init(grp) { return true; }, + exe(commands) { + for (let elm=false, i=commands.length; i && !elm; i--) // stop after first hit .. starting from list end ! + elm = this.hit(commands[i-1].a) + }, + selectable(elm) { + return elm && elm.draggable && elm.hit; + }, + hit(elm) { + if (!this.evt.inside // pointer not inside of canvas .. + || !this.selectable(elm) ) // no selectable elm .. + return false; + + if (!elm.state && this.elementHit(elm) && elm.draggable) { // no mode + if (!this.selection || this.selection && !(this.selection.state & g2.DRAG)) { + if (this.selection) this.selection.state ^= g2.OVER; + this.selection = elm; + elm.state = g2.OVER; // enter OVER mode .. + this.evt.hit = true; + } + } + else if (elm.state & g2.DRAG) { // in DRAG mode + if (!this.evt.btn) // leave DRAG mode .. + this.elementDragEnd(elm); + } + else if (elm.state & g2.OVER) { // in OVER mode + if (!this.elementHit(elm)) { // leave OVER mode .. + elm.state ^= g2.OVER; + this.evt.hit = false; + this.selection = false; + } + else if (this.evt.btn) // enter DRAG mode + this.elementDragBeg(elm); + } + + return elm.state && elm; // we definitely have a valid elm here ... + }, // ... but only return it depending on its state. + elementDragBeg(elm) { + elm.state |= g2.DRAG; + if (elm.dragBeg) elm.dragBeg(e); + }, + elementDragEnd(elm) { + elm.state ^= (g2.OVER | g2.DRAG); + this.selection = false; + if (elm.dragEnd) elm.dragEnd(e); + }, + elementHit(elm) { + return elm.hit && elm.hit({x:this.evt.xusr,y:this.evt.yusr,eps:this.evt.eps}); + } +};