wip
This commit is contained in:
1
playground/src/app.css
Normal file
1
playground/src/app.css
Normal file
@@ -0,0 +1 @@
|
||||
@import 'tailwindcss';
|
22
playground/src/app.d.ts
vendored
Normal file
22
playground/src/app.d.ts
vendored
Normal file
@@ -0,0 +1,22 @@
|
||||
// See https://svelte.dev/docs/kit/types#app.d.ts
|
||||
// for information about these interfaces
|
||||
declare global {
|
||||
namespace App {
|
||||
// interface Error {}
|
||||
// interface Locals {}
|
||||
// interface PageData {}
|
||||
// interface PageState {}
|
||||
// interface Platform {}
|
||||
}
|
||||
|
||||
export interface Window {
|
||||
Go: {
|
||||
new(): {
|
||||
run: (inst: WebAssembly.Instance) => Promise<void>;
|
||||
importObject: WebAssembly.Imports;
|
||||
};
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
export { };
|
12
playground/src/app.html
Normal file
12
playground/src/app.html
Normal file
@@ -0,0 +1,12 @@
|
||||
<!doctype html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="utf-8" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||
<script src="/wasm_exec.js"></script>
|
||||
%sveltekit.head%
|
||||
</head>
|
||||
<body data-sveltekit-preload-data="hover">
|
||||
<div style="display: contents">%sveltekit.body%</div>
|
||||
</body>
|
||||
</html>
|
1
playground/src/lib/assets/favicon.svg
Normal file
1
playground/src/lib/assets/favicon.svg
Normal file
@@ -0,0 +1 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" width="107" height="128" viewBox="0 0 107 128"><title>svelte-logo</title><path d="M94.157 22.819c-10.4-14.885-30.94-19.297-45.792-9.835L22.282 29.608A29.92 29.92 0 0 0 8.764 49.65a31.5 31.5 0 0 0 3.108 20.231 30 30 0 0 0-4.477 11.183 31.9 31.9 0 0 0 5.448 24.116c10.402 14.887 30.942 19.297 45.791 9.835l26.083-16.624A29.92 29.92 0 0 0 98.235 78.35a31.53 31.53 0 0 0-3.105-20.232 30 30 0 0 0 4.474-11.182 31.88 31.88 0 0 0-5.447-24.116" style="fill:#ff3e00"/><path d="M45.817 106.582a20.72 20.72 0 0 1-22.237-8.243 19.17 19.17 0 0 1-3.277-14.503 18 18 0 0 1 .624-2.435l.49-1.498 1.337.981a33.6 33.6 0 0 0 10.203 5.098l.97.294-.09.968a5.85 5.85 0 0 0 1.052 3.878 6.24 6.24 0 0 0 6.695 2.485 5.8 5.8 0 0 0 1.603-.704L69.27 76.28a5.43 5.43 0 0 0 2.45-3.631 5.8 5.8 0 0 0-.987-4.371 6.24 6.24 0 0 0-6.698-2.487 5.7 5.7 0 0 0-1.6.704l-9.953 6.345a19 19 0 0 1-5.296 2.326 20.72 20.72 0 0 1-22.237-8.243 19.17 19.17 0 0 1-3.277-14.502 17.99 17.99 0 0 1 8.13-12.052l26.081-16.623a19 19 0 0 1 5.3-2.329 20.72 20.72 0 0 1 22.237 8.243 19.17 19.17 0 0 1 3.277 14.503 18 18 0 0 1-.624 2.435l-.49 1.498-1.337-.98a33.6 33.6 0 0 0-10.203-5.1l-.97-.294.09-.968a5.86 5.86 0 0 0-1.052-3.878 6.24 6.24 0 0 0-6.696-2.485 5.8 5.8 0 0 0-1.602.704L37.73 51.72a5.42 5.42 0 0 0-2.449 3.63 5.79 5.79 0 0 0 .986 4.372 6.24 6.24 0 0 0 6.698 2.486 5.8 5.8 0 0 0 1.602-.704l9.952-6.342a19 19 0 0 1 5.295-2.328 20.72 20.72 0 0 1 22.237 8.242 19.17 19.17 0 0 1 3.277 14.503 18 18 0 0 1-8.13 12.053l-26.081 16.622a19 19 0 0 1-5.3 2.328" style="fill:#fff"/></svg>
|
After Width: | Height: | Size: 1.5 KiB |
35
playground/src/lib/components/EditorPanel.svelte
Normal file
35
playground/src/lib/components/EditorPanel.svelte
Normal file
@@ -0,0 +1,35 @@
|
||||
<script lang="ts">
|
||||
import type { Snippet } from "svelte";
|
||||
|
||||
interface Props {
|
||||
title: string;
|
||||
value: string;
|
||||
placeholder?: string;
|
||||
readonly?: boolean;
|
||||
children?: Snippet;
|
||||
}
|
||||
|
||||
let { title, value = $bindable(), placeholder = "", readonly = false, children }: Props = $props();
|
||||
</script>
|
||||
|
||||
<div class="flex flex-col h-full border-r border-gray-200 last:border-r-0">
|
||||
<div class="px-4 py-3 border-b border-gray-200 bg-gray-50/50">
|
||||
<h2 class="font-semibold text-gray-900 text-sm uppercase tracking-wide">{title}</h2>
|
||||
</div>
|
||||
<div class="flex-1 relative group">
|
||||
{#if readonly}
|
||||
<div class="absolute inset-0 p-4 overflow-auto">
|
||||
<pre class="text-sm font-mono text-gray-800 whitespace-pre-wrap leading-relaxed">{value}</pre>
|
||||
</div>
|
||||
{:else}
|
||||
<textarea bind:value {placeholder} class="absolute inset-0 w-full h-full p-4 text-sm font-mono resize-none border-0 outline-none bg-transparent text-gray-800 leading-relaxed placeholder:text-gray-400 focus:bg-gray-50/30 transition-colors" spellcheck="false"></textarea>
|
||||
{/if}
|
||||
|
||||
<!-- Subtle hover effect -->
|
||||
<div class="absolute inset-0 pointer-events-none opacity-0 group-hover:opacity-5 bg-black transition-opacity duration-200"></div>
|
||||
</div>
|
||||
|
||||
{#if children}
|
||||
{@render children()}
|
||||
{/if}
|
||||
</div>
|
26
playground/src/lib/components/Header.svelte
Normal file
26
playground/src/lib/components/Header.svelte
Normal file
@@ -0,0 +1,26 @@
|
||||
<script lang="ts">
|
||||
// import { CodeIcon, ArrowRightIcon } from "lucide-svelte";
|
||||
</script>
|
||||
|
||||
<header class="sticky top-0 z-10 border-b border-gray-200 bg-white/80 backdrop-blur-sm">
|
||||
<div class="container mx-auto px-6 py-4">
|
||||
<div class="flex items-center justify-between">
|
||||
<div class="flex items-center gap-3">
|
||||
<div class="rounded-lg bg-black p-2">
|
||||
<!-- <CodeIcon class="h-5 w-5 text-white" /> -->
|
||||
</div>
|
||||
<div>
|
||||
<h1 class="text-2xl font-bold text-black">Marka</h1>
|
||||
<p class="text-sm text-gray-600">Bidirectional Markdown ↔ JSON Parser</p>
|
||||
</div>
|
||||
</div>
|
||||
<div class="hidden items-center gap-2 text-sm text-gray-500 md:flex">
|
||||
<span>Template</span>
|
||||
<!-- <ArrowRightIcon class="w-4 h-4" /> -->
|
||||
<span>Markdown</span>
|
||||
<!-- <ArrowRightIcon class="w-4 h-4" /> -->
|
||||
<span>JSON</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</header>
|
84
playground/src/lib/components/Playground.svelte
Normal file
84
playground/src/lib/components/Playground.svelte
Normal file
@@ -0,0 +1,84 @@
|
||||
<script lang="ts">
|
||||
import EditorPanel from './EditorPanel.svelte';
|
||||
|
||||
// Sample data based on the provided example
|
||||
let templateValue = $state(`---
|
||||
_type: Recipe
|
||||
author.name: Max Richter
|
||||
---
|
||||
|
||||
# Baguette
|
||||
|
||||
My favourite baguette recipe
|
||||
|
||||
## Ingredients
|
||||
- Flour
|
||||
- Water
|
||||
- Salt
|
||||
|
||||
## Steps
|
||||
1. Mix Flour Water and Salt
|
||||
2. Bake the bread`);
|
||||
|
||||
let markdownValue = $state(`---
|
||||
_type: Recipe
|
||||
author.name: Max Richter
|
||||
---
|
||||
|
||||
# Baguette
|
||||
|
||||
My favourite baguette recipe
|
||||
|
||||
## Ingredients
|
||||
- Flour
|
||||
- Water
|
||||
- Salt
|
||||
|
||||
## Steps
|
||||
1. Mix Flour Water and Salt
|
||||
2. Bake the bread`);
|
||||
|
||||
// Simulated JSON output - in real implementation this would be generated from the parser
|
||||
let jsonOutput = $derived(() => {
|
||||
try {
|
||||
const output = {
|
||||
_type: 'Recipe',
|
||||
name: 'Baguette',
|
||||
author: {
|
||||
_type: 'Person',
|
||||
name: 'Max Richter'
|
||||
},
|
||||
description: 'My favourite baguette recipe',
|
||||
recipeIngredient: ['Flour', 'Water', 'Salt'],
|
||||
recipeInstructions: ['Mix Flour Water and Salt', 'Bake the bread']
|
||||
};
|
||||
return JSON.stringify(output, null, 2);
|
||||
} catch (e) {
|
||||
return 'Invalid input';
|
||||
}
|
||||
});
|
||||
</script>
|
||||
|
||||
<div class="grid min-h-0 flex-1 grid-cols-1 lg:grid-cols-3">
|
||||
<EditorPanel
|
||||
title="Template"
|
||||
bind:value={templateValue}
|
||||
placeholder="Enter your Marka template here..."
|
||||
/>
|
||||
|
||||
<EditorPanel
|
||||
title="Markdown"
|
||||
bind:value={markdownValue}
|
||||
placeholder="Enter your markdown content here..."
|
||||
/>
|
||||
|
||||
<EditorPanel title="Data" value={jsonOutput()} readonly={true}>
|
||||
{#snippet children()}
|
||||
<div class="absolute bottom-4 right-4 opacity-60">
|
||||
<div class="rounded border bg-white px-2 py-1 text-xs text-gray-500">
|
||||
Auto-generated JSON
|
||||
</div>
|
||||
</div>
|
||||
{/snippet}
|
||||
</EditorPanel>
|
||||
</div>
|
25
playground/src/lib/components/StatusBar.svelte
Normal file
25
playground/src/lib/components/StatusBar.svelte
Normal file
@@ -0,0 +1,25 @@
|
||||
<script lang="ts">
|
||||
// import { CheckCircleIcon, AlertCircleIcon } from "lucide-svelte";
|
||||
|
||||
let status = $state<'success' | 'error' | 'idle'>('success');
|
||||
let message = $state('Template parsed successfully');
|
||||
</script>
|
||||
|
||||
<div class="border-t border-gray-200 bg-gray-50/50 px-6 py-2">
|
||||
<div class="flex items-center justify-between text-sm">
|
||||
<div class="flex items-center gap-2">
|
||||
{#if status === 'success'}
|
||||
<!-- <CheckCircleIcon class="h-4 w-4 text-green-600" /> -->
|
||||
<span class="text-green-700">{message}</span>
|
||||
{:else if status === 'error'}
|
||||
<!-- <AlertCircleIcon class="h-4 w-4 text-red-600" /> -->
|
||||
<span class="text-red-700">{message}</span>
|
||||
{:else}
|
||||
<div class="h-4 w-4 rounded-full bg-gray-300"></div>
|
||||
<span class="text-gray-600">Ready</span>
|
||||
{/if}
|
||||
</div>
|
||||
|
||||
<div class="text-gray-500">Schema.org validation enabled</div>
|
||||
</div>
|
||||
</div>
|
1
playground/src/lib/index.ts
Normal file
1
playground/src/lib/index.ts
Normal file
@@ -0,0 +1 @@
|
||||
// place files you want to import through the `$lib` alias in this folder.
|
12
playground/src/routes/+layout.svelte
Normal file
12
playground/src/routes/+layout.svelte
Normal file
@@ -0,0 +1,12 @@
|
||||
<script lang="ts">
|
||||
import '../app.css';
|
||||
import favicon from '$lib/assets/favicon.svg';
|
||||
|
||||
let { children } = $props();
|
||||
</script>
|
||||
|
||||
<svelte:head>
|
||||
<link rel="icon" href={favicon} />
|
||||
</svelte:head>
|
||||
|
||||
{@render children?.()}
|
33
playground/src/routes/+page.svelte
Normal file
33
playground/src/routes/+page.svelte
Normal file
@@ -0,0 +1,33 @@
|
||||
<script lang="ts">
|
||||
import Header from '$lib/components/Header.svelte';
|
||||
import Playground from '$lib/components/Playground.svelte';
|
||||
import StatusBar from '$lib/components/StatusBar.svelte';
|
||||
</script>
|
||||
|
||||
<div class="bg-background text-foreground flex min-h-screen flex-col">
|
||||
<Header />
|
||||
<Playground />
|
||||
<StatusBar />
|
||||
</div>
|
||||
|
||||
<style>
|
||||
:global(body) {
|
||||
margin: 0;
|
||||
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
|
||||
background: white;
|
||||
}
|
||||
|
||||
:global(*) {
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
:global(textarea:focus) {
|
||||
outline: none;
|
||||
}
|
||||
|
||||
:global(pre) {
|
||||
margin: 0;
|
||||
font-family:
|
||||
'SF Mono', Monaco, 'Cascadia Code', 'Roboto Mono', Consolas, 'Courier New', monospace;
|
||||
}
|
||||
</style>
|
47
playground/src/routes/+page.svelte2
Normal file
47
playground/src/routes/+page.svelte2
Normal file
@@ -0,0 +1,47 @@
|
||||
<script lang="ts">
|
||||
import {onMount} from 'svelte';
|
||||
|
||||
const test = `---
|
||||
_type: Recipe
|
||||
author.name: Max Richter
|
||||
---
|
||||
|
||||
# Baguette
|
||||
|
||||
My favourite baguette recipe
|
||||
|
||||
## Ingredients
|
||||
- Flour
|
||||
- Water
|
||||
- Salt
|
||||
|
||||
## Steps
|
||||
1. Mix Flour Water and Salt
|
||||
2. Bake the bread
|
||||
`
|
||||
|
||||
function exec(){
|
||||
let start = performance.now();
|
||||
const res = markaParseFile(test);
|
||||
let end = performance.now();
|
||||
|
||||
console.log(JSON.parse(res))
|
||||
|
||||
start = performance.now();
|
||||
const res2 = markaMatchBlocks(test);
|
||||
end = performance.now();
|
||||
|
||||
console.log(JSON.parse(res2))
|
||||
}
|
||||
|
||||
onMount(async () => {
|
||||
const go = new Go();
|
||||
WebAssembly.instantiateStreaming(fetch("/main.wasm"), go.importObject).then((result) => {
|
||||
go.run(result.instance);
|
||||
});
|
||||
});
|
||||
|
||||
</script>
|
||||
|
||||
|
||||
<button onclick={exec}>test</button>
|
Reference in New Issue
Block a user