This commit is contained in:
Max Richter
2025-09-25 16:41:26 +02:00
parent 3f0d25f935
commit b13d5015f4
75 changed files with 3881 additions and 141 deletions

1
playground/src/app.css Normal file
View File

@@ -0,0 +1 @@
@import 'tailwindcss';

22
playground/src/app.d.ts vendored Normal file
View 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
View 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>

View 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

View 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>

View 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>

View 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>

View 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>

View File

@@ -0,0 +1 @@
// place files you want to import through the `$lib` alias in this folder.

View 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?.()}

View 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>

View 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>