2020-07-10 22:40:34 +02:00
|
|
|
const m = require('makerjs')
|
2020-07-16 23:55:37 +02:00
|
|
|
const u = require('./utils')
|
2020-06-27 20:13:06 +02:00
|
|
|
const Point = require('./point')
|
|
|
|
|
|
|
|
|
|
const assert = exports.assert = (exp, msg) => {
|
|
|
|
|
if (!exp) {
|
|
|
|
|
throw new Error(msg)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const type = exports.type = (val) => {
|
|
|
|
|
if (Array.isArray(val)) return 'array'
|
|
|
|
|
if (val === null) return 'null'
|
|
|
|
|
return typeof val
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const sane = exports.sane = (val, name, _type) => {
|
|
|
|
|
assert(type(val) == _type, `Field "${name}" should be of type ${_type}!`)
|
|
|
|
|
return val
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const detect_unexpected = exports.detect_unexpected = (obj, name, expected) => {
|
|
|
|
|
const sane_obj = sane(obj, name, 'object')
|
|
|
|
|
for (const key of Object.keys(sane_obj)) {
|
|
|
|
|
assert(expected.includes(key), `Unexpected key "${key}" within field "${name}"!`)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-30 22:29:14 +02:00
|
|
|
exports.in = (raw, name, arr) => {
|
|
|
|
|
assert(arr.includes(raw), `Field "${name}" should be one of [${arr.join(', ')}]!`)
|
|
|
|
|
return raw
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-28 22:35:53 +02:00
|
|
|
const numarr = exports.numarr = (raw, name, length) => {
|
|
|
|
|
assert(type(raw) == 'array' && raw.length == length, `Field "${name}" should be an array of length ${length}!`)
|
|
|
|
|
raw = raw.map(val => val || 0)
|
|
|
|
|
raw.map(val => assert(type(val) == 'number', `Field "${name}" should contain numbers!`))
|
|
|
|
|
return raw
|
2020-06-27 20:13:06 +02:00
|
|
|
}
|
|
|
|
|
|
2020-06-28 22:35:53 +02:00
|
|
|
const xy = exports.xy = (raw, name) => numarr(raw, name, 2)
|
|
|
|
|
|
2020-07-04 23:23:14 +02:00
|
|
|
const wh = exports.wh = (raw, name) => {
|
2020-06-28 22:35:53 +02:00
|
|
|
if (!Array.isArray(raw)) raw = [raw, raw]
|
2020-07-04 23:23:14 +02:00
|
|
|
return xy(raw, name)
|
2020-06-28 22:35:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
exports.trbl = (raw, name) => {
|
|
|
|
|
if (!Array.isArray(raw)) raw = [raw, raw, raw, raw]
|
|
|
|
|
if (raw.length == 2) raw = [raw[1], raw[0], raw[1], raw[0]]
|
|
|
|
|
return numarr(raw, name, 4)
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-04 23:23:14 +02:00
|
|
|
exports.anchor = (raw, name, points={}, check_unexpected=true, default_point=new Point()) => {
|
2020-06-28 22:35:53 +02:00
|
|
|
if (check_unexpected) detect_unexpected(raw, name, ['ref', 'shift', 'rotate'])
|
2020-07-14 23:03:44 +02:00
|
|
|
let point = default_point.clone()
|
2020-06-27 20:13:06 +02:00
|
|
|
if (raw.ref !== undefined) {
|
2020-07-14 23:03:44 +02:00
|
|
|
if (type(raw.ref) == 'array') {
|
|
|
|
|
// averaging multiple anchors
|
|
|
|
|
let x = 0, y = 0, r = 0
|
|
|
|
|
const len = raw.ref.length
|
|
|
|
|
for (const ref of raw.ref) {
|
|
|
|
|
assert(points[ref], `Unknown point reference "${ref}" in anchor "${name}"!`)
|
|
|
|
|
const resolved = points[ref]
|
|
|
|
|
x += resolved.x
|
|
|
|
|
y += resolved.y
|
|
|
|
|
r += resolved.r
|
|
|
|
|
}
|
|
|
|
|
point = new Point(x / len, y / len, r / len)
|
|
|
|
|
} else {
|
|
|
|
|
assert(points[raw.ref], `Unknown point reference "${raw.ref}" in anchor "${name}"!`)
|
|
|
|
|
point = points[raw.ref].clone()
|
|
|
|
|
}
|
2020-06-27 20:13:06 +02:00
|
|
|
}
|
|
|
|
|
if (raw.shift !== undefined) {
|
2020-07-10 22:40:34 +02:00
|
|
|
let xyval = wh(raw.shift || [0, 0], name + '.shift')
|
2020-07-14 23:03:44 +02:00
|
|
|
if (point.meta.mirrored) {
|
|
|
|
|
xyval[0] = -xyval[0]
|
|
|
|
|
}
|
|
|
|
|
point.shift(xyval, true)
|
2020-06-27 20:13:06 +02:00
|
|
|
}
|
|
|
|
|
if (raw.rotate !== undefined) {
|
2020-07-16 23:55:37 +02:00
|
|
|
point.r += sane(raw.rotate || 0, name + '.rotate', 'number')
|
2020-06-27 20:13:06 +02:00
|
|
|
}
|
2020-07-14 23:03:44 +02:00
|
|
|
return point
|
2020-07-16 23:55:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const extend_pair = exports.extend_pair = (to, from) => {
|
|
|
|
|
const to_type = type(to)
|
|
|
|
|
const from_type = type(from)
|
|
|
|
|
if (from === undefined || from === null) return to
|
|
|
|
|
if (from === '!!unset') return undefined
|
|
|
|
|
if (to_type != from_type) return from
|
|
|
|
|
if (from_type == 'object') {
|
|
|
|
|
const res = u.deepcopy(to)
|
|
|
|
|
for (const key of Object.keys(from)) {
|
|
|
|
|
res[key] = extend_pair(to[key], from[key])
|
2020-07-18 22:58:44 +02:00
|
|
|
if (res[key] === undefined) delete res[key]
|
2020-07-16 23:55:37 +02:00
|
|
|
}
|
|
|
|
|
return res
|
|
|
|
|
} else if (from_type == 'array') {
|
|
|
|
|
const res = u.deepcopy(to)
|
|
|
|
|
for (const [i, val] of from.entries()) {
|
|
|
|
|
res[i] = extend_pair(res[i], val)
|
|
|
|
|
}
|
|
|
|
|
return res
|
|
|
|
|
} else return from
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-17 23:20:49 +02:00
|
|
|
const extend = exports.extend = (...args) => {
|
2020-07-16 23:55:37 +02:00
|
|
|
let res = args[0]
|
|
|
|
|
for (const arg of args) {
|
|
|
|
|
if (res == arg) continue
|
|
|
|
|
res = extend_pair(res, arg)
|
|
|
|
|
}
|
|
|
|
|
return res
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-17 23:20:49 +02:00
|
|
|
const inherit = exports.inherit = (name_prefix, name, set) => {
|
|
|
|
|
let result = u.deepcopy(set[name])
|
|
|
|
|
if (result.extends !== undefined) {
|
|
|
|
|
let candidates = [name]
|
|
|
|
|
const list = []
|
|
|
|
|
while (candidates.length) {
|
|
|
|
|
const item = candidates.shift()
|
|
|
|
|
const other = u.deepcopy(set[item])
|
|
|
|
|
assert(other, `"${item}" (reached from "${name_prefix}.${name}.extends") does not name a valid target!`)
|
|
|
|
|
let parents = other.extends || []
|
|
|
|
|
if (type(parents) !== 'array') parents = [parents]
|
|
|
|
|
candidates = candidates.concat(parents)
|
|
|
|
|
delete other.extends
|
|
|
|
|
list.unshift(other)
|
2020-07-16 23:55:37 +02:00
|
|
|
}
|
2020-07-17 23:20:49 +02:00
|
|
|
result = extend.apply(this, list)
|
2020-07-16 23:55:37 +02:00
|
|
|
}
|
|
|
|
|
return result
|
2020-06-27 20:13:06 +02:00
|
|
|
}
|