mirror of
https://github.com/voson-wang/toon.git
synced 2026-01-29 15:24:10 +08:00
602 lines
15 KiB
Vue
602 lines
15 KiB
Vue
<script setup lang="ts">
|
||
import type { Delimiter } from '../../../../packages/toon/src'
|
||
import { useClipboard, useDebounceFn } from '@vueuse/core'
|
||
import { unzlibSync, zlibSync } from 'fflate'
|
||
import { base64ToUint8Array, stringToUint8Array, uint8ArrayToBase64, uint8ArrayToString } from 'uint8array-extras'
|
||
import { computed, onMounted, ref, shallowRef, watch } from 'vue'
|
||
import { DEFAULT_DELIMITER, encode } from '../../../../packages/toon/src'
|
||
import VPInput from './VPInput.vue'
|
||
|
||
interface PlaygroundState {
|
||
json: string
|
||
delimiter: Delimiter
|
||
indent: number
|
||
}
|
||
|
||
const PRESETS = {
|
||
hikes: {
|
||
context: {
|
||
task: 'Our favorite hikes together',
|
||
location: 'Boulder',
|
||
season: 'spring_2025',
|
||
},
|
||
friends: ['ana', 'luis', 'sam'],
|
||
hikes: [
|
||
{ id: 1, name: 'Blue Lake Trail', distanceKm: 7.5, elevationGain: 320, companion: 'ana', wasSunny: true },
|
||
{ id: 2, name: 'Ridge Overlook', distanceKm: 9.2, elevationGain: 540, companion: 'luis', wasSunny: false },
|
||
{ id: 3, name: 'Wildflower Loop', distanceKm: 5.1, elevationGain: 180, companion: 'sam', wasSunny: true },
|
||
],
|
||
},
|
||
orders: {
|
||
orders: [
|
||
{
|
||
orderId: 'ORD-001',
|
||
customer: { name: 'Alice Chen', email: 'alice@example.com' },
|
||
items: [
|
||
{ sku: 'WIDGET-A', quantity: 2, price: 29.99 },
|
||
{ sku: 'GADGET-B', quantity: 1, price: 49.99 },
|
||
],
|
||
total: 109.97,
|
||
status: 'shipped',
|
||
},
|
||
{
|
||
orderId: 'ORD-002',
|
||
customer: { name: 'Bob Smith', email: 'bob@example.com' },
|
||
items: [
|
||
{ sku: 'THING-C', quantity: 3, price: 15.00 },
|
||
],
|
||
total: 45.00,
|
||
status: 'delivered',
|
||
},
|
||
],
|
||
},
|
||
metrics: {
|
||
metrics: [
|
||
{ date: '2025-01-01', views: 5200, clicks: 180, conversions: 24, revenue: 2890.50 },
|
||
{ date: '2025-01-02', views: 6100, clicks: 220, conversions: 31, revenue: 3450.00 },
|
||
{ date: '2025-01-03', views: 4800, clicks: 165, conversions: 19, revenue: 2100.25 },
|
||
{ date: '2025-01-04', views: 5900, clicks: 205, conversions: 28, revenue: 3200.00 },
|
||
],
|
||
},
|
||
events: {
|
||
logs: [
|
||
{ timestamp: '2025-01-15T10:23:45Z', level: 'info', endpoint: '/api/users', statusCode: 200, responseTime: 45 },
|
||
{ timestamp: '2025-01-15T10:24:12Z', level: 'error', endpoint: '/api/orders', statusCode: 500, responseTime: 120, error: { message: 'Database timeout', retryable: true } },
|
||
{ timestamp: '2025-01-15T10:25:03Z', level: 'info', endpoint: '/api/products', statusCode: 200, responseTime: 32 },
|
||
{ timestamp: '2025-01-15T10:26:47Z', level: 'warn', endpoint: '/api/payment', statusCode: 429, responseTime: 5, error: { message: 'Rate limit exceeded', retryable: true } },
|
||
],
|
||
},
|
||
} as const
|
||
const DELIMITER_OPTIONS: { value: Delimiter, label: string }[] = [
|
||
{ value: ',', label: 'Comma (,)' },
|
||
{ value: '\t', label: 'Tab (\\t)' },
|
||
{ value: '|', label: 'Pipe (|)' },
|
||
]
|
||
const DEFAULT_JSON = JSON.stringify(PRESETS.hikes, undefined, 2)
|
||
|
||
const jsonInput = ref(DEFAULT_JSON)
|
||
const delimiter = ref<Delimiter>(DEFAULT_DELIMITER)
|
||
const indent = ref(2)
|
||
const hasCopiedUrl = ref(false)
|
||
|
||
const tokenizer = shallowRef<typeof import('gpt-tokenizer') | undefined>()
|
||
|
||
const encodingResult = computed(() => {
|
||
try {
|
||
const parsedInput = JSON.parse(jsonInput.value)
|
||
return {
|
||
output: encode(parsedInput, {
|
||
indent: indent.value,
|
||
delimiter: delimiter.value,
|
||
}),
|
||
error: undefined,
|
||
}
|
||
}
|
||
catch (e) {
|
||
return {
|
||
output: '',
|
||
error: e instanceof Error ? e.message : 'Invalid JSON',
|
||
}
|
||
}
|
||
})
|
||
|
||
const toonOutput = computed(() => encodingResult.value.output)
|
||
const error = computed(() => encodingResult.value.error)
|
||
|
||
const jsonTokens = computed(() =>
|
||
tokenizer.value?.encode(jsonInput.value).length,
|
||
)
|
||
const toonTokens = computed(() =>
|
||
tokenizer.value && toonOutput.value ? tokenizer.value.encode(toonOutput.value).length : undefined,
|
||
)
|
||
const tokenSavings = computed(() => {
|
||
if (!jsonTokens.value || !toonTokens.value)
|
||
return
|
||
|
||
const diff = jsonTokens.value - toonTokens.value
|
||
const percent = Math.abs((diff / jsonTokens.value) * 100).toFixed(1)
|
||
const sign = diff > 0 ? '−' : '+'
|
||
|
||
return { diff, percent, sign, isSavings: diff > 0 }
|
||
})
|
||
|
||
const { copy, copied } = useClipboard({ source: toonOutput })
|
||
|
||
async function copyShareUrl() {
|
||
await navigator.clipboard.writeText(window.location.href)
|
||
hasCopiedUrl.value = true
|
||
setTimeout(() => (hasCopiedUrl.value = false), 2000)
|
||
}
|
||
|
||
const updateUrl = useDebounceFn(() => {
|
||
if (typeof window === 'undefined')
|
||
return
|
||
|
||
const hash = encodeState()
|
||
window.history.replaceState(null, '', `#${hash}`)
|
||
}, 300)
|
||
|
||
watch([jsonInput, delimiter, indent], () => {
|
||
updateUrl()
|
||
})
|
||
|
||
onMounted(() => {
|
||
loadTokenizer()
|
||
|
||
const hash = window.location.hash.slice(1)
|
||
if (!hash)
|
||
return
|
||
|
||
const state = decodeState(hash)
|
||
if (state) {
|
||
jsonInput.value = state.json
|
||
delimiter.value = state.delimiter
|
||
indent.value = state.indent
|
||
}
|
||
})
|
||
|
||
function encodeState() {
|
||
const state: PlaygroundState = {
|
||
json: jsonInput.value,
|
||
delimiter: delimiter.value,
|
||
indent: indent.value,
|
||
}
|
||
|
||
const compressedData = zlibSync(stringToUint8Array(JSON.stringify(state)))
|
||
return uint8ArrayToBase64(compressedData, { urlSafe: true })
|
||
}
|
||
|
||
function decodeState(hash: string) {
|
||
try {
|
||
const bytes = base64ToUint8Array(hash)
|
||
const decompressedData = unzlibSync(bytes)
|
||
const decodedData = uint8ArrayToString(decompressedData)
|
||
if (decodedData)
|
||
return JSON.parse(decodedData) as PlaygroundState
|
||
}
|
||
catch {}
|
||
}
|
||
|
||
function loadPreset(name: keyof typeof PRESETS) {
|
||
jsonInput.value = JSON.stringify(PRESETS[name], undefined, 2)
|
||
}
|
||
|
||
async function loadTokenizer() {
|
||
tokenizer.value ??= await import('gpt-tokenizer')
|
||
}
|
||
</script>
|
||
|
||
<template>
|
||
<div class="playground">
|
||
<div class="playground-container">
|
||
<!-- Header -->
|
||
<header class="playground-header">
|
||
<h1>Playground</h1>
|
||
<p>Experiment with JSON to TOON encoding in real-time.</p>
|
||
</header>
|
||
|
||
<!-- Options Bar -->
|
||
<div class="options-bar">
|
||
<VPInput id="delimiter" label="Delimiter">
|
||
<select id="delimiter" v-model="delimiter">
|
||
<option v-for="opt in DELIMITER_OPTIONS" :key="opt.value" :value="opt.value">
|
||
{{ opt.label }}
|
||
</option>
|
||
</select>
|
||
</VPInput>
|
||
|
||
<VPInput id="indent" label="Indent">
|
||
<input
|
||
id="indent"
|
||
v-model.number="indent"
|
||
type="number"
|
||
min="0"
|
||
max="8"
|
||
>
|
||
</VPInput>
|
||
|
||
<VPInput id="preset" label="Preset">
|
||
<select id="preset" @change="(e) => loadPreset((e.target as HTMLSelectElement).value as keyof typeof PRESETS)">
|
||
<option value="" disabled selected>
|
||
Load example...
|
||
</option>
|
||
<option value="hikes">
|
||
Hikes (mixed structure)
|
||
</option>
|
||
<option value="orders">
|
||
Orders (nested objects)
|
||
</option>
|
||
<option value="metrics">
|
||
Metrics (tabular data)
|
||
</option>
|
||
<option value="events">
|
||
Events (semi-uniform)
|
||
</option>
|
||
</select>
|
||
</VPInput>
|
||
|
||
<button
|
||
class="share-button"
|
||
:class="[hasCopiedUrl && 'copied']"
|
||
:aria-label="hasCopiedUrl ? 'Link copied!' : 'Copy shareable URL'"
|
||
@click="copyShareUrl"
|
||
>
|
||
<span class="vpi-link" :class="[hasCopiedUrl && 'check']" aria-hidden="true" />
|
||
{{ hasCopiedUrl ? 'Copied!' : 'Share' }}
|
||
</button>
|
||
</div>
|
||
|
||
<!-- Editor Container -->
|
||
<div class="editor-container">
|
||
<!-- JSON Input -->
|
||
<div class="editor-pane">
|
||
<div class="pane-header">
|
||
<span class="pane-title">JSON Input</span>
|
||
<span class="pane-stats">
|
||
<span>{{ jsonTokens ?? '...' }} tokens</span>
|
||
<span>{{ jsonInput.length }} chars</span>
|
||
</span>
|
||
</div>
|
||
<textarea
|
||
id="json-input"
|
||
v-model="jsonInput"
|
||
class="editor-textarea"
|
||
spellcheck="false"
|
||
aria-label="JSON input"
|
||
:aria-describedby="error ? 'json-error' : undefined"
|
||
:aria-invalid="!!error"
|
||
placeholder="Enter JSON here..."
|
||
/>
|
||
</div>
|
||
|
||
<!-- TOON Output -->
|
||
<div class="editor-pane">
|
||
<div class="pane-header">
|
||
<span class="pane-title">
|
||
TOON Output
|
||
<span v-if="tokenSavings" class="savings-badge" :class="[!tokenSavings.isSavings && 'increase']">
|
||
{{ tokenSavings.sign }}{{ tokenSavings.percent }}%
|
||
</span>
|
||
</span>
|
||
<span class="pane-stats">
|
||
<span>{{ toonTokens ?? '...' }} tokens</span>
|
||
<span>{{ toonOutput.length }} chars</span>
|
||
</span>
|
||
</div>
|
||
<div class="editor-output">
|
||
<button
|
||
v-if="!error"
|
||
class="copy-button"
|
||
:class="[copied && 'copied']"
|
||
:aria-label="copied ? 'Copied to clipboard' : 'Copy to clipboard'"
|
||
:aria-pressed="copied"
|
||
@click="copy()"
|
||
/>
|
||
<pre v-if="!error"><code>{{ toonOutput }}</code></pre>
|
||
<div v-else id="json-error" role="alert" class="error-message">
|
||
{{ error }}
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
|
||
<style scoped>
|
||
.playground {
|
||
padding: 32px 24px 32px;
|
||
}
|
||
|
||
@media (min-width: 768px) {
|
||
.playground {
|
||
padding: 48px 32px 48px;
|
||
}
|
||
}
|
||
|
||
@media (min-width: 960px) {
|
||
.playground {
|
||
padding: 48px 32px 48px;
|
||
}
|
||
}
|
||
|
||
.playground-container {
|
||
max-width: 1400px;
|
||
margin: 0 auto;
|
||
}
|
||
|
||
.playground-header {
|
||
margin-bottom: 24px;
|
||
}
|
||
|
||
.playground-header h1 {
|
||
font-size: 28px;
|
||
font-weight: 600;
|
||
letter-spacing: -0.02em;
|
||
line-height: 40px;
|
||
color: var(--vp-c-text-1);
|
||
margin: 0 0 8px;
|
||
}
|
||
|
||
@media (min-width: 768px) {
|
||
.playground-header h1 {
|
||
font-size: 32px;
|
||
}
|
||
}
|
||
|
||
.playground-header p {
|
||
font-size: 16px;
|
||
line-height: 28px;
|
||
color: var(--vp-c-text-2);
|
||
}
|
||
|
||
.options-bar {
|
||
display: flex;
|
||
flex-wrap: wrap;
|
||
gap: 12px;
|
||
align-items: flex-end;
|
||
margin-bottom: 16px;
|
||
padding: 12px 16px;
|
||
background: var(--vp-c-bg-soft);
|
||
border-radius: 8px;
|
||
border: 1px solid var(--vp-c-divider);
|
||
}
|
||
|
||
.vpi-link {
|
||
--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='currentColor' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 24 24'%3E%3Cpath d='M10 13a5 5 0 0 0 7.54.54l3-3a5 5 0 0 0-7.07-7.07l-1.72 1.71'/%3E%3Cpath d='M14 11a5 5 0 0 0-7.54-.54l-3 3a5 5 0 0 0 7.07 7.07l1.71-1.71'/%3E%3C/svg%3E");
|
||
display: inline-block;
|
||
width: 1em;
|
||
height: 1em;
|
||
-webkit-mask: var(--icon) no-repeat;
|
||
mask: var(--icon) no-repeat;
|
||
-webkit-mask-size: 100% 100%;
|
||
mask-size: 100% 100%;
|
||
background-color: currentColor;
|
||
}
|
||
|
||
.vpi-link.check {
|
||
--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='currentColor' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 24 24'%3E%3Cpath d='M20 6 9 17l-5-5'/%3E%3C/svg%3E");
|
||
}
|
||
|
||
.share-button {
|
||
display: inline-flex;
|
||
align-items: center;
|
||
gap: 6px;
|
||
padding: 0 12px;
|
||
height: 32px;
|
||
font-size: 13px;
|
||
font-weight: 500;
|
||
color: var(--vp-c-text-1);
|
||
background: var(--vp-c-bg);
|
||
border: 1px solid var(--vp-c-border);
|
||
border-radius: 6px;
|
||
transition: border-color 0.25s, color 0.25s;
|
||
margin-left: auto;
|
||
}
|
||
|
||
.share-button:hover {
|
||
border-color: var(--vp-c-brand-1);
|
||
color: var(--vp-c-brand-1);
|
||
}
|
||
|
||
.share-button:focus-visible {
|
||
outline: 2px solid var(--vp-c-brand-1);
|
||
outline-offset: 2px;
|
||
}
|
||
|
||
.share-button.copied {
|
||
border-color: var(--vp-c-green-1);
|
||
color: var(--vp-c-green-1);
|
||
}
|
||
|
||
.editor-container {
|
||
display: grid;
|
||
grid-template-columns: 1fr 1fr;
|
||
gap: 16px;
|
||
}
|
||
|
||
@media (max-width: 768px) {
|
||
.editor-container {
|
||
grid-template-columns: 1fr;
|
||
}
|
||
}
|
||
|
||
.editor-pane {
|
||
display: flex;
|
||
flex-direction: column;
|
||
min-height: 500px;
|
||
border: 1px solid var(--vp-c-divider);
|
||
border-radius: 8px;
|
||
overflow: hidden;
|
||
background: var(--vp-c-bg-soft);
|
||
transition: border-color 0.25s;
|
||
}
|
||
|
||
@media (max-width: 768px) {
|
||
.editor-pane {
|
||
min-height: 400px;
|
||
}
|
||
}
|
||
|
||
.editor-pane:focus-within {
|
||
border-color: var(--vp-c-brand-1);
|
||
}
|
||
|
||
.pane-header {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 12px;
|
||
padding: 12px 16px;
|
||
background: var(--vp-c-bg-alt);
|
||
border-bottom: 1px solid var(--vp-c-divider);
|
||
}
|
||
|
||
.pane-title {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 8px;
|
||
font-size: 0.75rem;
|
||
font-weight: 600;
|
||
color: var(--vp-c-text-2);
|
||
text-transform: uppercase;
|
||
letter-spacing: 0.05em;
|
||
line-height: 1.5;
|
||
}
|
||
|
||
.pane-stats {
|
||
display: flex;
|
||
gap: 12px;
|
||
margin-left: auto;
|
||
font-size: 0.75rem;
|
||
font-weight: 400;
|
||
color: var(--vp-c-text-2);
|
||
text-transform: none;
|
||
letter-spacing: normal;
|
||
}
|
||
|
||
.savings-badge {
|
||
display: inline-flex;
|
||
padding: 2px 6px;
|
||
font-size: 0.625rem;
|
||
font-weight: 600;
|
||
color: var(--vp-c-green-1);
|
||
background: var(--vp-c-green-soft);
|
||
border-radius: 4px;
|
||
text-transform: none;
|
||
letter-spacing: normal;
|
||
}
|
||
|
||
.savings-badge.increase {
|
||
color: var(--vp-c-yellow-1);
|
||
background: var(--vp-c-yellow-soft);
|
||
}
|
||
|
||
.copy-button {
|
||
position: absolute;
|
||
top: 12px;
|
||
right: 12px;
|
||
z-index: 3;
|
||
border: 1px solid var(--vp-code-copy-code-border-color);
|
||
border-radius: 4px;
|
||
width: 40px;
|
||
height: 40px;
|
||
background-color: var(--vp-code-copy-code-bg);
|
||
opacity: 0;
|
||
cursor: pointer;
|
||
background-image: var(--vp-icon-copy);
|
||
background-position: 50%;
|
||
background-size: 20px;
|
||
background-repeat: no-repeat;
|
||
transition: border-color 0.25s, background-color 0.25s, opacity 0.25s;
|
||
}
|
||
|
||
.editor-output:hover .copy-button,
|
||
.copy-button:focus {
|
||
opacity: 1;
|
||
}
|
||
|
||
.copy-button:hover:not(:disabled),
|
||
.copy-button.copied {
|
||
border-color: var(--vp-code-copy-code-hover-border-color);
|
||
background-color: var(--vp-code-copy-code-hover-bg);
|
||
}
|
||
|
||
.copy-button:focus-visible {
|
||
outline: 2px solid var(--vp-c-brand-1);
|
||
outline-offset: 2px;
|
||
}
|
||
|
||
.copy-button:disabled {
|
||
opacity: 0.3;
|
||
cursor: not-allowed;
|
||
}
|
||
|
||
.copy-button.copied,
|
||
.copy-button:hover.copied {
|
||
border-radius: 0 4px 4px 0;
|
||
background-image: var(--vp-icon-copied);
|
||
}
|
||
|
||
.copy-button.copied::before,
|
||
.copy-button:hover.copied::before {
|
||
position: relative;
|
||
top: -1px;
|
||
transform: translateX(calc(-100% - 1px));
|
||
display: flex;
|
||
justify-content: center;
|
||
align-items: center;
|
||
border: 1px solid var(--vp-code-copy-code-hover-border-color);
|
||
border-right: 0;
|
||
border-radius: 4px 0 0 4px;
|
||
padding: 0 10px;
|
||
width: fit-content;
|
||
height: 40px;
|
||
text-align: center;
|
||
font-size: 12px;
|
||
font-weight: 500;
|
||
color: var(--vp-code-copy-code-active-text);
|
||
background-color: var(--vp-code-copy-code-hover-bg);
|
||
white-space: nowrap;
|
||
content: var(--vp-code-copy-copied-text-content);
|
||
}
|
||
|
||
.copy-button[aria-pressed="true"] {
|
||
opacity: 1;
|
||
}
|
||
|
||
.editor-textarea,
|
||
.editor-output {
|
||
flex: 1;
|
||
padding: 16px;
|
||
font-family: var(--vp-font-family-mono);
|
||
font-size: 0.875rem;
|
||
line-height: 1.7;
|
||
}
|
||
|
||
.editor-textarea {
|
||
resize: none;
|
||
color: var(--vp-c-text-1);
|
||
background: var(--vp-c-bg);
|
||
}
|
||
|
||
.editor-output {
|
||
position: relative;
|
||
overflow: auto;
|
||
background: var(--vp-code-block-bg);
|
||
}
|
||
|
||
.editor-output pre {
|
||
margin: 0;
|
||
white-space: pre;
|
||
}
|
||
|
||
.error-message {
|
||
color: var(--vp-c-danger-1);
|
||
padding: 8px 12px;
|
||
background: var(--vp-c-danger-soft);
|
||
border-radius: 4px;
|
||
font-size: 0.875rem;
|
||
font-family: var(--vp-font-family-base);
|
||
}
|
||
</style>
|