Fix light bulb errors (also add a dev mode to build script)

This commit is contained in:
prescientmoon 2024-11-28 14:58:57 +01:00
parent fbd05e4dd0
commit d6ba5609e8
Signed by: prescientmoon
SSH key fingerprint: SHA256:UUF9JT2s8Xfyv76b8ZuVL7XrmimH4o49p4b+iexbVH4
8 changed files with 170 additions and 161 deletions

View file

@ -5,15 +5,17 @@ import * as fs from 'fs'
const serve = process.env.ESBUILD_SERVE === '1' const serve = process.env.ESBUILD_SERVE === '1'
const baseurl = process.env.ESBUILD_BASEURL || '' const baseurl = process.env.ESBUILD_BASEURL || ''
const nodeEnv = process.env.NODE_ENV
const isProd = nodeEnv !== 'development'
console.log(`Building with baseurl ${baseurl}`) console.log(`Building with baseurl ${baseurl}`)
const ctx = await esbuild.context({ const ctx = await esbuild.context({
entryPoints: ['src/index.ts'], entryPoints: ['src/index.ts'],
minify: true, minify: isProd,
bundle: true, bundle: true,
metafile: true, metafile: true,
splitting: true, splitting: isProd,
outdir: 'dist', outdir: 'dist',
format: 'esm', format: 'esm',
target: ['es2020'], target: ['es2020'],
@ -23,7 +25,8 @@ const ctx = await esbuild.context({
'.svg': 'file' '.svg': 'file'
}, },
define: { define: {
'process.env.BASEURL': JSON.stringify(baseurl) 'process.env.BASEURL': JSON.stringify(baseurl),
'process.env.NODE_ENV': JSON.stringify(nodeEnv)
}, },
plugins: [ plugins: [
htmlPlugin({ htmlPlugin({

View file

@ -1,34 +1,34 @@
{ {
"name": "erratic-gate", "name": "erratic-gate",
"version": "1.0.0", "version": "1.0.0",
"type": "module", "type": "module",
"scripts": { "scripts": {
"dev": "ESBUILD_SERVE=1 node ./build.js", "dev": "NODE_ENV=development ESBUILD_SERVE=1 node ./build.js",
"build": "node ./build.js", "build": "node ./build.js",
"check": "tsc" "check": "tsc"
}, },
"devDependencies": { "devDependencies": {
"@craftamap/esbuild-plugin-html": "^0.6.1", "@craftamap/esbuild-plugin-html": "^0.6.1",
"@types/deepmerge": "^2.2.0", "@types/deepmerge": "^2.2.0",
"@types/mainloop.js": "^1.0.5", "@types/mainloop.js": "^1.0.5",
"@types/node": "^20.8.9", "@types/node": "^20.8.9",
"@types/react-dom": "^18.2.14", "@types/react-dom": "^18.2.14",
"@types/react-router-dom": "^4.3.4", "@types/react-router-dom": "^4.3.4",
"esbuild": "^0.19.5", "esbuild": "^0.19.5",
"esbuild-sass-plugin": "^2.16.0", "esbuild-sass-plugin": "^2.16.0",
"typescript": "^5.0.2" "typescript": "^5.0.2"
}, },
"dependencies": { "dependencies": {
"@eix-js/utils": "0.0.6", "@eix-js/utils": "0.0.6",
"@material-ui/core": "^4.2.1", "@material-ui/core": "^4.2.1",
"deepmerge": "^4.0.0", "deepmerge": "^4.0.0",
"keycode": "^2.2.0", "keycode": "^2.2.0",
"mainloop.js": "^1.0.4", "mainloop.js": "^1.0.4",
"react": "^16.8.6", "react": "^16.8.6",
"react-dom": "^16.8.6", "react-dom": "^16.8.6",
"react-router-dom": "^5.0.1", "react-router-dom": "^5.0.1",
"react-toastify": "^5.3.2", "react-toastify": "^5.3.2",
"rxjs": "^6.5.2", "rxjs": "^6.5.2",
"rxjs-hooks": "^0.5.1" "rxjs-hooks": "^0.5.1"
} }
} }

View file

@ -33,7 +33,11 @@ async function main() {
} }
} }
new EventSource('/esbuild').addEventListener('change', () => location.reload()) if (process.env.BASEURL === 'development') {
new EventSource('/esbuild').addEventListener('change', () =>
location.reload()
)
}
// Call entry // Call entry
main().catch((error) => { main().catch((error) => {

View file

@ -5,68 +5,68 @@ import { filter, take } from 'rxjs/operators'
* Keeps track of what a task should do and where the output should be delivered * Keeps track of what a task should do and where the output should be delivered
*/ */
export interface Task<T> { export interface Task<T> {
output: Subject<T> output: Subject<T>
execute: () => Promise<T> execute: () => Promise<T>
} }
/** /**
* Used to execute a number of async tasks * Used to execute a number of async tasks
*/ */
export class ExecutionQueue<T> { export class ExecutionQueue<T> {
/** /**
* An array of all the tasks wich need to be executed * An array of all the tasks wich need to be executed
*/ */
private tasks: Task<T>[] = [] private tasks: Task<T>[] = []
/** /**
* Keeps track of the current task * Keeps track of the current task
*/ */
private current: Promise<T> | null private current: Promise<T> | null
/** /**
* Wheather the tasks should continue beeing executed * Whether the tasks should continue being executed
*/ */
public active = true public active = true
/** /**
* Adds a new task to the queue * Adds a new task to the queue
* *
* @param task The task to add * @param task The task to add
*/ */
public push(task: () => Promise<T>) { public push(task: () => Promise<T>) {
const executionSubject = new Subject<T>() const executionSubject = new Subject<T>()
const executionPromise = executionSubject.pipe(take(1)).toPromise() const executionPromise = executionSubject.pipe(take(1)).toPromise()
this.tasks.push({ this.tasks.push({
output: executionSubject, output: executionSubject,
execute: task execute: task
}) })
if (!this.current) { if (!this.current) {
this.next() this.next()
}
return executionPromise
} }
/** return executionPromise
* Executes the next task in the queue }
*/
private next() {
const task = this.tasks.shift()
if (task) { /**
this.current = task.execute() * Executes the next task in the queue
*/
private next() {
const task = this.tasks.shift()
this.current.then(result => { if (task) {
task.output.next(result) this.current = task.execute()
if (this.active) { this.current.then((result) => {
this.next() task.output.next(result)
}
}) if (this.active) {
} else { this.next()
this.current = null
} }
})
} else {
this.current = null
} }
}
} }

View file

@ -24,7 +24,7 @@ const lightTemplate: PartialTemplate = {
activation: ` activation: `
const { main, active } = context.colors const { main, active } = context.colors
context.color(context.getBinary(2) ? active : main) context.color(context.getBinary(0) ? active : main)
` `
}, },
integration: { integration: {

View file

@ -295,8 +295,9 @@ export class Gate {
props: Property[] = this.template.properties.data, props: Property[] = this.template.properties.data,
target: GateProps = this.props, target: GateProps = this.props,
path: string[] = [] path: string[] = []
) { ): boolean {
// We don't want to update until every prop has been created // We don't want to update until every prop has been created
let shouldUpdate = false
let lockUpdates = true let lockUpdates = true
if (this.template.properties.enabled) { if (this.template.properties.enabled) {
@ -304,7 +305,7 @@ export class Gate {
if (isGroup(prop)) { if (isGroup(prop)) {
const { groupName } = prop const { groupName } = prop
target[groupName] = {} as GateProps target[groupName] = {} as GateProps
this.assignProps( shouldUpdate ||= this.assignProps(
typeof source[groupName] === 'object' typeof source[groupName] === 'object'
? (source[groupName] as PropsSave) ? (source[groupName] as PropsSave)
: {}, : {},
@ -312,36 +313,38 @@ export class Gate {
target[groupName] as GateProps, target[groupName] as GateProps,
[...path, groupName] [...path, groupName]
) )
} else {
const { name, base, needsUpdate } = prop
shouldUpdate ||= needsUpdate || false
continue const subject = new BehaviorSubject(
source.hasOwnProperty(name) ? source[name] : base
)
target[name] = subject
this.subscriptions.push(
subject.subscribe((value) => {
if (!lockUpdates && needsUpdate && path.length === 0) {
return this.update()
}
if (path.length === 0) {
return
}
this.updateNestedProp([...path, name], value)
})
)
} }
const { name, base, needsUpdate } = prop
const subject = new BehaviorSubject(
source.hasOwnProperty(name) ? source[name] : base
)
target[name] = subject
this.subscriptions.push(
subject.subscribe((value) => {
if (!lockUpdates && needsUpdate && path.length === 0) {
return this.update()
}
if (path.length === 0) {
return
}
this.updateNestedProp([...path, name], value)
})
)
} }
} }
lockUpdates = false lockUpdates = false
this.update()
if (path.length === 0 && shouldUpdate) this.update()
return shouldUpdate
} }
/** /**

View file

@ -1,6 +1,5 @@
import { GateTemplate, Property, RawProp } from './types/GateTemplate' import { GateTemplate, RawProp } from './types/GateTemplate'
import { categories } from '../saving/data/categories' import { categories } from '../saving/data/categories'
import { getRendererSafely } from '../logic-gates/helpers/getRendererSafely'
export const DefaultGateTemplate: GateTemplate = { export const DefaultGateTemplate: GateTemplate = {
metadata: { metadata: {

View file

@ -1,59 +1,59 @@
export class LocalStore<T> { export class LocalStore<T> {
public constructor(public name: string) { public constructor(public name: string) {
if (!localStorage.getItem(name)) { if (!localStorage.getItem(name)) {
localStorage.setItem(name, '{}') localStorage.setItem(name, '{}')
} }
}
public getAll(): Record<string, T> {
const raw = localStorage.getItem(this.name)
if (!raw)
throw new Error(
`An error occured when accesing ${this.name} in the local storage!`
)
else {
return JSON.parse(raw)
}
}
public ls(): string[] {
return Object.keys(this.getAll())
}
public *[Symbol.iterator](): Iterable<T> {
for (const item of this.ls()) {
return this.get(item)
}
}
public get(key = 'index'): T | undefined {
return this.getAll()[key]
}
public set(key: string | T, value?: T) {
let finalKey = key as string
let finalValue = value as T
if (typeof key !== 'string' || value === undefined) {
finalKey = 'index'
finalValue = key as T
} }
public getAll(): Record<string, T> { const currentData = this.getAll()
const raw = localStorage.getItem(this.name)
if (!raw) currentData[finalKey] = finalValue
throw new Error( localStorage.setItem(this.name, JSON.stringify(currentData))
`An error occured when accesing ${this.name} in the local storage!` }
)
else {
return JSON.parse(raw)
}
}
public ls(): string[] { public delete(key = 'index') {
return Object.keys(this.getAll()) const all = this.getAll()
} const last = all[key]
public *[Symbol.iterator](): Iterable<T> { delete all[key]
for (const item of this.ls()) {
return this.get(item)
}
}
public get(key = 'index'): T | undefined { localStorage.setItem(this.name, JSON.stringify(all))
return this.getAll()[key]
}
public set(key: string | T, value?: T) { return last
let finalKey = key as string }
let finalValue = value as T
if (typeof key !== 'string' || value === undefined) {
finalKey = 'index'
finalValue = key as T
}
const currentData = this.getAll()
currentData[finalKey] = finalValue
localStorage.setItem(this.name, JSON.stringify(currentData))
}
public delete(key = 'index') {
const all = this.getAll()
const last = all[key]
delete all[key]
localStorage.setItem(this.name, JSON.stringify(all))
return last
}
} }