mirror of
https://github.com/CharaChorder/DeviceManager.git
synced 2026-01-08 11:02:50 +00:00
165 lines
3.8 KiB
Svelte
165 lines
3.8 KiB
Svelte
<script lang="ts">
|
|
import { changes, ChangeType, chords } from "$lib/undo-redo.js";
|
|
import type { ChordInfo } from "$lib/undo-redo.js";
|
|
import ChordPhraseEdit from "./ChordPhraseEdit.svelte";
|
|
import ChordActionEdit from "./ChordActionEdit.svelte";
|
|
import type { Chord } from "$lib/serial/chord";
|
|
import { slide } from "svelte/transition";
|
|
import { charaFileToUriComponent } from "$lib/share/share-url";
|
|
import SharePopup from "../SharePopup.svelte";
|
|
import tippy from "tippy.js";
|
|
import { mount, unmount } from "svelte";
|
|
|
|
let { chord, onduplicate }: { chord: ChordInfo; onduplicate: () => void } =
|
|
$props();
|
|
|
|
function remove() {
|
|
changes.update((changes) => {
|
|
changes.push({
|
|
type: ChangeType.Chord,
|
|
id: chord.id,
|
|
actions: chord.actions,
|
|
phrase: chord.phrase,
|
|
deleted: true,
|
|
});
|
|
return changes;
|
|
});
|
|
}
|
|
|
|
function isSameChord(a: Chord, b: Chord) {
|
|
return (
|
|
a.actions.length === b.actions.length &&
|
|
a.actions.every((it, i) => it === b.actions[i])
|
|
);
|
|
}
|
|
|
|
function restore() {
|
|
changes.update((changes) =>
|
|
changes.filter(
|
|
(it) => !(it.type === ChangeType.Chord && isSameChord(it, chord)),
|
|
),
|
|
);
|
|
}
|
|
|
|
function duplicate() {
|
|
const id = [...chord.id];
|
|
id.splice(id.indexOf(0), 1);
|
|
id.push(0);
|
|
while ($chords.some((it) => JSON.stringify(it.id) === JSON.stringify(id))) {
|
|
id[id.length - 1]!++;
|
|
}
|
|
|
|
changes.update((changes) => {
|
|
changes.push({
|
|
type: ChangeType.Chord,
|
|
id,
|
|
actions: [...chord.actions],
|
|
phrase: [...chord.phrase],
|
|
});
|
|
return changes;
|
|
});
|
|
|
|
onduplicate();
|
|
}
|
|
|
|
async function share(event: Event) {
|
|
const url = new URL(window.location.href);
|
|
url.searchParams.set(
|
|
"import",
|
|
await charaFileToUriComponent({
|
|
charaVersion: 1,
|
|
type: "chords",
|
|
chords: [[chord.actions, chord.phrase]],
|
|
}),
|
|
);
|
|
await navigator.clipboard.writeText(url.toString());
|
|
let shareComponent = {};
|
|
tippy(event.target as HTMLElement, {
|
|
onCreate(instance) {
|
|
const target = instance.popper.querySelector(".tippy-content")!;
|
|
shareComponent = mount(SharePopup, { target });
|
|
},
|
|
onHidden(instance) {
|
|
instance.destroy();
|
|
},
|
|
onDestroy(_instance) {
|
|
unmount(shareComponent);
|
|
},
|
|
}).show();
|
|
}
|
|
</script>
|
|
|
|
<tr>
|
|
<th>
|
|
<ChordActionEdit {chord} onsubmit={() => {}} />
|
|
</th>
|
|
<td class="phrase-edit">
|
|
<ChordPhraseEdit {chord} />
|
|
</td>
|
|
<td>
|
|
<div class="table-buttons">
|
|
{#if !chord.deleted}
|
|
<button transition:slide class="icon compact" onclick={remove}
|
|
>delete</button
|
|
>
|
|
{:else}
|
|
<button transition:slide class="icon compact" onclick={restore}
|
|
>restore_from_trash</button
|
|
>
|
|
{/if}
|
|
<button disabled={chord.deleted} class="icon compact" onclick={duplicate}
|
|
>content_copy</button
|
|
>
|
|
<button
|
|
class="icon compact"
|
|
class:disabled={chord.isApplied}
|
|
onclick={restore}>undo</button
|
|
>
|
|
<div class="separator"></div>
|
|
<button class="icon compact" onclick={share}>share</button>
|
|
</div>
|
|
</td>
|
|
</tr>
|
|
|
|
<style lang="scss">
|
|
.separator {
|
|
display: inline-flex;
|
|
|
|
width: 1px;
|
|
height: 24px;
|
|
|
|
opacity: 0.2;
|
|
background: currentcolor;
|
|
}
|
|
|
|
button {
|
|
transition: opacity 75ms ease;
|
|
}
|
|
|
|
.phrase-edit {
|
|
position: relative;
|
|
}
|
|
|
|
tr {
|
|
position: relative;
|
|
}
|
|
|
|
.table-buttons {
|
|
opacity: 0;
|
|
transition: opacity 75ms ease;
|
|
position: absolute;
|
|
top: 0;
|
|
right: 0;
|
|
transform: translate(100%, -50%);
|
|
background: var(--md-sys-color-surface-variant);
|
|
}
|
|
|
|
.icon {
|
|
font-size: 18px;
|
|
}
|
|
|
|
tr:hover .table-buttons {
|
|
opacity: 1;
|
|
}
|
|
</style>
|