feat: editing

This commit is contained in:
2023-11-02 00:16:18 +01:00
parent fade2f978e
commit ef309d603e
22 changed files with 409 additions and 517 deletions

109
src/lib/undo-redo.ts Normal file
View File

@@ -0,0 +1,109 @@
import {persistentWritable} from "$lib/storage"
import {derived} from "svelte/store"
import {serializeActions} from "$lib/serial/chord"
import type {Chord} from "$lib/serial/chord"
import {deviceChords, deviceLayout, deviceSettings} from "$lib/serial/connection"
export enum ChangeType {
Layout,
Chord,
Setting,
}
export interface LayoutChange {
type: ChangeType.Layout
id: number
layer: number
action: number
}
export interface ChordChange {
type: ChangeType.Chord
actions: number[]
phrase: number[]
}
export interface SettingChange {
type: ChangeType.Setting
id: number
setting: number
}
export interface ChangeInfo {
isApplied: boolean
isCommitted?: boolean
}
export type Change = LayoutChange | ChordChange | SettingChange
export const changes = persistentWritable<Change[]>("changes", [])
export interface Overlay {
layout: [Map<number, number>, Map<number, number>, Map<number, number>]
chords: Map<bigint, number[]>
settings: Map<number, number>
}
export const overlay = derived(changes, changes => {
console.time("overlay building")
const overlay: Overlay = {
layout: [new Map(), new Map(), new Map()],
chords: new Map(),
settings: new Map(),
}
for (const change of changes) {
switch (change.type) {
case ChangeType.Layout:
overlay.layout[change.layer].set(change.id, change.action)
break
case ChangeType.Chord:
overlay.chords.set(serializeActions(change.actions), change.phrase)
break
case ChangeType.Setting:
overlay.settings.set(change.id, change.setting)
break
}
}
console.timeEnd("overlay building")
return overlay
})
export const settings = derived([overlay, deviceSettings], ([overlay, settings]) =>
settings.map<{value: number} & ChangeInfo>((value, id) => ({
value: overlay.settings.get(id) ?? value,
isApplied: !overlay.settings.has(id),
})),
)
export type KeyInfo = {action: number} & ChangeInfo
export const layout = derived([overlay, deviceLayout], ([overlay, layout]) =>
layout.map(
(actions, layer) =>
actions.map<KeyInfo>((action, id) => ({
action: overlay.layout[layer].get(id) ?? action,
isApplied: !overlay.layout[layer].has(id),
})) as [KeyInfo, KeyInfo, KeyInfo],
),
)
export type ChordInfo = Chord & ChangeInfo
export const chords = derived([overlay, deviceChords], ([overlay, chords]) =>
chords.map<ChordInfo>(chord => {
const key = serializeActions(chord.actions)
if (overlay.chords.has(key)) {
return {
actions: chord.actions,
phrase: overlay.chords.get(key)!,
isApplied: false,
}
} else {
return {
actions: chord.actions,
phrase: chord.phrase,
isApplied: true,
}
}
}),
)