Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 6 additions & 2 deletions eslint.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import typescriptPlugin from '@typescript-eslint/eslint-plugin';
import sveltePlugin from 'eslint-plugin-svelte';
import svelteParser from 'svelte-eslint-parser';
import globals from 'globals';
import designDojoPlugin from '@plures/eslint-plugin-design-dojo';

export default [
js.configs.recommended,
Expand Down Expand Up @@ -37,7 +38,8 @@ export default [
{
files: ['**/*.svelte'],
plugins: {
svelte: sveltePlugin
svelte: sveltePlugin,
'design-dojo': designDojoPlugin,
},
languageOptions: {
parser: svelteParser,
Expand All @@ -53,7 +55,9 @@ export default [
},
rules: {
'no-unused-vars': 'off',
'no-undef': 'off'
'no-undef': 'off',
'design-dojo/no-local-primitives': 'error',
'design-dojo/prefer-design-dojo-imports': 'warn',
}
},
{
Expand Down
119 changes: 33 additions & 86 deletions package-lock.json

Large diffs are not rendered by default.

2 changes: 2 additions & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -81,6 +81,7 @@
"@types/node": "^25.0.10",
"@typescript-eslint/eslint-plugin": "^8.53.0",
"@typescript-eslint/parser": "^8.53.0",
"@plures/eslint-plugin-design-dojo": "file:packages/eslint-plugin-design-dojo",
"chai": "^6.2.2",
"eslint": "^9.39.2",
"eslint-plugin-svelte": "^3.14.0",
Expand All @@ -103,6 +104,7 @@
"@financialadvisor/domain": "file:packages/domain",
"@financialadvisor/praxis": "file:.praxis",
"@financialadvisor/shared": "file:packages/shared",
"@plures/design-dojo": "file:packages/design-dojo",
"@modelcontextprotocol/sdk": "^1.25.2",
"@plures/pluresdb": "github:plures/pluresdb",
"@plures/praxis": "^1.2.0",
Expand Down
38 changes: 38 additions & 0 deletions packages/design-dojo/package.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
{
"name": "@plures/design-dojo",
"version": "0.1.0",
"description": "Plures design system — Svelte 5 UI component library",
"type": "module",
"svelte": "./src/index.ts",
"exports": {
".": {
"svelte": "./src/index.ts",
"import": "./src/index.ts",
"types": "./src/index.ts"
},
"./tokens.css": "./src/tokens.css"
},
"files": [
"src/**/*"
],
"scripts": {
"check": "svelte-check --tsconfig ./tsconfig.json"
},
"peerDependencies": {
"svelte": ">=5.0.0"
},
"devDependencies": {
"@types/node": "^25.0.10",
"svelte": "^5.46.4",
"svelte-check": "^4.3.5",
"typescript": "^5.9.3"
},
"keywords": [
"svelte",
"design-system",
"ui",
"components",
"plures"
],
"license": "MIT"
}
91 changes: 91 additions & 0 deletions packages/design-dojo/src/components/Alert.svelte
Original file line number Diff line number Diff line change
@@ -0,0 +1,91 @@
<script lang="ts">
import type { Snippet } from 'svelte';

type AlertVariant = 'info' | 'success' | 'warning' | 'danger';

interface Props {
variant?: AlertVariant;
title?: string;
class?: string;
children: Snippet;
}

const { variant = 'info', title, class: className = '', children }: Props = $props();

const icons: Record<AlertVariant, string> = {
info: 'ℹ',
success: '✓',
warning: '⚠',
danger: '✕',
};
</script>

<div class="dojo-alert dojo-alert--{variant} {className}" role="alert">
<span class="dojo-alert__icon" aria-hidden="true">{icons[variant]}</span>
<div class="dojo-alert__content">
{#if title}
<p class="dojo-alert__title">{title}</p>
{/if}
<div class="dojo-alert__body">
{@render children()}
</div>
</div>
</div>

<style>
.dojo-alert {
display: flex;
gap: var(--space-3);
padding: var(--space-3) var(--space-4);
border-radius: var(--radius-md);
border-width: 1px;
border-style: solid;
}

.dojo-alert__icon {
flex-shrink: 0;
font-size: var(--font-size-base);
line-height: var(--line-height-normal);
}

.dojo-alert__content {
flex: 1;
min-width: 0;
}

.dojo-alert__title {
margin: 0 0 var(--space-1) 0;
font-size: var(--font-size-sm);
font-weight: var(--font-weight-semibold);
line-height: var(--line-height-tight);
}

.dojo-alert__body {
font-size: var(--font-size-sm);
line-height: var(--line-height-normal);
}

.dojo-alert--info {
background-color: var(--color-primary-50);
border-color: var(--color-primary-200);
color: var(--color-primary-800);
}

.dojo-alert--success {
background-color: var(--color-success-50);
border-color: var(--color-success-100);
color: var(--color-success-700);
}

.dojo-alert--warning {
background-color: var(--color-warning-50);
border-color: var(--color-warning-100);
color: var(--color-warning-600);
}

.dojo-alert--danger {
background-color: var(--color-danger-50);
border-color: var(--color-danger-100);
color: var(--color-danger-700);
}
</style>
58 changes: 58 additions & 0 deletions packages/design-dojo/src/components/Badge.svelte
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
<script lang="ts">
import type { Snippet } from 'svelte';

type BadgeVariant = 'default' | 'primary' | 'success' | 'danger' | 'warning' | 'neutral';

interface Props {
variant?: BadgeVariant;
class?: string;
children: Snippet;
}

const { variant = 'default', class: className = '', children }: Props = $props();
</script>

<span class="dojo-badge dojo-badge--{variant} {className}">
{@render children()}
</span>

<style>
.dojo-badge {
display: inline-flex;
align-items: center;
gap: var(--space-1);
padding: 0 var(--space-2);
height: 1.375rem;
font-size: var(--font-size-xs);
font-weight: var(--font-weight-medium);
line-height: 1;
border-radius: var(--radius-full);
white-space: nowrap;
}

.dojo-badge--default,
.dojo-badge--neutral {
background-color: var(--color-neutral-100);
color: var(--color-neutral-700);
}

.dojo-badge--primary {
background-color: var(--color-primary-100);
color: var(--color-primary-700);
}

.dojo-badge--success {
background-color: var(--color-success-100);
color: var(--color-success-700);
}

.dojo-badge--danger {
background-color: var(--color-danger-100);
color: var(--color-danger-700);
}

.dojo-badge--warning {
background-color: var(--color-warning-100);
color: var(--color-warning-600);
}
</style>
114 changes: 114 additions & 0 deletions packages/design-dojo/src/components/Button.svelte
Original file line number Diff line number Diff line change
@@ -0,0 +1,114 @@
<script lang="ts">
import type { Snippet } from 'svelte';
import type { HTMLButtonAttributes } from 'svelte/elements';

interface Props extends HTMLButtonAttributes {
variant?: 'primary' | 'secondary' | 'danger' | 'ghost';
size?: 'sm' | 'md' | 'lg';
children: Snippet;
}

const { variant = 'primary', size = 'md', children, class: className = '', ...restProps }: Props =
$props();
</script>

<button class="dojo-btn dojo-btn--{variant} dojo-btn--{size} {className}" {...restProps}>
{@render children()}
</button>

<style>
.dojo-btn {
display: inline-flex;
align-items: center;
justify-content: center;
gap: var(--space-2);
border: none;
border-radius: var(--radius-md);
font-family: var(--font-family-sans);
font-weight: var(--font-weight-medium);
cursor: pointer;
text-decoration: none;
transition:
background-color var(--motion-duration-base) var(--motion-easing-default),
box-shadow var(--motion-duration-base) var(--motion-easing-default),
transform var(--motion-duration-fast) var(--motion-easing-default);
white-space: nowrap;
user-select: none;
}

.dojo-btn:focus-visible {
outline: 2px solid var(--color-border-focus);
outline-offset: 2px;
}

.dojo-btn:active:not(:disabled) {
transform: translateY(1px);
}

.dojo-btn:disabled {
opacity: 0.5;
cursor: not-allowed;
}

/* Sizes */
.dojo-btn--sm {
padding: var(--space-1) var(--space-3);
font-size: var(--font-size-sm);
line-height: var(--line-height-tight);
}

.dojo-btn--md {
padding: var(--space-2) var(--space-4);
font-size: var(--font-size-base);
line-height: var(--line-height-normal);
}

.dojo-btn--lg {
padding: var(--space-3) var(--space-6);
font-size: var(--font-size-lg);
line-height: var(--line-height-normal);
}

/* Variants */
.dojo-btn--primary {
background-color: var(--color-accent);
color: var(--color-text-inverse);
}

.dojo-btn--primary:hover:not(:disabled) {
background-color: var(--color-accent-hover);
}

.dojo-btn--primary:active:not(:disabled) {
background-color: var(--color-accent-active);
}

.dojo-btn--secondary {
background-color: var(--color-bg-base);
color: var(--color-text-primary);
border: 1px solid var(--color-border-strong);
}

.dojo-btn--secondary:hover:not(:disabled) {
background-color: var(--color-bg-muted);
}

.dojo-btn--danger {
background-color: var(--color-danger-600);
color: var(--color-text-inverse);
}

.dojo-btn--danger:hover:not(:disabled) {
background-color: var(--color-danger-700);
}

.dojo-btn--ghost {
background-color: transparent;
color: var(--color-text-primary);
border: 1px solid transparent;
}

.dojo-btn--ghost:hover:not(:disabled) {
background-color: var(--color-bg-muted);
}
</style>
Loading
Loading