2877 lines
77 KiB
TypeScript
Raw Normal View History

import { v4 as uuidv4 } from "uuid"
// import { Buffer } from 'buffer'
// weird hack for remix client-side logic
try {
window.process = {
env: {},
}
window.Buffer = () => {}
// window.Buffer = Buffer
} catch (err) {
// nothing
}
// const babel = require('@babel/standalone')
// babel.template = require('@babel/template').default
// babel.t = require('@babel/types')
// babel.generator = require('@babel/generator').default
// babel.babylon = require('@babel/parser')
// babel.prettier = require('prettier')
const objList = []
const stringList = []
// ES6?
// import { uuidv4 as uuid } from 'uuid'
// import babel from '@babel/standalone'
// import template from '@babel/template'
// babel.template = template
// import t from '@babel/types'
// babel.t = t
// import generator from '@babel/generator'
// babel.generator = generator
// import babylon from '@babel/parser'
// babel.babylon = babylon
// import prettier from 'prettier'
// babel.prettier = prettier
export const babel = () => {}
export const local = {}
export const t = () => {}
// export const t = babel?.t
export const getBabel = () => {
return babel
}
export const uuid = uuidv4
export const pause = (value, opts) => {
return stringify(value, opts)
}
export const save = (value, opts) => {
return stringify(value, opts)
}
// extract parent path from dot delimitted path
export const getParentPath = (path) => {
const parts = path?.split(".")
parts?.pop()
return parts?.join(".")
}
export const toJavascript = (value, opts = {}) => {
const defaultOpts = {
wrapInFunction: true,
declarations: [],
identifier: "obj",
keys: {
obj: 1,
},
dependancies: {},
mappings: [],
seen: [],
db: [],
}
Object.assign(defaultOpts, opts)
opts = defaultOpts
toJavascriptAux(value, opts)
// sort declarations by dependancies
for (const declaration of opts.declarations) {
if (declaration.type === "ExpressionStatement") {
// nothing
} else {
const key = declaration.declarations[0].id.name
const dependancies = opts.dependancies[key]
const dependancyCheck = {}
for (const dependancy of dependancies) {
dependancyCheck[dependancy] = false
}
let sortableDeclarationIndex = 0
let insertionIndex = 0
for (const sortableDeclaration of opts.declarations) {
if (sortableDeclaration.type === "ExpressionStatement") {
// nothing
} else {
const sortableDeclarationKey =
sortableDeclaration.declarations[0].id.name
if (dependancyCheck[sortableDeclarationKey] === false) {
insertionIndex = sortableDeclarationIndex + 1
dependancyCheck[sortableDeclarationKey] = true
}
}
// increment iterator index
sortableDeclarationIndex++
}
const declarationIndex = opts.declarations.indexOf(declaration)
opts.declarations.splice(declarationIndex, 1)
opts.declarations.splice(insertionIndex, 0, declaration)
}
}
let program
if (opts.wrapInFunction && !opts.moduleExport) {
opts.declarations.push(t.returnStatement(t.identifier("obj")))
const expression = t.expressionStatement(
t.callExpression(
t.functionExpression(null, [], t.blockStatement(opts.declarations)),
[]
)
)
program = t.program([expression])
} else {
if (opts.moduleExport) {
opts.declarations.push(
t.expressionStatement(
t.assignmentExpression(
"=",
t.memberExpression(t.identifier("module"), t.identifier("exports")),
t.identifier("obj")
)
)
)
}
program = t.program(opts.declarations)
}
const stringifiedProgram = getBabel().generator(program).code
return getBabel().prettier.format(stringifiedProgram, {
semi: false,
parser: "babel",
useTabs: true,
})
}
export const toJavascriptAux = (value, opts) => {
if (typeof value === "object") {
if (!opts.seen.includes(value)) {
// update map and seen
opts.seen.push(value)
const properties = createObjectProperties(value, opts)
const declaration = t.variableDeclaration("let", [
t.variableDeclarator(
t.identifier(opts.identifier),
t.objectExpression(properties)
),
])
opts.declarations.unshift(declaration)
for (const key of Object.keys(value)) {
const property = properties.find((v) => {
return v.value.name === key
})
if (property) {
const identifier = property.value.name
toJavascriptAux(value[key], { ...opts, identifier })
}
}
}
} else if (typeof value === "string") {
const declaration = t.variableDeclaration("let", [
t.variableDeclarator(
t.identifier(opts.identifier),
t.stringLiteral(value)
),
])
opts.declarations.unshift(declaration)
}
}
export const createObjectProperties = (value, opts) => {
const properties = []
let dependancies = opts.dependancies[opts.identifier]
if (!dependancies) dependancies = opts.dependancies[opts.identifier] = []
for (const key of Object.keys(value)) {
if (value[key] === value) {
opts.declarations.push(
t.expressionStatement(
t.assignmentExpression(
"=",
t.memberExpression(
t.identifier(opts.identifier),
t.identifier(key)
),
t.identifier(opts.identifier)
)
)
)
} else {
if (!opts.db.includes(value[key])) {
opts.db.push(value[key])
if (opts.keys[key] === undefined) {
opts.keys[key] = 1
}
const keyIncrement = opts.keys[key]
if (keyIncrement == 1) {
opts.mappings.push(key)
} else {
opts.mappings.push(key + keyIncrement)
}
opts.keys[key]++
}
const identifier = opts.mappings[opts.db.indexOf(value[key])]
dependancies.push(identifier)
properties.push(
t.objectProperty(
t.identifier(key),
t.identifier(identifier),
false,
key == identifier
)
)
}
}
return properties
}
export const serialize = (value, opts = {}) => {
opts.strictFunctions = false
opts.serializeArrayProps = true
return stringify(value, opts)
}
export const stringify = (value, opts = {}) => {
const schema = {
stringifier: stringifier,
replace(key, value) {
if (opts.firstRun) {
opts.firstRun = !opts.firstRun
return value
}
const after = opts.stringifier(key, value, opts)
const type = typeof after.value
if (type === "object") {
if (after === null || after.value === null) {
const ret = after.value
return ret
}
} else if (type === "string") {
const ret =
opts.known.get(after.key) || setKnown(opts.known, opts.input, after)
return ret
}
return after.value
},
strictFunctions: true,
firstRun: undefined,
known: new Map(),
input: [],
output: [],
}
Object.keys(schema).forEach((key) => {
if (getsmart(opts, epp(key), { undefined: true }, true).undefined == true) {
opts[key] = schema[key]
}
})
opts.virtual = opts.stringifier("", value, opts)
for (
let i = parseInt(setKnown(opts.known, opts.input, opts.virtual));
i < opts.input.length;
i++
) {
opts.firstRun = true
try {
opts.output[i] = JSON.stringify(opts.input[i], opts.replace, opts.space)
} catch (err) {
console.error(err)
}
}
return "[" + opts.output.join(",") + "]"
}
export const replacer = (key, value) => {
const opts = opts || {}
if (opts.firstRun) {
opts.firstRun = !opts.firstRun
return value
}
const after = opts.stringifier(key, value, opts)
// replace with if statements
const type = typeof after.value
if (type === "object") {
if (after === null) {
const ret = after.value
return ret
}
} else if (type === "string") {
const ret =
opts.known.get(after.key) || setKnown(opts.known, opts.input, after)
return ret
}
return after.value
}
export const setKnown = (known, input, virtual) => {
const index = String(input.push(virtual.value) - 1)
known.set(virtual.key, index)
return index
}
export const stringifier = (key, val, opts) => {
let ret = { value: val, key: val }
if (
val instanceof Function &&
typeof val.toString === "function" &&
(!opts.strictFunctions || typeof val.$scopes != "undefined")
) {
const known = opts.known.get(ret.key)
ret = {
value: known || {
type: "function",
js: val.toString(),
$scopes: val.$scopes,
$context: val.$context,
...val,
},
key: val,
}
if (ret.value.js == "function () { [native code] }") return
if (typeof known == "undefined") setKnown(opts.known, opts.input, ret)
} else if (ret.value === Infinity && typeof ret.value != "string") {
const known = opts.known.get(ret.key)
ret = {
value: known || {
type: "number",
js: "Infinity",
$scopes: [],
$context: {},
},
key: val,
}
if (typeof known == "undefined") setKnown(opts.known, opts.input, ret)
} else if (typeof ret.value === "undefined") {
ret = {
value: {
type: "undefined",
js: "undefined",
},
key: val,
}
} else if (ret.value instanceof Array && opts.serializeArrayProps) {
const known = opts.known.get(ret.key)
ret = {
value: known
? ret.value
: {
type: "Array",
js: ret.value,
uuid: ret.value.uuid,
},
key: val,
}
setKnown(opts.known, opts.input, ret)
}
return ret
}
export const primitives = (value) => {
return value instanceof String ? String(value) : value
}
export const Primitives = (key, value) => {
return typeof value === "string" ? new String(value) : value
}
export const play = (text, opts) => {
return parse(text, opts)
}
export const load = (text, opts = {}) => {
opts.strictFunctions = false
return parse(text, opts)
}
export const parse = (text, opts = {}) => {
const schema = {
// parser: eval('(function '+parser+')'),
parser: parser(opts),
value: {},
strictFunctions: true,
noFunctions: false,
firstPass: true,
output: new Map(),
}
Object.keys(schema).forEach((key) => {
if (getsmart(opts, epp(key), { undefined: true }, true).undefined == true) {
opts[key] = schema[key]
}
})
// opts.parser = opts.parser.bind(opts)
opts.input = JSON.parse(text, Primitives)
opts.firstPass = false
opts.input = opts.input.map(primitives)
opts.value = opts.input[0]
const isObject = typeof opts.value === "object" && opts.value
const tmp = isObject
? revive(opts.input, opts.output, opts.value, opts.parser, opts)
: opts.value
opts.replaceMode = true
let ret = revive(opts.input, new Map(), tmp, opts.parser, opts)
ret = opts.parser("", tmp, opts)
return ret
}
export const parser = (opts) => {
return function (key, val) {
if (val.js && val.type === "Array") {
const ret = opts.input[opts.output.get(val)].js
ret.uuid = opts.input[opts.output.get(val)].uuid
return ret
} else if (
val.js &&
val.type === "function" &&
opts.replaceMode &&
!opts.noFunctions
) {
let ret = opts.input[opts.output.get(val)]
if (typeof ret == val.type) return ret
const uuid = jsUUID()
let fn
let scopedEval
if (val.$scopedEval && typeof val.$scopedEval == "function") {
scopedEval = val.$scopedEval
} else {
const fns = createScopedEval(uuid)
try {
fn = eval(`(${fns})`)
const input = { val, smarts }
scopedEval = fn(input)
} catch (err) {
console.log("Caught error evaling createScopedEval", err)
}
}
ret = scopedEval({ val })
try {
Object.defineProperty(ret, "$scopes", {
value: val.$scopes,
enumerable: true,
})
if (val.uuid) {
ret.uuid = val.uuid
}
} catch (err) {
if (opts.verbose) console.error(err)
}
try {
Object.defineProperty(ret, "$context", {
value: val.$context,
enumerable: true,
})
} catch (err) {
if (opts.verbose) console.error(err)
}
try {
Object.defineProperty(ret, "$scopedEval", {
value: scopedEval,
enumerable: true,
})
} catch (err) {
if (opts.verbose) console.error(err)
}
opts.input[opts.output.get(val)] = ret
return ret
} else if (opts.replaceMode) {
return val
}
return Primitives(key, val)
}
}
export const revive = (input, parsed, output, parser, opts) => {
return Object.keys(output).reduce((output, key) => {
let value = output[key]
// if the value hasn't been revived yet
if (value instanceof String) {
const tmp = input[value]
if (typeof tmp === "object" && !parsed.get(tmp)) {
parsed.set(tmp, value)
output[key] = primitives(
parser(key, revive(input, parsed, tmp, parser, opts))
)
} else {
try {
output[key] = primitives(parser(key, tmp))
} catch (err) {
delete output[key]
}
}
} else {
try {
if (opts.replaceMode) {
// output[key] = primitives(parser(key, revive(input, parsed, value, parser, opts)))
value = parser(key, value)
if (typeof value === "object" && !parsed.get(value)) {
parsed.set(value, value)
output[key] = primitives(
parser(key, revive(input, parsed, value, parser, opts))
)
} else {
try {
output[key] = primitives(value)
} catch (err) {
delete output[key]
}
}
} else {
output[key] = primitives(parser(key, value))
}
} catch (err) {
delete output[key]
}
}
return output
}, output)
}
export const createScopedEval = (uuid) => {
const ret = /*javascript*/ `
function createScopedEval(${uuid}){
// scopeCode
${uuid}.scopeCode = ${uuid}.scopeCode || ${uuid}.getBabel().template.ast('try{}catch(err){console.log(err)}')
${uuid}.previousScopeCode = ${uuid}.currentScopeCode || ${uuid}.scopeCode
${uuid}.currentScopeCode = ${uuid}.scopeCode.block.body.length ? ${uuid}.getBabel().template.ast('try{}catch(err){console.log(err)}') : ${uuid}.scopeCode
if(${uuid}.previousScopeCode != ${uuid}.currentScopeCode){
${uuid}.previousScopeCode.block.body.push(
${uuid}.currentScopeCode
)
}
${uuid}.closureIndex = ${uuid}.closureIndex || 0
${uuid}.closure = ${uuid}.getsmart.bind(this)(${uuid}, ${
/*javascript*/ `\`val.$scopes.\${${uuid}.closureIndex}\``
}, {})
${uuid}.variableKeys = Object.keys(${uuid}.closure)
${uuid}.variableMap = ${uuid}.getsmart.bind(this)(${uuid}, ${
/*javascript*/ `\`val.$context.$variableMaps.\${${uuid}.closureIndex}\``
}, [])
${uuid}.allowedIdentifiers = ['let','var','const']
${uuid}.variableKeys.forEach((key)=>{
if(
typeof ${uuid}.variableMap[key] == 'string'
&& ${uuid}.allowedIdentifiers.indexOf(${uuid}.variableMap[key]) >= 0
){
try{
${uuid}.currentScopeCode.block.body.push(
${uuid}.getBabel().template.ast(
${
/*javascript*/ `\`
\${${uuid}.variableMap[key]} \${key} = ${uuid}.val.$scopes[\${${uuid}.closureIndex}]['\${key}']
\``
}
)
)
}catch(err){console.log(1,err)}
try{
${uuid}.currentScopeCode.block.body.push(
${uuid}.getBabel().template.ast(
${
/*javascript*/ `\`
Object.defineProperty(
${uuid}.val.$scopes[\${${uuid}.closureIndex}],
\${${uuid}.stringify(key)},
{
get(){
return \${key}
},
set(val){
\${key} = val
},
enumerable: true
}
)
\``
}
)
)
}catch(err){console.log(2,err)}
}
// console.log(${uuid}.scopeCode)
})
// console.log(${uuid}.scopeCode)
${uuid}.closureIndex++
if(${uuid}.closureIndex >= ${uuid}.getsmart.bind(this)(${uuid}, 'val.$scopes.length', -1)){
// console.log(${uuid}.scopeCode)
try{
${uuid}.currentScopeCode.block.body.push(
${uuid}.getBabel().template.ast(
${
/*javascript*/ `\`
return \${${uuid}.scopedEval('${uuid}')}
\``
}
)
)
}catch(err){console.log(3,err)}
try{
${uuid}.wrapper = ${uuid}.getBabel().template.ast(
${
/*javascript*/ `\`
function anonymous(){}
\``
}
)
}catch(err){console.log(4,err)}
// console.log(${uuid}.wrapper)
// console.log(${uuid}.scopeCode)
${uuid}.wrapper.body.body.push(${uuid}.scopeCode)
${uuid}.scopeCode = ${uuid}.wrapper
${uuid}.scopeCode = ${uuid}.getBabel().generator(
${uuid}.scopeCode
).code
// console.log(${uuid}.scopeCode)
${uuid}.scopeCode = eval("("+${uuid}.scopeCode+")")
// console.log(${uuid}.scopeCode.toString())
try {
${uuid}.val.$scopedEval = ${uuid}.scopeCode()
}catch(err){console.log(5,err)}
// console.log(${uuid}.val.$scopedEval)
// return ${uuid}.scopeCode.toString()
return ${uuid}.val.$scopedEval
} else {
return eval(${
/*javascript*/ `\`(\${${uuid}.createScopedEval('${uuid}')})\``
})(${uuid})
}
}
`
return ret
}
export const defineVariable = (uuid) => {
return /*javascript*/ `
${uuid.variableType} ${uuid.variableKey} = ${uuid}.$scope[${uuid}.variableKey]
Object.defineProperty(
${uuid}.$scope,
${uuid}.variableKey,
{
get(){
return ${uuid.variableKey}
},
set(val){
${uuid.variableKey} = val
},
enumerable: true
}
)
`
}
export const scopedEval = (uuid) => {
const ret = /*javascript*/ `function scopedEval(${uuid}){
if(typeof ${uuid} == 'string'){
${uuid} = {
val: {
js: ${uuid}
}
}
} else if(typeof ${uuid} == 'function' && typeof ${uuid}.toString == 'function'){
${uuid} = {
val: {
js: ${uuid}.toString()
}
}
}
try {
${uuid}.ret = eval('('+${uuid}.val.js+')')
} catch(err1){
try {
${uuid}.ret = eval('({'+${uuid}.val.js+'})')
${uuid}.keys = Object.keys(${uuid}.ret)
${uuid}.ret = ${uuid}.ret[${uuid}.keys[0]]
} catch(err2){
try {
${uuid}.ret = eval('({b:'+ ${uuid}.val.js +'})').b
} catch(err3){
console.error(err1)
console.error(err2)
console.error(err3)
}
}
}
return ${uuid}.ret
}
`
return ret
}
export const jsUUID = (prefix = "uuid") => {
return prefix + uuid().replace(/-/g, "")
}
export const context = (opts) => {
const uuid = gosmart.bind(this)(opts, "path.context.scope.uuid", jsUUID())
return eval(/*javascript*/ `
(
function(){
${contextObject(uuid)}
return ${uuid}
}
)()
`)
}
export const contextObject = (uuid) => {
return /*javascript*/ `
let ${uuid} = {
$$uuid: '${uuid}',
$closure: {},
$variableMap: {},
$functionScoper: (func)=>{
Object.defineProperty(
func,
'$scopes',
{
value: (function(arr){
for (var i = 0, arr2 = new Array(arr.length); i < arr.length; i++) {
arr2[i] = arr[i];
}
return arr2
})((typeof ${uuid} != 'undefined') ? ${uuid}.$scopes : []),
enumerable: true
}
)
Object.defineProperty(
func,
'$context',
{
value: ${uuid}
}
)
return func
},
$add: (type, name, value)=>{
${uuid}.$closure[name] = value
${uuid}.$variableMap[name] = type
},
$scopes: (function(arr){
for (var i = 0, arr2 = new Array(arr.length); i < arr.length; i++) {
arr2[i] = arr[i];
}
return arr2
})((typeof $context == 'object') ? $context.$scopes : []),
$variableMaps: (function(arr){
for (var i = 0, arr2 = new Array(arr.length); i < arr.length; i++) {
arr2[i] = arr[i];
}
return arr2
})((typeof $context == 'object') ? $context.$variableMaps : []),
$contexts: {},
$contextsList: [],
$parentContexts: [],
$contextStatus: "var",
$mode: (eval("var ${uuid}1 = null"), (typeof ${
uuid + "1"
} === "undefined")) ? "strict" : "non-strict",
}
${uuid}.$functionScoper = ${uuid}.$functionScoper(${uuid}.$functionScoper)
${uuid}.$scopes.splice(0,0,${uuid}.$closure)
${uuid}.$variableMaps.splice(0,0,${uuid}.$variableMap)
var globalThis = globalThis || global || window || {}
${uuid}.$contextStatus = ${uuid}.$mode == 'strict' ? '' : 'var'
try {
eval(${
/*javascript*/ `\`\${${uuid}.$contextStatus} $context = $context || ${uuid}\``
})
} catch(err){
${uuid}.$contextStatus = ''
}
eval(${
/*javascript*/ `\`\${${uuid}.$contextStatus} $context = $context || ${uuid}\``
})
if(typeof $context == 'object' && $context != ${uuid} && $context.$contexts instanceof Object){
$context.$contexts[${uuid}.$$uuid] = $context.$contexts[${uuid}.$$uuid] || []
${uuid}.$$instance = $context.$contexts[${uuid}.$$uuid].push(${uuid})-1
${uuid}.$parentContexts.push($context)
$context.$contextsList.push(${uuid})
}
if(!globalThis.$contexts){
globalThis.$contexts = {}
globalThis.$contexts[${uuid}.$$uuid] = [${uuid}]
globalThis.$contextsList = [${uuid}]
${uuid}.$$instance = 0
} else if(
globalThis.${uuid}s instanceof Object
&& ${uuid}.$parentContexts.length == 0
&& typeof ${uuid}.$$instance == 'undefined'
){
globalThis.${uuid}s[${uuid}.$$uuid] = globalThis.$contexts[${uuid}.$$uuid] || []
${uuid}.$$instance = globalThis.$contexts[${uuid}.$$uuid].push(${uuid})-1
globalThis.$contextsList.push(${uuid})
}
{
let $context = ${uuid}
}
`
}
export const createContext = (opts) => {
schema(opts, {
wrapBody: true,
})
const node = opts.aster(/*javascript*/ `
${contextObject(opts.uuid)}
`)
node[0].declarations[0].contextDeclaration = true
// so the $functionScoper function doesn't get wrapped or have $context inserted
const property3 = node[0].declarations[0].init.properties[3]
property3.value.scoperWrapped = true
property3.value.body.scopeInitialized = true
const property3ScopesValue =
property3.value.body.body[0].expression.arguments[2].properties[0].value
property3ScopesValue.callee.scoperWrapped = true
property3ScopesValue.callee.body.scopeInitialized = true
const property3ForStatement = property3ScopesValue.callee.body.body[0]
property3ForStatement.body.scopeInitialized = true
property3ForStatement.init.declarations[0].inScope = true
property3ForStatement.init.declarations[1].inScope = true
// so the $add function doesn't get wrapped or have $context inserted
const property4 = node[0].declarations[0].init.properties[4]
property4.value.scoperWrapped = true
property4.value.body.scopeInitialized = true
// so the $scopes self-invoking function doesn't get wrapped or have $context inserted
const property5 = node[0].declarations[0].init.properties[5]
property5.value.callee.scoperWrapped = true
property5.value.callee.body.scopeInitialized = true
const property5ForStatement = property5.value.callee.body.body[0]
property5ForStatement.body.scopeInitialized = true
property5ForStatement.init.declarations[0].inScope = true
property5ForStatement.init.declarations[1].inScope = true
// so the $variableMaps self-invoking function doesn't get wrapped or have $context inserted
const property6 = node[0].declarations[0].init.properties[6]
property6.value.callee.scoperWrapped = true
property6.value.callee.body.scopeInitialized = true
const property6ForStatement = property6.value.callee.body.body[0]
property6ForStatement.body.scopeInitialized = true
property6ForStatement.init.declarations[0].inScope = true
property6ForStatement.init.declarations[1].inScope = true
const node6 = node[6]
// make sure try statement block doesn't get scoped either
node6.block.scopeInitialized = true
// make sure catch statement block doesn't get scoped either
node6.handler.body.scopeInitialized = true
const node8 = node[8]
// make sure if statement block doesn't get scoped either
node8.consequent.scopeInitialized = true
const node9 = node[9]
// make sure if statement block doesn't get scoped either
node9.consequent.scopeInitialized = true
// make sure else if statement block doesn't get scoped either
node9.alternate.consequent.scopeInitialized = true
const node10 = node[10]
node10.scopeInitialized = true
node10.inheritScope = true
node[node.length - 1].lastContextNode = true
if (opts.wrapBody) {
const bodyWrapper = node[node.length - 1]
bodyWrapper.body.push(...opts.path.node.body)
}
addBindingsToContext({ ...opts, node })
// let addContextToScopeNode = scopeVar({
// uuid,
// key: '$context',
// type: 'let',
// aster
// })
// wrapper.body.splice(1,0,addContextToScopeNode)
return node
}
export const createInlineContext = (opts) => {
const wrapperString = /*javascript*/ `
for(let ${opts.uuid} = function(){
// node goes here
return ${opts.uuid}
}() ; a<1;a++){}
`
const inlineContextNode = opts.aster(wrapperString).init.declarations[0]
const contextBody = createContext({ ...opts, wrapBody: false })
inlineContextNode.init.callee.body.body.splice(0, 0, ...contextBody)
inlineContextNode.contextDeclaration = true
return inlineContextNode
}
export const addBindingsToContext = (opts) => {
for (const key in opts.path.scope.bindings) {
const binding = opts.path.scope.bindings[key]
if (binding.kind == "var") {
const newNode = scopeVar({
...opts,
key,
type: binding.kind,
})
opts.node.splice(opts.node.length - 1, 0, newNode)
}
}
}
export const scopeVarCode = (opts) => {
const ret = /*javascript*/ `
Object.defineProperty(
${opts.uuid}.$closure,
${stringify(opts.key)},
{
get(){
return ${opts.key}
},
set(val){
${opts.key} = val
},
enumerable: true
}
) &&
(${opts.uuid}.$variableMap["${opts.key}"] = "${opts.type}")
`
return ret
}
export const scopeVarInlineCode = (opts) => {
const ret = /*javascript*/ `
let ${jsUUID()} = (
${scopeVarCode(opts)}
)
`
return ret
}
export const scopeVar = (opts = {}) => {
let string
let thirdArg
let node
if (opts.inline) {
string = scopeVarInlineCode(opts)
node = opts.aster(string)
thirdArg = node.declarations[0].init.left.arguments[2]
node.declarations[0].inScope = true
} else {
string = scopeVarCode(opts)
node = opts.aster(string)
thirdArg = node.expression.left.arguments[2]
}
const getter = thirdArg.properties[0]
const setter = thirdArg.properties[1]
getter.body.scopeInitialized = true
setter.body.scopeInitialized = true
getter.body.scoperWrapped = true
setter.body.scoperWrapped = true
getter.scoperWrapped = true
setter.scoperWrapped = true
if (opts.inline) return node.declarations[0]
return node
}
export const functionWrapper = (uuid, path, aster) => {
const wrapper = aster(/*javascript*/ `
${uuid}.$functionScoper()
`)
wrapper.expression.arguments.push(path.node)
return wrapper
}
export const bodyInsert = (index, body, aster, ...things) => {
body.splice(index, 0, ...things)
return things.length
}
export const initBlock = (path, aster) => {
if (!path.node.scopeInitialized) {
path.node.scopeInitialized = true
const uuid = getPathUUID({ path })
const contextNode = createContext({ uuid, aster, path })
path.node.body = contextNode
}
}
export const getNodeUUID = (opts) => {
if (
opts.node &&
opts.node.type != "BlockStatement" &&
opts.node.type != "Program"
)
return getNodeUUID({
...opts,
node: opts.node.body || opts.node.block,
})
return gosmart.bind(this)(opts.node, "uuid", jsUUID())
}
export const getPathUUID = (opts) => {
if (
opts.path.context.scope.path.node.inheritScope ||
opts.path.scope.path.node.inheritScope
)
return getPathUUID({ ...opts, path: opts.path.parentPath })
return getNodeUUID({
...opts,
node: opts.path.context.scope.path.node,
})
}
export const babelPlugin = (babel) => {
const aster = babel.template.ast
const metaVisitor = {
Program(path) {
initBlock(path, aster)
},
BlockStatement(path) {
initBlock(path, aster)
},
ForInStatement() {
// nothing
},
// ForInStatement (path) {
// path = path
// },
ObjectMethod(path) {
const name = path.node.key.name
let replacement = aster(/*javascript*/ `
let a = {
${name}: function ${name}(){}
}
`)
replacement = replacement.declarations[0].init.properties[0]
replacement.value.body = path.node.body
replacement.value.params = path.node.params
path.replaceWith(replacement)
},
Function(path) {
if (
path.type != "FunctionDeclaration" &&
!path.node.scoperWrapped &&
!path.node.body.scoperWrapped
) {
path.node.scoperWrapped = true
path.node.body.scoperWrapped = true
const uuid = getPathUUID({ path })
const replacement = functionWrapper(uuid, path, aster)
path.replaceWith(replacement)
}
},
FunctionDeclaration(path) {
if (!path.node.scoped) {
path.node.scoped = true
const parentBlock = path.scope.parent
try {
parentBlock.block.body.forEach((node) => {
if (node.lastContextNode) {
const uuid = getPathUUID({ path })
const newNode = aster(/*javascript*/ `
${uuid}.$functionScoper(${path.node.id.name})
`)
node.body.splice(1, 0, newNode)
throw new Error("break foreach")
}
})
} catch (err) {
// nothing
}
}
},
VariableDeclarator(path) {
if (!path.node.inScope) {
path.node.inScope = true
const parentPath = getsmart.bind(this)(path, "parentPath", undefined)
if (
// this is for inline let and const declarations in normal
// js blocks
(parentPath.node.kind == "let" || parentPath.node.kind == "const") &&
// we check the length of declarations because we only have to do inline replacement
// if there's a chance another declaration might use a former one
parentPath.node.declarations.length > 1 &&
!(
parentPath.parentPath.node.type == "ForInStatement" ||
parentPath.parentPath.node.type == "ForOfStatement" ||
parentPath.parentPath.node.type == "ForStatement"
)
) {
const uuid = getPathUUID({ path })
if (uuid) {
const indexInParent = parentPath.node.declarations.indexOf(
path.node
)
const newDeclaration = scopeVar({
aster,
inline: true,
uuid,
key: parentPath.node.declarations[indexInParent].id.name,
type: parentPath.node.kind,
})
parentPath.node.declarations.splice(
indexInParent + 1,
0,
newDeclaration
)
}
} else if (
//
(parentPath.node.kind == "let" ||
parentPath.node.kind == "var" ||
parentPath.node.kind == "const") &&
// only do this for singular declarations
parentPath.node.declarations.length < 2 &&
// and check if variable is declared inside a ForX statement
(parentPath.parentPath.node.type == "ForInStatement" ||
parentPath.parentPath.node.type == "ForOfStatement" ||
parentPath.parentPath.node.type == "ForStatement")
) {
const uuid = getPathUUID({ path })
if (uuid) {
const indexInParent = parentPath.node.declarations.indexOf(
path.node
)
const newNode = scopeVar({
aster,
uuid,
key: parentPath.node.declarations[indexInParent].id.name,
type: parentPath.node.kind,
})
parentPath.parentPath.node.body.body.splice(0, 0, newNode)
}
} else if (
// this is a special case for when ForStatements get their own scope
(parentPath.node.kind == "let" || parentPath.node.kind == "const") &&
// we check the length of declarations because we only have to do inline replacement
// if there's a chance another declaration might use a former one
parentPath.node.declarations.length > 1 &&
parentPath.parentPath.node.type == "ForStatement"
) {
// if the first declaration isn't our context declaration, insert one
const uuid = gosmart.bind(this)(path, "scope.uuid", jsUUID())
if (!parentPath.node.declarations[0].contextDeclaration) {
const inlineContextDeclaration = createInlineContext({
path,
uuid,
aster,
})
parentPath.node.declarations.splice(0, 0, inlineContextDeclaration)
}
if (uuid) {
const indexInParent = parentPath.node.declarations.indexOf(
path.node
)
const newDeclaration = scopeVar({
aster,
inline: true,
uuid,
key: parentPath.node.declarations[indexInParent].id.name,
type: parentPath.node.kind,
})
parentPath.node.declarations.splice(
indexInParent + 1,
0,
newDeclaration
)
}
} else if (
parentPath.node.kind == "let" ||
parentPath.node.kind == "const"
) {
const uuid = getPathUUID({ path })
if (uuid) {
const indexInParent = parentPath.node.declarations.indexOf(
path.node
)
const newNode = scopeVar({
aster,
uuid,
key: parentPath.node.declarations[indexInParent].id.name,
type: parentPath.node.kind,
})
parentPath.insertAfter(newNode)
}
} else {
// let uuid = getPathUUID({path})
// if(uuid){
// let indexInParent = parentPath.node.declarations.indexOf(path.node)
// let newNode = scopeVar({
// aster,
// uuid,
// key: parentPath.node.declarations[indexInParent].id.name,
// type: parentPath.node.kind
// })
// parentPath.insertAfter(newNode)
// }
}
}
},
}
const ret = {
visitor: metaVisitor,
// visitor: {
// Program(path){
// path.traverse(metaVisitor)
// }
// }
}
return ret
}
export const transform = (src, opts = {}) => {
return getBabel().transform(src, {
plugins: [babelPlugin],
...opts,
})
}
/** non-parser stuff */
export const stripUuids = (thing, seen = []) => {
try {
delete thing.uuid
} catch {
// nothing
}
if (typeof thing === "object") {
const keys = Object.keys(thing)
for (const key of keys) {
const val = thing[key]
if (!seen.includes(val)) {
seen.push(val)
stripUuids(val, seen)
}
}
}
}
export const dupe = (obj, opts = {}) => {
return parse(stringify(obj, opts), opts)
}
export const clone = (obj, opts = {}) => {
return dupe(obj, opts)
}
export const schema = (obj1, obj2, opts = {}) => {
if (!opts.noSchemaClone) {
obj2 = clone(obj2, opts)
}
return merge(obj1, obj2, {
...opts,
})
}
export const create = (obj1, obj2, opts) => {
const ret = merge(obj1, obj2, {
clone: true,
...opts,
})
return ret
}
export const merge = (value1, value2, opts = {}, seen = new Map()) => {
if (seen.has(value1)) return seen.get(value1)
if (value1 instanceof Array && value2 instanceof Array) {
return value1
}
// base case non-basic value
if (basic(value1) && !opts.overwrite) {
return value1
}
if (opts.overwrite && basic(value2)) {
return value2
}
if (!basic(value1) && basic(value2)) {
return value1
}
if (opts.clone) {
value1 = clone(value1)
value2 = clone(value2)
}
const props = Object.keys(value2)
props.forEach((prop) => {
const propertyValue1 = value1[prop]
if (prop in value1 && basic(propertyValue1) && !opts.overwrite) {
return
}
const propertyValue2 = value2[prop]
seen.set(value1, value1)
let newVal = propertyValue2
if (prop in value1) {
newVal = merge.bind(this)(
propertyValue1,
propertyValue2,
{ ...opts, ...{ clone: false } },
seen
)
}
if (
getsmart.bind(this)(local.vue, "reactiveSetter", false) &&
getsmart.bind(this)(
this,
"$set",
getsmart.bind(this)(local.vue, "Vue.set", false)
) &&
value1
) {
const setToUse = this.$set || local.vue.Vue.set
setToUse?.(value1, prop, newVal)
if (
typeof getsmart.bind(this)(window, "$store.commit", undefined) ==
"function"
) {
window.$store.commit(local.vue.basePath || "graph/thing")
}
} else {
value1[prop] = newVal
if (
getsmart.bind(this)(local.vue, "store", false) &&
typeof getsmart.bind(this)(window, "$store.commit", undefined) ==
"function"
) {
window.$store.commit(local.vue.basePath || "graph/thing")
}
}
})
return value1
}
export const basic = (value) => {
const valueType = typeof value
const ret =
!(
valueType == "object" ||
valueType == "array" ||
valueType == "function"
) || value === null
return ret
}
export const mod = (args, mod) => {
return mod(args) || args
}
// transform(value, fn, path, ret={}){
// return forEach(value, fn, path, ret)
// },
export const deepForEach = (
value,
fn,
path,
ret = {},
seens = { originals: [], clones: [] },
first = true
) => {
path = path || ""
if (first) {
value = { "": value }
}
// if(!(typeof value == 'string' || typeof value == 'boolean' || typeof value == 'number')){
// seens.originals.push(value)
// }
if (Array.isArray(value)) {
forEachArray(value, fn, path, ret, seens)
} else if (typeof value == "object") {
forEachObject(value, fn, path, ret, seens)
}
return ret[""]
}
export const forEachObject = (obj, fn, path, ret, seens) => {
for (const key in obj) {
const deepPath = path ? `${path}.${key}` : key
const primitive =
typeof obj[key] == "string" ||
typeof obj[key] == "boolean" ||
typeof obj[key] == "number"
if (primitive || seens.originals.indexOf(obj[key]) < 0) {
if (!primitive) {
seens.originals.push(obj[key])
}
// Note that we always use obj[key] because it might be mutated by forEach
fn(obj[key], key, obj, deepPath, ret, seens)
deepForEach(obj[key], fn, deepPath, ret, seens, false)
}
}
}
export const forEachArray = (array, fn, path, ret = {}, seens) => {
array.forEach((value, index, arr) => {
const primitive =
typeof value == "string" ||
typeof value == "boolean" ||
typeof value == "number"
if (primitive || seens.originals.indexOf(value) < 0) {
if (!primitive) {
seens.originals.push(value)
}
const deepPath = `${path}.${index}`
fn(value, index, arr, deepPath, ret, seens)
// Note that we use arr[index] because it might be mutated by forEach
deepForEach(arr[index], fn, deepPath, ret, seens, false)
}
})
}
export const setThing = ({
option,
list = getsmart.bind(this)(objList),
obj = true,
keys = ["uuid", "_id", "id"],
keymatchtype,
push,
strings,
targets,
} = {}) => {
let index = thingIndex({
option,
list,
obj,
keys,
keymatchtype,
strings,
})
if (obj == "debug") {
console.log("index")
console.log(index)
console.log("list")
console.log(list)
}
if (index >= 0 && list) {
if (targets && targets.length && typeof targets.length == "number") {
for (let i = 0; i < targets.length; i++) {
const value = getsmart.bind(this)(option, targets[i], undefined)
if (value) {
setsmart.bind(this)(list[index], targets[i], value)
}
}
} else {
list.splice(index, 1, option)
if (
getsmart.bind(this)(local.vue, "reactiveSetter", false) &&
getsmart.bind(this)(
this,
"$set",
getsmart.bind(this)(local.vue, "Vue.set", false)
)
) {
if (
!localStorage.getItem("vuexWriteLock") &&
typeof getsmart.bind(this)(window, "$store.commit", undefined) ==
"function"
) {
window.$store.commit(local.vue.basePath || "graph/thing")
}
} else if (
getsmart.bind(this)(local.vue, "store", false) &&
!localStorage.getItem("vuexWriteLock") &&
typeof getsmart.bind(this)(window, "$store.commit", undefined) ==
"function"
) {
window.$store.commit(local.vue.basePath || "graph/thing")
}
}
// list[index] = option
} else if (push && list) {
if (
getsmart.bind(this)(local.vue, "reactiveSetter", false) ||
getsmart.bind(this)(local.vue, "store", false)
) {
list.splice(list.length, 0, option)
if (
!localStorage.getItem("vuexWriteLock") &&
typeof getsmart.bind(this)(window, "$store.commit", undefined) ==
"function"
) {
window.$store.commit(local.vue.basePath || "graph/thing")
}
} else {
list.push(option)
}
index = list.length - 1
}
return index
}
export const setThings = ({
options,
list = getsmart.bind(this)(objList),
obj = true,
keys = ["uuid", "_id", "id"],
keymatchtype,
push,
async,
} = {}) => {
if (options && options instanceof Array && list) {
for (const option of options) {
if (async) {
new Promise(() => {
setThing({
option,
list,
obj,
keys,
keymatchtype,
push,
})
})
} else {
setThing({
option,
list,
obj,
keys,
keymatchtype,
push,
})
}
}
}
return list
}
export const optIn = (
option,
list = getsmart.bind(this)(stringList),
obj,
keys = ["uuid", "_id", "id"],
keymatchtype,
index
) => {
if (typeof option === "object") {
obj = true
}
if (!obj && list && list.indexOf && list.indexOf(option) >= 0) {
return index ? list.indexOf(option) : true
} else if (obj && list && typeof list.length == "number") {
for (let i = 0; i < list.length; i++) {
if (!(keys && typeof keys.length == "number")) return
for (let indKey = 0; indKey < keys.length; indKey++) {
if (keymatchtype == "broad") {
if (
list[i] &&
getsmart.bind(this)(list[i], keys[indKey], undefined) ==
getsmart.bind(this)(option, keys[indKey], undefined) &&
getsmart.bind(this)(list[i], keys[indKey], undefined) !== undefined
) {
return index ? i : true
} else if (
list[i] &&
typeof list[i] == "string" &&
list[i] == getsmart.bind(this)(option, keys[indKey], undefined) &&
getsmart.bind(this)(option, keys[indKey], undefined) !== undefined
) {
return index ? i : true
}
} else {
if (
list[i] &&
getsmart.bind(this)(list[i], keys[indKey], undefined) ==
getsmart.bind(this)(option, keys[indKey], undefined) &&
getsmart.bind(this)(list[i], keys[indKey], undefined) !== undefined
) {
if (indKey == keys.length - 1) {
return index ? i : true
}
} else if (
list[i] &&
typeof list[i] == "string" &&
list[i] == getsmart.bind(this)(option, keys[indKey], undefined) &&
getsmart.bind(this)(option, keys[indKey], undefined) !== undefined
) {
if (indKey == keys.length - 1) {
return index ? i : true
}
}
}
}
}
}
return index ? -1 : false
}
export const thingIn = ({
option,
list = getsmart.bind(this)(objList),
obj = true,
keys = ["uuid", "_id", "id"],
keymatchtype,
retIndex,
// vue = local.vue
} = {}) => {
if (typeof option === "object") {
obj = true
}
if (!obj && list && list.indexOf && list.indexOf(option) >= 0) {
if (retIndex) {
return list.indexOf(option)
} else {
return true
}
} else if (obj && list && typeof list.length == "number") {
for (let i = 0; i < list.length; i++) {
if (!(keys && typeof keys.length == "number")) return
for (let indKey = 0; indKey < keys.length; indKey++) {
if (keymatchtype == "broad") {
if (
list[i] &&
getsmart.bind(this)(list[i], keys[indKey], undefined) ==
getsmart.bind(this)(option, keys[indKey], undefined) &&
getsmart.bind(this)(list[i], keys[indKey], undefined) !== undefined
) {
if (retIndex) {
return i
} else {
return true
}
} else if (
list[i] &&
typeof list[i] == "string" &&
list[i] == getsmart.bind(this)(option, keys[indKey], undefined) &&
getsmart.bind(this)(option, keys[indKey], undefined) !== undefined
) {
if (retIndex) {
return i
} else {
return true
}
}
} else {
if (
list[i] &&
getsmart.bind(this)(list[i], keys[indKey], undefined) ==
getsmart.bind(this)(option, keys[indKey], undefined) &&
getsmart.bind(this)(list[i], keys[indKey], undefined) !== undefined
) {
if (indKey == keys.length - 1) {
if (retIndex) {
return i
} else {
return true
}
}
} else if (
list[i] &&
typeof list[i] == "string" &&
list[i] == getsmart.bind(this)(option, keys[indKey], undefined) &&
getsmart.bind(this)(option, keys[indKey], undefined) !== undefined
) {
if (indKey == keys.length - 1) {
if (retIndex) {
return i
} else {
return true
}
}
}
}
}
}
}
if (retIndex) {
return -1
} else {
return false
}
}
export const optsIn = (
options,
list = getsmart.bind(this)(stringList),
obj,
keys = ["uuid", "_id", "id"],
keymatchtype
) => {
if (!(options instanceof Array)) return true
for (const option of options) {
// if(typeof option === 'object'){
// obj = true
// }
if (!obj && list && list.indexOf && list.indexOf(option) >= 0) {
// return true
} else if (obj && list) {
for (let i = 0; i < list.length; i++) {
if (!optIn(option, list[i], obj, keys, keymatchtype)) {
return false
}
}
} else {
return false
}
}
return true
}
export const thingsIn = ({
options,
list = getsmart.bind(this)(stringList),
obj,
keys = ["uuid", "_id", "id"],
keymatchtype,
} = {}) => {
if (!(options instanceof Array)) return true
for (const option of options) {
// if(typeof option === 'object'){
// obj = true
// }
if (!obj && list && list.indexOf && list.indexOf(option) >= 0) {
// return true
} else if (obj && list && typeof list.length == "number") {
for (let i = 0; i < list.length; i++) {
if (!optIn(option, list[i], obj, keys, keymatchtype)) {
return false
}
}
} else {
return false
}
}
return true
}
export const anyOptsIn = (
options,
list = getsmart.bind(this)(stringList),
obj,
keys = ["uuid", "_id", "id"],
keymatchtype
) => {
if (!(options instanceof Array)) return false
for (const option of options) {
// if(typeof option === 'object'){
// obj = true
// }
if (!obj && list && list.indexOf && list.indexOf(option) >= 0) {
return true
} else if (obj && list && typeof list.length == "number") {
for (let i = 0; i < list.length; i++) {
if (optIn(option, list[i], obj, keys, keymatchtype)) {
return true
}
}
}
}
return false
}
export const anyThingsIn = ({
options,
list = getsmart.bind(this)(stringList),
obj,
keys = ["uuid", "_id", "id"],
keymatchtype,
} = {}) => {
if (!(options instanceof Array)) return false
for (const option of options) {
// if(typeof option === 'object'){
// obj = true
// }
if (!obj && list && list.indexOf && list.indexOf(option) >= 0) {
return true
} else if (obj && list && typeof list.length == "number") {
for (let i = 0; i < list.length; i++) {
if (optIn(option, list[i], obj, keys, keymatchtype)) {
return true
}
}
}
}
return false
}
export const optIndex = (
option,
list = getsmart.bind(this)(stringList),
obj,
keys = ["uuid", "_id", "id"],
keymatchtype
) => {
if (typeof option === "object") {
obj = true
}
if (obj && list && keys && typeof list.length == "number") {
for (let i = 0; i < list.length; i++) {
if (optIn(option, list, obj, keys, keymatchtype)) {
return i
}
}
} else if (list) {
return list.indexOf(option)
}
return -1
}
export const thingIndex = ({
option,
list,
obj,
keys = ["uuid", "_id", "id"],
keymatchtype,
strings,
} = {}) => {
if (typeof option === "object") {
obj = true
}
if (obj && list && keys) {
const index = thingIn({
option,
list,
obj,
keys,
keymatchtype,
strings,
retIndex: true,
})
return index
} else if (list) {
return list.indexOf(option)
}
return -1
}
export const pushOpt = (
option,
list = getsmart.bind(this)(stringList),
obj,
keys = ["uuid", "_id", "id"],
keymatchtype,
index
) => {
if (
typeof list == "object" &&
!optIn(option, list, obj, keys, keymatchtype)
) {
if (
getsmart.bind(this)(local.vue, "reactiveSetter", false) ||
getsmart.bind(this)(local.vue, "store", false)
) {
list.splice(list.length, 0, option)
if (
!localStorage.getItem("vuexWriteLock") &&
typeof getsmart.bind(this)(window, "$store.commit", undefined) ==
"function"
) {
window.$store.commit(local.vue.basePath || "graph/thing")
}
} else {
list.push(option)
}
}
return index
? optIn(option, list, obj, keys, keymatchtype, index)
: optIn(option, list, obj, keys, keymatchtype, index)
}
export const addOpt = (
option,
list = getsmart.bind(this)(stringList),
obj,
keys = ["uuid", "_id", "id"],
keymatchtype,
index
) => {
if (typeof list == "object") {
if (
getsmart.bind(this)(local.vue, "reactiveSetter", false) ||
getsmart.bind(this)(local.vue, "store", false)
) {
list.splice(list.length, 0, option)
if (
!localStorage.getItem("vuexWriteLock") &&
typeof getsmart.bind(this)(window, "$store.commit", undefined) ==
"function"
) {
window.$store.commit(local.vue.basePath || "graph/thing")
}
} else {
list.push(option)
}
}
return index
? optIn(option, list, obj, keys, keymatchtype, index)
: optIn(option, list, obj, keys, keymatchtype, index)
}
export const pushThing = ({
option,
list = getsmart.bind(this)(stringList),
obj,
keys = ["uuid", "_id", "id"],
keymatchtype,
} = {}) => {
if (
typeof list == "object" &&
!thingIn({ option, list, obj, keys, keymatchtype })
) {
if (
getsmart.bind(this)(local.vue, "reactiveSetter", false) ||
getsmart.bind(this)(local.vue, "store", false)
) {
list.splice(list.length, 0, option)
if (
!localStorage.getItem("vuexWriteLock") &&
typeof getsmart.bind(this)(window, "$store.commit", undefined) ==
"function"
) {
window.$store.commit(local.vue.basePath || "graph/thing")
}
} else {
list.push(option)
}
}
}
export const pushOpts = (
options,
list = getsmart.bind(this)(stringList),
obj,
keys = ["uuid", "_id", "id"],
keymatchtype
) => {
if (!(options instanceof Array)) return
for (const option of options) {
pushOpt(option, list, obj, keys, keymatchtype)
}
}
export const pushThings = ({
options,
list = getsmart.bind(this)(stringList),
obj,
keys = ["uuid", "_id", "id"],
keymatchtype,
} = {}) => {
if (!(options instanceof Array)) return
for (const option of options) {
pushThing({ option, list, obj, keys, keymatchtype })
}
}
export const popOpt = (
option,
list = getsmart.bind(this)(stringList),
obj,
keys = ["uuid", "_id", "id"],
keymatchtype
) => {
if (typeof list == "object" && optIn(option, list, obj, keys, keymatchtype)) {
list.splice(optIndex(option, list, obj, keys, keymatchtype), 1)
if (
getsmart.bind(this)(local.vue, "reactiveSetter", false) ||
getsmart.bind(this)(local.vue, "store", false)
) {
if (
!localStorage.getItem("vuexWriteLock") &&
typeof getsmart.bind(this)(window, "$store.commit", undefined) ==
"function"
) {
window.$store.commit(local.vue.basePath || "graph/thing")
}
}
}
}
export const popThing = ({
option,
list = getsmart.bind(this)(stringList),
obj = true,
keys = ["uuid", "_id", "id"],
keymatchtype,
} = {}) => {
if (
typeof list == "object" &&
thingIn({
option,
list,
obj,
keys,
keymatchtype,
})
) {
list.splice(
thingIndex({
option,
list,
obj,
keys,
keymatchtype,
}),
1
)
if (
getsmart.bind(this)(local.vue, "reactiveSetter", false) ||
getsmart.bind(this)(local.vue, "store", false)
) {
if (
!localStorage.getItem("vuexWriteLock") &&
typeof getsmart.bind(this)(window, "$store.commit", undefined) ==
"function"
) {
window.$store.commit(local.vue.basePath || "graph/thing")
}
}
}
}
export const popOpts = (
options,
list = getsmart.bind(this)(stringList),
obj,
keys = ["uuid", "_id", "id"],
keymatchtype
) => {
if (!(options instanceof Array)) return
for (const option of options) {
popOpt(option, list, obj, keys, keymatchtype)
}
}
export const popThings = ({
options,
list = getsmart.bind(this)(stringList),
obj = true,
keys = ["uuid", "_id", "id"],
keymatchtype,
} = {}) => {
if (!(options instanceof Array)) return
for (const option of options) {
popOpt(option, list, obj, keys, keymatchtype)
}
}
export const toggleOpt = (
option,
list = getsmart.bind(this)(stringList),
obj,
keys = ["uuid", "_id", "id"],
keymatchtype
) => {
if (optIn(option, list, obj, keys, keymatchtype)) {
popOpt(option, list, obj, keys, keymatchtype)
} else {
pushOpt(option, list, obj, keys, keymatchtype)
}
}
export const toggleThing = ({
option,
list = getsmart.bind(this)(stringList),
obj = true,
keys = ["uuid", "_id", "id"],
keymatchtype,
} = {}) => {
if (optIn(option, list, obj, keys, keymatchtype)) {
popOpt(option, list, obj, keys, keymatchtype)
} else {
pushOpt(option, list, obj, keys, keymatchtype)
}
}
export const toggleOpts = (
options,
list = getsmart.bind(this)(stringList),
obj,
keys = ["uuid", "_id", "id"],
keymatchtype
) => {
if (!(options instanceof Array)) return
for (const option in options) {
toggleOpt(option, list, obj, keys, keymatchtype)
}
}
export const toggleThings = ({
options,
list = getsmart.bind(this)(stringList),
obj = true,
keys = ["uuid", "_id", "id"],
keymatchtype,
} = {}) => {
if (!(options instanceof Array)) return
for (const option in options) {
if (optIn(option, list, obj, keys, keymatchtype)) {
popOpt(option, list, obj, keys, keymatchtype)
} else {
pushOpt(option, list, obj, keys, keymatchtype)
}
}
}
// no use right now
// export const ratchetOpt = (
// option,
// list,
// obj,
// keys = ['uuid', '_id', 'id'],
// keymatchtype
// ) => {
// // find(obj, property, equals){
// // if(getsmart.bind(this)(obj, 'constructor', undefined) == Array){
// // for(var i=0; i<obj.length; i++){
// // find(obj[i], )
// // }
// // }
// // },
// }
export const getsmart = (
obj,
property,
defaultValue?: any,
context?: any,
schema?: any
) => {
if (!property && obj && typeof obj == "string") {
property = obj.split(".")
try {
obj = eval(property[0])
} catch (err) {
// console.error(err)
obj = property[0]
}
property = property.slice(1, property.length)
}
if (!property) {
if (context) {
return {
value: defaultValue,
undefined: true,
}
} else {
return defaultValue
}
}
// If the property list is in dot notation, convert to array
if (typeof property == "string") {
property = parsePropertyPath(property)
} else if (getsmart.bind(this)(property, "constructor", false) !== Array) {
if (context) {
return {
value: defaultValue,
undefined: true,
err: "properties path @property argument was not passed properly",
}
} else {
return defaultValue
}
}
const deepGetByArray = deepGetByArrayUnbound.bind(this)
return deepGetByArray(obj, property, defaultValue)
// In order to avoid constantly checking the type of the property
// we separate the real logic out into an inner function.
function deepGetByArrayUnbound(obj, propsArray, defaultValue) {
// This case getting to the last property but it not being ultimately defined
// Not just having a value of undefined
if (
propsArray.length > 0 &&
context &&
typeof obj == "object" &&
obj !== null &&
!(ee(propsArray[0]) in obj)
) {
return {
value: defaultValue,
undefined: true,
}
}
// If we have reached an undefined/null property
// then stop executing and return the default value.
// If no default was provided it will be undefined.
if (
typeof obj == "undefined" ||
obj == null ||
(schema && obj.constructor.name !== schema)
) {
if (context) {
let undef = true
if (propsArray.length === 0) {
undef = false
}
return {
value: defaultValue,
undefined: undef,
schema: schema && obj.constructor.name === schema,
}
} else {
return defaultValue
}
} // If the path array has no more elements, we've reached
// the intended property and return its value
if (propsArray.length === 0) {
if (context) {
return {
value: obj,
undefined: false,
}
} else {
return obj
}
} // Prepare our found property and path array for recursion
const nextObj = obj[ee(propsArray[0])]
const remainingProps = propsArray.slice(1)
return deepGetByArray(nextObj, remainingProps, defaultValue)
}
}
export const escapePropertyPath = (path = "") => {
const newPath = escapeEscapes(path)
return '["' + newPath + '"]'
}
export const epp = (path = "") => {
return escapePropertyPath(path)
}
export const escapeEscapes = (path = "") => {
let newPath = ""
for (let i in path) {
i = +i
const char = path[i]
if (i > 0 && i < path.length - 1) {
const prevChar = path[i - 1]
const nextChar = path[i + 1]
const openingArrayPath = char === '"' && prevChar === "["
// && (nextChar !== "\\" || i === path.length-1)
const closingArrayPath =
char === '"' && nextChar === "]" && prevChar !== "\\"
// let offset = 0
// if(openingArrayPath) offset = 1
if (openingArrayPath || closingArrayPath) {
newPath += "\\"
// path = path.slice(0,i+offset)+"\\"+path.slice(i+offset,path.length)
}
}
newPath += char
}
return newPath
}
export const ee = (path = "") => {
return escapeEscapes(path)
}
// TODO
// Make parsing use \" or \'
// Currently only uses \"
export const parsePropertyPath = (path = "") => {
const array = []
let readingArrayBasedPath = false
let i = 0
let push = false
let pushed = false
while (i < path.length) {
const arrayPathStart = path[i] == "[" && path[i + 1] == '"'
const escapedStart = !(path[i + 1] !== "\\" || i === 0)
if (readingArrayBasedPath) {
// we found the end of an array delimited path
const arrayPathEnd = path[i] == '"' && path[i + 1] == "]"
const escapedEnd = !(path[i - 1] !== "\\" || i == 0)
if (arrayPathEnd && !escapedEnd) {
i += 1
readingArrayBasedPath = false
if (!pushed) push = true
} else {
// if the path includes an "escaped" array based path begin or end value
// do not push the escape character
if (
(path[i] == "\\" && path[i + 1] == '"' && path[i + 2] == "]") ||
(path[i - 1] == "[" && path[i] == "\\" && path[i + 1] == '"')
) {
// nothing
} else {
array[array.length - 1] += path[i]
}
}
} else if (path[i] == ".") {
if (!pushed) push = true
}
// we found the start of an array delimited path
else if (arrayPathStart && !escapedStart) {
readingArrayBasedPath = true
if (!pushed) push = true
i += 1
} else {
if (i == 0) array.push("")
array[array.length - 1] += path[i]
}
i++
if (push && i < path.length) {
pushed = true
array.push("")
push = false
} else {
pushed = false
}
}
return array
}
export const ppp = (path = "") => {
return this.parsePropertyPath(path)
}
export const parsePropertyArray = (pathArray) => {
let path = ""
if (pathArray instanceof Array) {
pathArray.forEach((subPath) => {
path += epp(subPath)
})
} else if (typeof pathArray === "string") {
return path
}
return path
}
export const ppa = (pathArray) => {
return this.parsePropertyArray(pathArray)
}
export const pathToArray = (path) => {
if (typeof path == "string") {
return parsePropertyPath(path)
} else {
return path
}
}
export const pathToString = (path) => {
if (typeof path == "string") {
let ret = parsePropertyPath(path)
ret = parsePropertyArray(ret)
return ret
} else {
const ret = parsePropertyArray(path)
return ret
}
}
export const setsmart = (obj, property, value, context?: any) => {
if (!property && typeof obj == "string") {
property = obj.split(".")
try {
obj = eval(property[0])
} catch (err) {
obj = property[0]
}
property = property.slice(1, property.length)
}
// If the property list is in dot notation, convert to array
if (typeof property == "string") {
property = parsePropertyPath(property)
} else if (getsmart.bind(this)(property, "constructor", false) !== Array) {
if (context) {
return {
value: value,
undefined: true,
err: "properties path @property argument was not passed properly",
}
} else {
return value
}
}
// if no obj make obj
if (!obj || (typeof obj !== "object" && typeof obj !== "function")) obj = {}
const deepSetByArray = deepSetByArrayUnbound.bind(this)
if (property) {
return deepSetByArray(obj, property, value)
} else {
if (
getsmart.bind(this)(local.vue, "reactiveSetter", false) &&
getsmart.bind(this)(
this,
"$set",
getsmart.bind(this)(local.vue, "Vue.set", false)
) &&
obj
) {
const setToUse = this.$set || local.vue.Vue.set
setToUse(obj, undefined, value)
if (
typeof getsmart.bind(this)(window, "$store.commit", undefined) ==
"function"
) {
window.$store.commit(local.vue.basePath || "graph/thing")
}
} else {
obj = value
if (
getsmart.bind(this)(local.vue, "store", false) &&
typeof getsmart.bind(this)(window, "$store.commit", undefined) ==
"function"
) {
window.$store.commit(local.vue.basePath || "graph/thing")
}
}
if (context) {
return {
value: obj,
undefined: false,
err: "there were no properties passed",
}
} else {
return obj
}
}
// In order to avoid constantly checking the type of the property
// we separate the real logic out into an inner function.
function deepSetByArrayUnbound(obj, propsArray, value) {
// If the path array has only 1 more element, we've reached
// the intended property and set its value
if (propsArray.length == 1) {
if (
getsmart.bind(this)(local.vue, "reactiveSetter", false) &&
getsmart.bind(this)(
this,
"$set",
getsmart.bind(this)(local.vue, "Vue.set", false)
) &&
obj
) {
const setToUse = this.$set || local.vue.Vue.set
setToUse(obj, ee(propsArray[0]), value)
if (
typeof getsmart.bind(this)(window, "$store.commit", undefined) ==
"function"
) {
window.$store.commit(local.vue.basePath || "graph/thing")
}
} else {
// TODO: make parent object new object so that react can see the change
obj[ee(propsArray[0])] = value
if (
getsmart.bind(this)(local.vue, "store", false) &&
typeof getsmart.bind(this)(window, "$store.commit", undefined) ==
"function"
) {
window.$store.commit(local.vue.basePath || "graph/thing")
}
}
if (context) {
return {
value: obj[ee(propsArray[0])],
undefined: false,
}
} else {
return obj[ee(propsArray[0])]
}
}
// Prepare our path array for recursion
const remainingProps = propsArray.slice(1)
// check if next prop is object
if (typeof obj[ee(propsArray[0])] !== "object") {
// If we have reached an undefined/null property
if (
getsmart.bind(this)(local.vue, "reactiveSetter", false) &&
getsmart.bind(this)(
this,
"$set",
getsmart.bind(this)(local.vue, "Vue.set", false)
) &&
obj
) {
const setToUse = this.$set || local.vue.Vue.set
setToUse(obj, ee(propsArray[0]), {})
if (
typeof getsmart.bind(this)(window, "$store.commit", undefined) ==
"function"
) {
window.$store.commit(local.vue.basePath || "graph/thing")
}
} else {
obj[ee(propsArray[0])] = {}
if (
getsmart.bind(this)(local.vue, "store", false) &&
typeof getsmart.bind(this)(window, "$store.commit", undefined) ==
"function"
) {
window.$store.commit(local.vue.basePath || "graph/thing")
}
}
}
return deepSetByArray(obj[ee(propsArray[0])], remainingProps, value)
}
}
export const deletesmart = (obj, property) => {
if (!property && typeof obj == "string") {
property = obj.split(".")
try {
obj = eval(property[0])
} catch (err) {
// console.error(err)
obj = property[0]
}
property = property.slice(1, property.length)
}
// If the property list is in dot notation, convert to array
if (typeof property == "string") {
property = parsePropertyPath(property)
}
const parentPathArray = property.slice(0, property.length - 1)
const path = property[property.length - 1]
const parentObj = getsmart(obj, parentPathArray, {})
if (
getsmart.bind(this)(local.vue, "reactiveSetter", false) &&
getsmart.bind(this)(
this,
"$set",
getsmart.bind(this)(local.vue, "Vue.set", false)
) &&
obj
) {
this.$delete(parentObj, path)
} else {
delete parentObj[path]
}
}
export const pushSmart = (array, value) => {
if (
getsmart.bind(this)(local.vue, "reactiveSetter", false) &&
getsmart.bind(this)(
this,
"$set",
getsmart.bind(this)(local.vue, "Vue.set", false)
) &&
array
) {
array.push(value)
if (
typeof getsmart.bind(this)(window, "$store.commit", undefined) ==
"function"
) {
window.$store.commit(local.vue.basePath || "graph/thing")
}
} else {
array.push(value)
}
}
export const gosmart = (obj, property, value, context, schema) => {
// stands for get or set smart
let get = getsmart.bind(this)(
obj,
property,
value,
true,
schema ? absoluteType.bind(this)(value) : false
)
if (get.undefined || (schema && get.schema === false)) {
get = setsmart.bind(this)(obj, property, get.value, context)
if (context) {
return get
} else {
return get
}
} else {
return get.value
}
}
export const gosmarter = (obj, property, value, context, schema = true) => {
return gosmart.bind(this)(obj, property, value, context, schema)
}
export const absoluteType = (value) => {
let type
try {
type = value.constructor.name
} catch (e) {
if (typeof value === "undefined") type = "undefined"
if (value === null) type = "null"
}
return type
}
export const vgosmart = (obj, property, value, context) => {
// stands for v-model get or set smart
// return value from property path, either gotten or smartly set
return {
get: () => {
let get = getsmart.bind(this)(obj, property, value, true)
if (get.undefined) {
get = setsmart.bind(this)(obj, property, get.value, context)
}
if (context) {
return get
} else {
return getsmart.bind(this)(get, "value", get)
}
},
set: (val) => {
setsmart.bind(this)(obj, property, val)
},
}
}
export const getsmartval = (obj, property, defaultValue) => {
// get the value of a property path based off its type
const target = getsmart.bind(this)(obj, property, defaultValue)
if (target && target.type) {
if (target[target.type]) {
return target[target.type]
} else {
return defaultValue
}
} else if (target) {
return target
}
return defaultValue
}
export const safestring = (something) => {
return smarts.stringify(something || "")
}
export const safeparse = (something) => {
return smarts.parse(something || "")
}
export const flatten = (arrays, func = (i) => i) => {
const flat = []
arrays.forEach((array) => {
if (Array.isArray(array)) {
flat.push(...flatten(array))
} else {
flat.push(func(array))
}
})
return flat
}
export const mapsmart = (list, keyProperty = "title", returnExistant) => {
return new Promise((resolve, reject) => {
if (!keyProperty) {
reject()
} else if (list && typeof list.length == "number") {
if (list.length == 0) {
if (
(returnExistant &&
getsmart.bind(this)(list, "mapped." + returnExistant, false)) ||
!returnExistant
) {
resolve(true)
} else if (returnExistant) {
resolve(false)
} else {
resolve()
}
}
if (!list.mapped || typeof list.mapped === "boolean") {
if (
getsmart.bind(this)(local.vue, "reactiveSetter", false) &&
getsmart.bind(this)(
this,
"$set",
getsmart.bind(this)(local.vue, "Vue.set", false)
) &&
list
) {
const setToUse = this.$set || local.vue.Vue.set
setToUse(list, "mapped", {})
} else {
list["mapped"] = {}
}
}
for (let i = 0; i < list.length; i++) {
if (typeof list[i] !== "string") {
if (
getsmart.bind(this)(local.vue, "reactiveSetter", false) &&
getsmart.bind(this)(
this,
"$set",
getsmart.bind(this)(local.vue, "Vue.set", false)
) &&
list.mapped
) {
const setToUse = this.$set || local.vue.Vue.set
setToUse(list.mapped, list[i][keyProperty], list[i])
} else {
list["mapped"][list[i][keyProperty]] = list[i]
}
if (i == list.length - 1) {
if (
(returnExistant &&
getsmart.bind(this)(list, "mapped." + returnExistant, false)) ||
!returnExistant
) {
resolve(true)
} else if (returnExistant) {
resolve(false)
} else {
resolve()
}
}
}
// else if(populate){
// var funCounter = this.funCounter
// this.funCounter = this.funCounter + 1
// getThing({
// thing: list[i],
// clientId: this._uid,
// funCounter
// })
// this.$options.sockets['giveThing'] = data => {
// if(this._uid == data.clientId && data.funCounter == funCounter){
// (this.$set || local.vue.Vue.set)(list, i.toString(), data.thing)
// (this.$set || local.vue.Vue.set)(list.mapped, list[i][keyProperty], list[i])
// }
// if(i==list.length-1){
// if((returnExistant && getsmart.bind(this)(list, 'mapped.'+returnExistant, false)) || !returnExistant){
// resolve(true)
// } else if(returnExistant) {
// resolve(false)
// } else {
// resolve()
// }
// }
// }
// }
else if (i == list.length - 1) {
if (
(returnExistant &&
getsmart.bind(this)(list, "mapped." + returnExistant, false)) ||
!returnExistant
) {
resolve(true)
} else if (returnExistant) {
resolve(false)
} else {
resolve()
}
}
}
// if(list.mapped && !list.mapped['agora-client-mapped']){
// (this.$set || local.vue.Vue.set)(list.mapped, 'agora-client-mapped', true)
// }
}
})
}
export const domval = (thing) => {
return getsmart.bind(this)(thing, "properties.description", "")
}
export const getParent = (levels = Infinity) => {
if (typeof levels == "string") levels = (levels.match(/\.\./g) || []).length
if (levels >= this.pathAsArray.length - 1) {
return this.pathAsArray[0]
}
const level = this.pathAsArray.length - 1 - levels
return this.pathAsArray[level]
}
export const getThing = (props = {}) => {
const { list = getsmart.bind(this)(objList), defaultValue = undefined } =
props
const index = thingIn({
...props,
retIndex: true,
})
if (index >= 0) {
return list[index]
} else {
return defaultValue
}
}
export const equal = (obj1, obj2, seen = []) => {
if (obj1 && obj2 && typeof obj1 == "object" && typeof obj2 == "object") {
seen.push(obj1, obj2)
//Loop through properties in object 1
for (const p in obj1) {
//Check property exists on both objects
if (
typeof obj1.hasOwnProperty == "function" &&
typeof obj2.hasOwnProperty == "function" &&
Object.prototype.hasOwnProperty.call(obj1, p) !==
Object.prototype.hasOwnProperty.call(obj2, p)
)
return false
switch (typeof obj1[p]) {
//Deep compare objects
case "object":
if (seen.indexOf(obj1[p]) < 0 && !equal(obj1[p], obj2[p], seen))
return false
break
//Compare function code
case "function":
if (
typeof obj2[p] == "undefined" ||
obj1[p].toString() != obj2[p].toString()
)
return false
break
//Compare values
default:
if (obj1[p] != obj2[p]) return false
}
}
//Check object 2 for any extra properties
for (const p in obj2) {
if (!(p in obj1)) return false
}
return true
}
}
export const mergeall = (array, options) => {
if (!Array.isArray(array)) {
throw new Error("first argument should be an array")
}
return array.reduce(function (prev, next) {
return merge(prev, next, options)
}, {})
}
// export const smarts with all other exports
export const smarts = {
local,
t,
getBabel,
uuid,
pause,
save,
toJavascript,
toJavascriptAux,
createObjectProperties,
serialize,
stringify,
replacer,
setKnown,
stringifier,
primitives,
Primitives,
play,
load,
parse,
parser,
revive,
createScopedEval,
defineVariable,
scopedEval,
jsUUID,
context,
contextObject,
createContext,
createInlineContext,
addBindingsToContext,
scopeVarCode,
scopeVarInlineCode,
scopeVar,
functionWrapper,
bodyInsert,
initBlock,
getNodeUUID,
getPathUUID,
babelPlugin,
transform,
stripUuids,
dupe,
clone,
schema,
create,
merge,
basic,
mod,
deepForEach,
forEachObject,
forEachArray,
setThing,
setThings,
optIn,
thingIn,
optsIn,
thingsIn,
anyOptsIn,
anyThingsIn,
optIndex,
thingIndex,
pushOpt,
addOpt,
pushThing,
pushOpts,
pushThings,
popOpt,
popThing,
popOpts,
popThings,
toggleOpt,
toggleThing,
toggleOpts,
toggleThings,
// ratchetOpt,
getsmart,
escapePropertyPath,
epp,
escapeEscapes,
ee,
parsePropertyPath,
ppp,
parsePropertyArray,
ppa,
pathToArray,
pathToString,
setsmart,
deletesmart,
pushSmart,
gosmart,
gosmarter,
absoluteType,
vgosmart,
getsmartval,
safestring,
safeparse,
flatten,
mapsmart,
domval,
getParent,
getThing,
equal,
mergeall,
}