mirror of
https://github.com/CharaChorder/DeviceManager.git
synced 2026-01-18 16:02:57 +00:00
173 lines
4.1 KiB
Svelte
173 lines
4.1 KiB
Svelte
<script lang="ts">
|
|
import {highlightActions, layout} from "$lib/serial/connection"
|
|
import type {CharaLayout} from "$lib/serialization/layout"
|
|
import {KEYMAP_CODES} from "$lib/serial/keymap-codes"
|
|
import type {KeyInfo} from "$lib/serial/keymap-codes"
|
|
|
|
export let activeLayer = 0
|
|
export let keys: Record<"d" | "s" | "n" | "w" | "e", number>
|
|
export let type: "primary" | "secondary" | "tertiary" = "primary"
|
|
|
|
const layerNames = ["Primary Layer", "Number Layer", "Function Layer"]
|
|
|
|
const virtualLayerMap = [1, 0, 2]
|
|
const characterOffset = 8
|
|
|
|
function offsetDistance(quadrant: number, layer: number, activeLayer: number): number {
|
|
const layerOffsetIndex = virtualLayerMap[layer] - virtualLayerMap[activeLayer]
|
|
const layerOffset = quadrant > 2 ? -characterOffset : characterOffset
|
|
return 25 * quadrant + layerOffsetIndex * layerOffset
|
|
}
|
|
|
|
function getKeyDescriptions(keys: KeyInfo[]): string {
|
|
return keys
|
|
.filter(it => !!it)
|
|
.map(({title, id, code}, i) => `${title || id || code} (${layerNames[i]})`)
|
|
.join("\n")
|
|
}
|
|
|
|
function getActions(id: number, layout: CharaLayout): KeyInfo[] {
|
|
return Array.from({length: 3}).map((_, i) => {
|
|
const actionId = layout?.[i][id]
|
|
return KEYMAP_CODES[actionId]
|
|
})
|
|
}
|
|
</script>
|
|
|
|
<div class="radial {type}">
|
|
{#each [keys.n, keys.e, keys.s, keys.w] as id, quadrant}
|
|
{@const actions = getActions(id, $layout)}
|
|
<button
|
|
title={getKeyDescriptions(actions)}
|
|
class:active={actions.some(it => it && $highlightActions?.includes(it.code))}
|
|
>
|
|
{#each actions as keyInfo, layer}
|
|
{#if keyInfo}
|
|
<span
|
|
class:active={virtualLayerMap[activeLayer] === virtualLayerMap[layer]}
|
|
class:icon={!!keyInfo.icon}
|
|
style="offset-distance: {offsetDistance(quadrant, layer, activeLayer)}%"
|
|
>{keyInfo.icon || keyInfo.id || keyInfo.code}</span
|
|
>
|
|
{/if}
|
|
{/each}
|
|
</button>
|
|
{/each}
|
|
</div>
|
|
|
|
<style lang="scss">
|
|
@use "sass:math";
|
|
|
|
$border-width: 18px;
|
|
$gap: 6px;
|
|
$size: 96;
|
|
$offset: 14;
|
|
$scale-difference: 0.2;
|
|
$transition-time: 750ms;
|
|
|
|
.radial {
|
|
position: relative;
|
|
|
|
container: radial / size;
|
|
|
|
width: #{$size * 1px};
|
|
height: #{$size * 1px};
|
|
|
|
transition: all 250ms ease;
|
|
}
|
|
|
|
span {
|
|
$cr: math.div($size, 2) - 2 * $offset;
|
|
|
|
will-change: scale, offset-distance;
|
|
user-select: none;
|
|
|
|
scale: 0.9;
|
|
offset-path: path(
|
|
"M#{math.div($size, 2)} #{$offset}A#{$cr} #{$cr} 0 1 1 #{math.div($size, 2)} #{$size - $offset}A#{$cr} #{$cr} 0 1 1 #{math.div($size, 2)} #{$offset}Z"
|
|
);
|
|
offset-rotate: 0deg;
|
|
|
|
display: flex;
|
|
grid-column: 1;
|
|
grid-row: 1;
|
|
align-items: center;
|
|
justify-content: center;
|
|
|
|
width: 100%;
|
|
height: 100%;
|
|
|
|
font-size: 16px;
|
|
|
|
opacity: 0.2;
|
|
|
|
transition: scale $transition-time ease, opacity $transition-time ease,
|
|
offset-distance $transition-time ease;
|
|
|
|
&.active {
|
|
scale: 1;
|
|
opacity: 1;
|
|
}
|
|
|
|
&.icon {
|
|
font-size: 20px;
|
|
font-weight: 800;
|
|
}
|
|
}
|
|
|
|
button {
|
|
cursor: pointer;
|
|
|
|
position: absolute;
|
|
|
|
display: grid;
|
|
|
|
width: 100cqw;
|
|
height: 100cqh;
|
|
padding: 0;
|
|
|
|
font-family: "Noto Sans Mono", monospace;
|
|
font-size: 16px;
|
|
font-weight: 900;
|
|
color: var(--md-sys-color-on-surface-variant);
|
|
|
|
background: var(--md-sys-color-surface-variant);
|
|
border: none;
|
|
|
|
transition: all 250ms ease;
|
|
|
|
mask-image: url("$lib/assets/quater-ring.svg");
|
|
mask-size: 100% 100%;
|
|
|
|
&.active,
|
|
&:active {
|
|
color: var(--md-sys-color-on-tertiary);
|
|
background: var(--md-sys-color-tertiary);
|
|
}
|
|
|
|
&:nth-child(1) {
|
|
clip-path: polygon(50% 50%, 0 0, 100% 0);
|
|
}
|
|
|
|
&:nth-child(2) {
|
|
clip-path: polygon(50% 50%, 100% 0, 100% 100%);
|
|
}
|
|
|
|
&:nth-child(3) {
|
|
clip-path: polygon(50% 50%, 0 100%, 100% 100%);
|
|
}
|
|
|
|
&:nth-child(4) {
|
|
clip-path: polygon(50% 50%, 0 0, 0 100%);
|
|
}
|
|
}
|
|
|
|
.secondary > button {
|
|
filter: brightness(80%) contrast(120%);
|
|
}
|
|
|
|
.tertiary > button {
|
|
filter: brightness(80%) contrast(110%);
|
|
}
|
|
</style>
|