Merge branch 'master' into convolution
This commit is contained in:
		| @@ -31,11 +31,21 @@ function Commander (ronin) { | ||||
|     this.hide() | ||||
|   } | ||||
|  | ||||
|   this.cache = '' | ||||
|  | ||||
|   this.run = (txt = this._input.value) => { | ||||
|     if (txt.indexOf('$') > -1) { ronin.log('Present: $'); return } | ||||
|     const inter = new Lisp(txt, ronin.library) | ||||
|     inter.toPixels() | ||||
|     ronin.always === true && requestAnimationFrame(() => this.run(txt)) | ||||
|     this.cache = txt | ||||
|     if (ronin.always !== true) { | ||||
|       ronin.surface.maximize() | ||||
|       ronin.interpreter.run(this.cache) | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   this.loop = () => { | ||||
|     ronin.surface.maximize() | ||||
|     ronin.interpreter.run(this.cache) | ||||
|     ronin.always === true && requestAnimationFrame(() => this.loop()) | ||||
|   } | ||||
|  | ||||
|   this.load = function (txt) { | ||||
| @@ -47,6 +57,10 @@ function Commander (ronin) { | ||||
|   this.reindent = function () { | ||||
|     let val = this._input.value.replace(/\n/g, '').replace(/ +(?= )/g, '').replace(/\( \(/g, '((').replace(/\) \)/g, '))').trim() | ||||
|     let depth = 0 | ||||
|     if (val.split('(').length !== val.split(')').length) { | ||||
|       ronin.log('Uneven number of parens.') | ||||
|       return | ||||
|     } | ||||
|     for (let i = 0; i < val.length; i++) { | ||||
|       const c = val.charAt(i) | ||||
|       if (c === '(') { depth++ } else if (c === ')') { depth-- } | ||||
| @@ -68,7 +82,7 @@ function Commander (ronin) { | ||||
|     // Logs | ||||
|     if (msg && msg !== this._log.textContent) { | ||||
|       this._log.textContent = `${msg}` | ||||
|       console.log(msg) | ||||
|       // console.log(msg) | ||||
|     } | ||||
|     // Source | ||||
|     const _source = `${ronin.source} ${this._input.value.split('\n').length} lines` | ||||
|   | ||||
| @@ -7,7 +7,7 @@ function Library (ronin) { | ||||
|  | ||||
|   this.export = (path, format = 'image/png', quality = 1.0) => { // Exports a graphic file with format. | ||||
|     if (!path) { console.warn('Missing export path'); return path } | ||||
|     var dataUrl = ronin.surface.el.toDataURL(format, quality) | ||||
|     const dataUrl = ronin.surface.el.toDataURL(format, quality) | ||||
|     const data = dataUrl.replace(/^data:image\/png;base64,/, '').replace(/^data:image\/jpeg;base64,/, '') | ||||
|     fs.writeFileSync(path, data, 'base64') | ||||
|     return path | ||||
| @@ -31,8 +31,8 @@ function Library (ronin) { | ||||
|     return { x, y, w, h, t } | ||||
|   } | ||||
|  | ||||
|   this.circle = (x, y, r, t = 'circle') => { // Returns a circle shape. | ||||
|     return { x, y, r, t } | ||||
|   this.circle = (cx, cy, r, t = 'circle') => { // Returns a circle shape. | ||||
|     return { cx, cy, r, t } | ||||
|   } | ||||
|  | ||||
|   this.line = (a, b, t = 'line') => { // Returns a line shape. | ||||
| @@ -64,6 +64,83 @@ function Library (ronin) { | ||||
|     return rect | ||||
|   } | ||||
|  | ||||
|   // Frame | ||||
|  | ||||
|   this.frame = () => { // Returns a rect of the frame. | ||||
|     return ronin.surface.getFrame() | ||||
|   } | ||||
|  | ||||
|   this.center = () => { // Returns a position of the center of the frame. | ||||
|     const rect = this.frame() | ||||
|     return this.pos(rect.w / 2, rect.h / 2) | ||||
|   } | ||||
|  | ||||
|   this.resize = async (w, h, fit = true) => { // Resizes the canvas to target w and h, returns the rect. | ||||
|     const rect = { x: 0, y: 0, w, h } | ||||
|     const a = document.createElement('img') | ||||
|     const b = document.createElement('img') | ||||
|     a.src = ronin.surface.el.toDataURL() | ||||
|     await ronin.surface.resizeImage(a, b) | ||||
|     ronin.surface.resize(rect, fit) | ||||
|     return ronin.surface.draw(b, rect) | ||||
|   } | ||||
|  | ||||
|   this.rescale = async (w, h) => { // Rescales the canvas to target ratio of w and h, returns the rect. | ||||
|     const rect = { x: 0, y: 0, w: this.frame().w * w, h: this.frame().h * h } | ||||
|     const a = document.createElement('img') | ||||
|     const b = document.createElement('img') | ||||
|     a.src = ronin.surface.el.toDataURL() | ||||
|     await ronin.surface.resizeImage(a, b) | ||||
|     ronin.surface.resize(rect, true) | ||||
|     return ronin.surface.draw(b, rect) | ||||
|   } | ||||
|  | ||||
|   this.crop = async (rect) => { // Crop canvas to rect. | ||||
|     return ronin.surface.crop(rect) | ||||
|   } | ||||
|  | ||||
|   this.clone = (a, b) => { | ||||
|     ronin.surface.clone(a, b) | ||||
|     return [a, b] | ||||
|   } | ||||
|  | ||||
|   this.theme = (variable, el = document.documentElement) => { | ||||
|     // ex. (theme "f_main") -> :root { --f_main: "#fff" } | ||||
|     return getComputedStyle(el).getPropertyValue(`--${variable}`) | ||||
|   } | ||||
|  | ||||
|   // Gradients | ||||
|  | ||||
|   this.gradient = ([x1, y1, x2, y2], colors = ['white', 'black']) => { | ||||
|     return ronin.surface.linearGradient(x1, y1, x2, y2, colors) | ||||
|   } | ||||
|  | ||||
|   // Pixels | ||||
|  | ||||
|   this.pixels = (rect, fn, q) => { | ||||
|     const img = ronin.surface.context.getImageData(0, 0, rect.w, rect.h) | ||||
|     for (let i = 0, loop = img.data.length; i < loop; i += 4) { | ||||
|       const pixel = { r: img.data[i], g: img.data[i + 1], b: img.data[i + 2], a: img.data[i + 3] } | ||||
|       const processed = fn(pixel, q) | ||||
|       img.data[i] = processed[0] | ||||
|       img.data[i + 1] = processed[1] | ||||
|       img.data[i + 2] = processed[2] | ||||
|       img.data[i + 3] = processed[3] | ||||
|     } | ||||
|     ronin.surface.context.putImageData(img, 0, 0) | ||||
|     return rect | ||||
|   } | ||||
|  | ||||
|   this.saturation = (pixel, q = 1) => { | ||||
|     const color = 0.2126 * pixel.r + 0.7152 * pixel.g + 0.0722 * pixel.b | ||||
|     return [(color * (1 - q)) + (pixel.r * q), (color * (1 - q)) + (pixel.g * q), (color * (1 - q)) + (pixel.b * q), pixel.a] | ||||
|   } | ||||
|  | ||||
|   this.contrast = (pixel, q = 1) => { | ||||
|     const intercept = 128 * (1 - q) | ||||
|     return [pixel.r * q + intercept, pixel.g * q + intercept, pixel.b * q + intercept, pixel.a] | ||||
|   } | ||||
|  | ||||
|   // Strings | ||||
|  | ||||
|   this.concat = function (...items) { // Concat multiple strings. | ||||
| @@ -233,78 +310,12 @@ function Library (ronin) { | ||||
|     }, h) | ||||
|   } | ||||
|  | ||||
|   // Frame | ||||
|  | ||||
|   this.frame = () => { // Returns a rect of the frame. | ||||
|     return ronin.surface.getFrame() | ||||
|   this.keys = (item) => { // Returns a list of the object's keys | ||||
|     return Object.keys(item) | ||||
|   } | ||||
|  | ||||
|   this.center = () => { // Returns a position of the center of the frame. | ||||
|     const rect = this.frame() | ||||
|     return this.pos(rect.w / 2, rect.h / 2) | ||||
|   } | ||||
|  | ||||
|   this.resize = async (w, h, fit = true) => { // Resizes the canvas to target w and h, returns the rect. | ||||
|     const rect = { x: 0, y: 0, w, h } | ||||
|     const a = document.createElement('img') | ||||
|     const b = document.createElement('img') | ||||
|     a.src = ronin.surface.el.toDataURL() | ||||
|     await ronin.surface.resizeImage(a, b) | ||||
|     ronin.surface.resize(rect, fit) | ||||
|     return ronin.surface.draw(b, rect) | ||||
|   } | ||||
|  | ||||
|   this.rescale = async (w, h) => { // Rescales the canvas to target ratio of w and h, returns the rect. | ||||
|     const rect = { x: 0, y: 0, w: this.frame().w * w, h: this.frame().h * h } | ||||
|     const a = document.createElement('img') | ||||
|     const b = document.createElement('img') | ||||
|     a.src = ronin.surface.el.toDataURL() | ||||
|     await ronin.surface.resizeImage(a, b) | ||||
|     ronin.surface.resize(rect, true) | ||||
|     return ronin.surface.draw(b, rect) | ||||
|   } | ||||
|  | ||||
|   this.crop = async (rect) => { // Crop canvas to rect. | ||||
|     return ronin.surface.crop(rect) | ||||
|   } | ||||
|  | ||||
|   this.clone = (a, b) => { | ||||
|     ronin.surface.clone(a, b) | ||||
|     return [a, b] | ||||
|   } | ||||
|  | ||||
|   this.theme = (variable, el = document.documentElement) => { | ||||
|     // ex. (theme "f_main") -> :root { --f_main: "#fff" } | ||||
|     return getComputedStyle(el).getPropertyValue(`--${variable}`) | ||||
|   } | ||||
|  | ||||
|   // Gradients | ||||
|  | ||||
|   this.gradient = ([x1, y1, x2, y2], colors = ['white', 'black']) => { | ||||
|     return ronin.surface.linearGradient(x1, y1, x2, y2, colors) | ||||
|   } | ||||
|  | ||||
|   // Pixels | ||||
|  | ||||
|   this.pixels = (rect, fn, q) => { | ||||
|     const img = ronin.surface.context.getImageData(rect.x, rect.y, rect.w, rect.h) | ||||
|     for (let i = 0, loop = img.data.length; i < loop; i += 4) { | ||||
|       const pixel = { r: img.data[i], g: img.data[i + 1], b: img.data[i + 2], a: img.data[i + 3] } | ||||
|       const processed = fn(pixel, q) | ||||
|       img.data.set(processed, i); | ||||
|     } | ||||
|     ronin.surface.context.putImageData(img, rect.x, rect.y) | ||||
|     return rect | ||||
|   } | ||||
|  | ||||
|   this.saturation = (pixel, q = 1) => { | ||||
|     const color = 0.2126 * pixel.r + 0.7152 * pixel.g + 0.0722 * pixel.b * (1 - q) | ||||
|     return [color + (pixel.r * q), color + (pixel.g * q), color + (pixel.b * q), pixel.a] | ||||
|   } | ||||
|  | ||||
|   this.contrast = (pixel, q = 1) => { | ||||
|     const intercept = 128 * (1 - q) | ||||
|     return [pixel.r * q + intercept, pixel.g * q + intercept, pixel.b * q + intercept, pixel.a] | ||||
|   this.values = (item) => { // Returns a list of the object's values | ||||
|     return Object.values(item) | ||||
|   } | ||||
|  | ||||
|   // Convolve | ||||
| @@ -370,6 +381,16 @@ function Library (ronin) { | ||||
|     return args | ||||
|   } | ||||
|  | ||||
|   this.table = (arg) => { | ||||
|     console.table(arg) | ||||
|     return arg | ||||
|   } | ||||
|  | ||||
|   this.debug = (arg) => { | ||||
|     console.log(arg) | ||||
|     return arg | ||||
|   } | ||||
|  | ||||
|   this.time = (rate = 1) => { // Returns timestamp in milliseconds. | ||||
|     return (Date.now() * rate) | ||||
|   } | ||||
| @@ -397,4 +418,10 @@ function Library (ronin) { | ||||
|     console.log(`time taken: ${Date.now() - start}ms`) | ||||
|     return result | ||||
|   } | ||||
|  | ||||
|   // IO | ||||
|  | ||||
|   this.osc = (path) => { // Returns the latest osc message at path | ||||
|     return path ? ronin.osc.msg[path] : ronin.osc.msg | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,14 +1,14 @@ | ||||
| 'use strict' | ||||
|  | ||||
| function Lisp (input, lib) { | ||||
| function Lisp (lib = {}, includes = []) { | ||||
|   const path = require('path') | ||||
|   const fs = require('fs') | ||||
|  | ||||
|   const TYPES = { identifier: 0, number: 1, string: 2, bool: 3 } | ||||
|  | ||||
|   const Context = function (scope, parent) { | ||||
|     this.scope = scope | ||||
|     this.parent = parent | ||||
|  | ||||
|     this.get = function (identifier) { | ||||
|       if (identifier in this.scope) { | ||||
|         return this.scope[identifier] | ||||
| @@ -20,9 +20,9 @@ function Lisp (input, lib) { | ||||
|  | ||||
|   const special = { | ||||
|     include: (input, context) => { | ||||
|       if (!input[1].value || !fs.existsSync(input[1].value)) { console.warn('Source', input[1].value); return [] } | ||||
|       if (!input[1].value || !fs.existsSync(input[1].value)) { console.warn('Lisp', 'No file: ' + input[1].value); return [] } | ||||
|       const file = fs.readFileSync(input[1].value, { encoding: 'utf-8' }) | ||||
|       return interpret(this.parse(file), context) | ||||
|       return interpret(this.parse(`(${file})`), context) | ||||
|     }, | ||||
|     let: function (input, context) { | ||||
|       const letContext = input[1].reduce(function (acc, x) { | ||||
| @@ -60,21 +60,32 @@ function Lisp (input, lib) { | ||||
|         return interpret(input[2], new Context(lambdaScope, context)) | ||||
|       } | ||||
|     }, | ||||
|     __fn: function (input, context) { | ||||
|       return async function () { | ||||
|         const lambdaArguments = arguments | ||||
|         const lambdaScope = [].reduce(function (acc, x, i) { | ||||
|           acc[x.value] = lambdaArguments[i] | ||||
|           return acc | ||||
|         }, {}) | ||||
|         return interpret(input.slice(1), new Context(lambdaScope, context)) | ||||
|       } | ||||
|     }, | ||||
|     if: async function (input, context) { | ||||
|       if (await interpret(input[1], context)) { | ||||
|         return interpret(input[2], context) | ||||
|       } | ||||
|       return input[3] ? interpret(input[3], context) : [] | ||||
|     }, | ||||
|     __fn: function (input, context) { | ||||
|       return async function () { | ||||
|         const lambdaArguments = arguments | ||||
|         const keys = [...new Set(input.slice(2).flat(100).filter(i => | ||||
|           i.type === TYPES.identifier && | ||||
|           i.value[0] === '%' | ||||
|         ).map(x => x.value).sort())] | ||||
|         const lambdaScope = keys.reduce(function (acc, x, i) { | ||||
|           acc[x] = lambdaArguments[i] | ||||
|           return acc | ||||
|         }, {}) | ||||
|         return interpret(input.slice(1), new Context(lambdaScope, context)) | ||||
|       } | ||||
|     }, | ||||
|     __obj: async function (input, context) { | ||||
|       const obj = {} | ||||
|       for (let i = 1; i < input.length; i += 2) { | ||||
|         obj[await interpret(input[i], context)] = await interpret(input[i + 1], context) | ||||
|       } | ||||
|       return obj | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -90,7 +101,7 @@ function Lisp (input, lib) { | ||||
|   } | ||||
|  | ||||
|   const interpret = async function (input, context) { | ||||
|     if (!input) { console.warn('error', context.scope); return null } | ||||
|     if (!input) { console.warn('Lisp', 'error', context.scope); return null } | ||||
|  | ||||
|     if (context === undefined) { | ||||
|       return interpret(input, new Context(lib)) | ||||
| @@ -108,6 +119,8 @@ function Lisp (input, lib) { | ||||
|       return { type: TYPES.number, value: parseFloat(input) } | ||||
|     } else if (input[0] === '"' && input.slice(-1) === '"') { | ||||
|       return { type: TYPES.string, value: input.slice(1, -1) } | ||||
|     } else if (input[0] === ':') { | ||||
|       return { type: TYPES.string, value: input.slice(1) } | ||||
|     } else if (input === 'true' || input === 'false') { | ||||
|       return { type: TYPES.bool, value: input === 'true' } | ||||
|     } else { | ||||
| @@ -124,10 +137,14 @@ function Lisp (input, lib) { | ||||
|       input.unshift('__fn') | ||||
|       list.push(parenthesize(input, [])) | ||||
|       return parenthesize(input, list) | ||||
|     } else if (token === '{') { | ||||
|       input.unshift('__obj') | ||||
|       list.push(parenthesize(input, [])) | ||||
|       return parenthesize(input, list) | ||||
|     } else if (token === '(') { | ||||
|       list.push(parenthesize(input, [])) | ||||
|       return parenthesize(input, list) | ||||
|     } else if (token === ')') { | ||||
|     } else if (token === ')' || token === '}') { | ||||
|       return list | ||||
|     } else { | ||||
|       return parenthesize(input, list.concat(categorize(token))) | ||||
| @@ -135,21 +152,35 @@ function Lisp (input, lib) { | ||||
|   } | ||||
|  | ||||
|   const tokenize = function (input) { | ||||
|     const i = input.replace(/^\;.*\n?/gm, '').split('"') | ||||
|     return i.map(function (x, i) {  | ||||
|       return i % 2 === 0 ?  | ||||
|         x.replace(/\(/g, ' ( ').replace(/\)/g, ' ) ').replace(/' \( /g, ' \'( ') | ||||
|         : x.replace(/ /g, '!whitespace!')  | ||||
|     const i = input.replace(/^\;.*\n?/gm, '').replace('λ', 'lambda ').split('"') | ||||
|     return i.map(function (x, i) { | ||||
|       return i % 2 === 0 | ||||
|         ? x.replace(/\(/g, ' ( ') | ||||
|           .replace(/\)/g, ' ) ') | ||||
|           .replace(/' \( /g, ' \'( ') // '() | ||||
|           .replace(/\{/g, ' { ') // {} | ||||
|           .replace(/\}/g, ' } ') // {} | ||||
|         : x.replace(/ /g, '!whitespace!') | ||||
|     }) | ||||
|     .join('"').trim().split(/\s+/) | ||||
|     .map(function (x) { return x.replace(/!whitespace!/g, ' ') }) | ||||
|       .join('"').trim().split(/\s+/) | ||||
|       .map(function (x) { return x.replace(/!whitespace!/g, ' ') }) | ||||
|   } | ||||
|  | ||||
|   this.inc = function () { | ||||
|     return includes.reduce((acc, item) => { | ||||
|       const p = path.join(__dirname, `lisp/${item}.lisp`) | ||||
|       if (!fs.existsSync(p)) { console.warn('Lisp', `Missing include: ${p}`); return acc } | ||||
|       return `${acc}(include "${p}") ` | ||||
|     }, '') | ||||
|   } | ||||
|  | ||||
|   this.parse = function (input) { | ||||
|     return parenthesize(tokenize(`(${input})`)) | ||||
|     return parenthesize(tokenize(input)) | ||||
|   } | ||||
|  | ||||
|   this.toPixels = async function () { | ||||
|     return interpret(this.parse(input)) | ||||
|   this.run = async function (input) { | ||||
|     return interpret(this.parse(`( | ||||
|       ${this.inc()} | ||||
|       ${input})`)) | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										23
									
								
								desktop/sources/scripts/osc.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								desktop/sources/scripts/osc.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| 'use strict' | ||||
|  | ||||
| function Osc (ronin) { | ||||
|   const osc = require('osc') | ||||
|  | ||||
|   this.msg = {} | ||||
|  | ||||
|   this.start = function () { | ||||
|     const udpPort = new osc.UDPPort({ | ||||
|       localAddress: '0.0.0.0', | ||||
|       localPort: 49162, | ||||
|       metadata: true | ||||
|     }) | ||||
|     udpPort.on('message', this.onMsg) | ||||
|     udpPort.open() | ||||
|     ronin.log('OSC', 'Started.') | ||||
|   } | ||||
|  | ||||
|   this.onMsg = (msg, timeTag, info) => { | ||||
|     this.msg[msg.address] = msg.args | ||||
|     // ronin.log(`${info.address}:${info.port} > ${msg.args}`, info) | ||||
|   } | ||||
| } | ||||
| @@ -11,6 +11,8 @@ function Ronin () { | ||||
|     b_inv: '#ffb545' | ||||
|   } | ||||
|  | ||||
|   this.includes = ['prelude'] | ||||
|  | ||||
|   this.el = document.createElement('div') | ||||
|   this.el.id = 'ronin' | ||||
|  | ||||
| @@ -19,7 +21,8 @@ function Ronin () { | ||||
|   this.commander = new Commander(this) | ||||
|   this.surface = new Surface(this) | ||||
|   this.library = new Library(this) | ||||
|  | ||||
|   this.interpreter = new Lisp(this.library, this.includes) | ||||
|   this.osc = new Osc(this) | ||||
|   // Parameters | ||||
|  | ||||
|   this.always = false | ||||
| @@ -43,7 +46,7 @@ function Ronin () { | ||||
|     this.source.start() | ||||
|     this.commander.start() | ||||
|     this.surface.start() | ||||
|     console.log('Ronin', 'Started') | ||||
|     this.osc.start() | ||||
|   } | ||||
|  | ||||
|   this.reset = function () { | ||||
| @@ -61,7 +64,7 @@ function Ronin () { | ||||
|   this.animate = (b = true) => { | ||||
|     if (this.always === b) { return } | ||||
|     this.always = b | ||||
|     this.commander.run() | ||||
|     this.commander.loop() | ||||
|   } | ||||
|  | ||||
|   // Zoom | ||||
|   | ||||
| @@ -126,12 +126,6 @@ function Source (ronin) { | ||||
|     return `${str}` | ||||
|   } | ||||
|  | ||||
|   this.locate = function (name) { | ||||
|     if (!this.path) { return } | ||||
|     const loc = path.join(this.folder(), name) | ||||
|     return fs.existsSync(loc) ? loc : null | ||||
|   } | ||||
|  | ||||
|   // Etc | ||||
|  | ||||
|   this.name = function () { | ||||
|   | ||||
| @@ -100,7 +100,7 @@ function Surface (ronin) { | ||||
|   } | ||||
|  | ||||
|   this.traceCircle = function (circle, context) { | ||||
|     context.arc(circle.x, circle.y, circle.r, 0, 2 * Math.PI) | ||||
|     context.arc(circle.cx, circle.cy, circle.r, 0, 2 * Math.PI) | ||||
|   } | ||||
|  | ||||
|   this.traceText = function (text, context) { | ||||
| @@ -185,7 +185,7 @@ function Surface (ronin) { | ||||
|   } | ||||
|  | ||||
|   this.maximize = function () { | ||||
|     this.resize({ x: 0, y: 0, w: (window.innerWidth * this.ratio) - 60, h: (window.innerHeight * this.ratio) - 60, t: 'rect' }) | ||||
|     this.resize({ x: 0, y: 0, w: ((window.innerWidth - 60) * this.ratio), h: ((window.innerHeight - 60) * this.ratio), t: 'rect' }) | ||||
|   } | ||||
|  | ||||
|   this.onResize = function () { | ||||
|   | ||||
		Reference in New Issue
	
	Block a user