import { Connection, DeviceTemplate, LogicField, LogicFields, LogicType, Slot, SlotTemplate, SlotOccupant, SlotOccupantTemplate, SlotLogicType, ConnectionCableNetwork, SlotType, } from "ic10emu_wasm"; import { html, css, HTMLTemplateResult } from "lit"; import { customElement, property, query, state } from "lit/decorators.js"; import { BaseElement, defaultCss } from "../components"; import { VMDeviceMixin } from "./base_device"; import { default as uFuzzy } from "@leeoniya/ufuzzy"; import "@shoelace-style/shoelace/dist/components/card/card.js"; import "@shoelace-style/shoelace/dist/components/icon/icon.js"; import "@shoelace-style/shoelace/dist/components/tooltip/tooltip.js"; import "@shoelace-style/shoelace/dist/components/input/input.js"; import "@shoelace-style/shoelace/dist/components/details/details.js"; import "@shoelace-style/shoelace/dist/components/tab/tab.js"; import "@shoelace-style/shoelace/dist/components/tab-panel/tab-panel.js"; import "@shoelace-style/shoelace/dist/components/tab-group/tab-group.js"; import "@shoelace-style/shoelace/dist/components/copy-button/copy-button.js"; import "@shoelace-style/shoelace/dist/components/select/select.js"; import "@shoelace-style/shoelace/dist/components/badge/badge.js"; import "@shoelace-style/shoelace/dist/components/option/option.js"; import "@shoelace-style/shoelace/dist/components/drawer/drawer.js"; import "@shoelace-style/shoelace/dist/components/icon/icon.js"; import SlInput from "@shoelace-style/shoelace/dist/components/input/input.js"; import { parseIntWithHexOrBinary, parseNumber, structuralEqual, } from "../utils"; import SlSelect from "@shoelace-style/shoelace/dist/components/select/select.js"; import SlDrawer from "@shoelace-style/shoelace/dist/components/drawer/drawer.js"; import { DeviceDB, DeviceDBEntry } from "./device_db"; import { connectionFromDeviceDBConnection } from "./utils"; @customElement("vm-device-card") export class VMDeviceCard extends VMDeviceMixin(BaseElement) { image_err: boolean; @property({ type: Boolean }) open: boolean; constructor() { super(); this.open = false; } static styles = [ ...defaultCss, css` :host { display: block; box-sizing: border-box; } .card { width: 100%; box-sizing: border-box; } .image { width: 4rem; height: 4rem; } .header { display: flex; flex-direction: row; } .header-name { display: flex; flex-direction: row; width: 100%; flex-grow: 1; align-items: center; flex-wrap: wrap; } .device-card { --padding: var(--sl-spacing-small); } .device-name::part(input) { width: 10rem; } .device-id::part(input) { width: 7rem; } .device-name-hash::part(input) { width: 7rem; } .slot-header.image { width: 1.5rem; height: 1.5rem; border: var(--sl-panel-border-width) solid var(--sl-panel-border-color); border-radius: var(--sl-border-radius-medium); background-color: var(--sl-color-neutral-0); } sl-divider { --spacing: 0.25rem; } sl-button[variant="success"] { /* Changes the success theme color to purple using primitives */ --sl-color-success-600: var(--sl-color-purple-700); } sl-button[variant="primary"] { /* Changes the success theme color to purple using primitives */ --sl-color-primary-600: var(--sl-color-cyan-600); } sl-button[variant="warning"] { /* Changes the success theme color to purple using primitives */ --sl-color-warning-600: var(--sl-color-amber-600); } sl-tab-group { margin-left: 1rem; margin-right: 1rem; --indicator-color: var(--sl-color-purple-600); --sl-color-primary-600: var(--sl-color-purple-600); } sl-tab::part(base) { padding: var(--sl-spacing-small) var(--sl-spacing-medium); } sl-tab-group::part(base) { max-height: 20rem; overflow-y: auto; } `, ]; private _deviceDB: DeviceDB; get deviceDB(): DeviceDB { return this._deviceDB; } @state() set deviceDB(val: DeviceDB) { this._deviceDB = val; } connectedCallback(): void { super.connectedCallback(); window.VM!.addEventListener( "vm-device-db-loaded", this._handleDeviceDBLoad.bind(this), ); } _handleDeviceDBLoad(e: CustomEvent) { this.deviceDB = e.detail; } onImageErr(e: Event) { this.image_err = true; console.log("Image load error", e); } renderHeader(): HTMLTemplateResult { const activeIc = window.VM?.activeIC; const badges: HTMLTemplateResult[] = []; if (this.deviceID == activeIc?.id) { badges.push(html`db`); } activeIc?.pins?.forEach((id, index) => { if (this.deviceID == id) { badges.push( html`d${index}`, ); } }, this); return html`
Id Name Hash ${badges.map((badge) => badge)}
`; } renderFields(): HTMLTemplateResult { const fields = Array.from(this.fields.entries()); const inputIdBase = `vmDeviceCard${this.deviceID}Field`; return html` ${fields.map(([name, field], _index, _fields) => { return html` ${name} ${field.field_type} `; })} `; } lookupSlotOccupantImg( occupant: SlotOccupant | undefined, typ: SlotType, ): string { if (typeof occupant !== "undefined") { const hashLookup = (this.deviceDB ?? {}).names_by_hash ?? {}; const prefabName = hashLookup[occupant.prefab_hash] ?? "UnknownHash"; return `img/stationpedia/${prefabName}.png`; } else { return `img/stationpedia/SlotIcon_${typ}.png`; } } _onSlotImageErr(e: Event) { console.log("image_err", e); } static transparentImg = "" as const; renderSlot(slot: Slot, slotIndex: number): HTMLTemplateResult { const _fields = this.device.getSlotFields(slotIndex); const fields = Array.from(_fields.entries()); const inputIdBase = `vmDeviceCard${this.deviceID}Slot${slotIndex}Field`; const slotImg = this.lookupSlotOccupantImg(slot.occupant, slot.typ); return html` ${slotIndex} : ${slot.typ} ${ typeof slot.occupant !== "undefined" ? html` Occupant: ${slot.occupant.id} : ${slot.occupant.prefab_hash} Quantity: ${slot.occupant.quantity}/ ${slot.occupant.max_quantity} ` : "" }
${fields.map( ([name, field], _index, _fields) => html` ${name} ${field.field_type} `, )}
`; } renderSlots(): HTMLTemplateResult { return html`
${this.slots.map((slot, index, _slots) => this.renderSlot(slot, index))}
`; } renderReagents(): HTMLTemplateResult { return html``; } renderNetworks(): HTMLTemplateResult { const vmNetworks = window.VM!.networks; return html` < div class="networks" > ${ this.connections.map((connection, index, _conns) => { const conn = typeof connection === "object" ? connection.CableNetwork : null; return html` Connection:${index} ${vmNetworks.map( (net) => html`Network ${net}`, )} ${conn?.typ} `; }) } `; } renderPins(): HTMLTemplateResult { const pins = this.pins; const visibleDevices = window.VM!.visibleDevices(this.deviceID); return html` < div class="pins" > ${ pins?.map( (pin, index) => html` d${index} ${visibleDevices.map( (device, _index) => html` Device ${device.id} : ${device.name ?? device.prefabName} `, )} `, ) } `; } render(): HTMLTemplateResult { return html`
${this.renderHeader()}
Fields Slots Reagents Networks Pins ${this.renderFields()} ${this.renderSlots()} ${this.renderReagents()} ${this.renderNetworks()} ${this.renderPins()}
`; } _handleChangeID(e: CustomEvent) { const input = e.target as SlInput; const val = parseIntWithHexOrBinary(input.value); if (!isNaN(val)) { window.VM.changeDeviceId(this.deviceID, val); } else { input.value = this.deviceID.toString(); } } _handleChangeName(e: CustomEvent) { const input = e.target as SlInput; const name = input.value.length === 0 ? undefined : input.value; window.VM?.setDeviceName(this.deviceID, name); this.updateDevice(); } _handleChangeField(e: CustomEvent) { const input = e.target as SlInput; const field = input.getAttribute("key")!; const val = parseNumber(input.value); window.VM?.setDeviceField(this.deviceID, field, val); this.updateDevice(); } _handleChangeSlotField(e: CustomEvent) { const input = e.target as SlInput; const slot = parseInt(input.getAttribute("slotIndex")!); const field = input.getAttribute("key")!; const val = parseNumber(input.value); window.VM?.setDeviceSlotField(this.deviceID, slot, field, val); this.updateDevice(); } _handleChangeConnection(e: CustomEvent) { const select = e.target as SlSelect; const conn = parseInt(select.getAttribute("key")!); const val = select.value ? parseInt(select.value as string) : undefined; window.VM.setDeviceConnection(this.deviceID, conn, val); this.updateDevice(); } _handleChangePin(e: CustomEvent) { const select = e.target as SlSelect; const pin = parseInt(select.getAttribute("key")!); const val = select.value ? parseInt(select.value as string) : undefined; window.VM.setDevicePin(this.deviceID, pin, val); this.updateDevice(); } } @customElement("vm-device-list") export class VMDeviceList extends BaseElement { @state() accessor devices: number[]; static styles = [ ...defaultCss, css` .header { margin-bottom: 1rem; padding: 0.25rem 0.25rem; align-items: center; display: flex; flex-direction: row; width: 100%; box-sizing: border-box; } .device-list { display: flex; flex-direction: column; box-sizing: border-box; } .device-list-card { width: 100%; } .device-filter-input { margin-left: auto; } `, ]; constructor() { super(); this.devices = [...window.VM!.deviceIds]; } connectedCallback(): void { const root = super.connectedCallback(); window.VM?.addEventListener( "vm-devices-update", this._handleDevicesUpdate.bind(this), ); return root; } _handleDevicesUpdate(e: CustomEvent) { const ids = e.detail; if (!structuralEqual(this.devices, ids)) { this.devices = ids; this.devices.sort(); } } protected render(): HTMLTemplateResult { const deviceCards: HTMLTemplateResult[] = this.filteredDeviceIds.map( (id, _index, _ids) => html` `, ); const result = html`
Devices: ${this.devices.length} "
${deviceCards}
`; return result; } get filteredDeviceIds() { if (typeof this._filteredDeviceIds !== "undefined") { return this._filteredDeviceIds; } else { return this.devices; } } private _filteredDeviceIds: number[] | undefined; private _filter: string = ""; @query(".device-filter-input") accessor filterInput: SlInput; get filter() { return this._filter; } @state() set filter(val: string) { this._filter = val; this.performSearch(); } private filterTimeout: number | undefined; _handleFilterInput(_e: CustomEvent) { if (this.filterTimeout) { clearTimeout(this.filterTimeout); } const that = this; this.filterTimeout = setTimeout(() => { that.filter = that.filterInput.value; that.filterTimeout = undefined; }, 200); } performSearch() { if (this._filter) { const datapoints: [string, number][] = []; for (const device_id of this.devices) { const device = window.VM.devices.get(device_id); if (device) { if (typeof device.name !== "undefined") { datapoints.push([device.name, device.id]); } if (typeof device.prefabName !== "undefined") { datapoints.push([device.prefabName, device.id]); } } } const haystack: string[] = datapoints.map((data) => data[0]); const uf = new uFuzzy({}); const [_idxs, info, order] = uf.search(haystack, this._filter, 0, 1e3); const filtered = order?.map((infoIdx) => datapoints[info.idx[infoIdx]]); const deviceIds: number[] = filtered ?.map((data) => data[1]) ?.filter((val, index, arr) => arr.indexOf(val) === index) ?? []; this._filteredDeviceIds = deviceIds; } else { this._filteredDeviceIds = undefined; } } } @customElement("vm-add-device-button") export class VMAddDeviceButton extends BaseElement { static styles = [ ...defaultCss, css` .add-device-drawer { --size: 32rem; } .search-results { display: flex; flex-direction: row; overflow-x: auto; } .card { margin-top: var(--sl-spacing-small); margin-right: var(--sl-spacing-small); } .card + .card { } `, ]; @query("sl-drawer") accessor drawer: SlDrawer; @query(".device-search-input") accessor searchInput: SlInput; private _deviceDB: DeviceDB; private _strutures: Map = new Map(); private _datapoints: [string, string][] = []; private _haystack: string[] = []; get deviceDB() { return this._deviceDB; } @state() set deviceDB(val: DeviceDB) { this._deviceDB = val; this._strutures = new Map( Object.values(this.deviceDB.db) .filter((entry) => this.deviceDB.structures.includes(entry.name), this) .filter( (entry) => this.deviceDB.logic_enabled.includes(entry.name), this, ) .map((entry) => [entry.name, entry]), ); const datapoints: [string, string][] = []; for (const entry of this._strutures.values()) { datapoints.push( [entry.name, entry.name], [entry.title, entry.name], [entry.desc, entry.name], ); } const haystack: string[] = datapoints.map((data) => data[0]); this._datapoints = datapoints; this._haystack = haystack; this.performSearch(); } private _filter: string = ""; get filter() { return this._filter; } @state() set filter(val: string) { this._filter = val; this.performSearch(); } private _searchResults: DeviceDBEntry[]; private filterTimeout: number | undefined; performSearch() { if (this._filter) { const uf = new uFuzzy({}); const [_idxs, info, order] = uf.search( this._haystack, this._filter, 0, 1e3, ); const filtered = order?.map( (infoIdx) => this._datapoints[info.idx[infoIdx]], ); const names = filtered ?.map((data) => data[1]) ?.filter((val, index, arr) => arr.indexOf(val) === index) ?? []; this._searchResults = names.map((name) => this._strutures.get(name)!); } else { // clear our results and prefilter if the filter is empty this._searchResults = []; } } connectedCallback(): void { const root = super.connectedCallback(); window.VM!.addEventListener( "vm-device-db-loaded", this._handleDeviceDBLoad.bind(this), ); return root; } _handleDeviceDBLoad(e: CustomEvent) { this.deviceDB = e.detail; } renderSearchResults(): HTMLTemplateResult { const renderedResults: HTMLTemplateResult[] = this._searchResults?.map( (result) => html` `, ); return html`${renderedResults}`; } _handleDeviceAdd() { this.drawer.hide(); } render() { return html` Add Device Search Structures "
${this.renderSearchResults()}
{ this.drawer.hide(); }} > Close
`; } _handleSearchInput(e: CustomEvent) { if (this.filterTimeout) { clearTimeout(this.filterTimeout); } const that = this; this.filterTimeout = setTimeout(() => { that.filter = that.searchInput.value; that.filterTimeout = undefined; }, 200); } _handleAddButtonClick() { this.drawer.show(); } } @customElement("vm-device-template") export class VmDeviceTemplate extends BaseElement { private _deviceDB: DeviceDB; private image_err: boolean = false; static styles = [ ...defaultCss, css` .template-card { --padding: var(--sl-spacing-small); } .image { width: 3rem; height: 3rem; } .header { display: flex; flex-direction: row; } .card-body { // height: 18rem; overflow-y: auto; } sl-tab::part(base) { padding: var(--sl-spacing-small) var(--sl-spacing-medium); } sl-tab-group::part(base) { height: 14rem; overflow-y: auto; } `, ]; @state() fields: { [key in LogicType]?: LogicField }; @state() slots: SlotTemplate[]; @state() template: DeviceTemplate; @state() device_id: number | undefined; @state() device_name: string | undefined; @state() connections: Connection[]; constructor() { super(); this.deviceDB = window.VM!.db; } get deviceDB(): DeviceDB { return this._deviceDB; } @state() set deviceDB(val: DeviceDB) { this._deviceDB = val; this.setupState(); } private _prefab_name: string; get prefab_name(): string { return this._prefab_name; } @property({ type: String }) set prefab_name(val: string) { this._prefab_name = val; this.setupState(); } get dbDevice(): DeviceDBEntry { return this.deviceDB.db[this.prefab_name]; } setupState() { const slotlogicmap: { [key: number]: SlotLogicType[] } = {}; for (const [slt, slotIndexes] of Object.entries( this.dbDevice?.slotlogic ?? {}, )) { for (const slotIndex of slotIndexes) { const list = slotlogicmap[slotIndex] ?? []; list.push(slt as SlotLogicType); slotlogicmap[slotIndex] = list; } } this.fields = Object.fromEntries( Object.entries(this.dbDevice?.logic ?? {}).map(([lt, ft]) => { const value = lt === "PrefabHash" ? this.dbDevice.hash : 0.0; return [lt, { field_type: ft, value } as LogicField]; }), ); this.slots = (this.dbDevice?.slots ?? []).map( (slot, _index) => ({ typ: slot.typ, }) as SlotTemplate, ); const connections = Object.entries(this.dbDevice?.conn ?? {}).map( ([index, conn]) => [index, connectionFromDeviceDBConnection(conn)] as const, ); connections.sort((a, b) => { if (a[0] < b[0]) { return -1; } else if (a[0] > b[0]) { return 1; } else { return 0; } }); this.connections = connections.map((conn) => conn[1]); } connectedCallback(): void { super.connectedCallback(); window.VM!.addEventListener( "vm-device-db-loaded", this._handleDeviceDBLoad.bind(this), ); } _handleDeviceDBLoad(e: CustomEvent) { this.deviceDB = e.detail; } onImageErr(e: Event) { this.image_err = true; console.log("Image load error", e); } renderFields(): HTMLTemplateResult { const fields = Object.entries(this.fields); return html` ${fields.map(([name, field], _index, _fields) => { return html` ${name} ${field.field_type} `; })} `; } _handleChangeField(e: CustomEvent) { const input = e.target as SlInput; const field = input.getAttribute("key")! as LogicType; const val = parseNumber(input.value); this.fields[field].value = val; if (field === "ReferenceId" && val !== 0) { this.device_id = val; } this.requestUpdate(); } renderSlot(slot: Slot, slotIndex: number): HTMLTemplateResult { return html` `; } renderSlots(): HTMLTemplateResult { return html`
`; } renderReagents(): HTMLTemplateResult { return html``; } renderNetworks(): HTMLTemplateResult { const vmNetworks = window.VM!.networks; const connections = this.connections; return html`
${connections.map((connection, index, _conns) => { const conn = typeof connection === "object" ? connection.CableNetwork : null; return html` Connection:${index} ${vmNetworks.map( (net) => html`Network ${net}`, )} ${conn?.typ} `; })}
`; } _handleChangeConnection(e: CustomEvent) { const select = e.target as SlSelect; const conn = parseInt(select.getAttribute("key")!); const val = select.value ? parseInt(select.value as string) : undefined; (this.connections[conn] as ConnectionCableNetwork).CableNetwork.net = val; this.requestUpdate(); } renderPins(): HTMLTemplateResult { const device = this.deviceDB.db[this.prefab_name]; return html`
`; } render() { const device = this.dbDevice; return html`
${device?.name} ${device?.hash}
Add
Fields Slots Reagents Networks Pins ${this.renderFields()} ${this.renderSlots()} ${this.renderReagents()} ${this.renderNetworks()} ${this.renderPins()}
`; } _handleAddButtonClick() { this.dispatchEvent( new CustomEvent("add-device-template", { bubbles: true }), ); const template: DeviceTemplate = { id: this.device_id, name: this.device_name, prefab_name: this.prefab_name, slots: this.slots, connections: this.connections, fields: this.fields, }; window.VM.addDeviceFromTemplate(template); // reset state for new device this.setupState(); } }