diff --git a/-tmp/-wip/Image.Svg/Svg.ts b/-tmp/-wip/Image.Svg/Svg.ts new file mode 100644 index 0000000000..4119d19a18 --- /dev/null +++ b/-tmp/-wip/Image.Svg/Svg.ts @@ -0,0 +1,14 @@ +import { SvgRef } from './SvgRef'; +import { SvgElement } from './libs'; + +/** + * Helpers for working with SVG objects within the given container element. + * + * NOTE: + * This helper assumes <svg> data assembled via the webpack plugin + * (see [cell.compiler]). + */ +export const Svg = { + ref: SvgRef, + Element: SvgElement, +}; diff --git a/-tmp/-wip/Image.Svg/SvgRef.ts b/-tmp/-wip/Image.Svg/SvgRef.ts new file mode 100644 index 0000000000..feb5b67c0a --- /dev/null +++ b/-tmp/-wip/Image.Svg/SvgRef.ts @@ -0,0 +1,35 @@ +import * as t from './types'; +import { SVG } from './libs'; + +/** + * A helper for working with <SVG> objects within a given DOM container element. + * + * NOTE: + * This helper assumes <svg> data assembled via the webpack plugin + * (see [cell.compiler]). + */ +export function SvgRef(filename: string, elContainer: HTMLElement): t.SvgRef { + filename = (filename ?? '') + .trim() + .replace(/^\#/, '') + .replace(/\.svg$/, ''); + filename = `${filename}.svg`; + + return { + filename, + find(id: string) { + id = (id || '').trim().replace(/^\#/, ''); + const query = `#${Format.toId(filename)}__${Format.toId(id)}`; + const el = elContainer.querySelector(query); + return el ? SVG(el) : undefined; + }, + }; +} + +/** + * [Helpers] + */ + +const Format = { + toId: (value: string) => value.replace(/\./g, '_'), +}; diff --git a/-tmp/-wip/Image.Svg/dev/DEV.tsx b/-tmp/-wip/Image.Svg/dev/DEV.tsx new file mode 100644 index 0000000000..c4096fee40 --- /dev/null +++ b/-tmp/-wip/Image.Svg/dev/DEV.tsx @@ -0,0 +1,43 @@ +import React from 'react'; +import { DevActions } from 'sys.ui.dev'; + +import { Sample, SampleProps } from './Sample'; + +type Ctx = { props: SampleProps }; + +/** + * Actions + */ +export const actions = DevActions<Ctx>() + .namespace('ui.Image.Svg') + .context((e) => { + if (e.prev) return e.prev; + return { + props: { width: 600, color: 'dark' }, + }; + }) + + .items((e) => { + e.title('<Svg>'); + + e.button('width: 600', (e) => (e.ctx.props.width = 600)); + e.button('width: 200', (e) => (e.ctx.props.width = 200)); + + e.hr(1, 0.1); + + e.button('color: dark', (e) => (e.ctx.props.color = 'dark')); + e.button('color: blue', (e) => (e.ctx.props.color = 'blue')); + + e.hr(); + }) + + .subject((e) => { + e.settings({ + host: { background: -0.04 }, + layout: { cropmarks: -0.2 }, + }); + + e.render(<Sample {...e.ctx.props} />); + }); + +export default actions; diff --git a/-tmp/-wip/Image.Svg/dev/Sample.tsx b/-tmp/-wip/Image.Svg/dev/Sample.tsx new file mode 100644 index 0000000000..39ba27d054 --- /dev/null +++ b/-tmp/-wip/Image.Svg/dev/Sample.tsx @@ -0,0 +1,35 @@ +import '../types.declare'; + +import React, { useEffect } from 'react'; + +import { Svg } from '..'; +import { COLORS, css } from '../../../common'; +import Image from '../../../../static/images/sample/svg.sample.svg'; + +export type SampleProps = { color: 'dark' | 'blue'; width: number }; + +export const Sample: React.FC<SampleProps> = (props) => { + const { width } = props; + const ref = React.useRef<HTMLDivElement>(null); + + useEffect(() => { + const svg = Svg.ref('svg.sample', ref.current as HTMLElement); + + const isDark = props.color === 'dark'; + const color = isDark ? COLORS.DARK : COLORS.BLUE; + + const tick = svg.find('tick'); + const outline = svg.find('border-outline'); + + tick?.opacity(isDark ? 1 : 0.2); + outline?.stroke(color); + }, [props.color]); + + const styles = { base: css({}) }; + + return ( + <div ref={ref} {...css(styles.base)}> + <Image width={width} /> + </div> + ); +}; diff --git a/-tmp/-wip/Image.Svg/index.ts b/-tmp/-wip/Image.Svg/index.ts new file mode 100644 index 0000000000..6d98b36350 --- /dev/null +++ b/-tmp/-wip/Image.Svg/index.ts @@ -0,0 +1,3 @@ +export { Svg } from './Svg'; +export { SvgRef } from './SvgRef'; +export { SvgElement } from './libs'; diff --git a/-tmp/-wip/Image.Svg/libs.ts b/-tmp/-wip/Image.Svg/libs.ts new file mode 100644 index 0000000000..27e69bb61d --- /dev/null +++ b/-tmp/-wip/Image.Svg/libs.ts @@ -0,0 +1 @@ +export { SVG, Element as SvgElement } from '@svgdotjs/svg.js'; diff --git a/-tmp/-wip/Image.Svg/types.declare.ts b/-tmp/-wip/Image.Svg/types.declare.ts new file mode 100644 index 0000000000..499de023b4 --- /dev/null +++ b/-tmp/-wip/Image.Svg/types.declare.ts @@ -0,0 +1,11 @@ +/** + * Importing <SVG> assets. + * See: + * https://react-svgr.com + */ +declare module '*.svg' { + import React = require('react'); + type Svg = React.SVGProps<SVGSVGElement> & { width?: number; height?: number }; + export const ReactComponent: React.FC<Svg>; + export default ReactComponent; +} diff --git a/-tmp/-wip/Image.Svg/types.ts b/-tmp/-wip/Image.Svg/types.ts new file mode 100644 index 0000000000..d69b201621 --- /dev/null +++ b/-tmp/-wip/Image.Svg/types.ts @@ -0,0 +1,6 @@ +import { Element as SvgElement } from '@svgdotjs/svg.js'; + +export type SvgRef = { + filename: string; + find(id: string): SvgElement | undefined; +}; diff --git a/.github/workflows/Video.tsx b/.github/workflows/Video.tsx deleted file mode 100644 index 8d465ab698..0000000000 --- a/.github/workflows/Video.tsx +++ /dev/null @@ -1,54 +0,0 @@ -// @ts-types="@types/react" -import React from 'react'; - -// import '@vidstack/react/player/styles/base.css'; -// import '@vidstack/react/player/styles/plyr/theme.css'; - -import { MediaPlayer, MediaProvider } from '@vidstack/react'; -// import { PlyrLayout, plyrLayoutIcons } from '@vidstack/react/player/layouts/plyr'; - -console.log('MediaPlayer', MediaPlayer); - -/** - * TODO š· - add workspace/plugin refs for VitePress ā see: @sys/driver-vite - */ -import { Foo } from '@sys/tmp/ui'; - -export type VideoProps = { - title?: string; - src?: string; -}; - -/** - * Component - */ -export const Video: React.FC<VideoProps> = (props: VideoProps) => { - const src = props.src || DEFAULTS.src; - - // const elPlayer = ( - // <MediaPlayer title={props.title} src={src} playsInline={true}> - // <MediaProvider /> - // <PlyrLayout - // // thumbnails="https://files.vidstack.io/sprite-fight/thumbnails.vtt" - // icons={plyrLayoutIcons} - // /> - // </MediaPlayer> - // ); - - // TEMP š· - return ( - <div> - <div> - import: <Foo /> - </div> - {/* {elPlayer} */} - </div> - ); -}; - -/** - * Constants - */ -export const DEFAULTS = { - src: 'vimeo/499921561', // Tubes. -} as const; diff --git a/.github/workflows/jsr.yaml b/.github/workflows/jsr.yaml index c5323c288a..c0c292fb44 100644 --- a/.github/workflows/jsr.yaml +++ b/.github/workflows/jsr.yaml @@ -25,10 +25,21 @@ jobs: steps: - uses: actions/checkout@v3 - - uses: denoland/setup-deno@v1 + + - name: 'Install ESM Runtime: Deno 2.x' + uses: denoland/setup-deno@v1 with: deno-version: v2 + - name: 'Install ES Modules from JSR: https://jsr.io/@sys' + run: deno task info + + - name: Deno Info + run: deno info && deno --version + + - name: System Info + run: deno task info + - name: publish module ā "@sys/types" run: | cd code/sys/types @@ -137,6 +148,12 @@ jobs: deno task test --parallel deno publish --allow-dirty + - name: publish module ā "@sys/ui-react-components" + run: | + cd code/sys.ui/ui-react-components + deno task test --parallel + deno publish --allow-dirty + # š· # - name: publish module ā "@sys/driver-automerge" # run: | diff --git a/.github/workflows/jsr.yaml_ b/.github/workflows/jsr.yaml_ index 0333ddb396..41f1e48fd9 100644 --- a/.github/workflows/jsr.yaml_ +++ b/.github/workflows/jsr.yaml_ @@ -34,3 +34,15 @@ jobs: cd code/sys.driver/driver-vite deno task test --parallel --trace-leaks deno publish --allow-dirty + + - name: publish module ā "@sys/driver-vitepress" + run: | + cd code/sys.driver/driver-vitepress + deno task test --parallel --trace-leaks + deno publish --allow-dirty + + - name: publish module ā "@sys/tmp" + run: | + cd code/sys.tmp + deno task test --parallel --trace-leaks + deno publish --allow-dirty diff --git a/README.md b/README.md index 407b7b563f..3d125c0086 100644 --- a/README.md +++ b/README.md @@ -29,7 +29,6 @@ [jsr-ci-url]: https://github.com/sys-repo/sys/actions/workflows/jsr.yaml - Monorepo comprising the core set of shared `/sys` "system" modules that flexibly compose into varying arrangements of (1) extremely-late-bound, (2) strongly typed, (3) decentralised, "cell like" functional processes. - modules: [sys](/code/sys/) ā standard libs diff --git a/code/-tmpl/-test.ui.ts b/code/-tmpl/-test.ui.ts new file mode 100644 index 0000000000..6ef3bbc75e --- /dev/null +++ b/code/-tmpl/-test.ui.ts @@ -0,0 +1,7 @@ +/** + * @module + * Testing tools running in the browser/ui. + */ +export { expect } from '@sys/std/testing'; +export { Dev, Spec } from '@sys/ui-react-devharness'; +export * from './common.ts'; diff --git a/code/-tmpl/common.t.ts b/code/-tmpl/common.t.ts new file mode 100644 index 0000000000..65fc1d3b01 --- /dev/null +++ b/code/-tmpl/common.t.ts @@ -0,0 +1,10 @@ +/** + * š· + * NB: placeholder type exports to ensure template imports don't error. + */ +export type * from './deno/src/common/t.ts'; +export type * from './m.mod.ui/t.ts'; +export type * from './m.mod/t.ts'; + +export type * from '@sys/types/t'; +export type { CssInput } from '@sys/ui-css/t'; diff --git a/code/-tmpl/common.ts b/code/-tmpl/common.ts index c0fb3991aa..440e91203b 100644 --- a/code/-tmpl/common.ts +++ b/code/-tmpl/common.ts @@ -1,2 +1,13 @@ -// NB: placeholder to ensure template imports don't error. +/** + * š· + * NB: placeholder exports to ensure template imports don't error. + */ +export type * as t from './common.t.ts'; export * from './deno/src/common.ts'; + +/** + * UI Refs: + */ +export { Color, css } from '@sys/ui-css'; +export { Signal } from '@sys/ui-react'; +export { Button } from '@sys/ui-react-components'; diff --git a/code/-tmpl/deno/deno.json b/code/-tmpl/deno/deno.json index 0f09aee165..2a9035f348 100644 --- a/code/-tmpl/deno/deno.json +++ b/code/-tmpl/deno/deno.json @@ -5,6 +5,7 @@ "tasks": { "lint": "deno lint", "dry": "deno publish --allow-dirty --dry-run", + "check": "deno check src/*", "clean": "deno run -RWE ./-scripts/-clean.ts", "test": "deno test -RWNE --allow-run --allow-ffi", "dev": "deno run -RWNE --allow-run --allow-ffi ./-scripts/-dev.ts", diff --git a/code/-tmpl/deno/src/common/libs.ts b/code/-tmpl/deno/src/common/libs.ts index eb7b6f00c7..70c75258ea 100644 --- a/code/-tmpl/deno/src/common/libs.ts +++ b/code/-tmpl/deno/src/common/libs.ts @@ -1 +1 @@ -export { Err, Pkg, Time } from '@sys/std'; +export { Err, Pkg, rx, Signal, Time } from '@sys/std'; diff --git a/code/-tmpl/deno/src/pkg.ts b/code/-tmpl/deno/src/pkg.ts index 79cb3f9c80..68bbe9281e 100644 --- a/code/-tmpl/deno/src/pkg.ts +++ b/code/-tmpl/deno/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/name', version: '0.0.0' }; diff --git a/code/-tmpl/m.mod.ui/-SPEC.Debug.tsx b/code/-tmpl/m.mod.ui/-SPEC.Debug.tsx new file mode 100644 index 0000000000..e9f74789b8 --- /dev/null +++ b/code/-tmpl/m.mod.ui/-SPEC.Debug.tsx @@ -0,0 +1,66 @@ +import React from 'react'; +import { type t, Button, Color, css, Signal } from './common.ts'; + +type P = t.MyComponentProps; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + const s = Signal.create; + const props = { + theme: s<P['theme']>('Light'), + }; + const api = { + props, + listen() { + const p = props; + p.theme.value; + }, + }; + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => debug.listen()); + + /** + * Render: + */ + const styles = { + base: css({}), + title: css({ fontWeight: 'bold', marginBottom: 10 }), + cols: css({ display: 'grid', gridTemplateColumns: 'auto 1fr auto' }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={css(styles.title, styles.cols).class}> + <div>{'Title'}</div> + <div /> + <div></div> + </div> + + <Button + block + label={() => `theme: ${p.theme.value ?? '<undefined>'}`} + onClick={() => Signal.cycle<P['theme']>(p.theme, ['Light', 'Dark'])} + /> + + <hr /> + </div> + ); +}; diff --git a/code/-tmpl/m.mod.ui/-SPEC.tsx b/code/-tmpl/m.mod.ui/-SPEC.tsx new file mode 100644 index 0000000000..5a8a5966c3 --- /dev/null +++ b/code/-tmpl/m.mod.ui/-SPEC.tsx @@ -0,0 +1,28 @@ +import { Dev, Spec, Signal } from '../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { MyComponent } from './mod.ts'; + +export default Spec.describe('MyComponent', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + Dev.Theme.signalEffect(ctx, p.theme, 1); + Signal.effect(() => { + debug.listen(); + ctx.redraw(); + }); + + ctx.subject + .size() + .display('grid') + .render((e) => <MyComponent theme={p.theme.value} />); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/code/-tmpl/m.mod.ui/common.ts b/code/-tmpl/m.mod.ui/common.ts new file mode 100644 index 0000000000..a45006948c --- /dev/null +++ b/code/-tmpl/m.mod.ui/common.ts @@ -0,0 +1,7 @@ +export * from '../common.ts'; + +/** + * Constants: + */ +export const DEFAULTS = {} as const; +export const D = DEFAULTS; diff --git a/code/-tmpl/m.mod.ui/mod.ts b/code/-tmpl/m.mod.ui/mod.ts new file mode 100644 index 0000000000..e089acf488 --- /dev/null +++ b/code/-tmpl/m.mod.ui/mod.ts @@ -0,0 +1,4 @@ +/** + * @module + */ +export { MyComponent } from './ui.tsx'; diff --git a/code/-tmpl/m.mod.ui/t.ts b/code/-tmpl/m.mod.ui/t.ts new file mode 100644 index 0000000000..2b3310c354 --- /dev/null +++ b/code/-tmpl/m.mod.ui/t.ts @@ -0,0 +1,9 @@ +import type { t } from './common.ts'; + +/** + * <Component>: + */ +export type MyComponentProps = { + theme?: t.CommonTheme; + style?: t.CssInput; +}; diff --git a/code/-tmpl/m.mod.ui/ui.tsx b/code/-tmpl/m.mod.ui/ui.tsx new file mode 100644 index 0000000000..d9dd7c5c16 --- /dev/null +++ b/code/-tmpl/m.mod.ui/ui.tsx @@ -0,0 +1,23 @@ +import React from 'react'; +import { type t, Color, css, D } from './common.ts'; + +export const MyComponent: React.FC<t.MyComponentProps> = (props) => { + const {} = props; + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ + backgroundColor: 'rgba(255, 0, 0, 0.1)' /* RED */, + color: theme.fg, + }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div>{'š· Hello MyComponent'}</div> + </div> + ); +}; diff --git a/code/-tmpl/pkg/pkg.ts b/code/-tmpl/pkg/pkg.ts new file mode 100644 index 0000000000..d7df14757d --- /dev/null +++ b/code/-tmpl/pkg/pkg.ts @@ -0,0 +1,16 @@ +import type { Pkg } from '@sys/types'; + +/** + * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). + */ +export const pkg: Pkg = { name: '<NAME>', version: '<VERSION>' }; diff --git a/code/sys.driver/driver-automerge/deno.json b/code/sys.driver/driver-automerge/deno.json index dcd4edf0bb..c9eb0a0ba7 100644 --- a/code/sys.driver/driver-automerge/deno.json +++ b/code/sys.driver/driver-automerge/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/driver-automerge", - "version": "0.0.84", + "version": "0.0.94", "license": "MIT", "tasks": { "lint": "deno lint", diff --git a/code/sys.driver/driver-automerge/src/-test/ui.sample/App.tsx b/code/sys.driver/driver-automerge/src/-test/ui.sample/App.tsx index c524f7bb57..47a80ecd4a 100644 --- a/code/sys.driver/driver-automerge/src/-test/ui.sample/App.tsx +++ b/code/sys.driver/driver-automerge/src/-test/ui.sample/App.tsx @@ -29,7 +29,7 @@ export const App: React.FC<AppProps> = (props) => { }, []); /** - * Render + * Render: */ const theme = Color.theme(props.theme); const styles = { diff --git a/code/sys.driver/driver-automerge/src/pkg.ts b/code/sys.driver/driver-automerge/src/pkg.ts index 79cb3f9c80..2286978f45 100644 --- a/code/sys.driver/driver-automerge/src/pkg.ts +++ b/code/sys.driver/driver-automerge/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/driver-automerge', version: '0.0.94' }; diff --git a/code/sys.driver/driver-deno/deno.json b/code/sys.driver/driver-deno/deno.json index 75e47eefd1..f4216d2238 100644 --- a/code/sys.driver/driver-deno/deno.json +++ b/code/sys.driver/driver-deno/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/driver-deno", - "version": "0.0.86", + "version": "0.0.97", "license": "MIT", "tasks": { "dev": "deno run -RNE --watch ./-scripts/-dev.ts", diff --git a/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoFile/-.test.ts b/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoFile/-.test.ts index ae4283020b..ee07197fc3 100644 --- a/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoFile/-.test.ts +++ b/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoFile/-.test.ts @@ -56,7 +56,7 @@ describe('DenoFile', () => { describe('path', () => { it('from path: exists', async () => { const res = await DenoFile.workspace(rootPath); - const dirs = res.children.map((child) => Fs.dirname(child.path)); + const dirs = res.children.map((child) => child.path.dir); expect(res.exists).to.eql(true); expect(dirs.some((p) => p === 'code/sys/std')).to.be.true; expect(res.file).to.eql(rootPath); @@ -70,7 +70,7 @@ describe('DenoFile', () => { const dirs = root.data?.workspace?.map((p) => p.replace(/^\.\//, '')); expect(a.exists).to.eql(true); - expect(a.children.map((m) => Fs.dirname(m.path))).to.eql(dirs); + expect(a.children.map((m) => m.path.dir)).to.eql(dirs); expect(b.exists).to.eql(false); expect(b.children).to.eql([]); @@ -84,12 +84,27 @@ describe('DenoFile', () => { }); describe('workspace.children', () => { - it('children.files (paths)', async () => { + it('child.path', async () => { const ws = await DenoFile.workspace(); - const dirs = ws.children.map((child) => Fs.dirname(child.path)); + const dirs = ws.children.map((child) => child.path.dir); const paths = dirs.map((subdir) => Fs.join('./', subdir, 'deno.json')); ws.children.forEach((child) => { - expect(paths.includes(child.path)).to.eql(true); + expect(paths.includes(child.path.denofile)).to.eql(true); + }); + }); + + it('child.pkg', async () => { + const ws = await DenoFile.workspace(); + const match = ws.children.find((m) => m.denofile.name === pkg.name); + + expect(match).to.exist; + expect(match?.pkg).to.eql(pkg); + expect(match?.denofile.name).to.eql(pkg.name); + expect(match?.denofile.version).to.eql(pkg.version); + + ws.children.forEach((child) => { + expect(child.pkg.name).to.eql(child.denofile.name || '<unnamed>'); + expect(child.pkg.version).to.eql(child.denofile.version || '0.0.0'); }); }); }); @@ -100,7 +115,7 @@ describe('DenoFile', () => { expect(ws.modules.ok).to.eql(true); expect(ws.modules.error).to.eql(undefined); - const namesA = ws.children.map((m) => m.file.name ?? ''); + const namesA = ws.children.map((m) => m.denofile.name ?? ''); const namesB = ws.modules.items.map((m) => m.name); expect(namesA.filter(Boolean).toSorted()).to.eql(namesB.filter(Boolean).toSorted()); }); diff --git a/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoFile/t.Workspace.ts b/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoFile/t.Workspace.ts index abf42e21e0..11958b11eb 100644 --- a/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoFile/t.Workspace.ts +++ b/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoFile/t.Workspace.ts @@ -15,6 +15,7 @@ export type DenoWorkspace = { * Represents a single child of a workspace. */ export type DenoWorkspaceChild = { - readonly file: t.DenoFileJson; - readonly path: t.StringPath; + readonly path: { readonly dir: t.StringDir; readonly denofile: t.StringPath }; + readonly denofile: t.DenoFileJson; + readonly pkg: t.Pkg; }; diff --git a/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoFile/t.ts b/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoFile/t.ts index c0c95ffb3e..3bcc283ced 100644 --- a/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoFile/t.ts +++ b/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoFile/t.ts @@ -39,7 +39,15 @@ export type DenoFileJson = { licence?: string; tasks?: Record<string, string>; importMap?: t.StringPath; - imports?: Record<string, string>; + imports?: Record<string, t.StringModuleSpecifier>; exports?: Record<string, string>; workspace?: t.StringPath[]; }; + +/** + * A JSON file containing an import-map. + * Referenced by `importMap` path in `deno.json` file. + */ +export type DenoImportMapJson = { + imports?: Record<string, t.StringModuleSpecifier>; +}; diff --git a/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoFile/u.workspace.ts b/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoFile/u.workspace.ts index 7f10d27354..40c5312787 100644 --- a/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoFile/u.workspace.ts +++ b/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoFile/u.workspace.ts @@ -32,7 +32,7 @@ export const workspace: t.DenoFileLib['workspace'] = async (path, options = {}) file, children, get modules() { - return _modules || (_modules = Esm.modules(toModuleSpecifiers(children.map((m) => m.file)))); + return _modules || (_modules = Esm.modules(toSpecifiers(children.map((m) => m.denofile)))); }, }; return api; @@ -72,12 +72,22 @@ async function loadFiles(root: t.StringDir, subpaths: t.StringPath[]) { const promises = subpaths .map((subpath) => Path.join(root, subpath, 'deno.json')) .map((path) => load(path)); + + const toChild = (path: t.StringPath, denofile: t.DenoFileJson): t.DenoWorkspaceChild => { + const dir = Path.dirname(path); + return { + path: { dir, denofile: path }, + pkg: { name: denofile.name ?? '<unnamed>', version: denofile.version ?? '0.0.0' }, + denofile, + }; + }; + return (await Promise.all(promises)) - .map((m): t.DenoWorkspaceChild => ({ file: m.data!, path: trimPath(m.path) })) - .filter((m) => !!m.file); + .filter((m) => !!m.data) + .map((m) => toChild(trimPath(m.path), m.data!)); } -function toModuleSpecifiers(files: t.DenoFileJson[]) { +function toSpecifiers(files: t.DenoFileJson[]): t.StringModuleSpecifier[] { return files .filter((file) => !!file.name) .map((file) => { diff --git a/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoModule/-backup.test.ts b/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoModule/-backup.test.ts index 4fc1ebd531..e18ebb4b5c 100644 --- a/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoModule/-backup.test.ts +++ b/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoModule/-backup.test.ts @@ -29,7 +29,7 @@ describe('DenoModule.backup', () => { // Copy in sample project-files to create snapshot/backup of. const tmpl = SAMPLE.sample1.tmpl(); - await tmpl.copy(source); + await tmpl.write(source); await assertExists(Fs.join(source, PATHS.dist), true); await assertExists(Fs.join(source, PATHS.tmp), true); @@ -68,7 +68,7 @@ describe('DenoModule.backup', () => { const sample = sampleFs(); const source = sample.dir; - await SAMPLE.sample1.tmpl().copy(source); + await SAMPLE.sample1.tmpl().write(source); await assertExists(Fs.join(source, PATHS.dist), true); await assertExists(Fs.join(source, PATHS.tmp), true); @@ -100,7 +100,7 @@ describe('DenoModule.backup', () => { const sample = sampleFs(); const source = sample.dir; - await SAMPLE.sample1.tmpl().copy(source); + await SAMPLE.sample1.tmpl().write(source); await assertExists(Fs.join(source, PATHS.dist), true); await assertExists(Fs.join(source, PATHS.tmp), true); @@ -131,7 +131,7 @@ describe('DenoModule.backup', () => { it('{message} param ā commit details', async () => { const sample = sampleFs(); const source = sample.dir; - await SAMPLE.sample1.tmpl().copy(source); + await SAMPLE.sample1.tmpl().write(source); const message = 'š hello'; diff --git a/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoModule/u.upgrade.ts b/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoModule/u.upgrade.ts index c3b7a593e3..76e7f09fc0 100644 --- a/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoModule/u.upgrade.ts +++ b/code/sys.driver/driver-deno/src/ns.Runtime/m.DenoModule/u.upgrade.ts @@ -79,7 +79,7 @@ export const upgrade: t.DenoModuleLib['upgrade'] = async (args) => { /** * Finish up. */ - const fmtTargetVer = c.bold(c.green(Semver.toString(targetVersion))); + const fmtTargetVer = c.bold(c.brightCyan(Semver.toString(targetVersion))); console.info(c.green(`Project at version:`)); console.info(c.gray(`${c.white(c.bold(moduleName))}@${fmtTargetVer}`)); console.info(); diff --git a/code/sys.driver/driver-deno/src/pkg.ts b/code/sys.driver/driver-deno/src/pkg.ts index 79cb3f9c80..c7559a5430 100644 --- a/code/sys.driver/driver-deno/src/pkg.ts +++ b/code/sys.driver/driver-deno/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/driver-deno', version: '0.0.97' }; diff --git a/code/sys.driver/driver-immer/deno.json b/code/sys.driver/driver-immer/deno.json index f492539ba2..901fa8136e 100644 --- a/code/sys.driver/driver-immer/deno.json +++ b/code/sys.driver/driver-immer/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/driver-immer", - "version": "0.0.86", + "version": "0.0.96", "license": "MIT", "tasks": { "test": "deno test -RW", diff --git a/code/sys.driver/driver-immer/src/pkg.ts b/code/sys.driver/driver-immer/src/pkg.ts index 79cb3f9c80..3b547d69b0 100644 --- a/code/sys.driver/driver-immer/src/pkg.ts +++ b/code/sys.driver/driver-immer/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/driver-immer', version: '0.0.96' }; diff --git a/code/sys.driver/driver-obsidian/deno.json b/code/sys.driver/driver-obsidian/deno.json index 833d9b3c86..98b4711bdf 100644 --- a/code/sys.driver/driver-obsidian/deno.json +++ b/code/sys.driver/driver-obsidian/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/driver-obsidian", - "version": "0.0.73", + "version": "0.0.83", "license": "MIT", "tasks": { "lint": "deno lint", diff --git a/code/sys.driver/driver-obsidian/src/pkg.ts b/code/sys.driver/driver-obsidian/src/pkg.ts index 79cb3f9c80..7ebb194ea7 100644 --- a/code/sys.driver/driver-obsidian/src/pkg.ts +++ b/code/sys.driver/driver-obsidian/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/driver-obsidian', version: '0.0.83' }; diff --git a/code/sys.driver/driver-ollama/deno.json b/code/sys.driver/driver-ollama/deno.json index 389782e25e..ee24136e57 100644 --- a/code/sys.driver/driver-ollama/deno.json +++ b/code/sys.driver/driver-ollama/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/driver-ollama", - "version": "0.0.40", + "version": "0.0.50", "license": "MIT", "tasks": { "lint": "deno lint", diff --git a/code/sys.driver/driver-ollama/src/pkg.ts b/code/sys.driver/driver-ollama/src/pkg.ts index 79cb3f9c80..147f053698 100644 --- a/code/sys.driver/driver-ollama/src/pkg.ts +++ b/code/sys.driver/driver-ollama/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/driver-ollama', version: '0.0.50' }; diff --git a/code/sys.driver/driver-orbiter/deno.json b/code/sys.driver/driver-orbiter/deno.json index ed6855b66a..66cb50f16c 100644 --- a/code/sys.driver/driver-orbiter/deno.json +++ b/code/sys.driver/driver-orbiter/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/driver-orbiter", - "version": "0.0.63", + "version": "0.0.73", "license": "MIT", "tasks": { "lint": "deno lint", diff --git a/code/sys.driver/driver-orbiter/src/pkg.ts b/code/sys.driver/driver-orbiter/src/pkg.ts index 79cb3f9c80..4c9c503e67 100644 --- a/code/sys.driver/driver-orbiter/src/pkg.ts +++ b/code/sys.driver/driver-orbiter/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/driver-orbiter', version: '0.0.73' }; diff --git a/code/sys.driver/driver-quilibrium/deno.json b/code/sys.driver/driver-quilibrium/deno.json index 42b86ece8c..19c4ba27cb 100644 --- a/code/sys.driver/driver-quilibrium/deno.json +++ b/code/sys.driver/driver-quilibrium/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/driver-quilibrium", - "version": "0.0.76", + "version": "0.0.86", "license": "MIT", "tasks": { "test": "deno test -RW", diff --git a/code/sys.driver/driver-quilibrium/src/pkg.ts b/code/sys.driver/driver-quilibrium/src/pkg.ts index 79cb3f9c80..23f25f9ea0 100644 --- a/code/sys.driver/driver-quilibrium/src/pkg.ts +++ b/code/sys.driver/driver-quilibrium/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/driver-quilibrium', version: '0.0.86' }; diff --git a/code/sys.driver/driver-vite/-scripts/-build.ts b/code/sys.driver/driver-vite/-scripts/-build.ts deleted file mode 100644 index ab8d18ca3b..0000000000 --- a/code/sys.driver/driver-vite/-scripts/-build.ts +++ /dev/null @@ -1,14 +0,0 @@ -/** - * Run in a child-process (hence the `-allow-run` requirement). - */ -import { Vite } from '@sys/driver-vite'; -import { pkg } from '../src/pkg.ts'; - -// const input = './src/-test/vite.sample-1/main.ts'; -const input = './src/-test/vite.sample-2/index.html'; -// const input = './src/-test/vite.sample-3/main.ts'; - -const bundle = await Vite.build({ pkg, input }); - -console.log(`-------------------------------------------`); -console.info(bundle.toString({ pad: true })); diff --git a/code/sys.driver/driver-vite/-scripts/-main.ts b/code/sys.driver/driver-vite/-scripts/-main.ts index dbb18b8049..d19bf9754b 100644 --- a/code/sys.driver/driver-vite/-scripts/-main.ts +++ b/code/sys.driver/driver-vite/-scripts/-main.ts @@ -1,10 +1,5 @@ /** - * Sample used to pass "deno task" commands to the entry point. - * - * @example - * Invoked via deno run command: - * ```bash - * deno run jsr:@sys/http/server/start --port=1234 - * ``` + * Sample used to pass "deno task" commands to the entry point + * locally within the mono-repo rather than calling out to JSR. */ import '../src/-entry/-main.ts'; diff --git a/code/sys.driver/driver-vite/-scripts/-prep.ts b/code/sys.driver/driver-vite/-scripts/-prep.ts index b8dbaffbe4..3ff9a84aeb 100644 --- a/code/sys.driver/driver-vite/-scripts/-prep.ts +++ b/code/sys.driver/driver-vite/-scripts/-prep.ts @@ -1,25 +1,22 @@ import { Vite } from '@sys/driver-vite'; -import { type t, DenoDeps, DenoFile, Fs, PATHS, c, pkg } from './common.ts'; +import { c, Fs, PATHS, pkg, Semver } from './common.ts'; const resolve = (...parts: string[]) => Fs.join(import.meta.dirname ?? '', '..', ...parts); await Fs.remove(resolve('.tmp')); /** - * Save monorepo deps. + * Update to latest dependency versions. */ -const ws = await DenoFile.workspace(); -const deps: t.Dep[] = ws.modules.items.map((esm) => DenoDeps.toDep(esm)); - -const dir = resolve('src/-tmpl/.sys'); -await Fs.copy(Fs.join(ws.dir, 'deps.yaml'), Fs.join(dir, 'deps.yaml'), { force: true }); -await Fs.write(Fs.join(dir, 'deps.sys.yaml'), DenoDeps.toYaml(deps).text); +await Vite.Tmpl.prep(); /** - * Bundle files (for code-registry). + * Bundle files inline, base64-string FileMap (NB: so code can be referenfed within the registry). */ const bundle = Vite.Tmpl.Bundle; await bundle.toFilemap(); await bundle.writeToFile(resolve(PATHS.tmpl.tmp)); // NB: test output. -console.info(c.brightCyan('ā Prep Complete:'), `${pkg.name}@${c.brightCyan(pkg.version)}`); +const fmtVersion = Semver.Fmt.colorize(pkg.version); +const fmtModule = `${pkg.name}${c.dim('@')}${fmtVersion}`; +console.info(c.brightCyan('ā Prep Complete:'), fmtModule); console.info(); diff --git a/code/sys.driver/driver-vite/-scripts/-reset.ts b/code/sys.driver/driver-vite/-scripts/-reset.ts index c22fe5afba..55ab57446e 100644 --- a/code/sys.driver/driver-vite/-scripts/-reset.ts +++ b/code/sys.driver/driver-vite/-scripts/-reset.ts @@ -1,6 +1,7 @@ -import { Fs } from '@sys/fs'; +import { Fs } from './common.ts'; const remove = (path: string) => Fs.remove(Fs.resolve(path), { log: true }); await remove('./dist'); await remove('./.tmp'); +await remove('./.swc'); await remove('./node_modules'); diff --git a/code/sys.driver/driver-vite/-scripts/-tmp.ts b/code/sys.driver/driver-vite/-scripts/-tmp.ts index abe06e35d8..e133d73e74 100644 --- a/code/sys.driver/driver-vite/-scripts/-tmp.ts +++ b/code/sys.driver/driver-vite/-scripts/-tmp.ts @@ -1,4 +1,4 @@ -import { Fs } from '@sys/fs'; +import { Fs } from './common.ts'; console.log('tmp š·\n'); diff --git a/code/sys.driver/driver-vite/deno.json b/code/sys.driver/driver-vite/deno.json index f25a2cfe9d..93585b1acf 100644 --- a/code/sys.driver/driver-vite/deno.json +++ b/code/sys.driver/driver-vite/deno.json @@ -1,10 +1,11 @@ { "name": "@sys/driver-vite", - "version": "0.0.121", + "version": "0.0.137", "license": "MIT", "tasks": { "lint": "deno lint", "dry": "deno publish --allow-dirty --dry-run", + "check": "deno check src/*", "test": "deno test -RWNE --allow-run --allow-ffi --allow-sys", "reset": "deno run -RWE ./-scripts/-reset.ts", "init": "deno run -RWNE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=init --dir=./.tmp/sample", diff --git a/code/sys.driver/driver-vite/src/-entry/m.Entry.ts b/code/sys.driver/driver-vite/src/-entry/m.Entry.ts index b3f1006faa..021b777b41 100644 --- a/code/sys.driver/driver-vite/src/-entry/m.Entry.ts +++ b/code/sys.driver/driver-vite/src/-entry/m.Entry.ts @@ -9,17 +9,21 @@ import { build } from './u.build.ts'; import { dev } from './u.dev.ts'; import { serve } from './u.serve.ts'; +type O = Record<string, unknown>; + export const ViteEntry: t.ViteEntryLib = { dev, build, serve, async main(input) { - const args = wrangle.args(input ?? Deno.args); + const argsAsType = <T extends O>() => wrangle.args<T>((input ?? Deno.args) as string[]); + const args = argsAsType<t.ViteEntryArgs>(); const cmd = args.cmd; if (cmd === 'init') { const { init } = await import('./u.init.ts'); + console.log('args', args); await init(args); return; } @@ -94,8 +98,7 @@ export const ViteEntry: t.ViteEntryLib = { * Helpers */ const wrangle = { - args(argv: string[] | t.ViteEntryArgs) { - type T = t.ViteEntryArgs; + args<T extends O>(argv: string[] | T) { return Array.isArray(argv) ? Args.parse<T>(argv) : (argv as T); }, } as const; diff --git a/code/sys.driver/driver-vite/src/-entry/t.ts b/code/sys.driver/driver-vite/src/-entry/t.ts index a2041a22ba..2bb545c058 100644 --- a/code/sys.driver/driver-vite/src/-entry/t.ts +++ b/code/sys.driver/driver-vite/src/-entry/t.ts @@ -33,7 +33,12 @@ export type ViteEntryArgs = | ViteEntryArgsHelp; /** The `init` command. */ -export type ViteEntryArgsInit = { cmd: 'init'; dir?: P; silent?: boolean }; +export type ViteEntryArgsInit = { + cmd: 'init'; + dir?: P; + silent?: boolean; + tmpl?: t.ViteTmplKind | boolean; +}; /** The `clean` command. */ export type ViteEntryArgsClean = { cmd: 'clean'; dir?: P }; diff --git a/code/sys.driver/driver-vite/src/-entry/u.build.ts b/code/sys.driver/driver-vite/src/-entry/u.build.ts index d098bcbe48..c3a9c1c761 100644 --- a/code/sys.driver/driver-vite/src/-entry/u.build.ts +++ b/code/sys.driver/driver-vite/src/-entry/u.build.ts @@ -1,4 +1,4 @@ -import { type t, Cli, Path, pkg, Vite } from './common.ts'; +import { type t, Path, pkg, Vite } from './common.ts'; /** * Run a local HTTP server from entry command-args. @@ -8,12 +8,10 @@ export async function build(args: t.ViteEntryArgsBuild) { if (args.cmd !== 'build') return; if (!silent) console.info(); - const spinner = Cli.Spinner.create('building', { silent }); const cwd = args.dir ? Path.resolve(args.dir) : Path.cwd(); - const bundle = await Vite.build({ cwd, pkg, silent }); + const bundle = await Vite.build({ cwd, pkg, silent, spinner: true }); if (!silent) { - spinner.stop(); console.info(bundle.toString({ pad: true })); } } diff --git a/code/sys.driver/driver-vite/src/-entry/u.init.ts b/code/sys.driver/driver-vite/src/-entry/u.init.ts index 27396f1080..f450c54bab 100644 --- a/code/sys.driver/driver-vite/src/-entry/u.init.ts +++ b/code/sys.driver/driver-vite/src/-entry/u.init.ts @@ -1,25 +1,57 @@ import { Vite } from '../m.Vite/mod.ts'; -import { type t, c, pkg, ViteLog } from './common.ts'; +import { type t, c, pkg, Semver, ViteLog, V } from './common.ts'; + +/** + * Args validation: + */ +export const TmplKindSchema = V.union([V.literal('Default'), V.literal('ComponentLib')]); +export const InitSchema = V.object({ + cmd: V.literal('init'), + dir: V.optional(V.string()), + silent: V.optional(V.boolean()), + tmpl: V.optional(TmplKindSchema, 'Default'), +}); /** * Run the initialization templates. */ export async function init(args: t.ViteEntryArgsInit) { - const { silent = false } = args; if (args.cmd !== 'init') return; + const { silent = false } = args; if (!silent) { console.info(); console.info(`${pkg.name} ${c.gray(pkg.version)}`); } - await Vite.Tmpl.update({ in: args.dir, silent }); + await Vite.Tmpl.write({ + in: args.dir, + tmpl: wrangle.tmplKind(args), + silent, + }); if (!silent) { console.info(); ViteLog.API.log(); + + const fmtVersion = Semver.Fmt.colorize(pkg.version); + const fmtModule = `${pkg.name}${c.dim('@')}${fmtVersion}`; + console.info(); - console.info(c.brightCyan('ā Init Complete:'), `${pkg.name}@${c.brightCyan(pkg.version)}`); + console.info(c.brightCyan('ā Init Complete:'), `${fmtModule}`); console.info(); } } + +/** + * Helpers + */ +const wrangle = { + tmplKind(args: t.ViteEntryArgsInit): t.ViteTmplKind { + if (!args.tmpl || args.tmpl === true) return 'Default'; + const validated = V.safeParse(TmplKindSchema, args.tmpl); + const issues = validated.issues; + issues?.forEach((issue) => console.warn(c.yellow('Parse Error: --tmpl:'), issue.message)); + return issues ? 'Default' : validated.output; + }, +} as const; diff --git a/code/sys.driver/driver-vite/src/-entry/u.serve.ts b/code/sys.driver/driver-vite/src/-entry/u.serve.ts index 8d07e4a9d5..89bef6b7a5 100644 --- a/code/sys.driver/driver-vite/src/-entry/u.serve.ts +++ b/code/sys.driver/driver-vite/src/-entry/u.serve.ts @@ -15,9 +15,9 @@ export async function serve(args: t.ViteEntryArgsServe) { const app = HttpServer.create({ pkg, hash, static: ['/*', dir] }); const options = HttpServer.options({ port, pkg, hash, silent }); - const fmtDir = c.gray(dir.replace(/^\.\//, '')); + const fmtDir = c.gray(dir.replace(/^\.\//, '').replace(/\/$/, '')); const fmtDirExists = c.yellow(!dirExists ? c.bold('(does not exist)') : ''); - console.info(c.gray(`Folder: ${fmtDir} ${fmtDirExists}`)); + console.info(c.gray(`Static: ${fmtDir}/ ${fmtDirExists}`)); Deno.serve(options, app.fetch); await HttpServer.keyboard({ port, print: !silent }); diff --git a/code/sys.driver/driver-vite/src/-test/vite.sample-1/main.tsx b/code/sys.driver/driver-vite/src/-test/vite.sample-1/main.tsx index d1021fa8db..5ca3d65a92 100644 --- a/code/sys.driver/driver-vite/src/-test/vite.sample-1/main.tsx +++ b/code/sys.driver/driver-vite/src/-test/vite.sample-1/main.tsx @@ -4,7 +4,7 @@ import { createRoot } from 'react-dom/client'; /** * Sample: render react component. */ -const root = createRoot(document.getElementById('root')); +const root = createRoot(document.getElementById('root')!); root.render( <StrictMode> <div>Hello World š</div> diff --git a/code/sys.driver/driver-vite/src/-test/vite.sample-1/vite.config.ts b/code/sys.driver/driver-vite/src/-test/vite.sample-1/vite.config.ts index 7c6b9851d1..ecc0c9967d 100644 --- a/code/sys.driver/driver-vite/src/-test/vite.sample-1/vite.config.ts +++ b/code/sys.driver/driver-vite/src/-test/vite.sample-1/vite.config.ts @@ -1,11 +1,9 @@ -import { Vite } from '@sys/driver-vite'; -import { defineConfig } from 'vite'; - -export const paths = Vite.Config.paths(import.meta.url); +import { Vite } from 'jsr:@sys/driver-vite'; +import { defineConfig } from 'npm:vite'; export default defineConfig(() => Vite.Config.app({ - paths, + paths: Vite.Config.paths(), chunks(e) { e.chunk('react', 'react'); e.chunk('react.dom', 'react-dom'); diff --git a/code/sys.driver/driver-vite/src/-test/vite.sample-2/src/-entry/main.tsx b/code/sys.driver/driver-vite/src/-test/vite.sample-2/src/-entry/main.tsx index 6d8aa959d6..16ec9cc25a 100644 --- a/code/sys.driver/driver-vite/src/-test/vite.sample-2/src/-entry/main.tsx +++ b/code/sys.driver/driver-vite/src/-test/vite.sample-2/src/-entry/main.tsx @@ -16,7 +16,7 @@ dynamic.then((mod) => console.info('š¦ dynmaic import', mod)); /** * Sample: render react component. */ -const root = createRoot(document.getElementById('root')); +const root = createRoot(document.getElementById('root')!); root.render( <StrictMode> <View style={{ border: `solid 1px blue` }} /> diff --git a/code/sys.driver/driver-vite/src/-test/vite.sample-2/src/common/t.ts b/code/sys.driver/driver-vite/src/-test/vite.sample-2/src/common/t.ts index 9e4786784f..00b8efd553 100644 --- a/code/sys.driver/driver-vite/src/-test/vite.sample-2/src/common/t.ts +++ b/code/sys.driver/driver-vite/src/-test/vite.sample-2/src/common/t.ts @@ -1,2 +1,2 @@ export type { CommonTheme } from '@sys/types'; -export type { CssValue } from '@sys/ui-css/t'; +export type { CssInput } from '@sys/ui-css/t'; diff --git a/code/sys.driver/driver-vite/src/-test/vite.sample-2/src/mod.ts b/code/sys.driver/driver-vite/src/-test/vite.sample-2/src/mod.ts new file mode 100644 index 0000000000..8e8dc7889b --- /dev/null +++ b/code/sys.driver/driver-vite/src/-test/vite.sample-2/src/mod.ts @@ -0,0 +1 @@ +export { Foo } from './m.foo.ts'; diff --git a/code/sys.driver/driver-vite/src/-test/vite.sample-2/vite.config.ts b/code/sys.driver/driver-vite/src/-test/vite.sample-2/vite.config.ts index b876bdb85b..4b0084c50e 100644 --- a/code/sys.driver/driver-vite/src/-test/vite.sample-2/vite.config.ts +++ b/code/sys.driver/driver-vite/src/-test/vite.sample-2/vite.config.ts @@ -1,8 +1,8 @@ -import { Vite } from '@sys/driver-vite'; -import { defineConfig } from 'vite'; +import { Vite } from 'jsr:@sys/driver-vite'; +import { defineConfig } from 'npm:vite'; export const paths = Vite.Config.paths({ - cwd: import.meta.url, + // cwd: import.meta.url, app: { entry: 'src/-entry/index.html' }, }); diff --git a/code/sys.driver/driver-vite/src/-test/vite.sample-config/custom/vite.config.ts b/code/sys.driver/driver-vite/src/-test/vite.sample-config/custom/vite.config.ts index 2fb9fa59da..219c506fe2 100644 --- a/code/sys.driver/driver-vite/src/-test/vite.sample-config/custom/vite.config.ts +++ b/code/sys.driver/driver-vite/src/-test/vite.sample-config/custom/vite.config.ts @@ -2,17 +2,17 @@ import { Vite } from '@sys/driver-vite'; import { defineConfig } from 'vite'; -export const paths = Vite.Config.paths({ - app: { - entry: '.tmp/sample/src/-test/index.html', - outDir: '.tmp/sample/dist', - }, -}); - /** * SAMPLE: Custom plugin (no customization). */ export default defineConfig(async () => { + const paths = Vite.Config.paths({ + app: { + entry: '.tmp/sample/src/-test/index.html', + outDir: '.tmp/sample/dist', + }, + }); + const app = await Vite.Config.app({ paths, diff --git a/code/sys.driver/driver-vite/src/-test/vite.sample-config/simple/vite.config.ts b/code/sys.driver/driver-vite/src/-test/vite.sample-config/simple/vite.config.ts index 06cce588fa..b1d70cc137 100644 --- a/code/sys.driver/driver-vite/src/-test/vite.sample-config/simple/vite.config.ts +++ b/code/sys.driver/driver-vite/src/-test/vite.sample-config/simple/vite.config.ts @@ -1,7 +1,7 @@ -import { Vite } from '@sys/driver-vite'; -import { defineConfig } from 'vite'; +import { Vite } from 'jsr:@sys/driver-vite'; +import { defineConfig } from 'npm:vite'; -export const paths = Vite.Config.paths({ +const paths = Vite.Config.paths({ app: { entry: '.tmp/sample/src/-test/index.html', outDir: '.tmp/sample/dist', diff --git a/code/sys.driver/driver-vite/src/-tmpl/-scripts/-tmp.ts b/code/sys.driver/driver-vite/src/-tmpl/-scripts/-tmp.ts index 9e0f27eee7..ba662b31d4 100644 --- a/code/sys.driver/driver-vite/src/-tmpl/-scripts/-tmp.ts +++ b/code/sys.driver/driver-vite/src/-tmpl/-scripts/-tmp.ts @@ -1 +1 @@ -console.info('š', import.meta.url); +console.info(`\nš ${import.meta.url}\n`); diff --git a/code/sys.driver/driver-vite/src/-tmpl/.sys/deps.sys.yaml b/code/sys.driver/driver-vite/src/-tmpl/.sys/deps.sys.yaml deleted file mode 100644 index 290c766cd9..0000000000 --- a/code/sys.driver/driver-vite/src/-tmpl/.sys/deps.sys.yaml +++ /dev/null @@ -1,36 +0,0 @@ -deno.json: - - import: jsr:@sys/types@0.0.82 - - import: jsr:@sys/std@0.0.131 - - import: jsr:@sys/color@0.0.35 - - import: jsr:@sys/testing@0.0.75 - - import: jsr:@sys/fs@0.0.79 - - import: jsr:@sys/cli@0.0.64 - - import: jsr:@sys/process@0.0.65 - - import: jsr:@sys/crypto@0.0.64 - - import: jsr:@sys/http@0.0.46 - - import: jsr:@sys/text@0.0.73 - - import: jsr:@sys/tmpl@0.0.79 - - import: jsr:@sys/cmd@0.0.80 - - import: jsr:@sys/jsr@0.0.45 - - import: jsr:@sys/ui-css@0.0.70 - - import: jsr:@sys/ui-dom@0.0.77 - - import: jsr:@sys/ui-react@0.0.84 - - import: jsr:@sys/ui-react-devharness@0.0.82 - - import: jsr:@sys/ui-react-components@0.0.38 - - import: jsr:@sys/driver-automerge@0.0.84 - - import: jsr:@sys/driver-deno@0.0.86 - - import: jsr:@sys/driver-immer@0.0.86 - - import: jsr:@sys/driver-obsidian@0.0.73 - - import: jsr:@sys/driver-ollama@0.0.40 - - import: jsr:@sys/driver-orbiter@0.0.63 - - import: jsr:@sys/driver-quilibrium@0.0.76 - - import: jsr:@sys/driver-vite@0.0.121 - - import: jsr:@sys/driver-vitepress@0.0.284-alpha.2 - - import: jsr:@sys/sys@0.0.55 - - import: jsr:@sys/main@0.0.57 - - import: jsr:@tdb/api@0.0.72 - - import: jsr:@tdb/slc@0.0.68 - - import: jsr:@tdb/tmp@0.0.72 - - import: jsr:@sys/name@0.0.0 - - import: jsr:@sys/tmp@0.0.95 -package.json: [] diff --git a/code/sys.driver/driver-vite/src/-tmpl/.sys/deps.yaml b/code/sys.driver/driver-vite/src/-tmpl/.sys/deps.yaml deleted file mode 100644 index 9bd188f371..0000000000 --- a/code/sys.driver/driver-vite/src/-tmpl/.sys/deps.yaml +++ /dev/null @@ -1,117 +0,0 @@ -# -# System Dependencies ("imports") -# -# ./š¦ -# | deno.json -# |(write) ā deno.imports.json -# |(write) ā package.json -# -# This is the "single-source-of-truth" with regards to dependencies and versioning. -# Import maps (in the `deno.json` and `package.json` files) are auto-generated -# from this config definition. -# -# Also, as a programmatic API, other downstream dependencies -# (such as template generators, see `@sys/tmpl`) use this definition -# file to calculate the "latest" versions to inject into, say, -# a `package.json` file for a scaffolded project. -# - -groups: - std/deno: - # Deno standard libs ("std"). - - import: jsr:@std/async@1.0.10 - - import: jsr:@std/datetime@0.225.3 - - import: jsr:@std/dotenv@0.225.3 - - import: jsr:@std/encoding@1.0.7 - - import: jsr:@std/fs@1.0.13 - - import: jsr:@std/path@1.0.8 - - import: jsr:@std/semver@1.0.4 - - import: jsr:@std/testing@1.0.9 - - import: jsr:@std/uuid@1.0.4 - - automerge: - # https://automerge.org - - import: npm:@automerge/automerge@2.2.8 - - import: npm:@automerge/automerge-repo@1.2.1 - - import: npm:@automerge/automerge-repo-network-broadcastchannel@1.2.1 - - import: npm:@automerge/automerge-repo-storage-indexeddb@1.2.1 - - import: npm:@automerge/automerge-repo-storage-nodefs@1.2.1 - - import: npm:@onsetsoftware/automerge-patcher@0.14.0 - - crypto: - - import: npm:@noble/hashes@1.7.1 - wildcard: true - - build/tools: - - import: npm:@vitejs/plugin-react-swc@3.8.0 - - import: npm:rollup@4.34.8 - - import: npm:vite@6.1.1 - - import: npm:vite-plugin-wasm@3.4.1 - - ui/react: - - import: npm:@types/react@18.3.18 - - import: npm:@types/react-dom@18.3.5 - - import: npm:react@18.3.1 - - import: npm:react-dom@18.3.1 - -deno.json: - - group: std/deno - - group: crypto - - group: automerge - - # CLI tools - - import: jsr:@cliffy/keypress@1.0.0-rc.7 - - import: jsr:@cliffy/prompt@1.0.0-rc.7 - - import: jsr:@cliffy/table@1.0.0-rc.7 - - # Sundry: NPM - - import: npm:@types/diff@7.0.1 - - import: npm:chai@5 - - import: npm:approx-string-match@2 - - import: npm:date-fns@4 - - import: npm:subhosting@0.1.0-alpha.1 - - import: npm:diff@7 - - import: npm:fast-json-patch@3.1.1 - - import: npm:fake-indexeddb@6.0.0 - - import: npm:happy-dom@17.1.3 - - import: npm:hash-it@6.0.0 - - import: npm:ignore@7 - - import: npm:immer@10 - - import: npm:ora@8.2.0 - - import: npm:ollama@0.5.13 - - import: npm:pretty-bytes@6.1.1 - - import: npm:ramda@0.30.1 - - import: npm:rambda@9.4.2 - - import: npm:rxjs@7.8.2 - - import: npm:strip-ansi@7 - - import: npm:subhosting@0.1.0-alpha.1 - - import: npm:tinycolor2@1.6.0 - - import: npm:ts-essentials@10.0.4 - - import: npm:valibot@1.0.0-rc.1 - - import: npm:yaml@2.7.0 - - # Browser - - import: npm:csstype@3 - - import: npm:ua-parser-js@2.0.2 - - # UI - - import: npm:react-error-boundary@5 - - import: npm:react-inspector@6 - - import: npm:react-spinners@0.15.0 - -package.json: - - group: std/deno - - group: crypto - - group: build/tools - dev: true - - - import: npm:hono@4.7.2 - - # UI - - group: ui/react - - import: npm:react-icons@5.5.0 - - import: npm:@vidstack/react@1.12.12 - - # UI:Frameworks - - import: npm:vitepress@1.6.3 - - import: npm:vue@3.5.13 diff --git a/code/sys.driver/driver-vite/src/-tmpl/deno.json b/code/sys.driver/driver-vite/src/-tmpl/deno.json index f5408b5de7..77f03332f5 100644 --- a/code/sys.driver/driver-vite/src/-tmpl/deno.json +++ b/code/sys.driver/driver-vite/src/-tmpl/deno.json @@ -1,30 +1,27 @@ { "version": "0.0.0", - "license": "MIT", "tasks": { "dev": "deno run -RWNE --allow-run --allow-ffi <ENTRY> --cmd=dev --in=./src/-test/index.html", "build": "deno run -RWE --allow-run --allow-ffi <ENTRY> --cmd=build --in=./src/-test/index.html", "serve": "deno run -RNE --allow-run --allow-ffi <ENTRY> --cmd=serve", + "test": "deno test -RWNE --allow-run --allow-ffi --allow-sys", "clean": "deno run -RWE --allow-ffi <ENTRY> --cmd=clean", "upgrade": "deno run -RWNE --allow-run --allow-ffi <ENTRY> --cmd=upgrade", "backup": "deno run -RWE --allow-run --allow-ffi <ENTRY> --cmd=backup", "help": "deno run -RE --allow-ffi <ENTRY> --cmd=help", - "sys": "deno run -RWNE <ENTRY_SYS>", "tmp": "deno run -A ./-scripts/-tmp.ts" }, "compilerOptions": { "strict": true, "lib": ["deno.ns", "esnext", "dom", "dom.iterable", "dom.asynciterable"], "types": ["vite/client", "@types/react"], - "jsx": "react", - "jsxFactory": "React.createElement", - "jsxFragmentFactory": "React.Fragment" + "jsx": "react-jsx", + "jsxImportSource": "react" }, "workspace": [], + "importMap": "./imports.json", "nodeModulesDir": "auto", - "imports": { - "<SELF_IMPORT_NAME>": "<SELF_IMPORT_URI>" - } + "license": "MIT" } diff --git a/code/sys.driver/driver-vite/src/-tmpl/imports.json b/code/sys.driver/driver-vite/src/-tmpl/imports.json new file mode 100644 index 0000000000..68d5e7cd1f --- /dev/null +++ b/code/sys.driver/driver-vite/src/-tmpl/imports.json @@ -0,0 +1,17 @@ +{ + "imports": { + "@sys/std": "jsr:@sys/std@0.0.141", + "@sys/tmp": "jsr:@sys/tmp@0.0.110", + "@sys/types": "jsr:@sys/types@0.0.92", + "@sys/ui-css": "jsr:@sys/ui-css@0.0.81", + "@sys/driver-vite": "jsr:@sys/driver-vite@0.0.137", + "react": "npm:react@18.3.1", + "react-dom": "npm:react-dom@18.3.1", + "@types/react": "npm:@types/react@18.3.18", + "@types/react-dom": "npm:@types/react-dom@18.3.5", + "@vitejs/plugin-react-swc": "npm:@vitejs/plugin-react-swc@3.8.0", + "rollup": "npm:rollup@4.34.8", + "vite": "npm:vite@6.1.1", + "vite-plugin-wasm": "npm:vite-plugin-wasm@3.4.1" + } +} diff --git a/code/sys.driver/driver-vite/src/-tmpl/package.json b/code/sys.driver/driver-vite/src/-tmpl/package.json deleted file mode 100644 index 1eb5b9a268..0000000000 --- a/code/sys.driver/driver-vite/src/-tmpl/package.json +++ /dev/null @@ -1,19 +0,0 @@ -{ - "dependencies": { - "@sys/std": "npm:@jsr/sys__std", - "@sys/tmp": "npm:@jsr/sys__tmp", - "@sys/types": "npm:@jsr/sys__types", - "@sys/ui-css": "npm:@jsr/sys__ui-css", - "react": "", - "react-dom": "", - "valibot": "" - }, - "devDependencies": { - "@types/react": "", - "@types/react-dom": "", - "@vitejs/plugin-react-swc": "", - "rollup": "", - "vite": "", - "vite-plugin-wasm": "" - } -} diff --git a/code/sys.driver/driver-vite/src/-tmpl/src/-test/-sample/ui.Foo.tsx b/code/sys.driver/driver-vite/src/-tmpl/src/-test/-sample/ui.Foo.tsx index cc472ab2a9..056ddf70d8 100644 --- a/code/sys.driver/driver-vite/src/-tmpl/src/-test/-sample/ui.Foo.tsx +++ b/code/sys.driver/driver-vite/src/-tmpl/src/-test/-sample/ui.Foo.tsx @@ -1,6 +1,6 @@ import { useState } from 'react'; -import { Foo } from '@sys/tmp/ui'; +// import { Foo } from '@sys/tmp/ui'; import { Color, css } from '@sys/ui-css'; import type { t } from '../../common.ts'; @@ -70,7 +70,7 @@ export const FooSample: React.FC<FooComponent> = (props) => { <div>{`(see console for import samples)`}</div> <div style={{ paddingTop: 10 }}> {'Imported from ā '} - <Foo /> + {/* <Foo /> */} <div>{'š·š· TMP š·š· import WIP'}</div> <code>{'<JSX> ā Vite ā ESM.js.d.ts ā mod.ts ā JSR ā import'}</code> </div> diff --git a/code/sys.driver/driver-vite/src/-tmpl/src/-test/entry.lib.ts b/code/sys.driver/driver-vite/src/-tmpl/src/-test/entry.lib.ts deleted file mode 100644 index 4e406e6d67..0000000000 --- a/code/sys.driver/driver-vite/src/-tmpl/src/-test/entry.lib.ts +++ /dev/null @@ -1,8 +0,0 @@ -export const fn = async () => { - const { Foo } = await import('./-sample/m.Foo.ts'); - const { FooSample } = await import('./-sample/ui.Foo.tsx'); - return { - Foo, - FooSample, - }; -}; diff --git a/code/sys.driver/driver-vite/src/-tmpl/src/-test/entry.tsx b/code/sys.driver/driver-vite/src/-tmpl/src/-test/entry.tsx index f5e0c12336..92d0030698 100644 --- a/code/sys.driver/driver-vite/src/-tmpl/src/-test/entry.tsx +++ b/code/sys.driver/driver-vite/src/-tmpl/src/-test/entry.tsx @@ -1,4 +1,4 @@ -import { pkg } from '../common.ts'; +import { pkg } from '../pkg.ts'; /** * Render UI. @@ -14,7 +14,7 @@ import { FooSample } from './-sample/ui.Foo.tsx'; * š· Test " @sys " module imports from across the * namespace (monorepo/workspace). */ -import '@sys/tmp/sample-imports'; +// import '@sys/tmp/sample-imports'; /** * Sample: render react component. diff --git a/code/sys.driver/driver-vite/src/-tmpl/vite.config.ts b/code/sys.driver/driver-vite/src/-tmpl/vite.config.ts index b9a82aaf68..dd19226f00 100644 --- a/code/sys.driver/driver-vite/src/-tmpl/vite.config.ts +++ b/code/sys.driver/driver-vite/src/-tmpl/vite.config.ts @@ -1,15 +1,16 @@ -import { Vite } from '@sys/driver-vite'; -import { defineConfig } from 'vite'; - -export const paths = Vite.Config.paths({ app: { entry: './src/-test/index.html' } }); +import { Vite } from 'jsr:@sys/driver-vite'; +import { defineConfig } from 'npm:vite'; export default defineConfig(() => { + const entry = './src/-test/index.html'; + const paths = Vite.Config.paths({ app: { entry } }); return Vite.Config.app({ paths, chunks(e) { e.chunk('react', 'react'); e.chunk('react.dom', 'react-dom'); e.chunk('sys', ['@sys/std']); + e.chunk('css', ['@sys/ui-css']); }, }); }); diff --git a/code/sys.driver/driver-vite/src/common/libs.ts b/code/sys.driver/driver-vite/src/common/libs.ts index 055644b352..2a79093661 100644 --- a/code/sys.driver/driver-vite/src/common/libs.ts +++ b/code/sys.driver/driver-vite/src/common/libs.ts @@ -1,9 +1,9 @@ /** * System */ -export { Args, asArray, Delete, Err, isRecord, R, slug, Str, Time } from '@sys/std'; +export { Args, asArray, Delete, Err, isRecord, R, slug, Str, Time, V } from '@sys/std'; export { Esm } from '@sys/std/esm'; -export { Semver } from '@sys/std/semver'; +export { Semver } from '@sys/std/semver/server'; export { c, Cli, stripAnsi } from '@sys/cli'; export { HashFmt } from '@sys/crypto/fmt'; diff --git a/code/sys.driver/driver-vite/src/common/mod.ts b/code/sys.driver/driver-vite/src/common/mod.ts index f02c42540c..8556c7dd22 100644 --- a/code/sys.driver/driver-vite/src/common/mod.ts +++ b/code/sys.driver/driver-vite/src/common/mod.ts @@ -2,8 +2,10 @@ export type * as t from './t.ts'; export { pkg } from '../pkg.ts'; export * from './libs.ts'; +export * from './u.workspace.ts'; export const PATHS = { + base: './', dist: 'dist/', backup: '-backup/', tmp: '.tmp/', diff --git a/code/sys.driver/driver-vite/src/common/t.ts b/code/sys.driver/driver-vite/src/common/t.ts index 9d4c2d6f96..afa6294df3 100644 --- a/code/sys.driver/driver-vite/src/common/t.ts +++ b/code/sys.driver/driver-vite/src/common/t.ts @@ -20,6 +20,7 @@ export type * from '@sys/std/t'; export type { DenoFileJson, DenoFilePath, + DenoImportMapJson, DenoModuleBackup, DenoWorkspace, DenoWorkspaceChild, @@ -27,7 +28,7 @@ export type { } from '@sys/driver-deno/t'; export type { FsPathFilter } from '@sys/fs/t'; export type { ProcHandle, ProcOutput, ProcReadySignalFilter } from '@sys/process/t'; -export type { Tmpl, TmplCopyHandler, TmplFileOperation, TmplProcessFile } from '@sys/tmpl/t'; +export type { Tmpl, TmplWriteHandler, TmplFileOperation, TmplProcessFile } from '@sys/tmpl/t'; export type { CssValue } from '@sys/ui-css/t'; /** diff --git a/code/sys.driver/driver-vite/src/common/u.workspace.ts b/code/sys.driver/driver-vite/src/common/u.workspace.ts new file mode 100644 index 0000000000..146c44d576 --- /dev/null +++ b/code/sys.driver/driver-vite/src/common/u.workspace.ts @@ -0,0 +1,18 @@ +import { DenoDeps, DenoFile, Path, Esm } from './libs.ts'; + +/** + * Find and load the latest workspace data. + */ +export async function getWorkspaceModules() { + const ws = await DenoFile.workspace(); + const deps = (await DenoDeps.from(Path.join(ws.dir, 'deps.yaml'))).data; + const modules = Esm.modules([...(deps?.modules.items ?? []), ...ws.modules.items]); + return { + get ws() { + return ws; + }, + get modules() { + return modules; + }, + } as const; +} diff --git a/code/sys.driver/driver-vite/src/m.Log/u.Bundle.ts b/code/sys.driver/driver-vite/src/m.Log/u.Bundle.ts index ccb2b976b7..167e63b6cd 100644 --- a/code/sys.driver/driver-vite/src/m.Log/u.Bundle.ts +++ b/code/sys.driver/driver-vite/src/m.Log/u.Bundle.ts @@ -1,5 +1,4 @@ -import { type t, c, Path, Str, Time } from './common.ts'; - +import { type t, c, Path, Semver, Str, Time } from './common.ts'; import { digest, pad } from './u.ts'; export const Bundle: t.ViteLogLib['Bundle'] = { @@ -24,8 +23,9 @@ ${c.gray(`out: ${outDir.replace(/\/$/, '')}/dist.json`)} ${tx} `; text = text.trim(); if (pkg) { - const mod = c.white(c.bold(pkg.name)); - text += c.gray(`\npkg: ${mod} ${c.cyan(c.bold(pkg.version))}`); + const fmtVersion = Semver.Fmt.colorize(pkg.version); + const fmtModule = `${c.white(c.bold(pkg.name))}${c.dim('@')}${fmtVersion}`; + text += c.gray(`\npkg: ${fmtModule}`); } return pad(text, args.pad); }, diff --git a/code/sys.driver/driver-vite/src/m.Vite.Config.Workspace/-.test.ts b/code/sys.driver/driver-vite/src/m.Vite.Config.Workspace/-.test.ts index 335e180f31..54572f3504 100644 --- a/code/sys.driver/driver-vite/src/m.Vite.Config.Workspace/-.test.ts +++ b/code/sys.driver/driver-vite/src/m.Vite.Config.Workspace/-.test.ts @@ -8,7 +8,7 @@ describe('ViteConfig.workspace', () => { }); it('loads (via path)', async () => { - const map = (children: t.DenoWorkspaceChild[]) => children.map((m) => Fs.dirname(m.path)); + const map = (children: t.DenoWorkspaceChild[]) => children.map((m) => m.path.dir); const a = await workspace(); // NB: finds root workspace const b = await workspace({ denofile: ROOT.denofile.path }); diff --git a/code/sys.driver/driver-vite/src/m.Vite.Config.Workspace/mod.ts b/code/sys.driver/driver-vite/src/m.Vite.Config.Workspace/mod.ts index 34274f914d..415d57ba98 100644 --- a/code/sys.driver/driver-vite/src/m.Vite.Config.Workspace/mod.ts +++ b/code/sys.driver/driver-vite/src/m.Vite.Config.Workspace/mod.ts @@ -49,7 +49,7 @@ const wrangle = { }, async modules(base: t.StringDir, children: t.DenoWorkspaceChild[], filter?: t.WorkspaceFilter) { - const wait = children.map((child) => wrangle.exports(base, Fs.dirname(child.path), filter)); + const wait = children.map((child) => wrangle.exports(base, child.path.dir, filter)); const res = await Array.fromAsync(wait); return res .filter((item) => item.exists) diff --git a/code/sys.driver/driver-vite/src/m.Vite.Config/-.test.ts b/code/sys.driver/driver-vite/src/m.Vite.Config/-.test.ts index 0afbfb4f49..e1098861ba 100644 --- a/code/sys.driver/driver-vite/src/m.Vite.Config/-.test.ts +++ b/code/sys.driver/driver-vite/src/m.Vite.Config/-.test.ts @@ -1,4 +1,4 @@ -import { type t, c, describe, expect, it, Path, SAMPLE } from '../-test.ts'; +import { type t, c, describe, expect, it } from '../-test.ts'; import { Vite } from '../mod.ts'; import { Is } from './m.Is.ts'; import { ViteConfig } from './mod.ts'; diff --git a/code/sys.driver/driver-vite/src/m.Vite.Config/-app.test.ts b/code/sys.driver/driver-vite/src/m.Vite.Config/-app.test.ts index ceb8666d9a..ec7e5d5373 100644 --- a/code/sys.driver/driver-vite/src/m.Vite.Config/-app.test.ts +++ b/code/sys.driver/driver-vite/src/m.Vite.Config/-app.test.ts @@ -2,7 +2,7 @@ import { type t, Fs, c, describe, expect, it } from '../-test.ts'; import { ViteConfig } from './mod.ts'; describe('Config.Build', () => { - const { brightCyan: cyan, bold } = c; + const { brightCyan: cyan } = c; describe('app (application)', () => { const includesPlugin = (config: t.ViteUserConfig, name: string) => { @@ -10,44 +10,57 @@ describe('Config.Build', () => { return plugins.some((p) => p.name === name); }; - it('default', async () => { - const p = ViteConfig.paths(); - const config = await ViteConfig.app(); - - expect(config.root).to.eql(p.cwd); - expect(config.build?.rollupOptions?.input).to.eql(Fs.join(p.cwd, p.app.entry)); - expect(config.build?.outDir).to.eql(Fs.join(p.cwd, p.app.outDir)); + const print = (config: t.ViteUserConfig, titleSuffix?: string, paths?: t.ViteConfigPaths) => { + if (paths) { + console.info(); + console.info(cyan(c.bold('ā INPUT paths'))); + console.info(paths); + console.info(); + } console.info(); - console.info(bold(cyan('ViteConfig.app (default):'))); + console.info(cyan(c.bold('ā ViteConfig.app')), c.gray(titleSuffix ?? '')); console.info({ ...config, plugins: ((config.plugins ?? []) as t.VitePlugin[]).flat().map((m) => m.name), resolve: { ...config.resolve, - alias: `ā š³ ${config.resolve?.alias?.length} total aliases`, + alias: `ā š³ ${config.resolve?.alias?.length} aliases (across workspace)`, }, }); console.info(); + return config; + }; + + it('defaults', async () => { + const p = ViteConfig.paths(); + const config = await ViteConfig.app(); + print(config, '(defaults)', p); + + expect(config.root).to.eql(p.cwd); + expect(config.build?.rollupOptions?.input).to.eql(Fs.join(p.cwd, p.app.entry)); + expect(config.build?.outDir).to.eql(Fs.join(p.cwd, p.app.outDir)); expect(includesPlugin(config, 'vite-plugin-wasm')).to.be.true; expect(includesPlugin(config, 'vite:react-swc')).to.be.true; }); it('no plugins', async () => { - const config = await ViteConfig.app({ plugins: { wasm: false, react: false } }); + const config = await ViteConfig.app({ plugins: { wasm: false, react: false, deno: false } }); expect(config.plugins).to.eql([]); }); it('custom paths', async () => { const paths = ViteConfig.paths({ - cwd: ' /foo/ ', - app: { entry: 'src/-foo.html', outDir: 'bar' }, + cwd: ' /foo/ ', // NB: absolute path (trimmed internally). + app: { entry: 'src/-foo.html', outDir: 'foobar/out' }, }); const config = await ViteConfig.app({ paths }); + print(config, '(custom paths)', paths); + expect(config.root).to.eql('/foo/src'); - expect(config.build?.rollupOptions?.input).to.eql(Fs.join(paths.cwd, paths.app.entry)); - expect(config.build?.outDir).to.eql(Fs.join(paths.cwd, paths.app.outDir)); + expect(config.build?.rollupOptions?.input).to.eql(Fs.join(paths.cwd, 'src/-foo.html')); + expect(config.build?.outDir).to.eql(Fs.join(paths.cwd, 'foobar/out')); }); }); }); diff --git a/code/sys.driver/driver-vite/src/m.Vite.Config/-fromFile.test.ts b/code/sys.driver/driver-vite/src/m.Vite.Config/-fromFile.test.ts new file mode 100644 index 0000000000..b734393319 --- /dev/null +++ b/code/sys.driver/driver-vite/src/m.Vite.Config/-fromFile.test.ts @@ -0,0 +1,55 @@ +import { type t, c, describe, expect, it, Path, SAMPLE } from '../-test.ts'; +import { Vite } from '../mod.ts'; +import { ViteConfig } from './mod.ts'; + +describe('ViteConfig.fromFile', () => { + const { brightCyan: cyan } = c; + + const print = (res: t.ViteConfigFromFile) => { + console.info(); + console.info(cyan(`ā Type: ${c.bold('ViteConfigFromFile')}`)); + console.info(res); + console.info(); + }; + + it('from "/<root-dir>/"', async () => { + const rootDir = SAMPLE.Dirs.sample2; + const res = await ViteConfig.fromFile(rootDir); + print(res); + + expect(res.exists).to.eql(true); + expect(res.error).to.eql(undefined); + + expect(res.paths?.cwd).to.eql(Path.resolve(rootDir)); + expect(res.paths?.app.entry).to.eql('src/-entry/index.html'); + expect(res.paths?.app.base).to.eql('./'); + expect(res.paths?.app.outDir).to.eql('dist/'); + }); + + it('from "/<root-dir>/vite.config.ts" (with config filename)', async () => { + const rootDir = SAMPLE.Dirs.sample2; + const resA = await ViteConfig.fromFile(Path.join(rootDir, 'vite.config.ts')); + const resB = await ViteConfig.fromFile(rootDir); + expect(resA.exists).to.eql(true); + expect(resA.error).to.eql(undefined); + expect(resA).to.eql(resB); + }); + + it('loads main samples', async () => { + const test = async (path: t.StringPath) => { + const res = await Vite.Config.fromFile(path); + expect(res.error).to.eql(undefined); + expect(ViteConfig.Is.paths(res.paths)).to.be.true; + }; + + await test('src/-test/vite.sample-config/simple/vite.config.ts'); + await test('src/-test/vite.sample-config/custom/vite.config.ts'); + }); + + it('fail: not found', async () => { + const res = await ViteConfig.fromFile('/foo/404/vite.config.ts'); + expect(res.exists).to.eql(false); + expect(res.error?.message).to.include('A config file could not be found in directory'); + expect(res.error?.message).to.include(': /foo/404'); + }); +}); diff --git a/code/sys.driver/driver-vite/src/m.Vite.Config/-paths.test.ts b/code/sys.driver/driver-vite/src/m.Vite.Config/-paths.test.ts index 81457c9bbd..601bc81cbd 100644 --- a/code/sys.driver/driver-vite/src/m.Vite.Config/-paths.test.ts +++ b/code/sys.driver/driver-vite/src/m.Vite.Config/-paths.test.ts @@ -1,11 +1,10 @@ -import { type t, c, describe, expect, it, Path, SAMPLE } from '../-test.ts'; -import { Vite } from '../mod.ts'; +import { c, describe, expect, it, Path } from '../-test.ts'; import { ViteConfig } from './mod.ts'; -describe('ViteConfig: paths', () => { +describe('ViteConfig.paths', () => { const { brightCyan: cyan, bold } = c; - describe('ViteConfig.paths', () => { + describe('[ViteConfigPaths]: data structure', () => { it('default paths (empty params)', () => { const a = ViteConfig.paths(); const b = ViteConfig.paths(); @@ -65,52 +64,4 @@ describe('ViteConfig: paths', () => { expect(d.cwd).to.eql(b.cwd); }); }); - - describe('ViteConfig.fromFile', () => { - it('load from file path', async () => { - const rootDir = SAMPLE.Dirs.sample2; - const path = Path.join(rootDir, 'vite.config.ts'); - const res = await ViteConfig.fromFile(path); - - expect(res.error).to.eql(undefined); - expect(res.path).to.eql(Path.resolve(path)); - expect(res.exists).to.eql(true); - - const module = res.module; - expect(module.paths?.app.entry).to.eql('src/-entry/index.html'); - expect(res.module.paths?.cwd).to.eql(Path.resolve(rootDir)); - expect(typeof module.defineConfig === 'function').to.be.true; - }); - - it('no params: load from implicit {CWD}', async () => { - const res = await ViteConfig.fromFile(); - expect(res.path).to.eql(Path.resolve('vite.config.ts')); - expect(res.module.paths?.cwd).to.eql(Path.cwd()); - }); - - it('no `paths` | no `defineConfig`', async () => { - const path = Path.fromFileUrl(import.meta.url); // NB: not a `vite.config.ts` module. - const res = await ViteConfig.fromFile(path); - expect(res.error).to.eql(undefined); - expect(res.module).to.eql({}); - }); - - it('loads main samples', async () => { - const test = async (path: t.StringPath) => { - const res = await Vite.Config.fromFile(path); - expect(res.error).to.eql(undefined); - expect(ViteConfig.Is.paths(res.module.paths)).to.be.true; - }; - - await test('src/-test/vite.sample-config/simple/vite.config.ts'); - await test('src/-test/vite.sample-config/custom/vite.config.ts'); - }); - - it('fail: not found', async () => { - const res = await ViteConfig.fromFile('/foo/404/vite.config.ts'); - expect(res.error?.message).to.include('Module not found at path'); - expect(res.error?.cause?.name).to.eql('TypeError'); - expect(res.module).to.eql({}); - }); - }); }); diff --git a/code/sys.driver/driver-vite/src/m.Vite.Config/-plugins.test.ts b/code/sys.driver/driver-vite/src/m.Vite.Config/-plugins.test.ts index 7e63541fe6..3ff172db59 100644 --- a/code/sys.driver/driver-vite/src/m.Vite.Config/-plugins.test.ts +++ b/code/sys.driver/driver-vite/src/m.Vite.Config/-plugins.test.ts @@ -7,10 +7,12 @@ describe('ViteConfig: common plugins', () => { const includes = (name: string) => res.some((p) => p.name === name); expect(includes('vite-plugin-wasm')).to.be.true; expect(includes('vite:react-swc')).to.be.true; + expect(includes('deno')).to.be.true; + expect(includes('deno:prefix')).to.be.true; }); it('none (via options)', async () => { - const res = await commonPlugins({ wasm: false, react: false }); + const res = await commonPlugins({ wasm: false, react: false, deno: false }); expect(res.length).to.eql(0); }); }); diff --git a/code/sys.driver/driver-vite/src/m.Vite.Config/t.paths.ts b/code/sys.driver/driver-vite/src/m.Vite.Config/t.paths.ts index f64d748e66..9c948f93f3 100644 --- a/code/sys.driver/driver-vite/src/m.Vite.Config/t.paths.ts +++ b/code/sys.driver/driver-vite/src/m.Vite.Config/t.paths.ts @@ -1,9 +1,11 @@ import type { t } from './common.ts'; +/** + * Representation of paths for a Vite configuration. + */ export type ViteConfigPaths = { readonly cwd: t.StringDir; readonly app: t.ViteConfigPathsApp; - readonly lib: t.ViteConfigPathsLib; }; /** @@ -27,9 +29,3 @@ export type ViteConfigPathsApp = { */ readonly base: t.StringDir; }; - -/** - * Paths for "library mode" bundles. - * https://vite.dev/guide/build.html#library-mode - */ -export type ViteConfigPathsLib = {}; diff --git a/code/sys.driver/driver-vite/src/m.Vite.Config/t.ts b/code/sys.driver/driver-vite/src/m.Vite.Config/t.ts index 1277584547..587e91fba9 100644 --- a/code/sys.driver/driver-vite/src/m.Vite.Config/t.ts +++ b/code/sys.driver/driver-vite/src/m.Vite.Config/t.ts @@ -1,4 +1,3 @@ -import type { defineConfig } from 'vite'; import type { t } from './common.ts'; export type * from './t.app.ts'; @@ -38,7 +37,7 @@ export type ViteConfigLib = { /** * Attempts to dynamically load a `vite.config.ts` module. */ - fromFile(path?: t.StringPath): Promise<ViteConfigFromFile>; + fromFile(configDir?: t.StringDir): Promise<ViteConfigFromFile>; }; /** @@ -56,8 +55,12 @@ export type ViteBundleIO = { in: t.StringDir; out: t.StringDir }; * Common plugins (default: true). */ export type ViteConfigCommonPlugins = { + /** Flag indicating if the official `deno-vite` plugin should be included. */ + deno?: boolean; + /** Flag indicating if the "react+swc" plugin should be included. */ react?: boolean; + /** Flag indicating if the "wasm" plugin should be included. */ wasm?: boolean; }; @@ -74,10 +77,16 @@ export type ViteModuleChunksArgs = { /** * The result from the `Vite.Config.fromFile` method. + * See also: + * https://vite.dev/guide/api-javascript.html#loadconfigfromfile */ export type ViteConfigFromFile = { + /** Flag indicating if the config file exists on the filesystem. */ exists: boolean; - path: t.StringAbsolutePath; - module: { defineConfig?: typeof defineConfig; paths?: t.ViteConfigPaths }; + + /** The paths of the Vite configuration. */ + paths?: t.ViteConfigPaths; + + /** Any error details while loading. */ error?: t.StdError; }; diff --git a/code/sys.driver/driver-vite/src/m.Vite.Config/u.app.ts b/code/sys.driver/driver-vite/src/m.Vite.Config/u.app.ts index 8cb82e26a8..01effe5268 100644 --- a/code/sys.driver/driver-vite/src/m.Vite.Config/u.app.ts +++ b/code/sys.driver/driver-vite/src/m.Vite.Config/u.app.ts @@ -15,6 +15,7 @@ export const app: t.ViteConfigLib['app'] = async (options = {}) => { const input = Path.join(paths.cwd, paths.app.entry); const outDir = Path.join(paths.cwd, paths.app.outDir); + const publicDir = Path.join(paths.cwd, 'public'); const root = Path.dirname(input); /** @@ -55,6 +56,7 @@ export const app: t.ViteConfigLib['app'] = async (options = {}) => { const res: t.ViteUserConfig = { root, + publicDir, base: paths.app.base, server: { fs: { allow: ['..'] } }, // NB: allows stepping up out of the {cwd} and access other folders in the monorepo. worker: { format }, diff --git a/code/sys.driver/driver-vite/src/m.Vite.Config/u.fromFile.ts b/code/sys.driver/driver-vite/src/m.Vite.Config/u.fromFile.ts index 1cb24ed888..d508b03e88 100644 --- a/code/sys.driver/driver-vite/src/m.Vite.Config/u.fromFile.ts +++ b/code/sys.driver/driver-vite/src/m.Vite.Config/u.fromFile.ts @@ -1,56 +1,57 @@ -import { type t, Delete, Err, Path } from './common.ts'; -import { Is } from './m.Is.ts'; - -type R = t.ViteConfigFromFile; +import { loadConfigFromFile } from 'vite'; +import { type t, Delete, Err, Path, PATHS } from './common.ts'; /** * Attempts to dynamically load a `vite.config.ts` module. */ export const fromFile: t.ViteConfigLib['fromFile'] = async (input) => { const errors = Err.errors(); - const path = wrangle.path(input); - const res = await loadModule(path, errors); - return { - path, - exists: res.exists, - module: wrangle.module(res.mod), + const configRoot = wrangle.configDir(input); + + /** + * TODO š· change configRoot to ./.tmp/sample/<vite.config.ts> + */ + + const command = 'build'; + const mode = 'production'; + const fromFile = await loadConfigFromFile( + { command, mode }, // param: configEnv + undefined, // param: configFile + configRoot, // param: configRoot + undefined, // param: logLevel + undefined, // param: customLogger + 'native', // param: configLoader + ); + + const exists = fromFile !== null; + if (!exists) errors.push(`A config file could not be found in directory: ${configRoot}`); + + let paths: t.ViteConfigPaths | undefined; + if (exists) { + paths = { + cwd: Path.dirname(fromFile.path), + app: { + entry: Path.trimCwd(Path.join(fromFile.config.root ?? '', PATHS.html.index)), + outDir: Path.trimCwd(fromFile.config.build?.outDir ?? PATHS.dist), + base: fromFile.config.base ?? PATHS.base, + }, + }; + } + + return Delete.undefined<t.ViteConfigFromFile>({ + exists, + paths, error: errors.toError(), - }; + }); }; /** * Helpers */ -async function loadModule(path: string, errors: t.ErrorCollection) { - let exists = false; - path = `file://${path.replace(/^file\:\/\//, '')}`; - try { - const mod = await import(path); - exists = true; - return { mod, exists }; - } catch (cause: unknown) { - const unexpected = `Unexpected error while importing module from path: ${path}`; - if (!(cause instanceof Error)) { - errors.push(unexpected); - } else { - if (cause.message.includes('Module not found')) { - exists = false; - errors.push(`Module not found at path: ${path}`, cause); - } else { - errors.push(unexpected, cause); - } - } - return { exists }; - } -} - const wrangle = { - path(input?: string): t.StringAbsolutePath { - return typeof input === 'string' ? Path.resolve(input) : Path.resolve('vite.config.ts'); - }, - module(mod: any): R['module'] { - const defineConfig = typeof mod?.default === 'function' ? mod.default : undefined; - const paths = Is.paths(mod?.paths) ? mod.paths : undefined; - return Delete.undefined({ defineConfig, paths }); + configDir(input?: string) { + if (typeof input !== 'string') return Path.cwd(); + if (input.endsWith('vite.config.ts')) return Path.dirname(input); + return input; }, } as const; diff --git a/code/sys.driver/driver-vite/src/m.Vite.Config/u.paths.ts b/code/sys.driver/driver-vite/src/m.Vite.Config/u.paths.ts index d899b47f4b..2e2a0015a4 100644 --- a/code/sys.driver/driver-vite/src/m.Vite.Config/u.paths.ts +++ b/code/sys.driver/driver-vite/src/m.Vite.Config/u.paths.ts @@ -12,10 +12,9 @@ export const paths: F = (input) => { const app: t.DeepMutable<t.ViteConfigPathsApp> = { entry: PATHS.html.index, - base: './', + base: PATHS.base, outDir: PATHS.dist, }; - const lib: t.DeepMutable<t.ViteConfigPathsLib> = {}; if (valueExists(options.app?.entry)) app.entry = options.app?.entry; if (valueExists(options.app?.base)) app.base = options.app?.base; @@ -25,7 +24,7 @@ export const paths: F = (input) => { app.base = app.base.trim(); app.outDir = app.outDir.trim(); - return { cwd, app, lib }; + return { cwd, app }; }; /** diff --git a/code/sys.driver/driver-vite/src/m.Vite.Config/u.plugins.ts b/code/sys.driver/driver-vite/src/m.Vite.Config/u.plugins.ts index 4efbaec479..7586cd39aa 100644 --- a/code/sys.driver/driver-vite/src/m.Vite.Config/u.plugins.ts +++ b/code/sys.driver/driver-vite/src/m.Vite.Config/u.plugins.ts @@ -1,22 +1,35 @@ +import deno from '@deno/vite-plugin'; import react from '@vitejs/plugin-react-swc'; import wasm from 'vite-plugin-wasm'; -import { type t } from './common.ts'; + +import type { t } from './common.ts'; export async function commonPlugins(options: t.ViteConfigCommonPlugins = {}) { const plugins: t.VitePluginOption[] = []; - // WASM support. + /** + * The official Denoā¢ļø vite-plugin. + */ + if (options.deno ?? true) { + plugins.push(deno() as t.VitePlugin[]); + } + + /** + * WASM support. + */ if (options.wasm ?? true) { // deno-lint-ignore ban-ts-comment // @ts-ignore plugins.push(wasm()); } - // React (via the SWC compiler). - // - https://github.com/vitejs/vite-plugin-react-swc - // - https://swc.rs + /** + * React (via the SWC compiler). + * - https://github.com/vitejs/vite-plugin-react-swc + * - https://swc.rs + */ if (options.react ?? true) { - plugins.push(react() as any); + plugins.push(react() as t.VitePluginOption[]); } // Finish up. diff --git a/code/sys.driver/driver-vite/src/m.Vite.Tmpl/-.test.ts b/code/sys.driver/driver-vite/src/m.Vite.Tmpl/-.test.ts new file mode 100644 index 0000000000..512376e882 --- /dev/null +++ b/code/sys.driver/driver-vite/src/m.Vite.Tmpl/-.test.ts @@ -0,0 +1,95 @@ +import { type t, SAMPLE, describe, expect, it, pkg } from '../-test.ts'; +import { Vite } from '../mod.ts'; + +describe('Vite: Template Generation', () => { + const testFs = () => SAMPLE.fs('Vite.tmpl'); + + const pathAssertions = (paths: t.StringPath[]) => { + return { + paths, + exists(endsWith: t.StringPath) { + expect(paths.some((p) => p.endsWith(endsWith))).to.eql(true); + }, + } as const; + }; + + it('--tmpl: Default', async () => { + const fs = testFs(); + expect(await fs.ls()).to.eql([]); + + const tmpl = await Vite.Tmpl.create(); + const res = await tmpl.write(fs.dir); + expect(res.ctx).to.eql({ version: pkg.version, tmpl: 'Default' }); + + const a = (await fs.ls()).toSorted(); + const b = (await res.target.ls()).toSorted(); + expect(a).to.eql(b); + + const assert = pathAssertions(a); + [ + '-scripts/-tmp.ts', + '.gitignore', + '.npmrc', + '.vscode/settings.json', + 'README.md', + 'deno.json', + 'imports.json', + 'src/-test.ts', + 'src/-test/-sample/m.Foo.ts', + 'src/-test/-sample/t.ts', + 'src/-test/-sample/ui.Foo.tsx', + 'src/-test/entry.tsx', + 'src/-test/index.html', + 'src/-test/mod.ts', + 'src/.test.ts', + 'src/common.ts', + 'src/common/libs.ts', + 'src/common/mod.ts', + 'src/common/t.ts', + 'src/mod.ts', + 'src/pkg.ts', + 'src/types.ts', + 'vite.config.ts', + ].forEach((path) => assert.exists(path)); + }); + + it('--tmpl: ComponentLib', async () => { + const fs = testFs(); + expect(await fs.ls()).to.eql([]); + + const tmpl = await Vite.Tmpl.create({ tmpl: 'ComponentLib' }); + const res = await tmpl.write(fs.dir); + expect(res.ctx).to.eql({ version: pkg.version, tmpl: 'ComponentLib' }); + + const a = (await fs.ls()).toSorted(); + const b = (await res.target.ls()).toSorted(); + expect(a).to.eql(b); + + const assert = pathAssertions(a); + [ + '-scripts/-tmp.ts', + '.gitignore', + '.npmrc', + '.vscode/settings.json', + 'README.md', + 'deno.json', + 'imports.json', + 'src/-test.ts', + 'src/-test/-sample/m.Foo.ts', + 'src/-test/-sample/t.ts', + 'src/-test/-sample/ui.Foo.tsx', + 'src/-test/entry.tsx', + 'src/-test/index.html', + 'src/-test/mod.ts', + 'src/.test.ts', + 'src/common.ts', + 'src/common/libs.ts', + 'src/common/mod.ts', + 'src/common/t.ts', + 'src/mod.ts', + 'src/pkg.ts', + 'src/types.ts', + 'vite.config.ts', + ].forEach((path) => assert.exists(path)); + }); +}); diff --git a/code/sys.driver/driver-vite/src/m.Vite.Tmpl/-bundle.json b/code/sys.driver/driver-vite/src/m.Vite.Tmpl/-bundle.json index 714456a288..e41ec5fe8b 100644 --- a/code/sys.driver/driver-vite/src/m.Vite.Tmpl/-bundle.json +++ b/code/sys.driver/driver-vite/src/m.Vite.Tmpl/-bundle.json @@ -1,19 +1,16 @@ { - "-scripts/-tmp.ts": "data:application/typescript;base64,Y29uc29sZS5pbmZvKCfwn5GLJywgaW1wb3J0Lm1ldGEudXJsKTsK", + "-scripts/-tmp.ts": "data:application/typescript;base64,Y29uc29sZS5pbmZvKGBcbvCfkYsgJHtpbXBvcnQubWV0YS51cmx9XG5gKTsK", ".gitignore-": "data:text/plain;base64,bm9kZV9tb2R1bGVzCmRpc3QKLnRtcAouc3djCgouc3lzCi52c2NvZGUKCi1iYWNrdXAKLWJhY2t1cC8qKgo=", ".npmrc": "data:text/plain;base64,IyBEb2NzOiBodHRwczovL2pzci5pby9kb2NzL25wbS1jb21wYXRpYmlsaXR5I2luc3RhbGxpbmctYW5kLXVzaW5nLWpzci1wYWNrYWdlcwpAanNyOnJlZ2lzdHJ5PWh0dHBzOi8vbnBtLmpzci5pbwo=", - ".sys/deps.sys.yaml": "data:text/plain;base64,ZGVuby5qc29uOgogIC0gaW1wb3J0OiBqc3I6QHN5cy90eXBlc0AwLjAuODIKICAtIGltcG9ydDoganNyOkBzeXMvc3RkQDAuMC4xMzEKICAtIGltcG9ydDoganNyOkBzeXMvY29sb3JAMC4wLjM1CiAgLSBpbXBvcnQ6IGpzcjpAc3lzL3Rlc3RpbmdAMC4wLjc1CiAgLSBpbXBvcnQ6IGpzcjpAc3lzL2ZzQDAuMC43OQogIC0gaW1wb3J0OiBqc3I6QHN5cy9jbGlAMC4wLjY0CiAgLSBpbXBvcnQ6IGpzcjpAc3lzL3Byb2Nlc3NAMC4wLjY1CiAgLSBpbXBvcnQ6IGpzcjpAc3lzL2NyeXB0b0AwLjAuNjQKICAtIGltcG9ydDoganNyOkBzeXMvaHR0cEAwLjAuNDYKICAtIGltcG9ydDoganNyOkBzeXMvdGV4dEAwLjAuNzMKICAtIGltcG9ydDoganNyOkBzeXMvdG1wbEAwLjAuNzkKICAtIGltcG9ydDoganNyOkBzeXMvY21kQDAuMC44MAogIC0gaW1wb3J0OiBqc3I6QHN5cy9qc3JAMC4wLjQ1CiAgLSBpbXBvcnQ6IGpzcjpAc3lzL3VpLWNzc0AwLjAuNzAKICAtIGltcG9ydDoganNyOkBzeXMvdWktZG9tQDAuMC43NwogIC0gaW1wb3J0OiBqc3I6QHN5cy91aS1yZWFjdEAwLjAuODQKICAtIGltcG9ydDoganNyOkBzeXMvdWktcmVhY3QtZGV2aGFybmVzc0AwLjAuODIKICAtIGltcG9ydDoganNyOkBzeXMvdWktcmVhY3QtY29tcG9uZW50c0AwLjAuMzgKICAtIGltcG9ydDoganNyOkBzeXMvZHJpdmVyLWF1dG9tZXJnZUAwLjAuODQKICAtIGltcG9ydDoganNyOkBzeXMvZHJpdmVyLWRlbm9AMC4wLjg2CiAgLSBpbXBvcnQ6IGpzcjpAc3lzL2RyaXZlci1pbW1lckAwLjAuODYKICAtIGltcG9ydDoganNyOkBzeXMvZHJpdmVyLW9ic2lkaWFuQDAuMC43MwogIC0gaW1wb3J0OiBqc3I6QHN5cy9kcml2ZXItb2xsYW1hQDAuMC40MAogIC0gaW1wb3J0OiBqc3I6QHN5cy9kcml2ZXItb3JiaXRlckAwLjAuNjMKICAtIGltcG9ydDoganNyOkBzeXMvZHJpdmVyLXF1aWxpYnJpdW1AMC4wLjc2CiAgLSBpbXBvcnQ6IGpzcjpAc3lzL2RyaXZlci12aXRlQDAuMC4xMjEKICAtIGltcG9ydDoganNyOkBzeXMvZHJpdmVyLXZpdGVwcmVzc0AwLjAuMjg0LWFscGhhLjIKICAtIGltcG9ydDoganNyOkBzeXMvc3lzQDAuMC41NQogIC0gaW1wb3J0OiBqc3I6QHN5cy9tYWluQDAuMC41NwogIC0gaW1wb3J0OiBqc3I6QHRkYi9hcGlAMC4wLjcyCiAgLSBpbXBvcnQ6IGpzcjpAdGRiL3NsY0AwLjAuNjgKICAtIGltcG9ydDoganNyOkB0ZGIvdG1wQDAuMC43MgogIC0gaW1wb3J0OiBqc3I6QHN5cy9uYW1lQDAuMC4wCiAgLSBpbXBvcnQ6IGpzcjpAc3lzL3RtcEAwLjAuOTUKcGFja2FnZS5qc29uOiBbXQo=", - ".sys/deps.yaml": "data:text/plain;base64,IwojICBTeXN0ZW0gRGVwZW5kZW5jaWVzICgiaW1wb3J0cyIpCiMKIyAgICAgICAgLi/wn5KmCiMgICAgICAgICAgfCAgICAgICAgICAgIGRlbm8uanNvbgojICAgICAgICAgIHwod3JpdGUpICDihpIgIGRlbm8uaW1wb3J0cy5qc29uCiMgICAgICAgICAgfCh3cml0ZSkgIOKGkiAgcGFja2FnZS5qc29uCiMKIyAgVGhpcyBpcyB0aGUgInNpbmdsZS1zb3VyY2Utb2YtdHJ1dGgiIHdpdGggcmVnYXJkcyB0byBkZXBlbmRlbmNpZXMgYW5kIHZlcnNpb25pbmcuCiMgIEltcG9ydCBtYXBzIChpbiB0aGUgYGRlbm8uanNvbmAgYW5kIGBwYWNrYWdlLmpzb25gIGZpbGVzKSBhcmUgYXV0by1nZW5lcmF0ZWQKIyAgZnJvbSB0aGlzIGNvbmZpZyBkZWZpbml0aW9uLgojCiMgIEFsc28sIGFzIGEgcHJvZ3JhbW1hdGljIEFQSSwgb3RoZXIgZG93bnN0cmVhbSBkZXBlbmRlbmNpZXMKIyAgKHN1Y2ggYXMgdGVtcGxhdGUgZ2VuZXJhdG9ycywgc2VlIGBAc3lzL3RtcGxgKSB1c2UgdGhpcyBkZWZpbml0aW9uCiMgIGZpbGUgdG8gY2FsY3VsYXRlIHRoZSAibGF0ZXN0IiB2ZXJzaW9ucyB0byBpbmplY3QgaW50bywgc2F5LAojICBhIGBwYWNrYWdlLmpzb25gIGZpbGUgZm9yIGEgc2NhZmZvbGRlZCBwcm9qZWN0LgojCgpncm91cHM6CiAgc3RkL2Rlbm86CiAgICAjIERlbm8gc3RhbmRhcmQgbGlicyAoInN0ZCIpLgogICAgLSBpbXBvcnQ6IGpzcjpAc3RkL2FzeW5jQDEuMC4xMAogICAgLSBpbXBvcnQ6IGpzcjpAc3RkL2RhdGV0aW1lQDAuMjI1LjMKICAgIC0gaW1wb3J0OiBqc3I6QHN0ZC9kb3RlbnZAMC4yMjUuMwogICAgLSBpbXBvcnQ6IGpzcjpAc3RkL2VuY29kaW5nQDEuMC43CiAgICAtIGltcG9ydDoganNyOkBzdGQvZnNAMS4wLjEzCiAgICAtIGltcG9ydDoganNyOkBzdGQvcGF0aEAxLjAuOAogICAgLSBpbXBvcnQ6IGpzcjpAc3RkL3NlbXZlckAxLjAuNAogICAgLSBpbXBvcnQ6IGpzcjpAc3RkL3Rlc3RpbmdAMS4wLjkKICAgIC0gaW1wb3J0OiBqc3I6QHN0ZC91dWlkQDEuMC40CgogIGF1dG9tZXJnZToKICAgICMgaHR0cHM6Ly9hdXRvbWVyZ2Uub3JnCiAgICAtIGltcG9ydDogbnBtOkBhdXRvbWVyZ2UvYXV0b21lcmdlQDIuMi44CiAgICAtIGltcG9ydDogbnBtOkBhdXRvbWVyZ2UvYXV0b21lcmdlLXJlcG9AMS4yLjEKICAgIC0gaW1wb3J0OiBucG06QGF1dG9tZXJnZS9hdXRvbWVyZ2UtcmVwby1uZXR3b3JrLWJyb2FkY2FzdGNoYW5uZWxAMS4yLjEKICAgIC0gaW1wb3J0OiBucG06QGF1dG9tZXJnZS9hdXRvbWVyZ2UtcmVwby1zdG9yYWdlLWluZGV4ZWRkYkAxLjIuMQogICAgLSBpbXBvcnQ6IG5wbTpAYXV0b21lcmdlL2F1dG9tZXJnZS1yZXBvLXN0b3JhZ2Utbm9kZWZzQDEuMi4xCiAgICAtIGltcG9ydDogbnBtOkBvbnNldHNvZnR3YXJlL2F1dG9tZXJnZS1wYXRjaGVyQDAuMTQuMAoKICBjcnlwdG86CiAgICAtIGltcG9ydDogbnBtOkBub2JsZS9oYXNoZXNAMS43LjEKICAgICAgd2lsZGNhcmQ6IHRydWUKCiAgYnVpbGQvdG9vbHM6CiAgICAtIGltcG9ydDogbnBtOkB2aXRlanMvcGx1Z2luLXJlYWN0LXN3Y0AzLjguMAogICAgLSBpbXBvcnQ6IG5wbTpyb2xsdXBANC4zNC44CiAgICAtIGltcG9ydDogbnBtOnZpdGVANi4xLjEKICAgIC0gaW1wb3J0OiBucG06dml0ZS1wbHVnaW4td2FzbUAzLjQuMQoKICB1aS9yZWFjdDoKICAgIC0gaW1wb3J0OiBucG06QHR5cGVzL3JlYWN0QDE4LjMuMTgKICAgIC0gaW1wb3J0OiBucG06QHR5cGVzL3JlYWN0LWRvbUAxOC4zLjUKICAgIC0gaW1wb3J0OiBucG06cmVhY3RAMTguMy4xCiAgICAtIGltcG9ydDogbnBtOnJlYWN0LWRvbUAxOC4zLjEKCmRlbm8uanNvbjoKICAtIGdyb3VwOiBzdGQvZGVubwogIC0gZ3JvdXA6IGNyeXB0bwogIC0gZ3JvdXA6IGF1dG9tZXJnZQoKICAjIENMSSB0b29scwogIC0gaW1wb3J0OiBqc3I6QGNsaWZmeS9rZXlwcmVzc0AxLjAuMC1yYy43CiAgLSBpbXBvcnQ6IGpzcjpAY2xpZmZ5L3Byb21wdEAxLjAuMC1yYy43CiAgLSBpbXBvcnQ6IGpzcjpAY2xpZmZ5L3RhYmxlQDEuMC4wLXJjLjcKCiAgIyBTdW5kcnk6IE5QTQogIC0gaW1wb3J0OiBucG06QHR5cGVzL2RpZmZANy4wLjEKICAtIGltcG9ydDogbnBtOmNoYWlANQogIC0gaW1wb3J0OiBucG06YXBwcm94LXN0cmluZy1tYXRjaEAyCiAgLSBpbXBvcnQ6IG5wbTpkYXRlLWZuc0A0CiAgLSBpbXBvcnQ6IG5wbTpzdWJob3N0aW5nQDAuMS4wLWFscGhhLjEKICAtIGltcG9ydDogbnBtOmRpZmZANwogIC0gaW1wb3J0OiBucG06ZmFzdC1qc29uLXBhdGNoQDMuMS4xCiAgLSBpbXBvcnQ6IG5wbTpmYWtlLWluZGV4ZWRkYkA2LjAuMAogIC0gaW1wb3J0OiBucG06aGFwcHktZG9tQDE3LjEuMwogIC0gaW1wb3J0OiBucG06aGFzaC1pdEA2LjAuMAogIC0gaW1wb3J0OiBucG06aWdub3JlQDcKICAtIGltcG9ydDogbnBtOmltbWVyQDEwCiAgLSBpbXBvcnQ6IG5wbTpvcmFAOC4yLjAKICAtIGltcG9ydDogbnBtOm9sbGFtYUAwLjUuMTMKICAtIGltcG9ydDogbnBtOnByZXR0eS1ieXRlc0A2LjEuMQogIC0gaW1wb3J0OiBucG06cmFtZGFAMC4zMC4xCiAgLSBpbXBvcnQ6IG5wbTpyYW1iZGFAOS40LjIKICAtIGltcG9ydDogbnBtOnJ4anNANy44LjIKICAtIGltcG9ydDogbnBtOnN0cmlwLWFuc2lANwogIC0gaW1wb3J0OiBucG06c3ViaG9zdGluZ0AwLjEuMC1hbHBoYS4xCiAgLSBpbXBvcnQ6IG5wbTp0aW55Y29sb3IyQDEuNi4wCiAgLSBpbXBvcnQ6IG5wbTp0cy1lc3NlbnRpYWxzQDEwLjAuNAogIC0gaW1wb3J0OiBucG06dmFsaWJvdEAxLjAuMC1yYy4xCiAgLSBpbXBvcnQ6IG5wbTp5YW1sQDIuNy4wCgogICMgQnJvd3NlcgogIC0gaW1wb3J0OiBucG06Y3NzdHlwZUAzCiAgLSBpbXBvcnQ6IG5wbTp1YS1wYXJzZXItanNAMi4wLjIKCiAgIyBVSQogIC0gaW1wb3J0OiBucG06cmVhY3QtZXJyb3ItYm91bmRhcnlANQogIC0gaW1wb3J0OiBucG06cmVhY3QtaW5zcGVjdG9yQDYKICAtIGltcG9ydDogbnBtOnJlYWN0LXNwaW5uZXJzQDAuMTUuMAoKcGFja2FnZS5qc29uOgogIC0gZ3JvdXA6IHN0ZC9kZW5vCiAgLSBncm91cDogY3J5cHRvCiAgLSBncm91cDogYnVpbGQvdG9vbHMKICAgIGRldjogdHJ1ZQoKICAtIGltcG9ydDogbnBtOmhvbm9ANC43LjIKCiAgIyBVSQogIC0gZ3JvdXA6IHVpL3JlYWN0CiAgLSBpbXBvcnQ6IG5wbTpyZWFjdC1pY29uc0A1LjUuMAogIC0gaW1wb3J0OiBucG06QHZpZHN0YWNrL3JlYWN0QDEuMTIuMTIKCiAgIyBVSTpGcmFtZXdvcmtzCiAgLSBpbXBvcnQ6IG5wbTp2aXRlcHJlc3NAMS42LjMKICAtIGltcG9ydDogbnBtOnZ1ZUAzLjUuMTMK", ".vscode/settings.json": "data:application/json;base64,ewogICJkZW5vLmVuYWJsZSI6IHRydWUKfQo=", "README.md": "data:text/markdown;base64,IyBNb2R1bGUKCg==", - "deno.json": "data:application/json;base64,ewogICJ2ZXJzaW9uIjogIjAuMC4wIiwKICAibGljZW5zZSI6ICJNSVQiLAogICJ0YXNrcyI6IHsKICAgICJkZXYiOiAiZGVubyAgICAgIHJ1biAtUldORSAtLWFsbG93LXJ1biAtLWFsbG93LWZmaSA8RU5UUlk+IC0tY21kPWRldiAgICAtLWluPS4vc3JjLy10ZXN0L2luZGV4Lmh0bWwiLAogICAgImJ1aWxkIjogImRlbm8gICAgcnVuIC1SV0UgIC0tYWxsb3ctcnVuIC0tYWxsb3ctZmZpIDxFTlRSWT4gLS1jbWQ9YnVpbGQgIC0taW49Li9zcmMvLXRlc3QvaW5kZXguaHRtbCIsCiAgICAic2VydmUiOiAiZGVubyAgICBydW4gLVJORSAgLS1hbGxvdy1ydW4gLS1hbGxvdy1mZmkgPEVOVFJZPiAtLWNtZD1zZXJ2ZSIsCgogICAgImNsZWFuIjogImRlbm8gICAgcnVuIC1SV0UgIC0tYWxsb3ctZmZpICAgICAgICAgICAgIDxFTlRSWT4gLS1jbWQ9Y2xlYW4iLAogICAgInVwZ3JhZGUiOiAiZGVubyAgcnVuIC1SV05FIC0tYWxsb3ctcnVuIC0tYWxsb3ctZmZpIDxFTlRSWT4gLS1jbWQ9dXBncmFkZSIsCiAgICAiYmFja3VwIjogImRlbm8gICBydW4gLVJXRSAgLS1hbGxvdy1ydW4gLS1hbGxvdy1mZmkgPEVOVFJZPiAtLWNtZD1iYWNrdXAiLAogICAgImhlbHAiOiAiZGVubyAgICAgcnVuIC1SRSAgIC0tYWxsb3ctZmZpICAgICAgICAgICAgIDxFTlRSWT4gLS1jbWQ9aGVscCIsCgogICAgInN5cyI6ICJkZW5vICAgICAgcnVuIC1SV05FIDxFTlRSWV9TWVM+IiwKICAgICJ0bXAiOiAiZGVubyAgICAgIHJ1biAtQSAuLy1zY3JpcHRzLy10bXAudHMiCiAgfSwKICAiY29tcGlsZXJPcHRpb25zIjogewogICAgInN0cmljdCI6IHRydWUsCiAgICAibGliIjogWyJkZW5vLm5zIiwgImVzbmV4dCIsICJkb20iLCAiZG9tLml0ZXJhYmxlIiwgImRvbS5hc3luY2l0ZXJhYmxlIl0sCiAgICAidHlwZXMiOiBbInZpdGUvY2xpZW50IiwgIkB0eXBlcy9yZWFjdCJdLAogICAgImpzeCI6ICJyZWFjdCIsCiAgICAianN4RmFjdG9yeSI6ICJSZWFjdC5jcmVhdGVFbGVtZW50IiwKICAgICJqc3hGcmFnbWVudEZhY3RvcnkiOiAiUmVhY3QuRnJhZ21lbnQiCiAgfSwKICAid29ya3NwYWNlIjogW10sCiAgIm5vZGVNb2R1bGVzRGlyIjogImF1dG8iLAogICJpbXBvcnRzIjogewogICAgIjxTRUxGX0lNUE9SVF9OQU1FPiI6ICI8U0VMRl9JTVBPUlRfVVJJPiIKICB9Cn0K", - "package.json": "data:application/json;base64,ewogICJkZXBlbmRlbmNpZXMiOiB7CiAgICAiQHN5cy9zdGQiOiAibnBtOkBqc3Ivc3lzX19zdGQiLAogICAgIkBzeXMvdG1wIjogIm5wbTpAanNyL3N5c19fdG1wIiwKICAgICJAc3lzL3R5cGVzIjogIm5wbTpAanNyL3N5c19fdHlwZXMiLAogICAgIkBzeXMvdWktY3NzIjogIm5wbTpAanNyL3N5c19fdWktY3NzIiwKICAgICJyZWFjdCI6ICIiLAogICAgInJlYWN0LWRvbSI6ICIiLAogICAgInZhbGlib3QiOiAiIgogIH0sCiAgImRldkRlcGVuZGVuY2llcyI6IHsKICAgICJAdHlwZXMvcmVhY3QiOiAiIiwKICAgICJAdHlwZXMvcmVhY3QtZG9tIjogIiIsCiAgICAiQHZpdGVqcy9wbHVnaW4tcmVhY3Qtc3djIjogIiIsCiAgICAicm9sbHVwIjogIiIsCiAgICAidml0ZSI6ICIiLAogICAgInZpdGUtcGx1Z2luLXdhc20iOiAiIgogIH0KfQo=", + "deno.json": "data:application/json;base64,ewogICJ2ZXJzaW9uIjogIjAuMC4wIiwKICAidGFza3MiOiB7CiAgICAiZGV2IjogImRlbm8gICAgICBydW4gLVJXTkUgLS1hbGxvdy1ydW4gLS1hbGxvdy1mZmkgPEVOVFJZPiAtLWNtZD1kZXYgICAgLS1pbj0uL3NyYy8tdGVzdC9pbmRleC5odG1sIiwKICAgICJidWlsZCI6ICJkZW5vICAgIHJ1biAtUldFICAtLWFsbG93LXJ1biAtLWFsbG93LWZmaSA8RU5UUlk+IC0tY21kPWJ1aWxkICAtLWluPS4vc3JjLy10ZXN0L2luZGV4Lmh0bWwiLAogICAgInNlcnZlIjogImRlbm8gICAgcnVuIC1STkUgIC0tYWxsb3ctcnVuIC0tYWxsb3ctZmZpIDxFTlRSWT4gLS1jbWQ9c2VydmUiLAoKICAgICJ0ZXN0IjogImRlbm8gICAgIHRlc3QgLVJXTkUgLS1hbGxvdy1ydW4gLS1hbGxvdy1mZmkgLS1hbGxvdy1zeXMiLAogICAgImNsZWFuIjogImRlbm8gICAgcnVuIC1SV0UgIC0tYWxsb3ctZmZpICAgICAgICAgICAgIDxFTlRSWT4gLS1jbWQ9Y2xlYW4iLAogICAgInVwZ3JhZGUiOiAiZGVubyAgcnVuIC1SV05FIC0tYWxsb3ctcnVuIC0tYWxsb3ctZmZpIDxFTlRSWT4gLS1jbWQ9dXBncmFkZSIsCiAgICAiYmFja3VwIjogImRlbm8gICBydW4gLVJXRSAgLS1hbGxvdy1ydW4gLS1hbGxvdy1mZmkgPEVOVFJZPiAtLWNtZD1iYWNrdXAiLAogICAgImhlbHAiOiAiZGVubyAgICAgcnVuIC1SRSAgIC0tYWxsb3ctZmZpICAgICAgICAgICAgIDxFTlRSWT4gLS1jbWQ9aGVscCIsCgogICAgInRtcCI6ICJkZW5vICAgICAgcnVuIC1BIC4vLXNjcmlwdHMvLXRtcC50cyIKICB9LAogICJjb21waWxlck9wdGlvbnMiOiB7CiAgICAic3RyaWN0IjogdHJ1ZSwKICAgICJsaWIiOiBbImRlbm8ubnMiLCAiZXNuZXh0IiwgImRvbSIsICJkb20uaXRlcmFibGUiLCAiZG9tLmFzeW5jaXRlcmFibGUiXSwKICAgICJ0eXBlcyI6IFsidml0ZS9jbGllbnQiLCAiQHR5cGVzL3JlYWN0Il0sCiAgICAianN4IjogInJlYWN0LWpzeCIsCiAgICAianN4SW1wb3J0U291cmNlIjogInJlYWN0IgogIH0sCiAgIndvcmtzcGFjZSI6IFtdLAogICJpbXBvcnRNYXAiOiAiLi9pbXBvcnRzLmpzb24iLAogICJub2RlTW9kdWxlc0RpciI6ICJhdXRvIiwKICAibGljZW5zZSI6ICJNSVQiCn0K", + "imports.json": "data:application/json;base64,ewogICJpbXBvcnRzIjogewogICAgIkBzeXMvc3RkIjogImpzcjpAc3lzL3N0ZEAwLjAuMTQxIiwKICAgICJAc3lzL3RtcCI6ICJqc3I6QHN5cy90bXBAMC4wLjExMCIsCiAgICAiQHN5cy90eXBlcyI6ICJqc3I6QHN5cy90eXBlc0AwLjAuOTIiLAogICAgIkBzeXMvdWktY3NzIjogImpzcjpAc3lzL3VpLWNzc0AwLjAuODEiLAogICAgIkBzeXMvZHJpdmVyLXZpdGUiOiAianNyOkBzeXMvZHJpdmVyLXZpdGVAMC4wLjEzNyIsCiAgICAicmVhY3QiOiAibnBtOnJlYWN0QDE4LjMuMSIsCiAgICAicmVhY3QtZG9tIjogIm5wbTpyZWFjdC1kb21AMTguMy4xIiwKICAgICJAdHlwZXMvcmVhY3QiOiAibnBtOkB0eXBlcy9yZWFjdEAxOC4zLjE4IiwKICAgICJAdHlwZXMvcmVhY3QtZG9tIjogIm5wbTpAdHlwZXMvcmVhY3QtZG9tQDE4LjMuNSIsCiAgICAiQHZpdGVqcy9wbHVnaW4tcmVhY3Qtc3djIjogIm5wbTpAdml0ZWpzL3BsdWdpbi1yZWFjdC1zd2NAMy44LjAiLAogICAgInJvbGx1cCI6ICJucG06cm9sbHVwQDQuMzQuOCIsCiAgICAidml0ZSI6ICJucG06dml0ZUA2LjEuMSIsCiAgICAidml0ZS1wbHVnaW4td2FzbSI6ICJucG06dml0ZS1wbHVnaW4td2FzbUAzLjQuMSIKICB9Cn0K", "src/-test.ts": "data:application/typescript;base64,ZXhwb3J0ICogZnJvbSAnLi8tdGVzdC9tb2QudHMnOwo=", "src/-test/-sample/m.Foo.ts": "data:application/typescript;base64,aW1wb3J0IHR5cGUgKiBhcyB0IGZyb20gJy4vdC50cyc7CgovKioKICogQG1vZHVsZQogKiBTYW1wbGUgRm9vIG1vZHVsZS4KICovCmV4cG9ydCBjb25zdCBGb286IHQuRm9vID0geyBtc2c6ICfwn5GLJyB9Owo=", "src/-test/-sample/t.ts": "data:application/typescript;base64,ZXhwb3J0IHR5cGUgRm9vID0geyBtc2c/OiBzdHJpbmcgfTsK", - "src/-test/-sample/ui.Foo.tsx": "data:application/typescript+jsx;base64,aW1wb3J0IHsgdXNlU3RhdGUgfSBmcm9tICdyZWFjdCc7CgppbXBvcnQgeyBGb28gfSBmcm9tICdAc3lzL3RtcC91aSc7CmltcG9ydCB7IENvbG9yLCBjc3MgfSBmcm9tICdAc3lzL3VpLWNzcyc7CmltcG9ydCB0eXBlIHsgdCB9IGZyb20gJy4uLy4uL2NvbW1vbi50cyc7CgovKioKICogU2FtcGxlIENvbXBvbmVudCBkZW1vbnN0cmF0aW5nIHRoZSBmdW5kYW1lbnRhbHMgb2YgUmVhY3QKICogYW5kIHByb3ZpbmcgbW9kdWxlIGltcG9ydGluZyB3b3JrcyBhY3Jvc3MgdGhlIG1vbm9yZXBvLgogKgogKiAgIC0gTW9kdWxlICJpbXBvcnRzIiAocHJvdmUgQHN5cyBpbXBvcnRzIGZyb20gdGhlIHdvcmtzcGFjZSB3b3JrKQogKiAgIC0gU3R5bGU6IENTUyBwcmltaXRpdmVzCiAqICAgLSBTdHlsZTogQ29sb3IgcHJpbWl0aXZlcwogKgogKi8KZXhwb3J0IHR5cGUgRm9vQ29tcG9uZW50ID0gewogIHRoZW1lPzogdC5Db21tb25UaGVtZTsKICBzdHlsZT86IHQuQ3NzSW5wdXQ7Cn07CgpleHBvcnQgY29uc3QgRm9vU2FtcGxlOiBSZWFjdC5GQzxGb29Db21wb25lbnQ+ID0gKHByb3BzKSA9PiB7CiAgY29uc3QgW2lzT3Zlciwgc2V0T3Zlcl0gPSB1c2VTdGF0ZShmYWxzZSk7CiAgY29uc3Qgb3ZlciA9IChpc092ZXI6IGJvb2xlYW4pID0+ICgpID0+IHNldE92ZXIoaXNPdmVyKTsKCiAgY29uc3QgdGhlbWUgPSBDb2xvci50aGVtZShwcm9wcy50aGVtZSA/PyAnRGFyaycpOwoKICBjb25zdCBzdHlsZXMgPSB7CiAgICBiYXNlOiBjc3MoewogICAgICBNYXJnaW46IDIwLAogICAgICBwYWRkaW5nOiAyMCwKICAgICAgYmFja2dyb3VuZENvbG9yOiBpc092ZXIgPyAnaG90cGluaycgOiAnbGlnaHRncmVlbicsCiAgICAgIGZvbnRGYW1pbHk6ICdtb25vc3BhY2UnLAogICAgICBjb2xvcjogJ2JsdWUnLAogICAgfSksCiAgICB0aXRsZTogY3NzKHsKICAgICAgYmFja2dyb3VuZENvbG9yOiBDb2xvci5SVUJZLAogICAgICBmb250U2l6ZTogMzAsCiAgICAgIE1hcmdpblk6IDUsCiAgICAgIFBhZGRpbmdYOiAzMCwKICAgICAgUGFkZGluZ1k6IFszMCwgMTVdLAogICAgICBjb2xvcjogJ3JlZCcsCiAgICAgICc6aG92ZXInOiB7IGNvbG9yOiAnZ3JlZW4nIH0sCiAgICB9KSwKICAgIHRoZW1lU2FtcGxlOiBjc3MoewogICAgICBtYXJnaW5Ub3A6IDIwLAogICAgICBkaXNwbGF5OiAnZ3JpZCcsCiAgICAgIHBsYWNlSXRlbXM6ICdjZW50ZXInLAogICAgICBtaW5IZWlnaHQ6IDMwMCwKICAgICAgY29sb3I6IENvbG9yLmFscGhhKHRoZW1lLmZnLCBpc092ZXIgPyAxIDogMC4zKSwKICAgICAgYmFja2dyb3VuZENvbG9yOiB0aGVtZS5iZywKICAgICAgdHJhbnNpdGlvbjogYGNvbG9yIDIwMG1zYCwKICAgIH0pLAogIH07CgogIGNvbnN0IGVsVGhlbWVTYW1wbGUgPSAoCiAgICA8ZGl2IGNsYXNzTmFtZT17c3R5bGVzLnRoZW1lU2FtcGxlLmNsYXNzfT4KICAgICAgPGRpdj5IZWxsbzwvZGl2PgogICAgPC9kaXY+CiAgKTsKCiAgcmV0dXJuICgKICAgIDxkaXYKICAgICAgY2xhc3NOYW1lPXtjc3Moc3R5bGVzLmJhc2UsIHByb3BzLnN0eWxlKS5jbGFzc30KICAgICAgb25Nb3VzZUVudGVyPXtvdmVyKHRydWUpfQogICAgICBvbk1vdXNlTGVhdmU9e292ZXIoZmFsc2UpfQogICAgPgogICAgICA8ZGl2IGNsYXNzTmFtZT17c3R5bGVzLnRpdGxlLmNsYXNzfT4KICAgICAgICA8ZGl2PntgSGVsbG8gV29ybGQg8J+Ri2B9PC9kaXY+CiAgICAgIDwvZGl2PgogICAgICA8ZGl2PntgKHNlZSBjb25zb2xlIGZvciBpbXBvcnQgc2FtcGxlcylgfTwvZGl2PgogICAgICA8ZGl2IHN0eWxlPXt7IHBhZGRpbmdUb3A6IDEwIH19PgogICAgICAgIHsnSW1wb3J0ZWQgZnJvbSDihpAgJ30KICAgICAgICA8Rm9vIC8+CiAgICAgICAgPGRpdj57J/CfkLfwn5C3IFRNUCDwn5C38J+QtyBpbXBvcnQgV0lQJ308L2Rpdj4KICAgICAgICA8Y29kZT57JzxKU1g+IOKGkiBWaXRlIOKGkiBFU00uanMuZC50cyDihpIgbW9kLnRzIOKGkiBKU1Ig4oaSIGltcG9ydCd9PC9jb2RlPgogICAgICA8L2Rpdj4KICAgICAge2VsVGhlbWVTYW1wbGV9CiAgICA8L2Rpdj4KICApOwp9Owo=", - "src/-test/entry.lib.ts": "data:application/typescript;base64,ZXhwb3J0IGNvbnN0IGZuID0gYXN5bmMgKCkgPT4gewogIGNvbnN0IHsgRm9vIH0gPSBhd2FpdCBpbXBvcnQoJy4vLXNhbXBsZS9tLkZvby50cycpOwogIGNvbnN0IHsgRm9vU2FtcGxlIH0gPSBhd2FpdCBpbXBvcnQoJy4vLXNhbXBsZS91aS5Gb28udHN4Jyk7CiAgcmV0dXJuIHsKICAgIEZvbywKICAgIEZvb1NhbXBsZSwKICB9Owp9Owo=", - "src/-test/entry.tsx": "data:application/typescript+jsx;base64,aW1wb3J0IHsgcGtnIH0gZnJvbSAnLi4vY29tbW9uLnRzJzsKCi8qKgogKiBSZW5kZXIgVUkuCiAqLwpnbG9iYWxUaGlzLmRvY3VtZW50LnRpdGxlID0gcGtnLm5hbWU7CmNvbnNvbGUuaW5mbygn8J+QtyAuL2VudHJ5LnRzeCDihpIgUGtnOvCfkqYnLCBwa2cpOwoKaW1wb3J0IHsgU3RyaWN0TW9kZSB9IGZyb20gJ3JlYWN0JzsKaW1wb3J0IHsgY3JlYXRlUm9vdCB9IGZyb20gJ3JlYWN0LWRvbS9jbGllbnQnOwppbXBvcnQgeyBGb29TYW1wbGUgfSBmcm9tICcuLy1zYW1wbGUvdWkuRm9vLnRzeCc7CgovKioKICog8J+QtyBUZXN0ICIgQHN5cyAiIG1vZHVsZSBpbXBvcnRzIGZyb20gYWNyb3NzIHRoZQogKiAgICBuYW1lc3BhY2UgKG1vbm9yZXBvL3dvcmtzcGFjZSkuCiAqLwppbXBvcnQgJ0BzeXMvdG1wL3NhbXBsZS1pbXBvcnRzJzsKCi8qKgogKiBTYW1wbGU6IHJlbmRlciByZWFjdCBjb21wb25lbnQuCiAqLwpjb25zdCByb290ID0gY3JlYXRlUm9vdChkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgncm9vdCcpISk7CnJvb3QucmVuZGVyKAogIDxTdHJpY3RNb2RlPgogICAgPEZvb1NhbXBsZSBzdHlsZT17eyBib3JkZXI6IGBzb2xpZCAxcHggYmx1ZWAgfX0gLz4KICA8L1N0cmljdE1vZGU+LAopOwo=", + "src/-test/-sample/ui.Foo.tsx": "data:application/typescript+jsx;base64,aW1wb3J0IHsgdXNlU3RhdGUgfSBmcm9tICdyZWFjdCc7CgovLyBpbXBvcnQgeyBGb28gfSBmcm9tICdAc3lzL3RtcC91aSc7CmltcG9ydCB7IENvbG9yLCBjc3MgfSBmcm9tICdAc3lzL3VpLWNzcyc7CmltcG9ydCB0eXBlIHsgdCB9IGZyb20gJy4uLy4uL2NvbW1vbi50cyc7CgovKioKICogU2FtcGxlIENvbXBvbmVudCBkZW1vbnN0cmF0aW5nIHRoZSBmdW5kYW1lbnRhbHMgb2YgUmVhY3QKICogYW5kIHByb3ZpbmcgbW9kdWxlIGltcG9ydGluZyB3b3JrcyBhY3Jvc3MgdGhlIG1vbm9yZXBvLgogKgogKiAgIC0gTW9kdWxlICJpbXBvcnRzIiAocHJvdmUgQHN5cyBpbXBvcnRzIGZyb20gdGhlIHdvcmtzcGFjZSB3b3JrKQogKiAgIC0gU3R5bGU6IENTUyBwcmltaXRpdmVzCiAqICAgLSBTdHlsZTogQ29sb3IgcHJpbWl0aXZlcwogKgogKi8KZXhwb3J0IHR5cGUgRm9vQ29tcG9uZW50ID0gewogIHRoZW1lPzogdC5Db21tb25UaGVtZTsKICBzdHlsZT86IHQuQ3NzSW5wdXQ7Cn07CgpleHBvcnQgY29uc3QgRm9vU2FtcGxlOiBSZWFjdC5GQzxGb29Db21wb25lbnQ+ID0gKHByb3BzKSA9PiB7CiAgY29uc3QgW2lzT3Zlciwgc2V0T3Zlcl0gPSB1c2VTdGF0ZShmYWxzZSk7CiAgY29uc3Qgb3ZlciA9IChpc092ZXI6IGJvb2xlYW4pID0+ICgpID0+IHNldE92ZXIoaXNPdmVyKTsKCiAgY29uc3QgdGhlbWUgPSBDb2xvci50aGVtZShwcm9wcy50aGVtZSA/PyAnRGFyaycpOwoKICBjb25zdCBzdHlsZXMgPSB7CiAgICBiYXNlOiBjc3MoewogICAgICBNYXJnaW46IDIwLAogICAgICBwYWRkaW5nOiAyMCwKICAgICAgYmFja2dyb3VuZENvbG9yOiBpc092ZXIgPyAnaG90cGluaycgOiAnbGlnaHRncmVlbicsCiAgICAgIGZvbnRGYW1pbHk6ICdtb25vc3BhY2UnLAogICAgICBjb2xvcjogJ2JsdWUnLAogICAgfSksCiAgICB0aXRsZTogY3NzKHsKICAgICAgYmFja2dyb3VuZENvbG9yOiBDb2xvci5SVUJZLAogICAgICBmb250U2l6ZTogMzAsCiAgICAgIE1hcmdpblk6IDUsCiAgICAgIFBhZGRpbmdYOiAzMCwKICAgICAgUGFkZGluZ1k6IFszMCwgMTVdLAogICAgICBjb2xvcjogJ3JlZCcsCiAgICAgICc6aG92ZXInOiB7IGNvbG9yOiAnZ3JlZW4nIH0sCiAgICB9KSwKICAgIHRoZW1lU2FtcGxlOiBjc3MoewogICAgICBtYXJnaW5Ub3A6IDIwLAogICAgICBkaXNwbGF5OiAnZ3JpZCcsCiAgICAgIHBsYWNlSXRlbXM6ICdjZW50ZXInLAogICAgICBtaW5IZWlnaHQ6IDMwMCwKICAgICAgY29sb3I6IENvbG9yLmFscGhhKHRoZW1lLmZnLCBpc092ZXIgPyAxIDogMC4zKSwKICAgICAgYmFja2dyb3VuZENvbG9yOiB0aGVtZS5iZywKICAgICAgdHJhbnNpdGlvbjogYGNvbG9yIDIwMG1zYCwKICAgIH0pLAogIH07CgogIGNvbnN0IGVsVGhlbWVTYW1wbGUgPSAoCiAgICA8ZGl2IGNsYXNzTmFtZT17c3R5bGVzLnRoZW1lU2FtcGxlLmNsYXNzfT4KICAgICAgPGRpdj5IZWxsbzwvZGl2PgogICAgPC9kaXY+CiAgKTsKCiAgcmV0dXJuICgKICAgIDxkaXYKICAgICAgY2xhc3NOYW1lPXtjc3Moc3R5bGVzLmJhc2UsIHByb3BzLnN0eWxlKS5jbGFzc30KICAgICAgb25Nb3VzZUVudGVyPXtvdmVyKHRydWUpfQogICAgICBvbk1vdXNlTGVhdmU9e292ZXIoZmFsc2UpfQogICAgPgogICAgICA8ZGl2IGNsYXNzTmFtZT17c3R5bGVzLnRpdGxlLmNsYXNzfT4KICAgICAgICA8ZGl2PntgSGVsbG8gV29ybGQg8J+Ri2B9PC9kaXY+CiAgICAgIDwvZGl2PgogICAgICA8ZGl2PntgKHNlZSBjb25zb2xlIGZvciBpbXBvcnQgc2FtcGxlcylgfTwvZGl2PgogICAgICA8ZGl2IHN0eWxlPXt7IHBhZGRpbmdUb3A6IDEwIH19PgogICAgICAgIHsnSW1wb3J0ZWQgZnJvbSDihpAgJ30KICAgICAgICB7LyogPEZvbyAvPiAqL30KICAgICAgICA8ZGl2Pnsn8J+Qt/CfkLcgVE1QIPCfkLfwn5C3IGltcG9ydCBXSVAnfTwvZGl2PgogICAgICAgIDxjb2RlPnsnPEpTWD4g4oaSIFZpdGUg4oaSIEVTTS5qcy5kLnRzIOKGkiBtb2QudHMg4oaSIEpTUiDihpIgaW1wb3J0J308L2NvZGU+CiAgICAgIDwvZGl2PgogICAgICB7ZWxUaGVtZVNhbXBsZX0KICAgIDwvZGl2PgogICk7Cn07Cg==", + "src/-test/entry.tsx": "data:application/typescript+jsx;base64,aW1wb3J0IHsgcGtnIH0gZnJvbSAnLi4vcGtnLnRzJzsKCi8qKgogKiBSZW5kZXIgVUkuCiAqLwpnbG9iYWxUaGlzLmRvY3VtZW50LnRpdGxlID0gcGtnLm5hbWU7CmNvbnNvbGUuaW5mbygn8J+QtyAuL2VudHJ5LnRzeCDihpIgUGtnOvCfkqYnLCBwa2cpOwoKaW1wb3J0IHsgU3RyaWN0TW9kZSB9IGZyb20gJ3JlYWN0JzsKaW1wb3J0IHsgY3JlYXRlUm9vdCB9IGZyb20gJ3JlYWN0LWRvbS9jbGllbnQnOwppbXBvcnQgeyBGb29TYW1wbGUgfSBmcm9tICcuLy1zYW1wbGUvdWkuRm9vLnRzeCc7CgovKioKICog8J+QtyBUZXN0ICIgQHN5cyAiIG1vZHVsZSBpbXBvcnRzIGZyb20gYWNyb3NzIHRoZQogKiAgICBuYW1lc3BhY2UgKG1vbm9yZXBvL3dvcmtzcGFjZSkuCiAqLwovLyBpbXBvcnQgJ0BzeXMvdG1wL3NhbXBsZS1pbXBvcnRzJzsKCi8qKgogKiBTYW1wbGU6IHJlbmRlciByZWFjdCBjb21wb25lbnQuCiAqLwpjb25zdCByb290ID0gY3JlYXRlUm9vdChkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgncm9vdCcpISk7CnJvb3QucmVuZGVyKAogIDxTdHJpY3RNb2RlPgogICAgPEZvb1NhbXBsZSBzdHlsZT17eyBib3JkZXI6IGBzb2xpZCAxcHggYmx1ZWAgfX0gLz4KICA8L1N0cmljdE1vZGU+LAopOwo=", "src/-test/index.html": "data:text/plain;base64,PCFET0NUWVBFIGh0bWw+CjxodG1sIGxhbmc9ImVuIj4KICA8aGVhZD4KICAgIDxtZXRhIGNoYXJzZXQ9IlVURi04IiAvPgogICAgPG1ldGEgbmFtZT0idmlld3BvcnQiIGNvbnRlbnQ9IndpZHRoPWRldmljZS13aWR0aCwgaW5pdGlhbC1zY2FsZT0xLjAiIC8+CiAgICA8dGl0bGU+bG9hZGluZy4uLjwvdGl0bGU+CiAgPC9oZWFkPgogIDxib2R5PgogICAgPGRpdiBpZD0icm9vdCI+PC9kaXY+CiAgICA8c2NyaXB0IHR5cGU9Im1vZHVsZSIgc3JjPSIuL2VudHJ5LnRzeCI+PC9zY3JpcHQ+CiAgPC9ib2R5Pgo8L2h0bWw+Cg==", "src/-test/mod.ts": "data:application/typescript;base64,ZXhwb3J0IHsgZGVzY3JpYmUsIGV4cGVjdCwgZXhwZWN0RXJyb3IsIGl0LCBUZXN0aW5nIH0gZnJvbSAnQHN5cy90ZXN0aW5nL3NlcnZlcic7CmV4cG9ydCAqIGZyb20gJy4uL2NvbW1vbi50cyc7Cg==", "src/.test.ts": "data:application/typescript;base64,aW1wb3J0IHsgdHlwZSB0LCBkZXNjcmliZSwgaXQsIGV4cGVjdCwgUGtnLCBwa2cgfSBmcm9tICcuLy10ZXN0LnRzJzsKCmRlc2NyaWJlKGBtb2R1bGU6ICR7UGtnLnRvU3RyaW5nKHBrZyl9YCwgKCkgPT4gewogIGl0KCdleGlzdHMnLCAoKSA9PiB7CiAgICBjb25zb2xlLmluZm8oYPCfkqYgTW9kdWxlYCwgcGtnKTsKICAgIGV4cGVjdCh0eXBlb2YgcGtnLm5hbWUgPT09ICdzdHJpbmcnKS50by5iZS50cnVlOwogIH0pOwp9KTsK", @@ -24,5 +21,5 @@ "src/mod.ts": "data:application/typescript;base64,LyoqCiAqIEBtb2R1bGUKICogVG9vbHMgZm9yLi4u8J+QtwogKi8KZXhwb3J0IHsgcGtnIH0gZnJvbSAnLi9wa2cudHMnOwoKLyoqIE1vZHVsZSB0eXBlcy4gKi8KZXhwb3J0IHR5cGUgKiBhcyB0IGZyb20gJy4vdHlwZXMudHMnOwo=", "src/pkg.ts": "data:application/typescript;base64,aW1wb3J0IHsgUGtnLCB0eXBlIHQgfSBmcm9tICdAc3lzL3N0ZCc7CmltcG9ydCB7IGRlZmF1bHQgYXMgZGVubyB9IGZyb20gJy4uL2Rlbm8uanNvbicgd2l0aCB7IHR5cGU6ICdqc29uJyB9OwoKCi8qKgogKiBQYWNrYWdlIG1ldGEtZGF0YS4KICovCmV4cG9ydCBjb25zdCBwa2c6IHQuUGtnID0gUGtnLmZyb21Kc29uKGRlbm8pOwo=", "src/types.ts": "data:application/typescript;base64,LyoqCiAqIEBtb2R1bGUKICogTW9kdWxlIHR5cGVzLgogKi8KZXhwb3J0IHt9Owo=", - "vite.config.ts": "data:application/typescript;base64,aW1wb3J0IHsgVml0ZSB9IGZyb20gJ0BzeXMvZHJpdmVyLXZpdGUnOwppbXBvcnQgeyBkZWZpbmVDb25maWcgfSBmcm9tICd2aXRlJzsKCmV4cG9ydCBjb25zdCBwYXRocyA9IFZpdGUuQ29uZmlnLnBhdGhzKHsgYXBwOiB7IGVudHJ5OiAnLi9zcmMvLXRlc3QvaW5kZXguaHRtbCcgfSB9KTsKCmV4cG9ydCBkZWZhdWx0IGRlZmluZUNvbmZpZygoKSA9PiB7CiAgcmV0dXJuIFZpdGUuQ29uZmlnLmFwcCh7CiAgICBwYXRocywKICAgIGNodW5rcyhlKSB7CiAgICAgIGUuY2h1bmsoJ3JlYWN0JywgJ3JlYWN0Jyk7CiAgICAgIGUuY2h1bmsoJ3JlYWN0LmRvbScsICdyZWFjdC1kb20nKTsKICAgICAgZS5jaHVuaygnc3lzJywgWydAc3lzL3N0ZCddKTsKICAgIH0sCiAgfSk7Cn0pOwo=" + "vite.config.ts": "data:application/typescript;base64,aW1wb3J0IHsgVml0ZSB9IGZyb20gJ2pzcjpAc3lzL2RyaXZlci12aXRlJzsKaW1wb3J0IHsgZGVmaW5lQ29uZmlnIH0gZnJvbSAnbnBtOnZpdGUnOwoKZXhwb3J0IGRlZmF1bHQgZGVmaW5lQ29uZmlnKCgpID0+IHsKICBjb25zdCBlbnRyeSA9ICcuL3NyYy8tdGVzdC9pbmRleC5odG1sJzsKICBjb25zdCBwYXRocyA9IFZpdGUuQ29uZmlnLnBhdGhzKHsgYXBwOiB7IGVudHJ5IH0gfSk7CiAgcmV0dXJuIFZpdGUuQ29uZmlnLmFwcCh7CiAgICBwYXRocywKICAgIGNodW5rcyhlKSB7CiAgICAgIGUuY2h1bmsoJ3JlYWN0JywgJ3JlYWN0Jyk7CiAgICAgIGUuY2h1bmsoJ3JlYWN0LmRvbScsICdyZWFjdC1kb20nKTsKICAgICAgZS5jaHVuaygnc3lzJywgWydAc3lzL3N0ZCddKTsKICAgICAgZS5jaHVuaygnY3NzJywgWydAc3lzL3VpLWNzcyddKTsKICAgIH0sCiAgfSk7Cn0pOwo=" } diff --git a/code/sys.driver/driver-vite/src/m.Vite.Tmpl/.test.ts b/code/sys.driver/driver-vite/src/m.Vite.Tmpl/.test.ts deleted file mode 100644 index 5eecb6486d..0000000000 --- a/code/sys.driver/driver-vite/src/m.Vite.Tmpl/.test.ts +++ /dev/null @@ -1,18 +0,0 @@ -import { SAMPLE, describe, expect, it } from '../-test.ts'; -import { Vite } from '../mod.ts'; - -describe('Vite: Template Generation', () => { - it('inserts template', async () => { - const fs = SAMPLE.fs('Vite.tmpl'); - expect(await fs.ls()).to.eql([]); - - const tmpl = await Vite.Tmpl.create(); - const res = await tmpl.copy(fs.dir); - - const a = (await fs.ls()).toSorted(); - const b = (await res.target.ls()).toSorted(); - - expect(b.length).to.be.greaterThan(10); - expect(a).to.eql(b); - }); -}); diff --git a/code/sys.driver/driver-vite/src/m.Vite.Tmpl/m.Tmpl.ts b/code/sys.driver/driver-vite/src/m.Vite.Tmpl/m.Tmpl.ts index fd0bc421c5..a0ca24445a 100644 --- a/code/sys.driver/driver-vite/src/m.Vite.Tmpl/m.Tmpl.ts +++ b/code/sys.driver/driver-vite/src/m.Vite.Tmpl/m.Tmpl.ts @@ -1,10 +1,12 @@ import type { t } from './common.ts'; import { Bundle } from './m.Bundle.ts'; import { create } from './u.create.ts'; -import { update } from './u.update.ts'; +import { prep } from './u.prep.ts'; +import { write } from './u.write.ts'; export const ViteTmpl: t.ViteTmplLib = { Bundle, + prep, create, - update, + write, }; diff --git a/code/sys.driver/driver-vite/src/m.Vite.Tmpl/t.ts b/code/sys.driver/driver-vite/src/m.Vite.Tmpl/t.ts index 651c293acc..78c2000c5a 100644 --- a/code/sys.driver/driver-vite/src/m.Vite.Tmpl/t.ts +++ b/code/sys.driver/driver-vite/src/m.Vite.Tmpl/t.ts @@ -1,5 +1,8 @@ import type { t } from './common.ts'; +/** Index of strings representing templates variants. */ +export type ViteTmplKind = 'Default' | 'ComponentLib'; + /** * Template Library: * Create (and keep upated) a vanilla "Vite" project. @@ -11,13 +14,25 @@ export type ViteTmplLib = { /** Creates an instance of the template file generator. */ create(args?: t.ViteTmplCreateArgs): Promise<t.Tmpl>; - /** Initialize the local machine environment with latest templates */ - update(args?: t.ViteTmplUpdateArgs): Promise<t.ViteTmplUpdateResponse>; + /** Write and process the templates to the local file-system. */ + write(args?: t.ViteTmplWriteArgs): Promise<t.ViteTmplWriteResponse>; + + /** Prepare the template with latest state and dependency versions. */ + prep(options?: { silent?: boolean }): Promise<t.ViteTmplPrepResponse>; }; /** Arguments passed to the `ViteTmpl.create` method. */ export type ViteTmplCreateArgs = { version?: t.StringSemver; + tmpl?: t.ViteTmplKind; +}; + +/** + * The context object passed to the template file-processor. + */ +export type ViteTmplCtx = { + version: t.StringSemver; + tmpl: t.ViteTmplKind; }; /** @@ -32,14 +47,16 @@ export type ViteBundleLib = { }; /** Arguments passed to the `Vite.Tmpl.update` method. */ -export type ViteTmplUpdateArgs = { +export type ViteTmplWriteArgs = { force?: boolean; in?: t.StringDir; version?: t.StringSemver; + tmpl?: t.ViteTmplKind; silent?: boolean; }; -/** - * The response returned from an environment update. - */ -export type ViteTmplUpdateResponse = { readonly ops: t.TmplFileOperation[] }; +/** The response returned from an environment update. */ +export type ViteTmplWriteResponse = { readonly ops: t.TmplFileOperation[] }; + +/** The response returned from the `Vite.Tmpl.prep` method */ +export type ViteTmplPrepResponse = { readonly deps: t.EsmImportMap }; diff --git a/code/sys.driver/driver-vite/src/m.Vite.Tmpl/u.create.ts b/code/sys.driver/driver-vite/src/m.Vite.Tmpl/u.create.ts index bdfe6a69e6..32c9097268 100644 --- a/code/sys.driver/driver-vite/src/m.Vite.Tmpl/u.create.ts +++ b/code/sys.driver/driver-vite/src/m.Vite.Tmpl/u.create.ts @@ -1,4 +1,4 @@ -import { type t, Fs, PATHS, Tmpl } from './common.ts'; +import { type t, Fs, PATHS, Tmpl, pkg } from './common.ts'; import { Bundle } from './m.Bundle.ts'; import { createFileProcessor } from './u.process.file.ts'; @@ -6,12 +6,13 @@ import { createFileProcessor } from './u.process.file.ts'; * Create a new instance of the bundled file template. */ export const create: t.ViteTmplLib['create'] = async (args = {}) => { + const ctx = wrangle.ctx(args); const templatesDir = Fs.resolve(PATHS.tmpl.tmp); /** * Ensure the templates are hydrated and ready to use. */ - const beforeCopy: t.TmplCopyHandler = async () => { + const beforeWrite: t.TmplWriteHandler = async () => { await Fs.remove(templatesDir); await Bundle.writeToFile(templatesDir); }; @@ -19,11 +20,21 @@ export const create: t.ViteTmplLib['create'] = async (args = {}) => { /** * (š·) Perform additional setup here (as needed). */ - const afterCopy: t.TmplCopyHandler = async (e) => {}; + const afterWrite: t.TmplWriteHandler = async (e) => {}; /** * Template-engine instance. */ const processFile = createFileProcessor(args); - return Tmpl.create(templatesDir, { processFile, beforeCopy, afterCopy }); + return Tmpl.create(templatesDir, { processFile, beforeWrite, afterWrite, ctx }); }; + +/** + * Helpers + */ +const wrangle = { + ctx(args: t.ViteTmplCreateArgs): t.ViteTmplCtx { + const { version = pkg.version, tmpl = 'Default' } = args; + return { version, tmpl }; + }, +} as const; diff --git a/code/sys.driver/driver-vite/src/m.Vite.Tmpl/u.prep.ts b/code/sys.driver/driver-vite/src/m.Vite.Tmpl/u.prep.ts new file mode 100644 index 0000000000..086e256ac7 --- /dev/null +++ b/code/sys.driver/driver-vite/src/m.Vite.Tmpl/u.prep.ts @@ -0,0 +1,31 @@ +import { type t, c, Cli, Esm, Fs, getWorkspaceModules, Semver } from './common.ts'; + +/** + * Prepare the template with latest state, including making updates to deps/versions. + */ +export const prep: t.ViteTmplLib['prep'] = async (options = {}) => { + const { modules } = await getWorkspaceModules(); + const path = './src/-tmpl/imports.json'; + const current = (await Fs.readJson<t.DenoImportMapJson>(path)).data; + const imports = modules.latest(current?.imports ?? {}); + await Fs.writeJson(path, { ...current, imports }); + + if (!options.silent) { + const table = Cli.table([]); + Object.entries(imports).forEach(([key, value]) => { + const m = Esm.parse(value); + const pkg = c.gray(` ${key}`); + const registry = c.gray(m.registry.toUpperCase()); + const version = Semver.Fmt.colorize(m.version); + table.push([pkg, version, registry]); + }); + + console.info(); + console.info(c.italic(c.gray('imports.json'))); + console.info(c.brightGreen(`Dependencies:`)); + console.info(table.toString()); + console.info(); + } + + return { deps: imports }; +}; diff --git a/code/sys.driver/driver-vite/src/m.Vite.Tmpl/u.process.file.ts b/code/sys.driver/driver-vite/src/m.Vite.Tmpl/u.process.file.ts index 56c9c7af69..9ca2bfb19d 100644 --- a/code/sys.driver/driver-vite/src/m.Vite.Tmpl/u.process.file.ts +++ b/code/sys.driver/driver-vite/src/m.Vite.Tmpl/u.process.file.ts @@ -1,21 +1,13 @@ -import { Main } from '@sys/main/cmd'; -import { type t, c, DenoDeps, DenoFile, Esm, Fs, Path, pkg, Pkg } from './common.ts'; +import { type t, pkg } from './common.ts'; /** * File processing rules for the template. */ export function createFileProcessor(args: t.ViteTmplCreateArgs): t.TmplProcessFile { - console.log(`ā”ļøš¦š·š³š¦ šš§Øš¼āØš§« ššš§ ā ļø š„šļøš”ā⢠āāāāā`); - console.log('args', args); - - const getWorkspace = async () => { - const ws = await DenoFile.workspace(); - const deps = (await DenoDeps.from(Path.join(ws.dir, 'deps.yaml'))).data; - const modules = Esm.modules([...(deps?.modules.items ?? []), ...ws.modules.items]); - return { ws, modules }; - }; - return async (e) => { + const ctx = e.ctx as t.ViteTmplCtx; + if (!ctx) throw new Error(`Expected a {ctx} to be passed to the template file processor`); + if (e.target.exists && is.userspace(e.target.relative)) { /** * 𫵠DO NOT adjust user generated @@ -24,39 +16,18 @@ export function createFileProcessor(args: t.ViteTmplCreateArgs): t.TmplProcessFi return e.exclude('user-space'); } + if (e.contentType !== 'text') return; + if (e.target.relative === 'deno.json') { /** * Update versions in `deno.json`: */ const version = args.version ?? pkg.version; - const importUri = `jsr:${pkg.name}@${version}`; - const text = e.text.tmpl - .replace(/<ENTRY>/g, `${importUri}/main`) - .replace(/<ENTRY_SYS>/, `jsr:${Pkg.toString(Main.pkg)}`) - .replace(/<SELF_IMPORT_URI>/, importUri) - .replace(/<SELF_IMPORT_NAME>/, pkg.name); - + const entryUri = `jsr:${pkg.name}@${version}`; + const text = e.text.tmpl.replace(/<ENTRY>/g, `${entryUri}/main`); return e.modify(text); } - if (e.target.relative === 'package.json') { - const { modules } = await getWorkspace(); - const pkg = (await Fs.readJson<t.PkgJsonNode>(e.tmpl.absolute)).data; - const next = { - ...pkg, - dependencies: modules.latest(pkg?.dependencies ?? {}), - devDependencies: modules.latest(pkg?.devDependencies ?? {}), - }; - - console.info(c.gray(`Resolved versions:`)); - console.info(c.brightCyan(c.bold(`./package.json:`))); - console.info(next); - console.info(); - - const json = `${JSON.stringify(next, null, ' ')}\n`; - return e.modify(json); - } - if (e.target.file.name === '.gitignore-') { /** * Rename to ".gitignore" diff --git a/code/sys.driver/driver-vite/src/m.Vite.Tmpl/u.update.ts b/code/sys.driver/driver-vite/src/m.Vite.Tmpl/u.write.ts similarity index 71% rename from code/sys.driver/driver-vite/src/m.Vite.Tmpl/u.update.ts rename to code/sys.driver/driver-vite/src/m.Vite.Tmpl/u.write.ts index 2dd2ee990d..13068677a2 100644 --- a/code/sys.driver/driver-vite/src/m.Vite.Tmpl/u.update.ts +++ b/code/sys.driver/driver-vite/src/m.Vite.Tmpl/u.write.ts @@ -2,17 +2,17 @@ import { type t, c, Fs, Tmpl } from './common.ts'; import { create } from './u.create.ts'; /** - * Initialize the local machine environment with latest templates + * Write and process the templates to the local file-system. */ -export const update: t.ViteTmplLib['update'] = async (args = {}) => { +export const write: t.ViteTmplLib['write'] = async (args = {}) => { const { version, force = false, silent = false } = args; /** * Update template files. */ - const tmpl = await create({ version }); + const tmpl = await create({ version, tmpl: args.tmpl ?? 'Default' }); const dir = args.in ?? '.'; - const { ops } = await tmpl.copy(dir, { force }); + const { ops } = await tmpl.write(dir, { force }); /** * 𫵠Clean up helpers here (flesh out as needed: š·). diff --git a/code/sys.driver/driver-vite/src/m.Vite/-build.test.ts b/code/sys.driver/driver-vite/src/m.Vite/-build.test.ts index 40fac59a02..f292d51ec6 100644 --- a/code/sys.driver/driver-vite/src/m.Vite/-build.test.ts +++ b/code/sys.driver/driver-vite/src/m.Vite/-build.test.ts @@ -31,9 +31,10 @@ describe('Vite.build', () => { const testBuild = async (sample: t.StringDir) => { const fs = SAMPLE.fs('Vite.build'); + await Fs.copy(sample, fs.dir); + const cwd = fs.dir; - await Fs.copy(sample, cwd); - const m = await Vite.Config.fromFile(Fs.join(cwd, 'vite.config.ts')); + const fromFile = await Vite.Config.fromFile(Fs.join(cwd, 'vite.config.ts')); const res = await Vite.build({ cwd, pkg }); if (!res.ok) console.warn(res.toString()); @@ -42,7 +43,7 @@ describe('Vite.build', () => { expect(res.cmd.input).to.include('deno run'); expect(res.cmd.input).to.include('--node-modules-dir npm:vite'); expect(res.elapsed).to.be.greaterThan(0); - expect(res.paths).to.eql(m.module.paths); + expect(res.paths).to.eql(fromFile.paths); // Ensure the {pkg:name:version} data is included in the composite <digest> hash. const keys = Object.keys(res.dist.hash.parts); @@ -69,7 +70,6 @@ describe('Vite.build', () => { it('sample-1: simple', async () => { const { res, files, outDir } = await testBuild(SAMPLE.Dirs.sample1); - printHtml(files.html, 'sample-1', outDir); expect(files.html).to.include(`<title>Sample-1</title>`); expect(files.entry).to.include(`Hello World š`); diff --git a/code/sys.driver/driver-vite/src/m.Vite/t.ts b/code/sys.driver/driver-vite/src/m.Vite/t.ts index c47f4bcfcf..4b0c95db7f 100644 --- a/code/sys.driver/driver-vite/src/m.Vite/t.ts +++ b/code/sys.driver/driver-vite/src/m.Vite/t.ts @@ -41,8 +41,9 @@ export type ViteLib = { */ export type ViteBuildArgs = { cwd?: t.StringAbsoluteDir; - silent?: boolean; pkg?: t.Pkg; // Consumer module. + silent?: boolean; + spinner?: boolean; }; /** diff --git a/code/sys.driver/driver-vite/src/m.Vite/u.build.ts b/code/sys.driver/driver-vite/src/m.Vite/u.build.ts index 81c03955c0..ee79df04ad 100644 --- a/code/sys.driver/driver-vite/src/m.Vite/u.build.ts +++ b/code/sys.driver/driver-vite/src/m.Vite/u.build.ts @@ -1,4 +1,4 @@ -import { type t, Fs, Pkg, Process, Time } from './common.ts'; +import { type t, c, Cli, Fs, Pkg, Process, Time } from './common.ts'; import { Log, Wrangle } from './u.ts'; type B = t.ViteLib['build']; @@ -10,11 +10,28 @@ export const build: B = async (input) => { const timer = Time.timer(); const paths = await Wrangle.pathsFromConfigfile(input.cwd); - const { pkg, silent = true } = input; + const { pkg, silent = false } = input; const { cmd, args } = await Wrangle.command(paths, 'build'); const dir = Fs.join(paths.cwd, paths.app.outDir); + const cwd = paths.cwd; - const output = await Process.invoke({ args, silent }); + if (!silent) { + const table = Cli.table([]); + const push = (label: string, ...value: string[]) => table.push([c.gray(label), ...value]); + push('Directory:', c.gray(`${cwd.replace(/\/$/, '')}/`)); + push(' - entry:', paths.app.entry); + push(' - outDir:', paths.app.outDir); + push(' - base:', paths.app.base); + + console.info(c.bold(c.brightGreen('Paths'))); + console.info(table.toString().trim()); + console.info(); + } + + const spinner = Cli.Spinner.create('building', { silent, start: false }); + if ((input.spinner ?? true) && !silent) spinner.start(); + + const output = await Process.invoke({ cwd, args, silent: true }); const ok = output.success; if (pkg) { @@ -47,6 +64,7 @@ export const build: B = async (input) => { }, }; + spinner.stop(); return res; }; diff --git a/code/sys.driver/driver-vite/src/m.Vite/u.dev.ts b/code/sys.driver/driver-vite/src/m.Vite/u.dev.ts index 809fb98cda..738e05e0e2 100644 --- a/code/sys.driver/driver-vite/src/m.Vite/u.dev.ts +++ b/code/sys.driver/driver-vite/src/m.Vite/u.dev.ts @@ -4,12 +4,11 @@ import { Log, Wrangle } from './u.ts'; type D = t.ViteLib['dev']; -/** - * Matches (example): - * VITE v6.0.11 ready in 839 ms - */ export const REGEX = { - VITE_STARTED: /VITE v(?:\d+\.\d+\.\d+)\s+ready in\s+(\d+)\s+ms/, + /** + * Matches (example): "VITE v6.0.11 ready in 839 ms" + */ + STARTED: /VITE v(?:\d+\.\d+\.\d+)\s+ready in\s+(\d+)\s+ms/, } as const; /** @@ -32,7 +31,7 @@ export const dev: D = async (input) => { const readySignal: t.ProcReadySignalFilter = (e) => { const lines = stripAnsi(e.toString()).split('\n'); - return lines.some((line) => !!REGEX.VITE_STARTED.exec(line)); + return lines.some((line) => !!REGEX.STARTED.exec(line)); }; const proc = Process.spawn({ cwd, args, silent, readySignal, dispose$: input.dispose$ }); diff --git a/code/sys.driver/driver-vite/src/m.Vite/u.wrangle.ts b/code/sys.driver/driver-vite/src/m.Vite/u.wrangle.ts index 18be139141..cab1e64dfa 100644 --- a/code/sys.driver/driver-vite/src/m.Vite/u.wrangle.ts +++ b/code/sys.driver/driver-vite/src/m.Vite/u.wrangle.ts @@ -17,7 +17,7 @@ export const Wrangle = { const path = Path.join(rootDir, filename); const res = await ViteConfig.fromFile(path); - let paths = res.module.paths; + let paths = res.paths; if (!paths) { const err = `Failed to load paths from [${filename}], ensure it exports "paths". Source: ${path}`; diff --git a/code/sys.driver/driver-vite/src/pkg.ts b/code/sys.driver/driver-vite/src/pkg.ts index 79cb3f9c80..3ed982332f 100644 --- a/code/sys.driver/driver-vite/src/pkg.ts +++ b/code/sys.driver/driver-vite/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/driver-vite', version: '0.0.137' }; diff --git a/code/sys.driver/driver-vite/vite.config.ts b/code/sys.driver/driver-vite/vite.config.ts deleted file mode 100644 index 63c4880869..0000000000 --- a/code/sys.driver/driver-vite/vite.config.ts +++ /dev/null @@ -1,9 +0,0 @@ -/** - * @module - * š· (proxy export sample) - */ -export * from './src/-test/vite.sample-config/custom/vite.config.ts'; -export { default } from './src/-test/vite.sample-config/custom/vite.config.ts'; - -// export * from './src/-test/vite.sample-config/config.simple.ts'; -// export { default } from './src/-test/vite.sample-config/config.simple.ts'; diff --git a/code/sys.driver/driver-vitepress/-scripts/-prep.ts b/code/sys.driver/driver-vitepress/-scripts/-prep.ts index 9182a8614c..bb242d6f1c 100644 --- a/code/sys.driver/driver-vitepress/-scripts/-prep.ts +++ b/code/sys.driver/driver-vitepress/-scripts/-prep.ts @@ -1,25 +1,22 @@ -import { Vitepress } from '@sys/driver-vitepress'; -import { type t, c, DenoDeps, DenoFile, Fs, PATHS, pkg } from './common.ts'; +import { Vitepress } from '../src/mod.ts'; +import { c, Fs, PATHS, pkg, Semver } from './common.ts'; const resolve = (...parts: string[]) => Fs.join(import.meta.dirname ?? '', '..', ...parts); await Fs.remove(resolve('.tmp')); /** - * Save monorepo deps. + * Update to latest dependency versions. */ -const ws = await DenoFile.workspace(); -const deps: t.Dep[] = ws.modules.items.map((esm) => DenoDeps.toDep(esm)); - -const dir = resolve('src/-tmpl/.sys'); -await Fs.copy(Fs.join(ws.dir, 'deps.yaml'), Fs.join(dir, 'deps.yaml'), { force: true }); -await Fs.write(Fs.join(dir, 'deps.sys.yaml'), DenoDeps.toYaml(deps).text); +await Vitepress.Tmpl.prep(); /** - * Bundle files (for code-registry). + * Bundle files inline, base64-string FileMap (NB: so code can be referenfed within the registry). */ const Bundle = Vitepress.Tmpl.Bundle; await Bundle.toFilemap(); await Bundle.toFilesystem(resolve(PATHS.tmpl.tmp)); // NB: test output. -console.info(c.brightCyan('ā Prep Complete:'), `${pkg.name}@${c.brightCyan(pkg.version)}`); +const fmtVersion = Semver.Fmt.colorize(pkg.version); +const fmtModule = `${pkg.name}${c.dim('@')}${fmtVersion}`; +console.info(c.brightCyan('ā Prep Complete:'), fmtModule); console.info(); diff --git a/code/sys.driver/driver-vitepress/deno.json b/code/sys.driver/driver-vitepress/deno.json index 8d37a830b5..42f146bfee 100644 --- a/code/sys.driver/driver-vitepress/deno.json +++ b/code/sys.driver/driver-vitepress/deno.json @@ -1,21 +1,21 @@ { "name": "@sys/driver-vitepress", - "version": "0.0.284-alpha.2", + "version": "0.0.302", "license": "MIT", "tasks": { "lint": "deno lint", "dry": "deno publish --allow-dirty --dry-run", - "reset": "deno run -RWE ./-scripts/-reset.ts", - "test": "deno test -RWNE --allow-run --allow-ffi", + "test": "deno test -RWNE --allow-run --allow-ffi --allow-sys", "init": "deno run -RWNE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=init --dir=./.tmp/sample", "dev": "deno run -RWNE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=dev --dir=./.tmp/sample", "build": "deno run -RWE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=build --dir=./.tmp/sample", "serve": "deno run -RNE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=serve --dir=./.tmp/sample/dist", - "clean": "deno run -RWE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=clean --dir=./.tmp/sample", "upgrade": "deno run -RWNE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=upgrade --dir=./.tmp/sample", "backup": "deno run -RWE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=backup --dir=./.tmp/sample", "help": "deno run -RE --allow-ffi ./-scripts/-main.ts --cmd=help --dir=./.tmp/sample", "prep": "deno run -RWE --allow-ffi ./-scripts/-prep.ts", + "clean": "deno run -RWE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=clean --dir=./.tmp/sample", + "reset": "deno run -RWE ./-scripts/-reset.ts", "r": "deno task reset && deno task prep && deno task init", "rdev": "deno task r && deno task dev", "tmp": "deno run -A ./-scripts/-tmp.ts" @@ -25,7 +25,6 @@ "./t": "./src/types.ts", "./types": "./src/types.ts", "./main": "./src/-entry/-main.ts", - "./init": "./src/-entry/-init.ts", - "./ui": "./src/ui/mod.ts" + "./init": "./src/-entry/-init.ts" } } diff --git a/code/sys.driver/driver-vitepress/src/-entry/m.Entry.ts b/code/sys.driver/driver-vitepress/src/-entry/m.Entry.ts index a89b923102..072cf95b3d 100644 --- a/code/sys.driver/driver-vitepress/src/-entry/m.Entry.ts +++ b/code/sys.driver/driver-vitepress/src/-entry/m.Entry.ts @@ -8,8 +8,8 @@ import { PATHS, pkg, ViteEntry, - VitepressLog, ViteLog, + VitepressLog, } from './common.ts'; type F = t.VitepressEntryLib['main']; @@ -52,10 +52,10 @@ export const VitepressEntry: t.VitepressEntryLib = { if (args.cmd === 'build') { ViteLog.API.log({ cmd: 'build' }); console.info(); - const { dir = PATHS.inDir } = args; const res = await Vitepress.build({ inDir: dir, pkg, silent: false }); - console.info(res.toString({ pad: true })); + console.info(res.toString()); + console.info(); return; } diff --git a/code/sys.driver/driver-vitepress/src/-entry/u.init.ts b/code/sys.driver/driver-vitepress/src/-entry/u.init.ts index b482e4f8b7..17b8676832 100644 --- a/code/sys.driver/driver-vitepress/src/-entry/u.init.ts +++ b/code/sys.driver/driver-vitepress/src/-entry/u.init.ts @@ -1,5 +1,5 @@ import { Vitepress } from '../m.Vitepress/mod.ts'; -import { type t, c, PATHS, pkg, ViteLog } from './common.ts'; +import { type t, c, PATHS, pkg, Semver, ViteLog } from './common.ts'; /** * Run the initialization templates. @@ -13,13 +13,17 @@ export async function init(args: t.VitepressEntryArgsInit) { console.info(`${pkg.name} ${c.gray(pkg.version)}`); } - await Vitepress.Tmpl.update({ inDir: dir }); + await Vitepress.Tmpl.write({ inDir: dir }); if (!silent) { console.info(); ViteLog.API.log(); + + const fmtVersion = Semver.Fmt.colorize(pkg.version); + const fmtModule = `${pkg.name}${c.dim('@')}${fmtVersion}`; + console.info(); - console.info(c.brightCyan('ā Init Complete:'), `${pkg.name}@${c.brightCyan(pkg.version)}`); + console.info(c.brightCyan('ā Init Complete:'), `${fmtModule}`); console.info(); } } diff --git a/code/sys.driver/driver-vitepress/src/-test/tmpl.tests/-Global.test.ts b/code/sys.driver/driver-vitepress/src/-test/tmpl.tests/-Global.test.ts new file mode 100644 index 0000000000..22af1c9fd9 --- /dev/null +++ b/code/sys.driver/driver-vitepress/src/-test/tmpl.tests/-Global.test.ts @@ -0,0 +1,44 @@ +import { type t, describe, expect, it } from '../../-test.ts'; +import { Global } from '../../-tmpl/.sys/mod.ts'; + +import { type t as tt } from '../../-tmpl/.sys/common.ts'; + +describe('Global (State)', () => { + type G = tt.GlobalState; + + it('default (singleton)', () => { + const a = Global.state(); + const b = Global.state(); + expect(a).to.equal(b); // NB: same instance + }); + + it('custom: instance-id', () => { + const id = 'foo'; + const a = Global.state(id); + const b = Global.state(id); + const c = Global.state(); + const d = Global.state('something else'); + expect(a).to.equal(b); // NB: same instance + expect(a).to.not.equal(c); + expect(a).to.not.equal(d); + }); + + it('change', () => { + const a = Global.state(); + const b = Global.state(); + expect(a.current.tmp).to.eql(0); + a.change((d) => d.tmp++); + expect(b.current.tmp).to.eql(1); + }); + + it('shared events', () => { + const a = Global.state(); + const b = Global.state(); + const bEvents = a.events(); + const bFired: tt.GlobalStateEvent[] = []; + bEvents.changed$.pipe().subscribe((e) => bFired.push(e)); + + a.change((d) => d.tmp++); + expect(bFired[0].after).to.eql(a.current); + }); +}); diff --git a/code/sys.driver/driver-vitepress/src/-test/tmpl.tests/-Props.test.ts b/code/sys.driver/driver-vitepress/src/-test/tmpl.tests/-Props.test.ts new file mode 100644 index 0000000000..165b8f170f --- /dev/null +++ b/code/sys.driver/driver-vitepress/src/-test/tmpl.tests/-Props.test.ts @@ -0,0 +1,37 @@ +import { describe, expect, it } from '../../-test.ts'; +import { Props } from '../../-tmpl/.sys/mod.ts'; + +describe('Props', () => { + it('encode ā decode', () => { + const props = { foo: 123, bar: { msg: 'š hello' } }; + const a = Props.encode(props); + const b = Props.decode(a); + expect(a).to.include('123,34,102,111,111,34,58,49,50,51,44,34,98,97,114,34,58'); + expect(b).to.eql(props); + }); + + describe('decode', () => { + it('invalid input', () => { + const NON = [123, true, null, undefined, BigInt(0), Symbol('foo'), {}, []]; + NON.forEach((value: any) => expect(Props.decode(value)).to.eql({})); + }); + + it('empty string', () => { + const test = (input: string) => { + expect(Props.decode(input)).to.eql({}); + }; + test(''); + test(' '); + }); + + it('corrupt binary', () => { + const res = Props.decode('123,34,102,111,111,34,58'); + expect(res).to.eql({}); + }); + + it('not JSON', () => { + const res = Props.decode('foobar'); + expect(res).to.eql({}); + }); + }); +}); diff --git a/code/sys.driver/driver-vitepress/src/-test/u.assert.ts b/code/sys.driver/driver-vitepress/src/-test/u.assert.ts index b71dfc2858..23c87eb225 100644 --- a/code/sys.driver/driver-vitepress/src/-test/u.assert.ts +++ b/code/sys.driver/driver-vitepress/src/-test/u.assert.ts @@ -21,7 +21,6 @@ export const assertEnvExists = async (dir: t.StringDir, expected = true) => { '.vitepress/config.ts', '.vitepress/config.markdown.ts', '.vitepress/theme/index.ts', - '.sys/-main.ts', '.sys/components/index.ts', '.sys/components/Video.vue', '.sys/components/Video.tsx', diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/-main.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/-main.ts deleted file mode 100644 index 31dd5981c3..0000000000 --- a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/-main.ts +++ /dev/null @@ -1 +0,0 @@ -import 'jsr:@sys/driver-vitepress/main'; diff --git a/code/sys.driver/driver-vitepress/src/ui/common.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/common.ts similarity index 100% rename from code/sys.driver/driver-vitepress/src/ui/common.ts rename to code/sys.driver/driver-vitepress/src/-tmpl/.sys/common.ts diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/common/libs.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/common/libs.ts new file mode 100644 index 0000000000..27b12bc00e --- /dev/null +++ b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/common/libs.ts @@ -0,0 +1 @@ +export { Immutable, slug } from '@sys/std'; diff --git a/deploy/slc.db.team/src/common/mod.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/common/mod.ts similarity index 64% rename from deploy/slc.db.team/src/common/mod.ts rename to code/sys.driver/driver-vitepress/src/-tmpl/.sys/common/mod.ts index 1c5667a854..8cf7cac86f 100644 --- a/deploy/slc.db.team/src/common/mod.ts +++ b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/common/mod.ts @@ -1,3 +1,2 @@ -export { pkg } from '../pkg.ts'; export * from './libs.ts'; export type * as t from './t.ts'; diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/common/t.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/common/t.ts new file mode 100644 index 0000000000..3f57d44e25 --- /dev/null +++ b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/common/t.ts @@ -0,0 +1,2 @@ +export type * from '@sys/types'; +export type * from '../types.ts'; diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/components/React.Wrapper.Sample.tsx b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/components/React.Wrapper.Sample.tsx deleted file mode 100644 index 141de488de..0000000000 --- a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/components/React.Wrapper.Sample.tsx +++ /dev/null @@ -1,16 +0,0 @@ -// @ts-types="@types/react" -import React from 'react'; - -export type MyComponentProps = { - count?: number; -}; - -export const MyComponent: React.FC<MyComponentProps> = (props) => { - console.log('MyComponent.props:', props); - return ( - <div style={{ marginTop: 5, padding: 10, backgroundColor: 'rgba(255, 0, 0, 0.1)' /* RED */ }}> - <div>Hello from React š</div> - </div> - ); -}; -// diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/components/React.Wrapper.vue b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/components/React.Wrapper.vue deleted file mode 100644 index 72c089742d..0000000000 --- a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/components/React.Wrapper.vue +++ /dev/null @@ -1,14 +0,0 @@ - -<template> - <div ref="root"></div> -</template> - -<script setup lang="ts"> -import { setup, ref } from './React.setup'; -import { MyComponent } from './React.Wrapper.Sample'; - -const root = ref(); -setup(root, MyComponent, { count: 1234 }); -</script> - -<style scoped></style> diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/components/React.setup.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/components/React.setup.ts deleted file mode 100644 index 13f934e060..0000000000 --- a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/components/React.setup.ts +++ /dev/null @@ -1,31 +0,0 @@ - -import React from 'react'; -import ReactDOM from 'react-dom/client'; -import { onBeforeUnmount, onMounted, ref as vueRef, type Ref } from 'vue'; - -type O = Record<string, unknown>; - -export const ref = () => vueRef<HTMLElement | undefined>(); - -/** - * Setup a react-in-vue wrapper. - */ -export function setup<P extends O>( - refRoot: Ref<HTMLElement | undefined>, - Component: React.FC<P>, - props?: P, -) { - let root: ReactDOM.Root | undefined; - - onMounted(() => { - if (refRoot.value) { - const el = React.createElement(Component, props); - root = ReactDOM.createRoot(refRoot.value); - root.render(el); - } - }); - - onBeforeUnmount(() => { - root?.unmount(); - }); -} diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/components/Video.tsx b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/components/Video.tsx deleted file mode 100644 index c4fc667f69..0000000000 --- a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/components/Video.tsx +++ /dev/null @@ -1,27 +0,0 @@ -// @ts-types="@types/react" -import React from 'react'; - -import '@sys/tmp/sample-imports'; -import { Foo, VideoPlayer } from '@sys/tmp/ui'; - -export const DEFAULTS = { - src: 'vimeo/499921561', // Tubes. -} as const; - -export type VideoProps = { - title?: string; - src?: string; -}; - -/** - * Component - */ -export const Video: React.FC<VideoProps> = (props: VideoProps) => { - const src = props.src || DEFAULTS.src; - return ( - <div> - <Foo /> - <VideoPlayer /> - </div> - ); -}; diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/components/Video.vue b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/components/Video.vue deleted file mode 100644 index 8abf06d334..0000000000 --- a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/components/Video.vue +++ /dev/null @@ -1,20 +0,0 @@ - -<template> - <div ref="root" class="root"></div> -</template> - -<script setup lang="ts"> -import { setup, ref } from './React.setup'; -import { Video } from './Video'; - -type VideoProps = { title?: string; src?: string }; -const root = ref(); -const props = defineProps<VideoProps>(); -setup(root, Video, props); -</script> - -<style scoped> -.root { - padding-bottom: 10px; -} -</style> diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/components/index.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/components/index.ts deleted file mode 100644 index 2dbf0434e9..0000000000 --- a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/components/index.ts +++ /dev/null @@ -1,9 +0,0 @@ -import type { EnhanceAppContext } from 'vitepress'; - -import ReactWrapper from './React.Wrapper.vue'; -import Video from './Video.vue'; - -export function registerComponents(ctx: EnhanceAppContext) { - ctx.app.component('Video', Video); - ctx.app.component('ReactWrapper', ReactWrapper); -} diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/deps.sys.yaml b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/deps.sys.yaml deleted file mode 100644 index 290c766cd9..0000000000 --- a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/deps.sys.yaml +++ /dev/null @@ -1,36 +0,0 @@ -deno.json: - - import: jsr:@sys/types@0.0.82 - - import: jsr:@sys/std@0.0.131 - - import: jsr:@sys/color@0.0.35 - - import: jsr:@sys/testing@0.0.75 - - import: jsr:@sys/fs@0.0.79 - - import: jsr:@sys/cli@0.0.64 - - import: jsr:@sys/process@0.0.65 - - import: jsr:@sys/crypto@0.0.64 - - import: jsr:@sys/http@0.0.46 - - import: jsr:@sys/text@0.0.73 - - import: jsr:@sys/tmpl@0.0.79 - - import: jsr:@sys/cmd@0.0.80 - - import: jsr:@sys/jsr@0.0.45 - - import: jsr:@sys/ui-css@0.0.70 - - import: jsr:@sys/ui-dom@0.0.77 - - import: jsr:@sys/ui-react@0.0.84 - - import: jsr:@sys/ui-react-devharness@0.0.82 - - import: jsr:@sys/ui-react-components@0.0.38 - - import: jsr:@sys/driver-automerge@0.0.84 - - import: jsr:@sys/driver-deno@0.0.86 - - import: jsr:@sys/driver-immer@0.0.86 - - import: jsr:@sys/driver-obsidian@0.0.73 - - import: jsr:@sys/driver-ollama@0.0.40 - - import: jsr:@sys/driver-orbiter@0.0.63 - - import: jsr:@sys/driver-quilibrium@0.0.76 - - import: jsr:@sys/driver-vite@0.0.121 - - import: jsr:@sys/driver-vitepress@0.0.284-alpha.2 - - import: jsr:@sys/sys@0.0.55 - - import: jsr:@sys/main@0.0.57 - - import: jsr:@tdb/api@0.0.72 - - import: jsr:@tdb/slc@0.0.68 - - import: jsr:@tdb/tmp@0.0.72 - - import: jsr:@sys/name@0.0.0 - - import: jsr:@sys/tmp@0.0.95 -package.json: [] diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/deps.yaml b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/deps.yaml deleted file mode 100644 index 9bd188f371..0000000000 --- a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/deps.yaml +++ /dev/null @@ -1,117 +0,0 @@ -# -# System Dependencies ("imports") -# -# ./š¦ -# | deno.json -# |(write) ā deno.imports.json -# |(write) ā package.json -# -# This is the "single-source-of-truth" with regards to dependencies and versioning. -# Import maps (in the `deno.json` and `package.json` files) are auto-generated -# from this config definition. -# -# Also, as a programmatic API, other downstream dependencies -# (such as template generators, see `@sys/tmpl`) use this definition -# file to calculate the "latest" versions to inject into, say, -# a `package.json` file for a scaffolded project. -# - -groups: - std/deno: - # Deno standard libs ("std"). - - import: jsr:@std/async@1.0.10 - - import: jsr:@std/datetime@0.225.3 - - import: jsr:@std/dotenv@0.225.3 - - import: jsr:@std/encoding@1.0.7 - - import: jsr:@std/fs@1.0.13 - - import: jsr:@std/path@1.0.8 - - import: jsr:@std/semver@1.0.4 - - import: jsr:@std/testing@1.0.9 - - import: jsr:@std/uuid@1.0.4 - - automerge: - # https://automerge.org - - import: npm:@automerge/automerge@2.2.8 - - import: npm:@automerge/automerge-repo@1.2.1 - - import: npm:@automerge/automerge-repo-network-broadcastchannel@1.2.1 - - import: npm:@automerge/automerge-repo-storage-indexeddb@1.2.1 - - import: npm:@automerge/automerge-repo-storage-nodefs@1.2.1 - - import: npm:@onsetsoftware/automerge-patcher@0.14.0 - - crypto: - - import: npm:@noble/hashes@1.7.1 - wildcard: true - - build/tools: - - import: npm:@vitejs/plugin-react-swc@3.8.0 - - import: npm:rollup@4.34.8 - - import: npm:vite@6.1.1 - - import: npm:vite-plugin-wasm@3.4.1 - - ui/react: - - import: npm:@types/react@18.3.18 - - import: npm:@types/react-dom@18.3.5 - - import: npm:react@18.3.1 - - import: npm:react-dom@18.3.1 - -deno.json: - - group: std/deno - - group: crypto - - group: automerge - - # CLI tools - - import: jsr:@cliffy/keypress@1.0.0-rc.7 - - import: jsr:@cliffy/prompt@1.0.0-rc.7 - - import: jsr:@cliffy/table@1.0.0-rc.7 - - # Sundry: NPM - - import: npm:@types/diff@7.0.1 - - import: npm:chai@5 - - import: npm:approx-string-match@2 - - import: npm:date-fns@4 - - import: npm:subhosting@0.1.0-alpha.1 - - import: npm:diff@7 - - import: npm:fast-json-patch@3.1.1 - - import: npm:fake-indexeddb@6.0.0 - - import: npm:happy-dom@17.1.3 - - import: npm:hash-it@6.0.0 - - import: npm:ignore@7 - - import: npm:immer@10 - - import: npm:ora@8.2.0 - - import: npm:ollama@0.5.13 - - import: npm:pretty-bytes@6.1.1 - - import: npm:ramda@0.30.1 - - import: npm:rambda@9.4.2 - - import: npm:rxjs@7.8.2 - - import: npm:strip-ansi@7 - - import: npm:subhosting@0.1.0-alpha.1 - - import: npm:tinycolor2@1.6.0 - - import: npm:ts-essentials@10.0.4 - - import: npm:valibot@1.0.0-rc.1 - - import: npm:yaml@2.7.0 - - # Browser - - import: npm:csstype@3 - - import: npm:ua-parser-js@2.0.2 - - # UI - - import: npm:react-error-boundary@5 - - import: npm:react-inspector@6 - - import: npm:react-spinners@0.15.0 - -package.json: - - group: std/deno - - group: crypto - - group: build/tools - dev: true - - - import: npm:hono@4.7.2 - - # UI - - group: ui/react - - import: npm:react-icons@5.5.0 - - import: npm:@vidstack/react@1.12.12 - - # UI:Frameworks - - import: npm:vitepress@1.6.3 - - import: npm:vue@3.5.13 diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/mod.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/mod.ts new file mode 100644 index 0000000000..f137b8fe9c --- /dev/null +++ b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/mod.ts @@ -0,0 +1,2 @@ +export { Props } from './u/u.Props.ts'; +export { Global } from './u/u.Global.ts'; diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/types.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/types.ts new file mode 100644 index 0000000000..077dd7213a --- /dev/null +++ b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/types.ts @@ -0,0 +1,5 @@ +/** + * @module + * System types + */ +export type * from './u/t.ts'; diff --git a/code/sys.driver/driver-vitepress/src/ui/components/common.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/u/common.ts similarity index 100% rename from code/sys.driver/driver-vitepress/src/ui/components/common.ts rename to code/sys.driver/driver-vitepress/src/-tmpl/.sys/u/common.ts diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/u/t.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/u/t.ts new file mode 100644 index 0000000000..4b1abcee50 --- /dev/null +++ b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/u/t.ts @@ -0,0 +1,14 @@ +import type { t } from './common.ts'; + +export type GlobalState = { + tmp: number; +}; + +/** + * Immutable wrapper. + */ +type P = t.PatchOperation; +export type GlobalStateImmutable = t.ImmutableRef<t.GlobalState, P, GlobalStateEvents>; + +export type GlobalStateEvents = t.ImmutableEvents<GlobalState, P>; +export type GlobalStateEvent = t.InferImmutableEvent<t.GlobalStateEvents>; diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/u/u.Global.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/u/u.Global.ts new file mode 100644 index 0000000000..3d91316744 --- /dev/null +++ b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/u/u.Global.ts @@ -0,0 +1,27 @@ +import { type t, Immutable, slug } from './common.ts'; + +const defaultId = `default:${slug()}`; +const refs = new Map<string, t.GlobalStateImmutable>(); + +/** + * Global state interface. + */ +export const Global = { + /** + * Retrieve an instance of the global state object. + * + * @param instance - Optional unique identifier for the global state instance. + * If not provided, the default instance is returned. + * No param is the equivalent of retrieving the "singlton" instance. + * + * @returns The global state instance corresponding to the provided identifier. + */ + state(instance?: t.StringId): t.GlobalStateImmutable { + const id = instance ?? defaultId; + if (refs.has(id)) return refs.get(id)!; + + const model = Immutable.clonerRef<t.GlobalState>({ tmp: 0 }); + refs.set(id, model); + return model; + }, +} as const; diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/u/u.Props.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/u/u.Props.ts new file mode 100644 index 0000000000..f0435495d4 --- /dev/null +++ b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/u/u.Props.ts @@ -0,0 +1,23 @@ +type O = Record<string, unknown>; + +/** + * Helper for passing props between Vue and React. + */ +export const Props = { + encode<P extends O>(props: P): string { + const json = JSON.stringify(props); + const binary = new TextEncoder().encode(json); + return String(binary); + }, + + decode(encoded: string): O { + if (typeof encoded !== 'string') return {}; + try { + const binary = Uint8Array.from(encoded.split(',')); + const json = new TextDecoder().decode(binary); + return JSON.parse(json); + } catch (error) { + return {}; + } + }, +} as const; diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/ui/React.NotFound.tsx b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/ui/React.NotFound.tsx new file mode 100644 index 0000000000..50ea6f0bdd --- /dev/null +++ b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/ui/React.NotFound.tsx @@ -0,0 +1,21 @@ +import { css } from '@sys/ui-css'; +import React from 'react'; + +export type NotFoundProps = {}; + +export const NotFound: React.FC<NotFoundProps> = (props) => { + const {} = props; + + const styles = { + base: css({ + backgroundColor: 'rgba(255, 0, 0, 0.1)' /* RED */, + padding: 10, + }), + }; + + return ( + <div className={styles.base.class}> + <div>{`š· Component Not Found`}</div> + </div> + ); +}; diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/ui/React.components.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/ui/React.components.ts new file mode 100644 index 0000000000..32f727ed08 --- /dev/null +++ b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/ui/React.components.ts @@ -0,0 +1,26 @@ +import React from 'react'; + +/** + * Component factory lookup. + */ +export async function lookup(kind: string): Promise<React.FC | undefined> { + if (kind === 'sys/tmp/ui:Foo') { + const { Foo } = await import('@sys/tmp/ui'); + return Foo; + } + + const players = ['ConceptPlayer', 'VideoPlayer', 'Panel']; + + if (players.includes(kind)) { + const { Player } = await import('@sys/ui-react-components'); + if (kind === 'ConceptPlayer') return Player.Concept.View; + if (kind === 'VideoPlayer') return Player.Video.View; + } + + if (kind === 'Panel') { + const { Panel } = await import('@sys/ui-react-components'); + return Panel; + } + + return; // NB: no-match. +} diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/ui/React.setup.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/ui/React.setup.ts new file mode 100644 index 0000000000..97f72617d8 --- /dev/null +++ b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/ui/React.setup.ts @@ -0,0 +1,30 @@ +import React from 'react'; +import { createRoot, type Root } from 'react-dom/client'; +import { onBeforeUnmount, onMounted, ref as vueRef, type Ref } from 'vue'; +import { NotFound } from './React.NotFound.tsx'; + +type O = Record<string, unknown>; +export const ref = () => vueRef<HTMLElement | undefined>(); + +/** + * Setup a <react-in-vue> wrapper component. + */ +export function setup<P extends O>( + refRoot: Ref<HTMLElement | undefined>, + props: P, + getComponent: () => Promise<React.FC<P> | undefined>, +) { + type C = React.FC<P>; + let root: Root | undefined; + + onMounted(async () => { + if (refRoot.value) { + const Component = ((await getComponent()) || NotFound) as C; + const el = React.createElement(Component, props); + root = createRoot(refRoot.value); + root.render(el); + } + }); + + onBeforeUnmount(() => root?.unmount()); +} diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/ui/React.vue b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/ui/React.vue new file mode 100644 index 0000000000..88efa40c5d --- /dev/null +++ b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/ui/React.vue @@ -0,0 +1,19 @@ +<template> + <div ref="root"></div> +</template> + +<script setup lang="ts"> +import { Props } from '../u/u.Props'; +import { lookup } from './React.components'; +import { ref, setup } from './React.setup'; + +type Base64EncodedJsonString = string; +type InputProps = { kind: string; props?: Base64EncodedJsonString }; + +const root = ref(); +const input = defineProps<InputProps>(); +const props = Props.decode(input.props ?? ''); +setup(root, props, async () => lookup(input.kind)); +</script> + +<style scoped></style> diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.sys/ui/mod.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/ui/mod.ts new file mode 100644 index 0000000000..ae254fc0d0 --- /dev/null +++ b/code/sys.driver/driver-vitepress/src/-tmpl/.sys/ui/mod.ts @@ -0,0 +1,6 @@ +import type { EnhanceAppContext } from 'vitepress'; +import React from './React.vue'; + +export function registerComponents(ctx: EnhanceAppContext) { + ctx.app.component('React', React); +} diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.vitepress/config.alias.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.vitepress/config.alias.ts deleted file mode 100644 index 5f2a833357..0000000000 --- a/code/sys.driver/driver-vitepress/src/-tmpl/.vitepress/config.alias.ts +++ /dev/null @@ -1,52 +0,0 @@ -import type { Alias } from 'vite'; - -import { DenoDeps } from '@sys/driver-deno/runtime'; -import { ViteConfig } from '@sys/driver-vite'; -import { Err, Path, R } from '@sys/std'; - -/** - * Generate "import" statement alias map for: - * - * - jsr:@sys System modules within the workspace. - * - npm:<deps> The upstream dependencies imported from the NPM registry. - */ -export async function getAliases() { - const deps = (await loadDeps('./.sys/deps.yaml')).deps; - const depsSys = (await loadDeps('./.sys/deps.sys.yaml')).deps; - - const isSys = (name: string) => name.startsWith('@sys/'); - const ws = await ViteConfig.workspace(); - const a = depsSys.map((m) => m.name).filter(isSys); - const b = ws.modules.items.map((m) => m.name).filter(isSys); - const isSystemMonorepo = a.every((item) => b.includes(item)); - if (isSystemMonorepo) return ws.aliases; - - const modules = [...deps, ...depsSys]; - const aliases = modules - .filter((m) => !!m.registry) - .filter((m) => m.version !== '0.0.0') - .map((m) => ViteConfig.alias(m.registry, m.name)); - - const unique = R.uniqBy((item: Alias) => `${item.replacement}:${item.find.toString()}`); - return unique(aliases); -} - -/** - * Helpers - */ -async function loadDeps(path: string) { - const errors = Err.errors(); - path = Path.resolve(import.meta.dirname ?? '', '..', path); - - const res = await DenoDeps.from(path); - if (res.error || !res.data?.deps) { - const err = `Failed to load system dependencies from: ${path}`; - console.warn(err, { cause: res.error }); - errors.push(err); - } - - return { - deps: (res.data?.deps ?? []).map((d) => d.module), - error: errors.toError(), - } as const; -} diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.vitepress/config.markdown.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.vitepress/config.markdown.ts index 4952b6a5ab..2eddaf7fde 100644 --- a/code/sys.driver/driver-vitepress/src/-tmpl/.vitepress/config.markdown.ts +++ b/code/sys.driver/driver-vitepress/src/-tmpl/.vitepress/config.markdown.ts @@ -1,42 +1,34 @@ import type { MarkdownRenderer } from 'vitepress'; import { parse } from 'yaml'; +import { Props } from '../.sys/mod.ts'; export const markdown = { config(md: MarkdownRenderer) { const rules = md.renderer.rules; - // Store the original fence rule. - const defaultFence = - rules.fence || - function (tokens, idx, options, _env, self) { - return self.renderToken(tokens, idx, options); - }; + // NB: Store the original fence rule. + const originalFenceRule = + rules.fence || ((tokens, idx, options, env, self) => self.renderToken(tokens, idx, options)); // Override the fence rule looking for YAML structures. rules.fence = (tokens, idx, options, env, self) => { const token = tokens[idx]; + const renderOriginal = () => originalFenceRule(tokens, idx, options, env, self); if (token.info.trim() === 'yaml') { const yaml = parse(token.content) || undefined; - if (yaml?.component === 'Video') { - const defaultHtml = defaultFence(tokens, idx, options, env, self); - const src = yaml.src || ''; - let html = `<Video src="${src}"/>`; - if (yaml.debug) html = `${html}\n${defaultHtml}`; - return html; - } + const formatHtml = (html: string) => (yaml.debug ? `${html}\n${renderOriginal()}` : html); - if (yaml?.component === 'ConceptPlayer') { - const defaultHtml = defaultFence(tokens, idx, options, env, self); - const src = yaml.video || ''; - let html = `<Video src="${src}"/>`; - if (yaml.debug) html = `${html}\n${defaultHtml}`; - return html; + if (typeof yaml?.component === 'string') { + const data = { ...yaml }; + delete data.component; + const html = `<React kind="${yaml.component}" props="${Props.encode(data)}" />`; + return formatHtml(html); } } - // No overriden matches found, return default rendering. - return defaultFence(tokens, idx, options, env, self); + // No match found, return default. + return renderOriginal(); }; }, }; diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.vitepress/config.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.vitepress/config.ts index 510baefefb..58cd8feea0 100644 --- a/code/sys.driver/driver-vitepress/src/-tmpl/.vitepress/config.ts +++ b/code/sys.driver/driver-vitepress/src/-tmpl/.vitepress/config.ts @@ -1,21 +1,31 @@ import type { ConfigEnv } from 'vite'; +import { ViteConfig } from '@sys/driver-vite'; import { defineConfig } from 'vitepress'; + import { Config } from '../src/config.ts'; import { sidebar } from '../src/nav.ts'; -import { getAliases } from './config.alias.ts'; import { markdown } from './config.markdown.ts'; export default async (env: ConfigEnv) => { const { title, description } = Config; - const alias = (await getAliases()) as any; // NB: type-hack ("vitepress" vs. "vite" fighting). + const ws = await ViteConfig.workspace(); + const alias = ws.aliases; return defineConfig({ title, description, - srcDir: '<SRC_DIR>', + base: '/', + srcDir: './docs', markdown, - themeConfig: { sidebar, search: { provider: 'local' } }, - vite: { resolve: { alias }, plugins: [] }, + themeConfig: { + sidebar, + // search: { provider: 'local' }, + }, + appearance: false, // NB: "light/dark" mode switch. + vite: { + plugins: [], + resolve: { alias }, + }, }); }; diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.vitepress/theme/Layout.vue b/code/sys.driver/driver-vitepress/src/-tmpl/.vitepress/theme/Layout.vue new file mode 100644 index 0000000000..c9fdecc1bc --- /dev/null +++ b/code/sys.driver/driver-vitepress/src/-tmpl/.vitepress/theme/Layout.vue @@ -0,0 +1,29 @@ +<template> + <Layout> + <template v-for="slot in slots" #[slot]> + <React kind="Panel" /> + </template> + </Layout> +</template> + +<script setup> +import DefaultTheme from 'vitepress/theme'; +const { Layout } = DefaultTheme; + +/** + * Docs: https://vitepress.dev/guide/extending-default-theme#layout-slots + */ +const slots = [ + 'nav-bar-title-before', + 'nav-bar-title-after', + 'nav-bar-content-before', + 'nav-bar-content-after', + 'nav-screen-content-before', + 'nav-screen-content-after', + 'sidebar-nav-before', + 'sidebar-nav-after', + 'aside-top', + 'aside-outline-before', + 'aside-outline-after', +]; +</script> diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/.vitepress/theme/index.ts b/code/sys.driver/driver-vitepress/src/-tmpl/.vitepress/theme/index.ts index 25c9325c16..bd91712324 100644 --- a/code/sys.driver/driver-vitepress/src/-tmpl/.vitepress/theme/index.ts +++ b/code/sys.driver/driver-vitepress/src/-tmpl/.vitepress/theme/index.ts @@ -1,10 +1,12 @@ import type { Theme as ThemeType } from 'vitepress'; import DefaultTheme from 'vitepress/theme'; -import { registerComponents as registerSystemComponents } from '../../.sys/components/index.ts'; +import { registerComponents as registerSystemComponents } from '../../.sys/ui/mod.ts'; +import Layout from './Layout.vue'; export const Theme: ThemeType = { extends: DefaultTheme, + Layout, enhanceApp(ctx) { registerSystemComponents(ctx); }, diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/deno.json b/code/sys.driver/driver-vitepress/src/-tmpl/deno.json index 80f651cc2c..314dc467d5 100644 --- a/code/sys.driver/driver-vitepress/src/-tmpl/deno.json +++ b/code/sys.driver/driver-vitepress/src/-tmpl/deno.json @@ -1,14 +1,13 @@ { "version": "0.0.0", "tasks": { - "dev": "deno run -RWNE --allow-run --allow-ffi <ENTRY> --cmd=dev", - "build": "deno run -RWE --allow-run --allow-ffi <ENTRY> --cmd=build", - "serve": "deno run -RNE --allow-run --allow-ffi <ENTRY> --cmd=serve", - "clean": "deno run -RWE --allow-ffi <ENTRY> --cmd=clean", - "upgrade": "deno run -RWNE --allow-run --allow-ffi <ENTRY> --cmd=upgrade", - "backup": "deno run -RWE --allow-run --allow-ffi <ENTRY> --cmd=backup", - "help": "deno run -RE --allow-ffi <ENTRY> --cmd=help", - "sys": "deno run -RWNE <ENTRY_SYS>" + "dev": "deno run -RWNE --allow-run --allow-ffi <ENTRY> --cmd=dev", + "build": "deno run -RWE --allow-run --allow-ffi <ENTRY> --cmd=build", + "serve": "deno run -RNE --allow-run --allow-ffi <ENTRY> --cmd=serve", + "clean": "deno run -RWE --allow-ffi <ENTRY> --cmd=clean", + "upgrade": "deno run -RWNE --allow-run --allow-ffi <ENTRY> --cmd=upgrade", + "backup": "deno run -RWE --allow-run --allow-ffi <ENTRY> --cmd=backup", + "help": "deno run -RE --allow-ffi <ENTRY> --cmd=help" }, "compilerOptions": { "strict": true, @@ -18,9 +17,5 @@ }, "workspace": [], "nodeModulesDir": "auto", - "imports": { - "@sys/driver-deno": "<DRIVER_DENO>", - "@sys/driver-vite": "<DRIVER_VITE>", - "@sys/driver-vitepress": "<DRIVER_VITEPRESS>" - } + "importMap": "./imports.json" } diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/docs/index.md b/code/sys.driver/driver-vitepress/src/-tmpl/docs/index.md index 59a89e4200..939a4ab1ce 100644 --- a/code/sys.driver/driver-vitepress/src/-tmpl/docs/index.md +++ b/code/sys.driver/driver-vitepress/src/-tmpl/docs/index.md @@ -19,11 +19,15 @@ notes: | debug: true component: ConceptPlayer video: vimeo/727951677 +thumbnails: true timestamps: + '00:00:00.000': + image: /images/volcano.jpg '00:03:58.215': image: https://wrpcd.net/cdn-cgi/imagedelivery/BXluQx4ige9GuW0Ia56BHw/28f5b7ed-67d1-419d-8db0-d95ae90e8100/rectcontain3 ``` + ```yaml diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/docs/public/images/volcano.jpg b/code/sys.driver/driver-vitepress/src/-tmpl/docs/public/images/volcano.jpg new file mode 100644 index 0000000000..ab0cef116f Binary files /dev/null and b/code/sys.driver/driver-vitepress/src/-tmpl/docs/public/images/volcano.jpg differ diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/imports.json b/code/sys.driver/driver-vitepress/src/-tmpl/imports.json new file mode 100644 index 0000000000..ee0fdf438d --- /dev/null +++ b/code/sys.driver/driver-vitepress/src/-tmpl/imports.json @@ -0,0 +1,9 @@ +{ + "imports": { + "@sys/driver-deno": "jsr:@sys/driver-deno@0.0.97", + "@sys/driver-vite": "jsr:@sys/driver-vite@0.0.137", + "@sys/driver-vitepress": "jsr:@sys/driver-vitepress@0.0.302", + "@deno/vite-plugin": "npm:@deno/vite-plugin@1.0.4", + "@vitejs/plugin-react-swc": "npm:@vitejs/plugin-react-swc@3.8.0" + } +} diff --git a/code/sys.driver/driver-vitepress/src/-tmpl/package.json b/code/sys.driver/driver-vitepress/src/-tmpl/package.json index b0cd38d014..96831fd6f2 100644 --- a/code/sys.driver/driver-vitepress/src/-tmpl/package.json +++ b/code/sys.driver/driver-vitepress/src/-tmpl/package.json @@ -1,17 +1,18 @@ { "dependencies": { - "@vidstack/react": "", - "@sys/std": "npm:@jsr/sys__std", - "@sys/tmp": "npm:@jsr/sys__tmp", - "@sys/ui-css": "npm:@jsr/sys__ui-css", - "react": "", - "react-dom": "", - "yaml": "" + "@vidstack/react": "npm:@vidstack/react@1.12.12", + "@sys/std": "npm:@jsr/sys__std@0.0.141", + "@sys/tmp": "npm:@jsr/sys__tmp@0.0.110", + "@sys/ui-css": "npm:@jsr/sys__ui-css@0.0.81", + "react": "npm:react@18.3.1", + "react-dom": "npm:react-dom@18.3.1", + "yaml": "npm:yaml@2.7.0" }, "devDependencies": { - "@types/react": "", - "@types/react-dom": "", - "vitepress": "", - "vue": "" + "@types/react": "npm:@types/react@18.3.18", + "@types/react-dom": "npm:@types/react-dom@18.3.5", + "vite": "npm:vite@6.1.1", + "vitepress": "npm:vitepress@1.6.3", + "vue": "npm:vue@3.5.13" } } diff --git a/code/sys.driver/driver-vitepress/src/common/libs.ts b/code/sys.driver/driver-vitepress/src/common/libs.ts index 7530c9bb48..ab907d1152 100644 --- a/code/sys.driver/driver-vitepress/src/common/libs.ts +++ b/code/sys.driver/driver-vitepress/src/common/libs.ts @@ -7,6 +7,7 @@ export { Process } from '@sys/process'; export { Args, Date, Err, Is, rx, slug, Str, Time } from '@sys/std'; export { Esm } from '@sys/std/esm'; export { Ignore } from '@sys/std/ignore'; +export { Semver } from '@sys/std/semver/server'; export { HashFmt } from '@sys/crypto/fmt'; export { Hash } from '@sys/crypto/hash'; diff --git a/code/sys.driver/driver-vitepress/src/common/mod.ts b/code/sys.driver/driver-vitepress/src/common/mod.ts index 5bfc83caa3..694597b1d7 100644 --- a/code/sys.driver/driver-vitepress/src/common/mod.ts +++ b/code/sys.driver/driver-vitepress/src/common/mod.ts @@ -1,8 +1,9 @@ export { pkg } from '../pkg.ts'; -export * from './libs.ts'; - export type * as t from './t.ts'; +export * from './libs.ts'; +export * from './u.workspace.ts'; + export const PATHS = { inDir: './', srcDir: './docs', diff --git a/code/sys.driver/driver-vitepress/src/common/t.ts b/code/sys.driver/driver-vitepress/src/common/t.ts index 92b4f5e820..8573e04e35 100644 --- a/code/sys.driver/driver-vitepress/src/common/t.ts +++ b/code/sys.driver/driver-vitepress/src/common/t.ts @@ -1,10 +1,10 @@ export type * from '@sys/driver-vite/t'; export type * from '@sys/types/t'; -export type { DenoFileJson, DenoModuleBackup, Dep } from '@sys/driver-deno/t'; +export type { DenoFileJson, DenoImportMapJson, DenoModuleBackup, Dep } from '@sys/driver-deno/t'; export type { DirSnapshot, FileMap, FsPathFilter } from '@sys/fs/t'; export type { ProcHandle, ProcReadySignalFilter } from '@sys/process/t'; export type { EsmModules, Ignore } from '@sys/std/t'; -export type { Tmpl, TmplCopyHandler, TmplFileOperation, TmplProcessFile } from '@sys/tmpl/t'; +export type { Tmpl, TmplWriteHandler, TmplFileOperation, TmplProcessFile } from '@sys/tmpl/t'; export type * from '../types.ts'; diff --git a/code/sys.driver/driver-vitepress/src/common/u.workspace.ts b/code/sys.driver/driver-vitepress/src/common/u.workspace.ts new file mode 100644 index 0000000000..146c44d576 --- /dev/null +++ b/code/sys.driver/driver-vitepress/src/common/u.workspace.ts @@ -0,0 +1,18 @@ +import { DenoDeps, DenoFile, Path, Esm } from './libs.ts'; + +/** + * Find and load the latest workspace data. + */ +export async function getWorkspaceModules() { + const ws = await DenoFile.workspace(); + const deps = (await DenoDeps.from(Path.join(ws.dir, 'deps.yaml'))).data; + const modules = Esm.modules([...(deps?.modules.items ?? []), ...ws.modules.items]); + return { + get ws() { + return ws; + }, + get modules() { + return modules; + }, + } as const; +} diff --git a/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/-bundle.json b/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/-bundle.json index e1ab17915d..5da74aed66 100644 --- a/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/-bundle.json +++ b/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/-bundle.json @@ -1,26 +1,34 @@ { ".gitignore-": "data:text/plain;base64,bm9kZV9tb2R1bGVzCmRpc3QKLnRtcAoKLnN5cwoudnNjb2RlCi52aXRlcHJlc3MKCi1iYWNrdXAKLWJhY2t1cC8qKgoudml0ZXByZXNzL2NhY2hlLyoqCi52aXRlcHJlc3MvZGlzdC8qKgo=", ".npmrc": "data:text/plain;base64,IyBEb2NzOiBodHRwczovL2pzci5pby9kb2NzL25wbS1jb21wYXRpYmlsaXR5I2luc3RhbGxpbmctYW5kLXVzaW5nLWpzci1wYWNrYWdlcwpAanNyOnJlZ2lzdHJ5PWh0dHBzOi8vbnBtLmpzci5pbwo=", - ".sys/-main.ts": "data:application/typescript;base64,aW1wb3J0ICdqc3I6QHN5cy9kcml2ZXItdml0ZXByZXNzL21haW4nOwo=", - ".sys/components/React.Wrapper.Sample.tsx": "data:application/typescript+jsx;base64,Ly8gQHRzLXR5cGVzPSJAdHlwZXMvcmVhY3QiCmltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCc7CgpleHBvcnQgdHlwZSBNeUNvbXBvbmVudFByb3BzID0gewogIGNvdW50PzogbnVtYmVyOwp9OwoKZXhwb3J0IGNvbnN0IE15Q29tcG9uZW50OiBSZWFjdC5GQzxNeUNvbXBvbmVudFByb3BzPiA9IChwcm9wcykgPT4gewogIGNvbnNvbGUubG9nKCdNeUNvbXBvbmVudC5wcm9wczonLCBwcm9wcyk7CiAgcmV0dXJuICgKICAgIDxkaXYgc3R5bGU9e3sgbWFyZ2luVG9wOiA1LCBwYWRkaW5nOiAxMCwgYmFja2dyb3VuZENvbG9yOiAncmdiYSgyNTUsIDAsIDAsIDAuMSknIC8qIFJFRCAqLyB9fT4KICAgICAgPGRpdj5IZWxsbyBmcm9tIFJlYWN0IPCfkYs8L2Rpdj4KICAgIDwvZGl2PgogICk7Cn07Ci8vCg==", - ".sys/components/React.Wrapper.vue": "data:text/plain;base64,Cjx0ZW1wbGF0ZT4KICA8ZGl2IHJlZj0icm9vdCI+PC9kaXY+CjwvdGVtcGxhdGU+Cgo8c2NyaXB0IHNldHVwIGxhbmc9InRzIj4KaW1wb3J0IHsgc2V0dXAsIHJlZiB9IGZyb20gJy4vUmVhY3Quc2V0dXAnOwppbXBvcnQgeyBNeUNvbXBvbmVudCB9IGZyb20gJy4vUmVhY3QuV3JhcHBlci5TYW1wbGUnOwoKY29uc3Qgcm9vdCA9IHJlZigpOwpzZXR1cChyb290LCBNeUNvbXBvbmVudCwgeyBjb3VudDogMTIzNCB9KTsKPC9zY3JpcHQ+Cgo8c3R5bGUgc2NvcGVkPjwvc3R5bGU+Cg==", - ".sys/components/React.setup.ts": "data:application/typescript;base64,CmltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCc7CmltcG9ydCBSZWFjdERPTSBmcm9tICdyZWFjdC1kb20vY2xpZW50JzsKaW1wb3J0IHsgb25CZWZvcmVVbm1vdW50LCBvbk1vdW50ZWQsIHJlZiBhcyB2dWVSZWYsIHR5cGUgUmVmIH0gZnJvbSAndnVlJzsKCnR5cGUgTyA9IFJlY29yZDxzdHJpbmcsIHVua25vd24+OwoKZXhwb3J0IGNvbnN0IHJlZiA9ICgpID0+IHZ1ZVJlZjxIVE1MRWxlbWVudCB8IHVuZGVmaW5lZD4oKTsKCi8qKgogKiBTZXR1cCBhIHJlYWN0LWluLXZ1ZSB3cmFwcGVyLgogKi8KZXhwb3J0IGZ1bmN0aW9uIHNldHVwPFAgZXh0ZW5kcyBPPigKICByZWZSb290OiBSZWY8SFRNTEVsZW1lbnQgfCB1bmRlZmluZWQ+LAogIENvbXBvbmVudDogUmVhY3QuRkM8UD4sCiAgcHJvcHM/OiBQLAopIHsKICBsZXQgcm9vdDogUmVhY3RET00uUm9vdCB8IHVuZGVmaW5lZDsKCiAgb25Nb3VudGVkKCgpID0+IHsKICAgIGlmIChyZWZSb290LnZhbHVlKSB7CiAgICAgIGNvbnN0IGVsID0gUmVhY3QuY3JlYXRlRWxlbWVudChDb21wb25lbnQsIHByb3BzKTsKICAgICAgcm9vdCA9IFJlYWN0RE9NLmNyZWF0ZVJvb3QocmVmUm9vdC52YWx1ZSk7CiAgICAgIHJvb3QucmVuZGVyKGVsKTsKICAgIH0KICB9KTsKCiAgb25CZWZvcmVVbm1vdW50KCgpID0+IHsKICAgIHJvb3Q/LnVubW91bnQoKTsKICB9KTsKfQo=", - ".sys/components/Video.tsx": "data:application/typescript+jsx;base64,Ly8gQHRzLXR5cGVzPSJAdHlwZXMvcmVhY3QiCmltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCc7CgppbXBvcnQgJ0BzeXMvdG1wL3NhbXBsZS1pbXBvcnRzJzsKaW1wb3J0IHsgRm9vLCBWaWRlb1BsYXllciB9IGZyb20gJ0BzeXMvdG1wL3VpJzsKCmV4cG9ydCBjb25zdCBERUZBVUxUUyA9IHsKICBzcmM6ICd2aW1lby80OTk5MjE1NjEnLCAvLyBUdWJlcy4KfSBhcyBjb25zdDsKCmV4cG9ydCB0eXBlIFZpZGVvUHJvcHMgPSB7CiAgdGl0bGU/OiBzdHJpbmc7CiAgc3JjPzogc3RyaW5nOwp9OwoKLyoqCiAqIENvbXBvbmVudAogKi8KZXhwb3J0IGNvbnN0IFZpZGVvOiBSZWFjdC5GQzxWaWRlb1Byb3BzPiA9IChwcm9wczogVmlkZW9Qcm9wcykgPT4gewogIGNvbnN0IHNyYyA9IHByb3BzLnNyYyB8fCBERUZBVUxUUy5zcmM7CiAgcmV0dXJuICgKICAgIDxkaXY+CiAgICAgIDxGb28gLz4KICAgICAgPFZpZGVvUGxheWVyIC8+CiAgICA8L2Rpdj4KICApOwp9Owo=", - ".sys/components/Video.vue": "data:text/plain;base64,Cjx0ZW1wbGF0ZT4KICA8ZGl2IHJlZj0icm9vdCIgY2xhc3M9InJvb3QiPjwvZGl2Pgo8L3RlbXBsYXRlPgoKPHNjcmlwdCBzZXR1cCBsYW5nPSJ0cyI+CmltcG9ydCB7IHNldHVwLCByZWYgfSBmcm9tICcuL1JlYWN0LnNldHVwJzsKaW1wb3J0IHsgVmlkZW8gfSBmcm9tICcuL1ZpZGVvJzsKCnR5cGUgVmlkZW9Qcm9wcyA9IHsgdGl0bGU/OiBzdHJpbmc7IHNyYz86IHN0cmluZyB9Owpjb25zdCByb290ID0gcmVmKCk7CmNvbnN0IHByb3BzID0gZGVmaW5lUHJvcHM8VmlkZW9Qcm9wcz4oKTsKc2V0dXAocm9vdCwgVmlkZW8sIHByb3BzKTsKPC9zY3JpcHQ+Cgo8c3R5bGUgc2NvcGVkPgoucm9vdCB7CiAgcGFkZGluZy1ib3R0b206IDEwcHg7Cn0KPC9zdHlsZT4K", - ".sys/components/index.ts": "data:application/typescript;base64,aW1wb3J0IHR5cGUgeyBFbmhhbmNlQXBwQ29udGV4dCB9IGZyb20gJ3ZpdGVwcmVzcyc7CgppbXBvcnQgUmVhY3RXcmFwcGVyIGZyb20gJy4vUmVhY3QuV3JhcHBlci52dWUnOwppbXBvcnQgVmlkZW8gZnJvbSAnLi9WaWRlby52dWUnOwoKZXhwb3J0IGZ1bmN0aW9uIHJlZ2lzdGVyQ29tcG9uZW50cyhjdHg6IEVuaGFuY2VBcHBDb250ZXh0KSB7CiAgY3R4LmFwcC5jb21wb25lbnQoJ1ZpZGVvJywgVmlkZW8pOwogIGN0eC5hcHAuY29tcG9uZW50KCdSZWFjdFdyYXBwZXInLCBSZWFjdFdyYXBwZXIpOwp9Cg==", - ".sys/deps.sys.yaml": "data:text/plain;base64,ZGVuby5qc29uOgogIC0gaW1wb3J0OiBqc3I6QHN5cy90eXBlc0AwLjAuODIKICAtIGltcG9ydDoganNyOkBzeXMvc3RkQDAuMC4xMzEKICAtIGltcG9ydDoganNyOkBzeXMvY29sb3JAMC4wLjM1CiAgLSBpbXBvcnQ6IGpzcjpAc3lzL3Rlc3RpbmdAMC4wLjc1CiAgLSBpbXBvcnQ6IGpzcjpAc3lzL2ZzQDAuMC43OQogIC0gaW1wb3J0OiBqc3I6QHN5cy9jbGlAMC4wLjY0CiAgLSBpbXBvcnQ6IGpzcjpAc3lzL3Byb2Nlc3NAMC4wLjY1CiAgLSBpbXBvcnQ6IGpzcjpAc3lzL2NyeXB0b0AwLjAuNjQKICAtIGltcG9ydDoganNyOkBzeXMvaHR0cEAwLjAuNDYKICAtIGltcG9ydDoganNyOkBzeXMvdGV4dEAwLjAuNzMKICAtIGltcG9ydDoganNyOkBzeXMvdG1wbEAwLjAuNzkKICAtIGltcG9ydDoganNyOkBzeXMvY21kQDAuMC44MAogIC0gaW1wb3J0OiBqc3I6QHN5cy9qc3JAMC4wLjQ1CiAgLSBpbXBvcnQ6IGpzcjpAc3lzL3VpLWNzc0AwLjAuNzAKICAtIGltcG9ydDoganNyOkBzeXMvdWktZG9tQDAuMC43NwogIC0gaW1wb3J0OiBqc3I6QHN5cy91aS1yZWFjdEAwLjAuODQKICAtIGltcG9ydDoganNyOkBzeXMvdWktcmVhY3QtZGV2aGFybmVzc0AwLjAuODIKICAtIGltcG9ydDoganNyOkBzeXMvdWktcmVhY3QtY29tcG9uZW50c0AwLjAuMzgKICAtIGltcG9ydDoganNyOkBzeXMvZHJpdmVyLWF1dG9tZXJnZUAwLjAuODQKICAtIGltcG9ydDoganNyOkBzeXMvZHJpdmVyLWRlbm9AMC4wLjg2CiAgLSBpbXBvcnQ6IGpzcjpAc3lzL2RyaXZlci1pbW1lckAwLjAuODYKICAtIGltcG9ydDoganNyOkBzeXMvZHJpdmVyLW9ic2lkaWFuQDAuMC43MwogIC0gaW1wb3J0OiBqc3I6QHN5cy9kcml2ZXItb2xsYW1hQDAuMC40MAogIC0gaW1wb3J0OiBqc3I6QHN5cy9kcml2ZXItb3JiaXRlckAwLjAuNjMKICAtIGltcG9ydDoganNyOkBzeXMvZHJpdmVyLXF1aWxpYnJpdW1AMC4wLjc2CiAgLSBpbXBvcnQ6IGpzcjpAc3lzL2RyaXZlci12aXRlQDAuMC4xMjEKICAtIGltcG9ydDoganNyOkBzeXMvZHJpdmVyLXZpdGVwcmVzc0AwLjAuMjg0LWFscGhhLjIKICAtIGltcG9ydDoganNyOkBzeXMvc3lzQDAuMC41NQogIC0gaW1wb3J0OiBqc3I6QHN5cy9tYWluQDAuMC41NwogIC0gaW1wb3J0OiBqc3I6QHRkYi9hcGlAMC4wLjcyCiAgLSBpbXBvcnQ6IGpzcjpAdGRiL3NsY0AwLjAuNjgKICAtIGltcG9ydDoganNyOkB0ZGIvdG1wQDAuMC43MgogIC0gaW1wb3J0OiBqc3I6QHN5cy9uYW1lQDAuMC4wCiAgLSBpbXBvcnQ6IGpzcjpAc3lzL3RtcEAwLjAuOTUKcGFja2FnZS5qc29uOiBbXQo=", - ".sys/deps.yaml": "data:text/plain;base64,IwojICBTeXN0ZW0gRGVwZW5kZW5jaWVzICgiaW1wb3J0cyIpCiMKIyAgICAgICAgLi/wn5KmCiMgICAgICAgICAgfCAgICAgICAgICAgIGRlbm8uanNvbgojICAgICAgICAgIHwod3JpdGUpICDihpIgIGRlbm8uaW1wb3J0cy5qc29uCiMgICAgICAgICAgfCh3cml0ZSkgIOKGkiAgcGFja2FnZS5qc29uCiMKIyAgVGhpcyBpcyB0aGUgInNpbmdsZS1zb3VyY2Utb2YtdHJ1dGgiIHdpdGggcmVnYXJkcyB0byBkZXBlbmRlbmNpZXMgYW5kIHZlcnNpb25pbmcuCiMgIEltcG9ydCBtYXBzIChpbiB0aGUgYGRlbm8uanNvbmAgYW5kIGBwYWNrYWdlLmpzb25gIGZpbGVzKSBhcmUgYXV0by1nZW5lcmF0ZWQKIyAgZnJvbSB0aGlzIGNvbmZpZyBkZWZpbml0aW9uLgojCiMgIEFsc28sIGFzIGEgcHJvZ3JhbW1hdGljIEFQSSwgb3RoZXIgZG93bnN0cmVhbSBkZXBlbmRlbmNpZXMKIyAgKHN1Y2ggYXMgdGVtcGxhdGUgZ2VuZXJhdG9ycywgc2VlIGBAc3lzL3RtcGxgKSB1c2UgdGhpcyBkZWZpbml0aW9uCiMgIGZpbGUgdG8gY2FsY3VsYXRlIHRoZSAibGF0ZXN0IiB2ZXJzaW9ucyB0byBpbmplY3QgaW50bywgc2F5LAojICBhIGBwYWNrYWdlLmpzb25gIGZpbGUgZm9yIGEgc2NhZmZvbGRlZCBwcm9qZWN0LgojCgpncm91cHM6CiAgc3RkL2Rlbm86CiAgICAjIERlbm8gc3RhbmRhcmQgbGlicyAoInN0ZCIpLgogICAgLSBpbXBvcnQ6IGpzcjpAc3RkL2FzeW5jQDEuMC4xMAogICAgLSBpbXBvcnQ6IGpzcjpAc3RkL2RhdGV0aW1lQDAuMjI1LjMKICAgIC0gaW1wb3J0OiBqc3I6QHN0ZC9kb3RlbnZAMC4yMjUuMwogICAgLSBpbXBvcnQ6IGpzcjpAc3RkL2VuY29kaW5nQDEuMC43CiAgICAtIGltcG9ydDoganNyOkBzdGQvZnNAMS4wLjEzCiAgICAtIGltcG9ydDoganNyOkBzdGQvcGF0aEAxLjAuOAogICAgLSBpbXBvcnQ6IGpzcjpAc3RkL3NlbXZlckAxLjAuNAogICAgLSBpbXBvcnQ6IGpzcjpAc3RkL3Rlc3RpbmdAMS4wLjkKICAgIC0gaW1wb3J0OiBqc3I6QHN0ZC91dWlkQDEuMC40CgogIGF1dG9tZXJnZToKICAgICMgaHR0cHM6Ly9hdXRvbWVyZ2Uub3JnCiAgICAtIGltcG9ydDogbnBtOkBhdXRvbWVyZ2UvYXV0b21lcmdlQDIuMi44CiAgICAtIGltcG9ydDogbnBtOkBhdXRvbWVyZ2UvYXV0b21lcmdlLXJlcG9AMS4yLjEKICAgIC0gaW1wb3J0OiBucG06QGF1dG9tZXJnZS9hdXRvbWVyZ2UtcmVwby1uZXR3b3JrLWJyb2FkY2FzdGNoYW5uZWxAMS4yLjEKICAgIC0gaW1wb3J0OiBucG06QGF1dG9tZXJnZS9hdXRvbWVyZ2UtcmVwby1zdG9yYWdlLWluZGV4ZWRkYkAxLjIuMQogICAgLSBpbXBvcnQ6IG5wbTpAYXV0b21lcmdlL2F1dG9tZXJnZS1yZXBvLXN0b3JhZ2Utbm9kZWZzQDEuMi4xCiAgICAtIGltcG9ydDogbnBtOkBvbnNldHNvZnR3YXJlL2F1dG9tZXJnZS1wYXRjaGVyQDAuMTQuMAoKICBjcnlwdG86CiAgICAtIGltcG9ydDogbnBtOkBub2JsZS9oYXNoZXNAMS43LjEKICAgICAgd2lsZGNhcmQ6IHRydWUKCiAgYnVpbGQvdG9vbHM6CiAgICAtIGltcG9ydDogbnBtOkB2aXRlanMvcGx1Z2luLXJlYWN0LXN3Y0AzLjguMAogICAgLSBpbXBvcnQ6IG5wbTpyb2xsdXBANC4zNC44CiAgICAtIGltcG9ydDogbnBtOnZpdGVANi4xLjEKICAgIC0gaW1wb3J0OiBucG06dml0ZS1wbHVnaW4td2FzbUAzLjQuMQoKICB1aS9yZWFjdDoKICAgIC0gaW1wb3J0OiBucG06QHR5cGVzL3JlYWN0QDE4LjMuMTgKICAgIC0gaW1wb3J0OiBucG06QHR5cGVzL3JlYWN0LWRvbUAxOC4zLjUKICAgIC0gaW1wb3J0OiBucG06cmVhY3RAMTguMy4xCiAgICAtIGltcG9ydDogbnBtOnJlYWN0LWRvbUAxOC4zLjEKCmRlbm8uanNvbjoKICAtIGdyb3VwOiBzdGQvZGVubwogIC0gZ3JvdXA6IGNyeXB0bwogIC0gZ3JvdXA6IGF1dG9tZXJnZQoKICAjIENMSSB0b29scwogIC0gaW1wb3J0OiBqc3I6QGNsaWZmeS9rZXlwcmVzc0AxLjAuMC1yYy43CiAgLSBpbXBvcnQ6IGpzcjpAY2xpZmZ5L3Byb21wdEAxLjAuMC1yYy43CiAgLSBpbXBvcnQ6IGpzcjpAY2xpZmZ5L3RhYmxlQDEuMC4wLXJjLjcKCiAgIyBTdW5kcnk6IE5QTQogIC0gaW1wb3J0OiBucG06QHR5cGVzL2RpZmZANy4wLjEKICAtIGltcG9ydDogbnBtOmNoYWlANQogIC0gaW1wb3J0OiBucG06YXBwcm94LXN0cmluZy1tYXRjaEAyCiAgLSBpbXBvcnQ6IG5wbTpkYXRlLWZuc0A0CiAgLSBpbXBvcnQ6IG5wbTpzdWJob3N0aW5nQDAuMS4wLWFscGhhLjEKICAtIGltcG9ydDogbnBtOmRpZmZANwogIC0gaW1wb3J0OiBucG06ZmFzdC1qc29uLXBhdGNoQDMuMS4xCiAgLSBpbXBvcnQ6IG5wbTpmYWtlLWluZGV4ZWRkYkA2LjAuMAogIC0gaW1wb3J0OiBucG06aGFwcHktZG9tQDE3LjEuMwogIC0gaW1wb3J0OiBucG06aGFzaC1pdEA2LjAuMAogIC0gaW1wb3J0OiBucG06aWdub3JlQDcKICAtIGltcG9ydDogbnBtOmltbWVyQDEwCiAgLSBpbXBvcnQ6IG5wbTpvcmFAOC4yLjAKICAtIGltcG9ydDogbnBtOm9sbGFtYUAwLjUuMTMKICAtIGltcG9ydDogbnBtOnByZXR0eS1ieXRlc0A2LjEuMQogIC0gaW1wb3J0OiBucG06cmFtZGFAMC4zMC4xCiAgLSBpbXBvcnQ6IG5wbTpyYW1iZGFAOS40LjIKICAtIGltcG9ydDogbnBtOnJ4anNANy44LjIKICAtIGltcG9ydDogbnBtOnN0cmlwLWFuc2lANwogIC0gaW1wb3J0OiBucG06c3ViaG9zdGluZ0AwLjEuMC1hbHBoYS4xCiAgLSBpbXBvcnQ6IG5wbTp0aW55Y29sb3IyQDEuNi4wCiAgLSBpbXBvcnQ6IG5wbTp0cy1lc3NlbnRpYWxzQDEwLjAuNAogIC0gaW1wb3J0OiBucG06dmFsaWJvdEAxLjAuMC1yYy4xCiAgLSBpbXBvcnQ6IG5wbTp5YW1sQDIuNy4wCgogICMgQnJvd3NlcgogIC0gaW1wb3J0OiBucG06Y3NzdHlwZUAzCiAgLSBpbXBvcnQ6IG5wbTp1YS1wYXJzZXItanNAMi4wLjIKCiAgIyBVSQogIC0gaW1wb3J0OiBucG06cmVhY3QtZXJyb3ItYm91bmRhcnlANQogIC0gaW1wb3J0OiBucG06cmVhY3QtaW5zcGVjdG9yQDYKICAtIGltcG9ydDogbnBtOnJlYWN0LXNwaW5uZXJzQDAuMTUuMAoKcGFja2FnZS5qc29uOgogIC0gZ3JvdXA6IHN0ZC9kZW5vCiAgLSBncm91cDogY3J5cHRvCiAgLSBncm91cDogYnVpbGQvdG9vbHMKICAgIGRldjogdHJ1ZQoKICAtIGltcG9ydDogbnBtOmhvbm9ANC43LjIKCiAgIyBVSQogIC0gZ3JvdXA6IHVpL3JlYWN0CiAgLSBpbXBvcnQ6IG5wbTpyZWFjdC1pY29uc0A1LjUuMAogIC0gaW1wb3J0OiBucG06QHZpZHN0YWNrL3JlYWN0QDEuMTIuMTIKCiAgIyBVSTpGcmFtZXdvcmtzCiAgLSBpbXBvcnQ6IG5wbTp2aXRlcHJlc3NAMS42LjMKICAtIGltcG9ydDogbnBtOnZ1ZUAzLjUuMTMK", - ".vitepress/config.alias.ts": "data:application/typescript;base64,aW1wb3J0IHR5cGUgeyBBbGlhcyB9IGZyb20gJ3ZpdGUnOwoKaW1wb3J0IHsgRGVub0RlcHMgfSBmcm9tICdAc3lzL2RyaXZlci1kZW5vL3J1bnRpbWUnOwppbXBvcnQgeyBWaXRlQ29uZmlnIH0gZnJvbSAnQHN5cy9kcml2ZXItdml0ZSc7CmltcG9ydCB7IEVyciwgUGF0aCwgUiB9IGZyb20gJ0BzeXMvc3RkJzsKCi8qKgogKiBHZW5lcmF0ZSAiaW1wb3J0IiBzdGF0ZW1lbnQgYWxpYXMgbWFwIGZvcjoKICoKICogICAgLSBqc3I6QHN5cyAgICAgICAgIFN5c3RlbSBtb2R1bGVzIHdpdGhpbiB0aGUgd29ya3NwYWNlLgogKiAgICAtIG5wbTo8ZGVwcz4gICAgICAgVGhlIHVwc3RyZWFtIGRlcGVuZGVuY2llcyBpbXBvcnRlZCBmcm9tIHRoZSBOUE0gcmVnaXN0cnkuCiAqLwpleHBvcnQgYXN5bmMgZnVuY3Rpb24gZ2V0QWxpYXNlcygpIHsKICBjb25zdCBkZXBzID0gKGF3YWl0IGxvYWREZXBzKCcuLy5zeXMvZGVwcy55YW1sJykpLmRlcHM7CiAgY29uc3QgZGVwc1N5cyA9IChhd2FpdCBsb2FkRGVwcygnLi8uc3lzL2RlcHMuc3lzLnlhbWwnKSkuZGVwczsKCiAgY29uc3QgaXNTeXMgPSAobmFtZTogc3RyaW5nKSA9PiBuYW1lLnN0YXJ0c1dpdGgoJ0BzeXMvJyk7CiAgY29uc3Qgd3MgPSBhd2FpdCBWaXRlQ29uZmlnLndvcmtzcGFjZSgpOwogIGNvbnN0IGEgPSBkZXBzU3lzLm1hcCgobSkgPT4gbS5uYW1lKS5maWx0ZXIoaXNTeXMpOwogIGNvbnN0IGIgPSB3cy5tb2R1bGVzLml0ZW1zLm1hcCgobSkgPT4gbS5uYW1lKS5maWx0ZXIoaXNTeXMpOwogIGNvbnN0IGlzU3lzdGVtTW9ub3JlcG8gPSBhLmV2ZXJ5KChpdGVtKSA9PiBiLmluY2x1ZGVzKGl0ZW0pKTsKICBpZiAoaXNTeXN0ZW1Nb25vcmVwbykgcmV0dXJuIHdzLmFsaWFzZXM7CgogIGNvbnN0IG1vZHVsZXMgPSBbLi4uZGVwcywgLi4uZGVwc1N5c107CiAgY29uc3QgYWxpYXNlcyA9IG1vZHVsZXMKICAgIC5maWx0ZXIoKG0pID0+ICEhbS5yZWdpc3RyeSkKICAgIC5maWx0ZXIoKG0pID0+IG0udmVyc2lvbiAhPT0gJzAuMC4wJykKICAgIC5tYXAoKG0pID0+IFZpdGVDb25maWcuYWxpYXMobS5yZWdpc3RyeSwgbS5uYW1lKSk7CgogIGNvbnN0IHVuaXF1ZSA9IFIudW5pcUJ5KChpdGVtOiBBbGlhcykgPT4gYCR7aXRlbS5yZXBsYWNlbWVudH06JHtpdGVtLmZpbmQudG9TdHJpbmcoKX1gKTsKICByZXR1cm4gdW5pcXVlKGFsaWFzZXMpOwp9CgovKioKICogSGVscGVycwogKi8KYXN5bmMgZnVuY3Rpb24gbG9hZERlcHMocGF0aDogc3RyaW5nKSB7CiAgY29uc3QgZXJyb3JzID0gRXJyLmVycm9ycygpOwogIHBhdGggPSBQYXRoLnJlc29sdmUoaW1wb3J0Lm1ldGEuZGlybmFtZSA/PyAnJywgJy4uJywgcGF0aCk7CgogIGNvbnN0IHJlcyA9IGF3YWl0IERlbm9EZXBzLmZyb20ocGF0aCk7CiAgaWYgKHJlcy5lcnJvciB8fCAhcmVzLmRhdGE/LmRlcHMpIHsKICAgIGNvbnN0IGVyciA9IGBGYWlsZWQgdG8gbG9hZCBzeXN0ZW0gZGVwZW5kZW5jaWVzIGZyb206ICR7cGF0aH1gOwogICAgY29uc29sZS53YXJuKGVyciwgeyBjYXVzZTogcmVzLmVycm9yIH0pOwogICAgZXJyb3JzLnB1c2goZXJyKTsKICB9CgogIHJldHVybiB7CiAgICBkZXBzOiAocmVzLmRhdGE/LmRlcHMgPz8gW10pLm1hcCgoZCkgPT4gZC5tb2R1bGUpLAogICAgZXJyb3I6IGVycm9ycy50b0Vycm9yKCksCiAgfSBhcyBjb25zdDsKfQo=", - ".vitepress/config.markdown.ts": "data:application/typescript;base64,aW1wb3J0IHR5cGUgeyBNYXJrZG93blJlbmRlcmVyIH0gZnJvbSAndml0ZXByZXNzJzsKaW1wb3J0IHsgcGFyc2UgfSBmcm9tICd5YW1sJzsKCmV4cG9ydCBjb25zdCBtYXJrZG93biA9IHsKICBjb25maWcobWQ6IE1hcmtkb3duUmVuZGVyZXIpIHsKICAgIGNvbnN0IHJ1bGVzID0gbWQucmVuZGVyZXIucnVsZXM7CgogICAgLy8gU3RvcmUgdGhlIG9yaWdpbmFsIGZlbmNlIHJ1bGUuCiAgICBjb25zdCBkZWZhdWx0RmVuY2UgPQogICAgICBydWxlcy5mZW5jZSB8fAogICAgICBmdW5jdGlvbiAodG9rZW5zLCBpZHgsIG9wdGlvbnMsIF9lbnYsIHNlbGYpIHsKICAgICAgICByZXR1cm4gc2VsZi5yZW5kZXJUb2tlbih0b2tlbnMsIGlkeCwgb3B0aW9ucyk7CiAgICAgIH07CgogICAgLy8gT3ZlcnJpZGUgdGhlIGZlbmNlIHJ1bGUgbG9va2luZyBmb3IgWUFNTCBzdHJ1Y3R1cmVzLgogICAgcnVsZXMuZmVuY2UgPSAodG9rZW5zLCBpZHgsIG9wdGlvbnMsIGVudiwgc2VsZikgPT4gewogICAgICBjb25zdCB0b2tlbiA9IHRva2Vuc1tpZHhdOwoKICAgICAgaWYgKHRva2VuLmluZm8udHJpbSgpID09PSAneWFtbCcpIHsKICAgICAgICBjb25zdCB5YW1sID0gcGFyc2UodG9rZW4uY29udGVudCkgfHwgdW5kZWZpbmVkOwogICAgICAgIGlmICh5YW1sPy5jb21wb25lbnQgPT09ICdWaWRlbycpIHsKICAgICAgICAgIGNvbnN0IGRlZmF1bHRIdG1sID0gZGVmYXVsdEZlbmNlKHRva2VucywgaWR4LCBvcHRpb25zLCBlbnYsIHNlbGYpOwogICAgICAgICAgY29uc3Qgc3JjID0geWFtbC5zcmMgfHwgJyc7CiAgICAgICAgICBsZXQgaHRtbCA9IGA8VmlkZW8gc3JjPSIke3NyY30iLz5gOwogICAgICAgICAgaWYgKHlhbWwuZGVidWcpIGh0bWwgPSBgJHtodG1sfVxuJHtkZWZhdWx0SHRtbH1gOwogICAgICAgICAgcmV0dXJuIGh0bWw7CiAgICAgICAgfQoKICAgICAgICBpZiAoeWFtbD8uY29tcG9uZW50ID09PSAnQ29uY2VwdFBsYXllcicpIHsKICAgICAgICAgIGNvbnN0IGRlZmF1bHRIdG1sID0gZGVmYXVsdEZlbmNlKHRva2VucywgaWR4LCBvcHRpb25zLCBlbnYsIHNlbGYpOwogICAgICAgICAgY29uc3Qgc3JjID0geWFtbC52aWRlbyB8fCAnJzsKICAgICAgICAgIGxldCBodG1sID0gYDxWaWRlbyBzcmM9IiR7c3JjfSIvPmA7CiAgICAgICAgICBpZiAoeWFtbC5kZWJ1ZykgaHRtbCA9IGAke2h0bWx9XG4ke2RlZmF1bHRIdG1sfWA7CiAgICAgICAgICByZXR1cm4gaHRtbDsKICAgICAgICB9CiAgICAgIH0KCiAgICAgIC8vIE5vIG92ZXJyaWRlbiBtYXRjaGVzIGZvdW5kLCByZXR1cm4gZGVmYXVsdCByZW5kZXJpbmcuCiAgICAgIHJldHVybiBkZWZhdWx0RmVuY2UodG9rZW5zLCBpZHgsIG9wdGlvbnMsIGVudiwgc2VsZik7CiAgICB9OwogIH0sCn07Cg==", - ".vitepress/config.ts": "data:application/typescript;base64,aW1wb3J0IHR5cGUgeyBDb25maWdFbnYgfSBmcm9tICd2aXRlJzsKCmltcG9ydCB7IGRlZmluZUNvbmZpZyB9IGZyb20gJ3ZpdGVwcmVzcyc7CmltcG9ydCB7IENvbmZpZyB9IGZyb20gJy4uL3NyYy9jb25maWcudHMnOwppbXBvcnQgeyBzaWRlYmFyIH0gZnJvbSAnLi4vc3JjL25hdi50cyc7CmltcG9ydCB7IGdldEFsaWFzZXMgfSBmcm9tICcuL2NvbmZpZy5hbGlhcy50cyc7CmltcG9ydCB7IG1hcmtkb3duIH0gZnJvbSAnLi9jb25maWcubWFya2Rvd24udHMnOwoKZXhwb3J0IGRlZmF1bHQgYXN5bmMgKGVudjogQ29uZmlnRW52KSA9PiB7CiAgY29uc3QgeyB0aXRsZSwgZGVzY3JpcHRpb24gfSA9IENvbmZpZzsKICBjb25zdCBhbGlhcyA9IChhd2FpdCBnZXRBbGlhc2VzKCkpIGFzIGFueTsgLy8gTkI6IHR5cGUtaGFjayAoInZpdGVwcmVzcyIgdnMuICJ2aXRlIiBmaWdodGluZykuCgogIHJldHVybiBkZWZpbmVDb25maWcoewogICAgdGl0bGUsCiAgICBkZXNjcmlwdGlvbiwKICAgIHNyY0RpcjogJzxTUkNfRElSPicsCiAgICBtYXJrZG93biwKICAgIHRoZW1lQ29uZmlnOiB7IHNpZGViYXIsIHNlYXJjaDogeyBwcm92aWRlcjogJ2xvY2FsJyB9IH0sCiAgICB2aXRlOiB7IHJlc29sdmU6IHsgYWxpYXMgfSwgcGx1Z2luczogW10gfSwKICB9KTsKfTsK", + ".sys/common.ts": "data:application/typescript;base64,ZXhwb3J0ICogZnJvbSAnLi9jb21tb24vbW9kLnRzJzsK", + ".sys/common/libs.ts": "data:application/typescript;base64,ZXhwb3J0IHsgSW1tdXRhYmxlLCBzbHVnIH0gZnJvbSAnQHN5cy9zdGQnOwo=", + ".sys/common/mod.ts": "data:application/typescript;base64,ZXhwb3J0ICogZnJvbSAnLi9saWJzLnRzJzsKZXhwb3J0IHR5cGUgKiBhcyB0IGZyb20gJy4vdC50cyc7Cg==", + ".sys/common/t.ts": "data:application/typescript;base64,ZXhwb3J0IHR5cGUgKiBmcm9tICdAc3lzL3R5cGVzJzsKZXhwb3J0IHR5cGUgKiBmcm9tICcuLi90eXBlcy50cyc7Cg==", + ".sys/mod.ts": "data:application/typescript;base64,ZXhwb3J0IHsgUHJvcHMgfSBmcm9tICcuL3UvdS5Qcm9wcy50cyc7CmV4cG9ydCB7IEdsb2JhbCB9IGZyb20gJy4vdS91Lkdsb2JhbC50cyc7Cg==", + ".sys/types.ts": "data:application/typescript;base64,LyoqCiAqIEBtb2R1bGUKICogU3lzdGVtIHR5cGVzCiAqLwpleHBvcnQgdHlwZSAqIGZyb20gJy4vdS90LnRzJzsK", + ".sys/u/common.ts": "data:application/typescript;base64,ZXhwb3J0ICogZnJvbSAnLi4vY29tbW9uLnRzJzsK", + ".sys/u/t.ts": "data:application/typescript;base64,aW1wb3J0IHR5cGUgeyB0IH0gZnJvbSAnLi9jb21tb24udHMnOwoKZXhwb3J0IHR5cGUgR2xvYmFsU3RhdGUgPSB7CiAgdG1wOiBudW1iZXI7Cn07CgovKioKICogSW1tdXRhYmxlIHdyYXBwZXIuCiAqLwp0eXBlIFAgPSB0LlBhdGNoT3BlcmF0aW9uOwpleHBvcnQgdHlwZSBHbG9iYWxTdGF0ZUltbXV0YWJsZSA9IHQuSW1tdXRhYmxlUmVmPHQuR2xvYmFsU3RhdGUsIFAsIEdsb2JhbFN0YXRlRXZlbnRzPjsKCmV4cG9ydCB0eXBlIEdsb2JhbFN0YXRlRXZlbnRzID0gdC5JbW11dGFibGVFdmVudHM8R2xvYmFsU3RhdGUsIFA+OwpleHBvcnQgdHlwZSBHbG9iYWxTdGF0ZUV2ZW50ID0gdC5JbmZlckltbXV0YWJsZUV2ZW50PHQuR2xvYmFsU3RhdGVFdmVudHM+Owo=", + ".sys/u/u.Global.ts": "data:application/typescript;base64,aW1wb3J0IHsgdHlwZSB0LCBJbW11dGFibGUsIHNsdWcgfSBmcm9tICcuL2NvbW1vbi50cyc7Cgpjb25zdCBkZWZhdWx0SWQgPSBgZGVmYXVsdDoke3NsdWcoKX1gOwpjb25zdCByZWZzID0gbmV3IE1hcDxzdHJpbmcsIHQuR2xvYmFsU3RhdGVJbW11dGFibGU+KCk7CgovKioKICogR2xvYmFsIHN0YXRlIGludGVyZmFjZS4KICovCmV4cG9ydCBjb25zdCBHbG9iYWwgPSB7CiAgLyoqCiAgICogUmV0cmlldmUgYW4gaW5zdGFuY2Ugb2YgdGhlIGdsb2JhbCBzdGF0ZSBvYmplY3QuCiAgICoKICAgKiBAcGFyYW0gaW5zdGFuY2UgLSBPcHRpb25hbCB1bmlxdWUgaWRlbnRpZmllciBmb3IgdGhlIGdsb2JhbCBzdGF0ZSBpbnN0YW5jZS4KICAgKiAgICAgICAgICAgICAgICAgICBJZiBub3QgcHJvdmlkZWQsIHRoZSBkZWZhdWx0IGluc3RhbmNlIGlzIHJldHVybmVkLgogICAqICAgICAgICAgICAgICAgICAgIE5vIHBhcmFtIGlzIHRoZSBlcXVpdmFsZW50IG9mIHJldHJpZXZpbmcgdGhlICJzaW5nbHRvbiIgaW5zdGFuY2UuCiAgICoKICAgKiBAcmV0dXJucyBUaGUgZ2xvYmFsIHN0YXRlIGluc3RhbmNlIGNvcnJlc3BvbmRpbmcgdG8gdGhlIHByb3ZpZGVkIGlkZW50aWZpZXIuCiAgICovCiAgc3RhdGUoaW5zdGFuY2U/OiB0LlN0cmluZ0lkKTogdC5HbG9iYWxTdGF0ZUltbXV0YWJsZSB7CiAgICBjb25zdCBpZCA9IGluc3RhbmNlID8/IGRlZmF1bHRJZDsKICAgIGlmIChyZWZzLmhhcyhpZCkpIHJldHVybiByZWZzLmdldChpZCkhOwoKICAgIGNvbnN0IG1vZGVsID0gSW1tdXRhYmxlLmNsb25lclJlZjx0Lkdsb2JhbFN0YXRlPih7IHRtcDogMCB9KTsKICAgIHJlZnMuc2V0KGlkLCBtb2RlbCk7CiAgICByZXR1cm4gbW9kZWw7CiAgfSwKfSBhcyBjb25zdDsK", + ".sys/u/u.Props.ts": "data:application/typescript;base64,dHlwZSBPID0gUmVjb3JkPHN0cmluZywgdW5rbm93bj47CgovKioKICogSGVscGVyIGZvciBwYXNzaW5nIHByb3BzIGJldHdlZW4gVnVlIGFuZCBSZWFjdC4KICovCmV4cG9ydCBjb25zdCBQcm9wcyA9IHsKICBlbmNvZGU8UCBleHRlbmRzIE8+KHByb3BzOiBQKTogc3RyaW5nIHsKICAgIGNvbnN0IGpzb24gPSBKU09OLnN0cmluZ2lmeShwcm9wcyk7CiAgICBjb25zdCBiaW5hcnkgPSBuZXcgVGV4dEVuY29kZXIoKS5lbmNvZGUoanNvbik7CiAgICByZXR1cm4gU3RyaW5nKGJpbmFyeSk7CiAgfSwKCiAgZGVjb2RlKGVuY29kZWQ6IHN0cmluZyk6IE8gewogICAgaWYgKHR5cGVvZiBlbmNvZGVkICE9PSAnc3RyaW5nJykgcmV0dXJuIHt9OwogICAgdHJ5IHsKICAgICAgY29uc3QgYmluYXJ5ID0gVWludDhBcnJheS5mcm9tKGVuY29kZWQuc3BsaXQoJywnKSk7CiAgICAgIGNvbnN0IGpzb24gPSBuZXcgVGV4dERlY29kZXIoKS5kZWNvZGUoYmluYXJ5KTsKICAgICAgcmV0dXJuIEpTT04ucGFyc2UoanNvbik7CiAgICB9IGNhdGNoIChlcnJvcikgewogICAgICByZXR1cm4ge307CiAgICB9CiAgfSwKfSBhcyBjb25zdDsK", + ".sys/ui/React.NotFound.tsx": "data:application/typescript+jsx;base64,aW1wb3J0IHsgY3NzIH0gZnJvbSAnQHN5cy91aS1jc3MnOwppbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnOwoKZXhwb3J0IHR5cGUgTm90Rm91bmRQcm9wcyA9IHt9OwoKZXhwb3J0IGNvbnN0IE5vdEZvdW5kOiBSZWFjdC5GQzxOb3RGb3VuZFByb3BzPiA9IChwcm9wcykgPT4gewogIGNvbnN0IHt9ID0gcHJvcHM7CgogIGNvbnN0IHN0eWxlcyA9IHsKICAgIGJhc2U6IGNzcyh7CiAgICAgIGJhY2tncm91bmRDb2xvcjogJ3JnYmEoMjU1LCAwLCAwLCAwLjEpJyAvKiBSRUQgKi8sCiAgICAgIHBhZGRpbmc6IDEwLAogICAgfSksCiAgfTsKCiAgcmV0dXJuICgKICAgIDxkaXYgY2xhc3NOYW1lPXtzdHlsZXMuYmFzZS5jbGFzc30+CiAgICAgIDxkaXY+e2Dwn5C3IENvbXBvbmVudCBOb3QgRm91bmRgfTwvZGl2PgogICAgPC9kaXY+CiAgKTsKfTsK", + ".sys/ui/React.components.ts": "data:application/typescript;base64,aW1wb3J0IFJlYWN0IGZyb20gJ3JlYWN0JzsKCi8qKgogKiBDb21wb25lbnQgZmFjdG9yeSBsb29rdXAuCiAqLwpleHBvcnQgYXN5bmMgZnVuY3Rpb24gbG9va3VwKGtpbmQ6IHN0cmluZyk6IFByb21pc2U8UmVhY3QuRkMgfCB1bmRlZmluZWQ+IHsKICBpZiAoa2luZCA9PT0gJ3N5cy90bXAvdWk6Rm9vJykgewogICAgY29uc3QgeyBGb28gfSA9IGF3YWl0IGltcG9ydCgnQHN5cy90bXAvdWknKTsKICAgIHJldHVybiBGb287CiAgfQoKICBjb25zdCBwbGF5ZXJzID0gWydDb25jZXB0UGxheWVyJywgJ1ZpZGVvUGxheWVyJywgJ1BhbmVsJ107CgogIGlmIChwbGF5ZXJzLmluY2x1ZGVzKGtpbmQpKSB7CiAgICBjb25zdCB7IFBsYXllciB9ID0gYXdhaXQgaW1wb3J0KCdAc3lzL3VpLXJlYWN0LWNvbXBvbmVudHMnKTsKICAgIGlmIChraW5kID09PSAnQ29uY2VwdFBsYXllcicpIHJldHVybiBQbGF5ZXIuQ29uY2VwdC5WaWV3OwogICAgaWYgKGtpbmQgPT09ICdWaWRlb1BsYXllcicpIHJldHVybiBQbGF5ZXIuVmlkZW8uVmlldzsKICB9CgogIGlmIChraW5kID09PSAnUGFuZWwnKSB7CiAgICBjb25zdCB7IFBhbmVsIH0gPSBhd2FpdCBpbXBvcnQoJ0BzeXMvdWktcmVhY3QtY29tcG9uZW50cycpOwogICAgcmV0dXJuIFBhbmVsOwogIH0KCiAgcmV0dXJuOyAvLyBOQjogbm8tbWF0Y2guCn0K", + ".sys/ui/React.setup.ts": "data:application/typescript;base64,aW1wb3J0IFJlYWN0IGZyb20gJ3JlYWN0JzsKaW1wb3J0IHsgY3JlYXRlUm9vdCwgdHlwZSBSb290IH0gZnJvbSAncmVhY3QtZG9tL2NsaWVudCc7CmltcG9ydCB7IG9uQmVmb3JlVW5tb3VudCwgb25Nb3VudGVkLCByZWYgYXMgdnVlUmVmLCB0eXBlIFJlZiB9IGZyb20gJ3Z1ZSc7CmltcG9ydCB7IE5vdEZvdW5kIH0gZnJvbSAnLi9SZWFjdC5Ob3RGb3VuZC50c3gnOwoKdHlwZSBPID0gUmVjb3JkPHN0cmluZywgdW5rbm93bj47CmV4cG9ydCBjb25zdCByZWYgPSAoKSA9PiB2dWVSZWY8SFRNTEVsZW1lbnQgfCB1bmRlZmluZWQ+KCk7CgovKioKICogU2V0dXAgYSA8cmVhY3QtaW4tdnVlPiB3cmFwcGVyIGNvbXBvbmVudC4KICovCmV4cG9ydCBmdW5jdGlvbiBzZXR1cDxQIGV4dGVuZHMgTz4oCiAgcmVmUm9vdDogUmVmPEhUTUxFbGVtZW50IHwgdW5kZWZpbmVkPiwKICBwcm9wczogUCwKICBnZXRDb21wb25lbnQ6ICgpID0+IFByb21pc2U8UmVhY3QuRkM8UD4gfCB1bmRlZmluZWQ+LAopIHsKICB0eXBlIEMgPSBSZWFjdC5GQzxQPjsKICBsZXQgcm9vdDogUm9vdCB8IHVuZGVmaW5lZDsKCiAgb25Nb3VudGVkKGFzeW5jICgpID0+IHsKICAgIGlmIChyZWZSb290LnZhbHVlKSB7CiAgICAgIGNvbnN0IENvbXBvbmVudCA9ICgoYXdhaXQgZ2V0Q29tcG9uZW50KCkpIHx8IE5vdEZvdW5kKSBhcyBDOwogICAgICBjb25zdCBlbCA9IFJlYWN0LmNyZWF0ZUVsZW1lbnQoQ29tcG9uZW50LCBwcm9wcyk7CiAgICAgIHJvb3QgPSBjcmVhdGVSb290KHJlZlJvb3QudmFsdWUpOwogICAgICByb290LnJlbmRlcihlbCk7CiAgICB9CiAgfSk7CgogIG9uQmVmb3JlVW5tb3VudCgoKSA9PiByb290Py51bm1vdW50KCkpOwp9Cg==", + ".sys/ui/React.vue": "data:text/plain;base64,PHRlbXBsYXRlPgogIDxkaXYgcmVmPSJyb290Ij48L2Rpdj4KPC90ZW1wbGF0ZT4KCjxzY3JpcHQgc2V0dXAgbGFuZz0idHMiPgppbXBvcnQgeyBQcm9wcyB9IGZyb20gJy4uL3UvdS5Qcm9wcyc7CmltcG9ydCB7IGxvb2t1cCB9IGZyb20gJy4vUmVhY3QuY29tcG9uZW50cyc7CmltcG9ydCB7IHJlZiwgc2V0dXAgfSBmcm9tICcuL1JlYWN0LnNldHVwJzsKCnR5cGUgQmFzZTY0RW5jb2RlZEpzb25TdHJpbmcgPSBzdHJpbmc7CnR5cGUgSW5wdXRQcm9wcyA9IHsga2luZDogc3RyaW5nOyBwcm9wcz86IEJhc2U2NEVuY29kZWRKc29uU3RyaW5nIH07Cgpjb25zdCByb290ID0gcmVmKCk7CmNvbnN0IGlucHV0ID0gZGVmaW5lUHJvcHM8SW5wdXRQcm9wcz4oKTsKY29uc3QgcHJvcHMgPSBQcm9wcy5kZWNvZGUoaW5wdXQucHJvcHMgPz8gJycpOwpzZXR1cChyb290LCBwcm9wcywgYXN5bmMgKCkgPT4gbG9va3VwKGlucHV0LmtpbmQpKTsKPC9zY3JpcHQ+Cgo8c3R5bGUgc2NvcGVkPjwvc3R5bGU+Cg==", + ".sys/ui/mod.ts": "data:application/typescript;base64,aW1wb3J0IHR5cGUgeyBFbmhhbmNlQXBwQ29udGV4dCB9IGZyb20gJ3ZpdGVwcmVzcyc7CmltcG9ydCBSZWFjdCBmcm9tICcuL1JlYWN0LnZ1ZSc7CgpleHBvcnQgZnVuY3Rpb24gcmVnaXN0ZXJDb21wb25lbnRzKGN0eDogRW5oYW5jZUFwcENvbnRleHQpIHsKICBjdHguYXBwLmNvbXBvbmVudCgnUmVhY3QnLCBSZWFjdCk7Cn0K", + ".vitepress/config.markdown.ts": "data:application/typescript;base64,aW1wb3J0IHR5cGUgeyBNYXJrZG93blJlbmRlcmVyIH0gZnJvbSAndml0ZXByZXNzJzsKaW1wb3J0IHsgcGFyc2UgfSBmcm9tICd5YW1sJzsKaW1wb3J0IHsgUHJvcHMgfSBmcm9tICcuLi8uc3lzL21vZC50cyc7CgpleHBvcnQgY29uc3QgbWFya2Rvd24gPSB7CiAgY29uZmlnKG1kOiBNYXJrZG93blJlbmRlcmVyKSB7CiAgICBjb25zdCBydWxlcyA9IG1kLnJlbmRlcmVyLnJ1bGVzOwoKICAgIC8vIE5COiBTdG9yZSB0aGUgb3JpZ2luYWwgZmVuY2UgcnVsZS4KICAgIGNvbnN0IG9yaWdpbmFsRmVuY2VSdWxlID0KICAgICAgcnVsZXMuZmVuY2UgfHwgKCh0b2tlbnMsIGlkeCwgb3B0aW9ucywgZW52LCBzZWxmKSA9PiBzZWxmLnJlbmRlclRva2VuKHRva2VucywgaWR4LCBvcHRpb25zKSk7CgogICAgLy8gT3ZlcnJpZGUgdGhlIGZlbmNlIHJ1bGUgbG9va2luZyBmb3IgWUFNTCBzdHJ1Y3R1cmVzLgogICAgcnVsZXMuZmVuY2UgPSAodG9rZW5zLCBpZHgsIG9wdGlvbnMsIGVudiwgc2VsZikgPT4gewogICAgICBjb25zdCB0b2tlbiA9IHRva2Vuc1tpZHhdOwogICAgICBjb25zdCByZW5kZXJPcmlnaW5hbCA9ICgpID0+IG9yaWdpbmFsRmVuY2VSdWxlKHRva2VucywgaWR4LCBvcHRpb25zLCBlbnYsIHNlbGYpOwoKICAgICAgaWYgKHRva2VuLmluZm8udHJpbSgpID09PSAneWFtbCcpIHsKICAgICAgICBjb25zdCB5YW1sID0gcGFyc2UodG9rZW4uY29udGVudCkgfHwgdW5kZWZpbmVkOwogICAgICAgIGNvbnN0IGZvcm1hdEh0bWwgPSAoaHRtbDogc3RyaW5nKSA9PiAoeWFtbC5kZWJ1ZyA/IGAke2h0bWx9XG4ke3JlbmRlck9yaWdpbmFsKCl9YCA6IGh0bWwpOwoKICAgICAgICBpZiAodHlwZW9mIHlhbWw/LmNvbXBvbmVudCA9PT0gJ3N0cmluZycpIHsKICAgICAgICAgIGNvbnN0IGRhdGEgPSB7IC4uLnlhbWwgfTsKICAgICAgICAgIGRlbGV0ZSBkYXRhLmNvbXBvbmVudDsKICAgICAgICAgIGNvbnN0IGh0bWwgPSBgPFJlYWN0IGtpbmQ9IiR7eWFtbC5jb21wb25lbnR9IiBwcm9wcz0iJHtQcm9wcy5lbmNvZGUoZGF0YSl9IiAvPmA7CiAgICAgICAgICByZXR1cm4gZm9ybWF0SHRtbChodG1sKTsKICAgICAgICB9CiAgICAgIH0KCiAgICAgIC8vIE5vIG1hdGNoIGZvdW5kLCByZXR1cm4gZGVmYXVsdC4KICAgICAgcmV0dXJuIHJlbmRlck9yaWdpbmFsKCk7CiAgICB9OwogIH0sCn07Cg==", + ".vitepress/config.ts": "data:application/typescript;base64,aW1wb3J0IHR5cGUgeyBDb25maWdFbnYgfSBmcm9tICd2aXRlJzsKCmltcG9ydCB7IFZpdGVDb25maWcgfSBmcm9tICdAc3lzL2RyaXZlci12aXRlJzsKaW1wb3J0IHsgZGVmaW5lQ29uZmlnIH0gZnJvbSAndml0ZXByZXNzJzsKCmltcG9ydCB7IENvbmZpZyB9IGZyb20gJy4uL3NyYy9jb25maWcudHMnOwppbXBvcnQgeyBzaWRlYmFyIH0gZnJvbSAnLi4vc3JjL25hdi50cyc7CmltcG9ydCB7IG1hcmtkb3duIH0gZnJvbSAnLi9jb25maWcubWFya2Rvd24udHMnOwoKZXhwb3J0IGRlZmF1bHQgYXN5bmMgKGVudjogQ29uZmlnRW52KSA9PiB7CiAgY29uc3QgeyB0aXRsZSwgZGVzY3JpcHRpb24gfSA9IENvbmZpZzsKICBjb25zdCB3cyA9IGF3YWl0IFZpdGVDb25maWcud29ya3NwYWNlKCk7CiAgY29uc3QgYWxpYXMgPSB3cy5hbGlhc2VzOwoKICByZXR1cm4gZGVmaW5lQ29uZmlnKHsKICAgIHRpdGxlLAogICAgZGVzY3JpcHRpb24sCiAgICBiYXNlOiAnLycsCiAgICBzcmNEaXI6ICcuL2RvY3MnLAogICAgbWFya2Rvd24sCiAgICB0aGVtZUNvbmZpZzogewogICAgICBzaWRlYmFyLAogICAgICAvLyBzZWFyY2g6IHsgcHJvdmlkZXI6ICdsb2NhbCcgfSwKICAgIH0sCiAgICBhcHBlYXJhbmNlOiBmYWxzZSwgLy8gTkI6ICJsaWdodC9kYXJrIiBtb2RlIHN3aXRjaC4KICAgIHZpdGU6IHsKICAgICAgcGx1Z2luczogW10sCiAgICAgIHJlc29sdmU6IHsgYWxpYXMgfSwKICAgIH0sCiAgfSk7Cn07Cg==", + ".vitepress/theme/Layout.vue": "data:text/plain;base64,PHRlbXBsYXRlPgogIDxMYXlvdXQ+CiAgICA8dGVtcGxhdGUgdi1mb3I9InNsb3QgaW4gc2xvdHMiICNbc2xvdF0+CiAgICAgIDxSZWFjdCBraW5kPSJQYW5lbCIgLz4KICAgIDwvdGVtcGxhdGU+CiAgPC9MYXlvdXQ+CjwvdGVtcGxhdGU+Cgo8c2NyaXB0IHNldHVwPgppbXBvcnQgRGVmYXVsdFRoZW1lIGZyb20gJ3ZpdGVwcmVzcy90aGVtZSc7CmNvbnN0IHsgTGF5b3V0IH0gPSBEZWZhdWx0VGhlbWU7CgovKioKICogRG9jczogaHR0cHM6Ly92aXRlcHJlc3MuZGV2L2d1aWRlL2V4dGVuZGluZy1kZWZhdWx0LXRoZW1lI2xheW91dC1zbG90cwogKi8KY29uc3Qgc2xvdHMgPSBbCiAgJ25hdi1iYXItdGl0bGUtYmVmb3JlJywKICAnbmF2LWJhci10aXRsZS1hZnRlcicsCiAgJ25hdi1iYXItY29udGVudC1iZWZvcmUnLAogICduYXYtYmFyLWNvbnRlbnQtYWZ0ZXInLAogICduYXYtc2NyZWVuLWNvbnRlbnQtYmVmb3JlJywKICAnbmF2LXNjcmVlbi1jb250ZW50LWFmdGVyJywKICAnc2lkZWJhci1uYXYtYmVmb3JlJywKICAnc2lkZWJhci1uYXYtYWZ0ZXInLAogICdhc2lkZS10b3AnLAogICdhc2lkZS1vdXRsaW5lLWJlZm9yZScsCiAgJ2FzaWRlLW91dGxpbmUtYWZ0ZXInLApdOwo8L3NjcmlwdD4K", ".vitepress/theme/index.css": "data:text/plain;base64,LyogcGxhY2Vob2xkZXIgKi8K", - ".vitepress/theme/index.ts": "data:application/typescript;base64,aW1wb3J0IHR5cGUgeyBUaGVtZSBhcyBUaGVtZVR5cGUgfSBmcm9tICd2aXRlcHJlc3MnOwppbXBvcnQgRGVmYXVsdFRoZW1lIGZyb20gJ3ZpdGVwcmVzcy90aGVtZSc7CgppbXBvcnQgeyByZWdpc3RlckNvbXBvbmVudHMgYXMgcmVnaXN0ZXJTeXN0ZW1Db21wb25lbnRzIH0gZnJvbSAnLi4vLi4vLnN5cy9jb21wb25lbnRzL2luZGV4LnRzJzsKCmV4cG9ydCBjb25zdCBUaGVtZTogVGhlbWVUeXBlID0gewogIGV4dGVuZHM6IERlZmF1bHRUaGVtZSwKICBlbmhhbmNlQXBwKGN0eCkgewogICAgcmVnaXN0ZXJTeXN0ZW1Db21wb25lbnRzKGN0eCk7CiAgfSwKfTsKCmV4cG9ydCBkZWZhdWx0IFRoZW1lOwo=", + ".vitepress/theme/index.ts": "data:application/typescript;base64,aW1wb3J0IHR5cGUgeyBUaGVtZSBhcyBUaGVtZVR5cGUgfSBmcm9tICd2aXRlcHJlc3MnOwppbXBvcnQgRGVmYXVsdFRoZW1lIGZyb20gJ3ZpdGVwcmVzcy90aGVtZSc7CgppbXBvcnQgeyByZWdpc3RlckNvbXBvbmVudHMgYXMgcmVnaXN0ZXJTeXN0ZW1Db21wb25lbnRzIH0gZnJvbSAnLi4vLi4vLnN5cy91aS9tb2QudHMnOwppbXBvcnQgTGF5b3V0IGZyb20gJy4vTGF5b3V0LnZ1ZSc7CgpleHBvcnQgY29uc3QgVGhlbWU6IFRoZW1lVHlwZSA9IHsKICBleHRlbmRzOiBEZWZhdWx0VGhlbWUsCiAgTGF5b3V0LAogIGVuaGFuY2VBcHAoY3R4KSB7CiAgICByZWdpc3RlclN5c3RlbUNvbXBvbmVudHMoY3R4KTsKICB9LAp9OwoKZXhwb3J0IGRlZmF1bHQgVGhlbWU7Cg==", ".vscode/settings.json": "data:application/json;base64,ewogICJkZW5vLmVuYWJsZSI6IHRydWUsCn0K", - "deno.json": "data:application/json;base64,ewogICJ2ZXJzaW9uIjogIjAuMC4wIiwKICAidGFza3MiOiB7CiAgICAiZGV2IjogImRlbm8gICAgICBydW4gLVJXTkUgLS1hbGxvdy1ydW4gLS1hbGxvdy1mZmkgPEVOVFJZPiAtLWNtZD1kZXYiLAogICAgImJ1aWxkIjogImRlbm8gICAgcnVuIC1SV0UgLS1hbGxvdy1ydW4gLS1hbGxvdy1mZmkgPEVOVFJZPiAtLWNtZD1idWlsZCIsCiAgICAic2VydmUiOiAiZGVubyAgICBydW4gLVJORSAtLWFsbG93LXJ1biAtLWFsbG93LWZmaSA8RU5UUlk+IC0tY21kPXNlcnZlIiwKICAgICJjbGVhbiI6ICJkZW5vICAgIHJ1biAtUldFIC0tYWxsb3ctZmZpIDxFTlRSWT4gLS1jbWQ9Y2xlYW4iLAogICAgInVwZ3JhZGUiOiAiZGVubyAgcnVuIC1SV05FIC0tYWxsb3ctcnVuIC0tYWxsb3ctZmZpIDxFTlRSWT4gLS1jbWQ9dXBncmFkZSIsCiAgICAiYmFja3VwIjogImRlbm8gICBydW4gLVJXRSAtLWFsbG93LXJ1biAtLWFsbG93LWZmaSA8RU5UUlk+IC0tY21kPWJhY2t1cCIsCiAgICAiaGVscCI6ICJkZW5vICAgICBydW4gLVJFIC0tYWxsb3ctZmZpIDxFTlRSWT4gLS1jbWQ9aGVscCIsCiAgICAic3lzIjogImRlbm8gICAgICBydW4gLVJXTkUgPEVOVFJZX1NZUz4iCiAgfSwKICAiY29tcGlsZXJPcHRpb25zIjogewogICAgInN0cmljdCI6IHRydWUsCiAgICAibGliIjogWyJkZW5vLm5zIiwgImVzbmV4dCIsICJkb20iLCAiZG9tLml0ZXJhYmxlIiwgImRvbS5hc3luY2l0ZXJhYmxlIl0sCiAgICAidHlwZXMiOiBbInZpdGUvY2xpZW50Il0sCiAgICAianN4IjogInJlYWN0LWpzeCIKICB9LAogICJ3b3Jrc3BhY2UiOiBbXSwKICAibm9kZU1vZHVsZXNEaXIiOiAiYXV0byIsCiAgImltcG9ydHMiOiB7CiAgICAiQHN5cy9kcml2ZXItZGVubyI6ICI8RFJJVkVSX0RFTk8+IiwKICAgICJAc3lzL2RyaXZlci12aXRlIjogIjxEUklWRVJfVklURT4iLAogICAgIkBzeXMvZHJpdmVyLXZpdGVwcmVzcyI6ICI8RFJJVkVSX1ZJVEVQUkVTUz4iCiAgfQp9Cg==", - "docs/index.md": "data:text/markdown;base64,IyDwn5GLIEhlbGxvCgpHZW5lcmF0ZWQgd2l0aCBbYEBzeXMvZHJpdmVyLXZpdGVwcmVzc0A8RFJJVkVSX1ZFUj5gXShodHRwczovL2pzci5pby9Ac3lzL2RyaXZlci12aXRlcHJlc3NAPERSSVZFUl9WRVI+KQoKCmBgYHlhbWwKZGVidWc6IHRydWUKY29tcG9uZW50OiBzeXMvdG1wL3VpOkZvbwpub3RlczogfAogIFNob3VsZCBpbXBvcnQ6IEBzeXMvdG1wQDAuMC41Ny91aTo8Rm9vPiAoV0lQKQogIFByb3ZlczoKICAtIHJlYWN0IHJlbmRlcmluZwogIC0gaW1wb3J0IGZyb20gc3lzdGVtIG1vZHVsZSAobW9ub3JlcG8pIHZpYSBKU1IuCiAgLSBjb3JyZWN0IGFsaWFzaW5nIG9mIGZ1bGx5LXF1YWxpZmllZCAiaW1wb3J0IiB0ZXh0IChlZzogIm5wbTpyZWFjdDoxOC4wLjAiIOKGkiAicmVhY3QiKQpgYGAKCgpgYGB5YW1sCmRlYnVnOiB0cnVlCmNvbXBvbmVudDogQ29uY2VwdFBsYXllcgp2aWRlbzogdmltZW8vNzI3OTUxNjc3CnRpbWVzdGFtcHM6IAogICcwMDowMzo1OC4yMTUnOiAKICAgIGltYWdlOiBodHRwczovL3dycGNkLm5ldC9jZG4tY2dpL2ltYWdlZGVsaXZlcnkvQlhsdVF4NGlnZTlHdVcwSWE1NkJIdy8yOGY1YjdlZC02N2QxLTQxOWQtOGRiMC1kOTVhZTkwZTgxMDAvcmVjdGNvbnRhaW4zCmBgYAoKCgpgYGB5YW1sCmRlYnVnOiB0cnVlCmNvbXBvbmVudDogVmlkZW8Kc3JjOiB2aW1lby83Mjc5NTE2NzcKYGBgCgo8cD4mbmJzcDs8L3A+CgotLS0KCgojIE1hcmtkb3duIFtTeW50YXhdKGh0dHBzOi8vbWFya2Rvd24taXQuZ2l0aHViLmlvLykKQSBzYW1wbGUgYnJlYWtkb3duIG9mIHRoZSBtYWluIG1hcmtkb3duIHN5bnRheCBvcHRpb25zIGF2YWlsYWJsZSB0byB5b3UuLi4oIPCfkLcgKToKCgojIyBIMSBIZWFkZXIKIyMjIEgyIEhlYWRlcgojIyMjIEgzIEhlYWRlcgojIyMjIyBINCBIZWFkZXIKCi0gKipCb2xkIHRleHQqKiBhbmQgKml0YWxpYyB0ZXh0KiBjYW4gYmUgZW1waGFzaXplZC4KLSBMaXN0czoKICAtIFVub3JkZXJlZCBpdGVtCiAgLSBBbm90aGVyIGl0ZW0KICAgIC0gU3ViLWl0ZW0KICAgIC0gQW5vdGhlciBzdWItaXRlbQogIC0gVGhpcmQgaXRlbQotIE9yZGVyZWQgbGlzdDoKICAxLiBGaXJzdCBpdGVtCiAgMi4gU2Vjb25kIGl0ZW0KICAzLiBUaGlyZCBpdGVtCgo+IEJsb2NrcXVvdGVzIGFyZSB1c2VmdWwgZm9yIGhpZ2hsaWdodGluZyB0ZXh0LgoKSW5saW5lIGBjb2RlYCBhbmQgY29kZSBibG9ja3M6CgpgYGB0cwpjb25zdCBoZWxsbyA9IChzdWJqZWN0OiBzdHJpbmcpID0+IGDwn5GLIEhlbGxvLCAke3N1YmplY3R9IWA7CmNvbnNvbGUubG9nKGhlbGxvKCdXb3JsZCcpKTsKYGBgCgoKQ29sb3Igc3ludGF4IGhpZ2hsaWdodGluZyBhbGwgbWFqb3IgbW9kZXJuIGxhbmd1YWdlczoKCmBgYHB5dGhvbgojIPCfkYsKcHJpbnQoIkhlbGxvLCBXb3JsZCEiKQpgYGAKCgpMaW5rczogIApbSW50ZXJuYWwgbGlua10oI3NlY3Rpb24pIHwgW0V4dGVybmFsIGxpbmtdKGh0dHBzOi8vZXhhbXBsZS5jb20pICAKCkZvb3Rub3RlIHJlZmVyZW5jZVteMV0uCgpUYWJsZXM6CnwgSGVhZGVyIDEgfCBIZWFkZXIgMiB8CnwtLS0tLS0tLS0tfC0tLS0tLS0tLS18CnwgUm93IDEgICAgfCBEYXRhIDEgICB8CnwgUm93IDIgICAgfCBEYXRhIDIgICB8CgpbXjFdOiBUaGlzIGlzIGEgZm9vdG5vdGUuCgpUaGlzIGlzIGEgcGFyYWdyYXBoIHdpdGggYSBbcmVmZXJlbmNlLXN0eWxlIGxpbmtdW3JlZl0gYW5kIGFub3RoZXIgW2lubGluZSBsaW5rXShodHRwczovL2V4YW1wbGUuY29tKS4KCltyZWZdOiBodHRwczovL2V4YW1wbGUuY29tICJPcHRpb25hbCB0aXRsZSIK", + "deno.json": "data:application/json;base64,ewogICJ2ZXJzaW9uIjogIjAuMC4wIiwKICAidGFza3MiOiB7CiAgICAiZGV2IjogImRlbm8gICAgICBydW4gLVJXTkUgLS1hbGxvdy1ydW4gLS1hbGxvdy1mZmkgICA8RU5UUlk+IC0tY21kPWRldiIsCiAgICAiYnVpbGQiOiAiZGVubyAgICBydW4gLVJXRSAtLWFsbG93LXJ1biAtLWFsbG93LWZmaSAgICA8RU5UUlk+IC0tY21kPWJ1aWxkIiwKICAgICJzZXJ2ZSI6ICJkZW5vICAgIHJ1biAtUk5FIC0tYWxsb3ctcnVuIC0tYWxsb3ctZmZpICAgIDxFTlRSWT4gLS1jbWQ9c2VydmUiLAogICAgImNsZWFuIjogImRlbm8gICAgcnVuIC1SV0UgLS1hbGxvdy1mZmkgICAgICAgICAgICAgICAgPEVOVFJZPiAtLWNtZD1jbGVhbiIsCiAgICAidXBncmFkZSI6ICJkZW5vICBydW4gLVJXTkUgLS1hbGxvdy1ydW4gLS1hbGxvdy1mZmkgICA8RU5UUlk+IC0tY21kPXVwZ3JhZGUiLAogICAgImJhY2t1cCI6ICJkZW5vICAgcnVuIC1SV0UgLS1hbGxvdy1ydW4gLS1hbGxvdy1mZmkgICAgPEVOVFJZPiAtLWNtZD1iYWNrdXAiLAogICAgImhlbHAiOiAiZGVubyAgICAgcnVuIC1SRSAtLWFsbG93LWZmaSAgICAgICAgICAgICAgICAgPEVOVFJZPiAtLWNtZD1oZWxwIgogIH0sCiAgImNvbXBpbGVyT3B0aW9ucyI6IHsKICAgICJzdHJpY3QiOiB0cnVlLAogICAgImxpYiI6IFsiZGVuby5ucyIsICJlc25leHQiLCAiZG9tIiwgImRvbS5pdGVyYWJsZSIsICJkb20uYXN5bmNpdGVyYWJsZSJdLAogICAgInR5cGVzIjogWyJ2aXRlL2NsaWVudCJdLAogICAgImpzeCI6ICJyZWFjdC1qc3giCiAgfSwKICAid29ya3NwYWNlIjogW10sCiAgIm5vZGVNb2R1bGVzRGlyIjogImF1dG8iLAogICJpbXBvcnRNYXAiOiAiLi9pbXBvcnRzLmpzb24iCn0K", + "docs/index.md": "data:text/markdown;base64,IyDwn5GLIEhlbGxvCgpHZW5lcmF0ZWQgd2l0aCBbYEBzeXMvZHJpdmVyLXZpdGVwcmVzc0A8RFJJVkVSX1ZFUj5gXShodHRwczovL2pzci5pby9Ac3lzL2RyaXZlci12aXRlcHJlc3NAPERSSVZFUl9WRVI+KQoKCmBgYHlhbWwKZGVidWc6IHRydWUKY29tcG9uZW50OiBzeXMvdG1wL3VpOkZvbwpub3RlczogfAogIFNob3VsZCBpbXBvcnQ6IEBzeXMvdG1wQDAuMC41Ny91aTo8Rm9vPiAoV0lQKQogIFByb3ZlczoKICAtIHJlYWN0IHJlbmRlcmluZwogIC0gaW1wb3J0IGZyb20gc3lzdGVtIG1vZHVsZSAobW9ub3JlcG8pIHZpYSBKU1IuCiAgLSBjb3JyZWN0IGFsaWFzaW5nIG9mIGZ1bGx5LXF1YWxpZmllZCAiaW1wb3J0IiB0ZXh0IChlZzogIm5wbTpyZWFjdDoxOC4wLjAiIOKGkiAicmVhY3QiKQpgYGAKCgpgYGB5YW1sCmRlYnVnOiB0cnVlCmNvbXBvbmVudDogQ29uY2VwdFBsYXllcgp2aWRlbzogdmltZW8vNzI3OTUxNjc3CnRodW1ibmFpbHM6IHRydWUKdGltZXN0YW1wczogCiAgJzAwOjAwOjAwLjAwMCc6IAogICAgaW1hZ2U6IC9pbWFnZXMvdm9sY2Fuby5qcGcKICAnMDA6MDM6NTguMjE1JzogCiAgICBpbWFnZTogaHR0cHM6Ly93cnBjZC5uZXQvY2RuLWNnaS9pbWFnZWRlbGl2ZXJ5L0JYbHVReDRpZ2U5R3VXMElhNTZCSHcvMjhmNWI3ZWQtNjdkMS00MTlkLThkYjAtZDk1YWU5MGU4MTAwL3JlY3Rjb250YWluMwpgYGAKCiFbaW1hZ2VdKC9pbWFnZXMvdm9sY2Fuby5qcGcpCgoKYGBgeWFtbApkZWJ1ZzogdHJ1ZQpjb21wb25lbnQ6IFZpZGVvCnNyYzogdmltZW8vNzI3OTUxNjc3CmBgYAoKPHA+Jm5ic3A7PC9wPgoKLS0tCgoKIyBNYXJrZG93biBbU3ludGF4XShodHRwczovL21hcmtkb3duLWl0LmdpdGh1Yi5pby8pCkEgc2FtcGxlIGJyZWFrZG93biBvZiB0aGUgbWFpbiBtYXJrZG93biBzeW50YXggb3B0aW9ucyBhdmFpbGFibGUgdG8geW91Li4uKCDwn5C3ICk6CgoKIyMgSDEgSGVhZGVyCiMjIyBIMiBIZWFkZXIKIyMjIyBIMyBIZWFkZXIKIyMjIyMgSDQgSGVhZGVyCgotICoqQm9sZCB0ZXh0KiogYW5kICppdGFsaWMgdGV4dCogY2FuIGJlIGVtcGhhc2l6ZWQuCi0gTGlzdHM6CiAgLSBVbm9yZGVyZWQgaXRlbQogIC0gQW5vdGhlciBpdGVtCiAgICAtIFN1Yi1pdGVtCiAgICAtIEFub3RoZXIgc3ViLWl0ZW0KICAtIFRoaXJkIGl0ZW0KLSBPcmRlcmVkIGxpc3Q6CiAgMS4gRmlyc3QgaXRlbQogIDIuIFNlY29uZCBpdGVtCiAgMy4gVGhpcmQgaXRlbQoKPiBCbG9ja3F1b3RlcyBhcmUgdXNlZnVsIGZvciBoaWdobGlnaHRpbmcgdGV4dC4KCklubGluZSBgY29kZWAgYW5kIGNvZGUgYmxvY2tzOgoKYGBgdHMKY29uc3QgaGVsbG8gPSAoc3ViamVjdDogc3RyaW5nKSA9PiBg8J+RiyBIZWxsbywgJHtzdWJqZWN0fSFgOwpjb25zb2xlLmxvZyhoZWxsbygnV29ybGQnKSk7CmBgYAoKCkNvbG9yIHN5bnRheCBoaWdobGlnaHRpbmcgYWxsIG1ham9yIG1vZGVybiBsYW5ndWFnZXM6CgpgYGBweXRob24KIyDwn5GLCnByaW50KCJIZWxsbywgV29ybGQhIikKYGBgCgoKTGlua3M6ICAKW0ludGVybmFsIGxpbmtdKCNzZWN0aW9uKSB8IFtFeHRlcm5hbCBsaW5rXShodHRwczovL2V4YW1wbGUuY29tKSAgCgpGb290bm90ZSByZWZlcmVuY2VbXjFdLgoKVGFibGVzOgp8IEhlYWRlciAxIHwgSGVhZGVyIDIgfAp8LS0tLS0tLS0tLXwtLS0tLS0tLS0tfAp8IFJvdyAxICAgIHwgRGF0YSAxICAgfAp8IFJvdyAyICAgIHwgRGF0YSAyICAgfAoKW14xXTogVGhpcyBpcyBhIGZvb3Rub3RlLgoKVGhpcyBpcyBhIHBhcmFncmFwaCB3aXRoIGEgW3JlZmVyZW5jZS1zdHlsZSBsaW5rXVtyZWZdIGFuZCBhbm90aGVyIFtpbmxpbmUgbGlua10oaHR0cHM6Ly9leGFtcGxlLmNvbSkuCgpbcmVmXTogaHR0cHM6Ly9leGFtcGxlLmNvbSAiT3B0aW9uYWwgdGl0bGUiCg==", + "docs/public/images/volcano.jpg": "", "docs/section-a/item-a.md": "data:text/markdown;base64,IyBUaXRsZS1BICAKCioqTG9yZW0gaXBzdW0qKiBkb2xvciBzaXQgYW1ldCwgY29uc2VjdGV0dXIgYWRpcGlzY2luZyBlbGl0LiBRdWlzcXVlIG5lYyBxdWFtIGxvcmVtLiBQcmFlc2VudCBmZXJtZW50dW0sIGF1Z3VlIHV0IHBvcnRhIHZhcml1cywgZXJvcyBuaXNsIGV1aXNtb2QgYW50ZSwgYWMgc3VzY2lwaXQgZWxpdCBsaWJlcm8gbmVjIGRvbG9yLiBNb3JiaSBtYWduYSBlbmltLCBtb2xlc3RpZSBub24gYXJjdSBpZCwgdmFyaXVzIHNvbGxpY2l0dWRpbiBuZXF1ZS4gSW4gc2VkIHF1YW0gbWF1cmlzLiBBZW5lYW4gbWkgbmlzbCwgZWxlbWVudHVtIG5vbiBhcmN1IHF1aXMsIHVsdHJpY2VzIHRpbmNpZHVudCBhdWd1ZS4gVml2YW11cyBmZXJtZW50dW0gaWFjdWxpcyB0ZWxsdXMgZmluaWJ1cyBwb3J0dGl0b3IuIE51bGxhIGV1IHB1cnVzIGlkIGRvbG9yIGF1Y3RvciBzdXNjaXBpdC4gSW50ZWdlciBsYWNpbmlhIHNhcGllbiBhdCBhbnRlIHRlbXB1cyB2b2x1dHBhdC4KICA=", "docs/section-a/item-b.md": "data:text/markdown;base64,IyBUaXRsZS1CICAKCioqTG9yZW0gaXBzdW0qKiBkb2xvciBzaXQgYW1ldCwgY29uc2VjdGV0dXIgYWRpcGlzY2luZyBlbGl0LiBRdWlzcXVlIG5lYyBxdWFtIGxvcmVtLiBQcmFlc2VudCBmZXJtZW50dW0sIGF1Z3VlIHV0IHBvcnRhIHZhcml1cywgZXJvcyBuaXNsIGV1aXNtb2QgYW50ZSwgYWMgc3VzY2lwaXQgZWxpdCBsaWJlcm8gbmVjIGRvbG9yLiBNb3JiaSBtYWduYSBlbmltLCBtb2xlc3RpZSBub24gYXJjdSBpZCwgdmFyaXVzIHNvbGxpY2l0dWRpbiBuZXF1ZS4gSW4gc2VkIHF1YW0gbWF1cmlzLiBBZW5lYW4gbWkgbmlzbCwgZWxlbWVudHVtIG5vbiBhcmN1IHF1aXMsIHVsdHJpY2VzIHRpbmNpZHVudCBhdWd1ZS4gVml2YW11cyBmZXJtZW50dW0gaWFjdWxpcyB0ZWxsdXMgZmluaWJ1cyBwb3J0dGl0b3IuIE51bGxhIGV1IHB1cnVzIGlkIGRvbG9yIGF1Y3RvciBzdXNjaXBpdC4gSW50ZWdlciBsYWNpbmlhIHNhcGllbiBhdCBhbnRlIHRlbXB1cyB2b2x1dHBhdC4KICA=", - "package.json": "data:application/json;base64,ewogICJkZXBlbmRlbmNpZXMiOiB7CiAgICAiQHZpZHN0YWNrL3JlYWN0IjogIiIsCiAgICAiQHN5cy9zdGQiOiAibnBtOkBqc3Ivc3lzX19zdGQiLAogICAgIkBzeXMvdG1wIjogIm5wbTpAanNyL3N5c19fdG1wIiwKICAgICJAc3lzL3VpLWNzcyI6ICJucG06QGpzci9zeXNfX3VpLWNzcyIsCiAgICAicmVhY3QiOiAiIiwKICAgICJyZWFjdC1kb20iOiAiIiwKICAgICJ5YW1sIjogIiIKICB9LAogICJkZXZEZXBlbmRlbmNpZXMiOiB7CiAgICAiQHR5cGVzL3JlYWN0IjogIiIsCiAgICAiQHR5cGVzL3JlYWN0LWRvbSI6ICIiLAogICAgInZpdGVwcmVzcyI6ICIiLAogICAgInZ1ZSI6ICIiCiAgfQp9Cg==", + "imports.json": "data:application/json;base64,ewogICJpbXBvcnRzIjogewogICAgIkBzeXMvZHJpdmVyLWRlbm8iOiAianNyOkBzeXMvZHJpdmVyLWRlbm9AMC4wLjk3IiwKICAgICJAc3lzL2RyaXZlci12aXRlIjogImpzcjpAc3lzL2RyaXZlci12aXRlQDAuMC4xMzciLAogICAgIkBzeXMvZHJpdmVyLXZpdGVwcmVzcyI6ICJqc3I6QHN5cy9kcml2ZXItdml0ZXByZXNzQDAuMC4zMDIiLAogICAgIkBkZW5vL3ZpdGUtcGx1Z2luIjogIm5wbTpAZGVuby92aXRlLXBsdWdpbkAxLjAuNCIsCiAgICAiQHZpdGVqcy9wbHVnaW4tcmVhY3Qtc3djIjogIm5wbTpAdml0ZWpzL3BsdWdpbi1yZWFjdC1zd2NAMy44LjAiCiAgfQp9Cg==", + "package.json": "data:application/json;base64,ewogICJkZXBlbmRlbmNpZXMiOiB7CiAgICAiQHZpZHN0YWNrL3JlYWN0IjogIm5wbTpAdmlkc3RhY2svcmVhY3RAMS4xMi4xMiIsCiAgICAiQHN5cy9zdGQiOiAibnBtOkBqc3Ivc3lzX19zdGRAMC4wLjE0MSIsCiAgICAiQHN5cy90bXAiOiAibnBtOkBqc3Ivc3lzX190bXBAMC4wLjExMCIsCiAgICAiQHN5cy91aS1jc3MiOiAibnBtOkBqc3Ivc3lzX191aS1jc3NAMC4wLjgxIiwKICAgICJyZWFjdCI6ICJucG06cmVhY3RAMTguMy4xIiwKICAgICJyZWFjdC1kb20iOiAibnBtOnJlYWN0LWRvbUAxOC4zLjEiLAogICAgInlhbWwiOiAibnBtOnlhbWxAMi43LjAiCiAgfSwKICAiZGV2RGVwZW5kZW5jaWVzIjogewogICAgIkB0eXBlcy9yZWFjdCI6ICJucG06QHR5cGVzL3JlYWN0QDE4LjMuMTgiLAogICAgIkB0eXBlcy9yZWFjdC1kb20iOiAibnBtOkB0eXBlcy9yZWFjdC1kb21AMTguMy41IiwKICAgICJ2aXRlIjogIm5wbTp2aXRlQDYuMS4xIiwKICAgICJ2aXRlcHJlc3MiOiAibnBtOnZpdGVwcmVzc0AxLjYuMyIsCiAgICAidnVlIjogIm5wbTp2dWVAMy41LjEzIgogIH0KfQo=", "src/config.ts": "data:application/typescript;base64,CmV4cG9ydCBjb25zdCBDb25maWcgPSB7CiAgdGl0bGU6ICdVbnRpdGxlZCcsCiAgZGVzY3JpcHRpb246ICcnLCAvLyBSZW5kZXJlZCBpbiB0aGUgaGVhZCBvZiBlYWNoIHBhZ2UsIHVzZWZ1bCBmb3IgU0VPLgp9IGFzIGNvbnN0Owo=", "src/nav.ts": "data:application/typescript;base64,aW1wb3J0IHR5cGUgeyBEZWZhdWx0VGhlbWUgfSBmcm9tICd2aXRlcHJlc3MnOwoKLyoqCiAqIExlZnQgc2lkZWJhciBuYXZpZ2F0aW9uLgogKgogKiBEb2N1bWVudGF0aW9uOgogKiBodHRwczovL3ZpdGVwcmVzcy5kZXYvcmVmZXJlbmNlL2RlZmF1bHQtdGhlbWUtc2lkZWJhcgogKi8KZXhwb3J0IGNvbnN0IHNpZGViYXI6IERlZmF1bHRUaGVtZS5TaWRlYmFyID0gWwogIHsKICAgIHRleHQ6ICdTZWN0aW9uIFRpdGxlIEEnLAogICAgaXRlbXM6IFsKICAgICAgeyB0ZXh0OiAnSXRlbS1BJywgbGluazogJ3NlY3Rpb24tYS9pdGVtLWEnIH0sCiAgICAgIHsgdGV4dDogJ0l0ZW0tQicsIGxpbms6ICdzZWN0aW9uLWEvaXRlbS1iJyB9LAogICAgXSwKICB9LApdOwo=" } diff --git a/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/m.Tmpl.ts b/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/m.Tmpl.ts index 46b17591d0..09d195664a 100644 --- a/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/m.Tmpl.ts +++ b/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/m.Tmpl.ts @@ -1,10 +1,12 @@ -import { type t } from './common.ts'; -import { create } from './m.Tmpl.create.ts'; +import type { t } from './common.ts'; import { Bundle } from './m.Bundle.ts'; -import { update } from './m.Tmpl.update.ts'; +import { create } from './u.create.ts'; +import { prep } from './u.prep.ts'; +import { write } from './u.write.ts'; export const VitepressTmpl: t.VitepressTmplLib = { Bundle, + prep, create, - update, + write, }; diff --git a/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/t.ts b/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/t.ts index 9e738fbd17..f0dbf60cc9 100644 --- a/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/t.ts +++ b/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/t.ts @@ -11,8 +11,11 @@ export type VitepressTmplLib = { /** Creates an instance of the template file generator. */ create(args: t.VitepressTmplCreateArgs): Promise<t.Tmpl>; - /** Initialize the local machine environment with latest templates */ - update(args?: t.VitepressTmplUpdateArgs): Promise<t.VitepressTmplUpdateResponse>; + /** Write and process the templates to the local file-system. */ + write(args?: t.VitepressTmplWriteArgs): Promise<t.VitepressTmplWriteResponse>; + + /** Prepare the template with latest state and dependency versions. */ + prep(options?: { silent?: boolean }): Promise<t.ViteTmplPrepResponse>; }; /** Arguments passed to the `VitepressTmpl.create` method. */ @@ -34,7 +37,7 @@ export type VitepressBundleLib = { }; /** Arguments passed to the `VitePress.Tmpl.update` method. */ -export type VitepressTmplUpdateArgs = { +export type VitepressTmplWriteArgs = { force?: boolean; inDir?: t.StringDir; srcDir?: t.StringDir; @@ -45,4 +48,4 @@ export type VitepressTmplUpdateArgs = { /** * The response returned from an environment update. */ -export type VitepressTmplUpdateResponse = { readonly ops: t.TmplFileOperation[] }; +export type VitepressTmplWriteResponse = { readonly ops: t.TmplFileOperation[] }; diff --git a/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/m.Tmpl.create.ts b/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/u.create.ts similarity index 70% rename from code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/m.Tmpl.create.ts rename to code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/u.create.ts index 35e62c8188..b5476effa2 100644 --- a/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/m.Tmpl.create.ts +++ b/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/u.create.ts @@ -1,6 +1,6 @@ import { type t, Fs, PATHS, Tmpl } from './common.ts'; import { Bundle } from './m.Bundle.ts'; -import { createFileProcessor } from './u.file.ts'; +import { createFileProcessor } from './u.process.file.ts'; /** * Create a new instance of the bundled file template. @@ -11,7 +11,7 @@ export const create: t.VitepressTmplLib['create'] = async (args) => { /** * Ensure the templates are hydrated and ready to use. */ - const beforeCopy: t.TmplCopyHandler = async () => { + const beforeWrite: t.TmplWriteHandler = async () => { await Fs.remove(templatesDir); await Bundle.toFilesystem(templatesDir); }; @@ -19,11 +19,11 @@ export const create: t.VitepressTmplLib['create'] = async (args) => { /** * (š·) Perform additional setup here (as needed). */ - const afterCopy: t.TmplCopyHandler = async (e) => {}; + const afterWrite: t.TmplWriteHandler = async (e) => {}; /** * Template-engine instance. */ const processFile = createFileProcessor(args); - return Tmpl.create(templatesDir, { processFile, beforeCopy, afterCopy }); + return Tmpl.create(templatesDir, { processFile, beforeWrite, afterWrite }); }; diff --git a/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/u.prep.ts b/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/u.prep.ts new file mode 100644 index 0000000000..9c68126af6 --- /dev/null +++ b/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/u.prep.ts @@ -0,0 +1,55 @@ +import { type t, c, Cli, Esm, Fs, getWorkspaceModules, Semver } from './common.ts'; + +type O = Record<string, string>; + +/** + * Prepare the template with latest state, including making updates to deps/versions. + */ +export const prep: t.VitepressTmplLib['prep'] = async (options = {}) => { + const { modules } = await getWorkspaceModules(); + let deps: O = {}; + + const updateDenoJson = async () => { + const path = './src/-tmpl/imports.json'; + const current = (await Fs.readJson<t.DenoImportMapJson>(path)).data; + const imports = modules.latest(current?.imports ?? {}); + const next = { ...current, imports }; + await Fs.writeJson(path, next); + deps = { ...deps, ...imports }; + }; + + const updatePackageJson = async () => { + const path = './src/-tmpl/package.json'; + const current = (await Fs.readJson<t.PkgJsonNode>(path)).data; + const next: t.PkgJsonNode = { + ...current, + dependencies: modules.latest(current?.dependencies ?? {}), + devDependencies: modules.latest(current?.devDependencies ?? {}), + }; + + await Fs.writeJson(path, next); + deps = { ...deps, ...next.dependencies, ...next.devDependencies }; + }; + + await updateDenoJson(); + await updatePackageJson(); + + if (!options.silent) { + const table = Cli.table([]); + Object.entries(deps).forEach(([key, value]) => { + const m = Esm.parse(value); + const pkg = c.gray(` ${key}`); + const registry = c.gray(m.registry.toUpperCase()); + const version = Semver.Fmt.colorize(m.version); + table.push([pkg, version, registry]); + }); + + console.info(); + console.info(c.italic(c.gray('imports.json'))); + console.info(c.brightGreen(`Dependencies:`)); + console.info(table.toString()); + console.info(); + } + + return { deps }; +}; diff --git a/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/u.file.ts b/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/u.process.file.ts similarity index 55% rename from code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/u.file.ts rename to code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/u.process.file.ts index 56a91e4d76..5f9b0250c7 100644 --- a/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/u.file.ts +++ b/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/u.process.file.ts @@ -1,7 +1,4 @@ -import { pkg as pkgDeno } from '@sys/driver-deno'; -import { pkg as pkgVite } from '@sys/driver-vite'; -import { Main } from '@sys/main/cmd'; -import { type t, c, DenoDeps, Esm, Fs, PATHS, pkg, Pkg } from './common.ts'; +import { type t, PATHS, pkg } from './common.ts'; /** * File processing rules for the template. @@ -9,18 +6,6 @@ import { type t, c, DenoDeps, Esm, Fs, PATHS, pkg, Pkg } from './common.ts'; export function createFileProcessor(args: t.VitepressTmplCreateArgs): t.TmplProcessFile { const { srcDir = PATHS.srcDir } = args; - const getDeps = async (base: t.StringDir) => { - const from = DenoDeps.from; - const join = (...parts: string[]) => Fs.join(base, ...parts); - const load = async (path: string) => (await from(join(path))).data?.modules.items ?? []; - - const m1 = await load('.sys/deps.yaml'); - const m2 = await load('.sys/deps.sys.yaml'); - const modules = Esm.modules([...m1, ...m2]); - - return { modules }; - }; - return async (e) => { if (e.target.exists && is.userspace(e.target.relative)) { /** @@ -30,40 +15,18 @@ export function createFileProcessor(args: t.VitepressTmplCreateArgs): t.TmplProc return e.exclude('user-space'); } + if (e.contentType !== 'text') return; + if (e.target.relative === 'deno.json') { /** * Update versions in `deno.json`: */ const version = args.version ?? pkg.version; const importUri = `jsr:${pkg.name}@${version}`; - const text = e.text.tmpl - .replace(/<ENTRY>/g, `${importUri}/main`) - .replace(/<ENTRY_MAIN>/, `jsr:${Pkg.toString(Main.pkg)}`) - .replace(/<DRIVER_DENO>/, `jsr:${Pkg.toString(pkgDeno)}`) - .replace(/<DRIVER_VITE>/, `jsr:${Pkg.toString(pkgVite)}`) - .replace(/<DRIVER_VITEPRESS>/, `jsr:${Pkg.toString(pkg)}`); - + const text = e.text.tmpl.replace(/<ENTRY>/g, `${importUri}/main`); return e.modify(text); } - if (e.target.relative === 'package.json') { - const { modules } = await getDeps(e.target.base); - const pkg = (await Fs.readJson<t.PkgJsonNode>(e.tmpl.absolute)).data; - const next = { - ...pkg, - dependencies: modules.latest(pkg?.dependencies ?? {}), - devDependencies: modules.latest(pkg?.devDependencies ?? {}), - }; - - console.info(c.gray(`Resolved versions:`)); - console.info(c.brightCyan(c.bold(`./package.json:`))); - console.info(next); - console.info(); - - const json = `${JSON.stringify(next, null, ' ')}\n`; - return e.modify(json); - } - if (e.target.relative === 'docs/index.md') { const text = e.text.tmpl.replace(/\<DRIVER_VER\>/g, pkg.version); return e.modify(text); diff --git a/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/m.Tmpl.update.ts b/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/u.write.ts similarity index 68% rename from code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/m.Tmpl.update.ts rename to code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/u.write.ts index 995f8a8fb0..c2783638be 100644 --- a/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/m.Tmpl.update.ts +++ b/code/sys.driver/driver-vitepress/src/m.Vitepress.Tmpl/u.write.ts @@ -1,17 +1,17 @@ import { type t, c, Fs, Tmpl } from './common.ts'; -import { create } from './m.Tmpl.create.ts'; +import { create } from './u.create.ts'; /** - * Initialize the local machine environment with latest templates + * Write and process the templates to the local file-system. */ -export const update: t.VitepressTmplLib['update'] = async (args = {}) => { +export const write: t.VitepressTmplLib['write'] = async (args = {}) => { const { inDir = '', srcDir, version, force = false, silent = false } = args; /** * Update template files. */ const tmpl = await create({ inDir, srcDir, version }); - const copied = await tmpl.copy(inDir, { force }); + const copied = await tmpl.write(inDir, { force }); const { ops } = copied; /** @@ -19,7 +19,7 @@ export const update: t.VitepressTmplLib['update'] = async (args = {}) => { * eg. migration change patching. */ const remove = (...path: string[]) => Fs.remove(Fs.join(inDir, ...path)); - // await remove('./path/to/obsolete/file'); + // await remove('./<path>'); /** * Finish up. diff --git a/code/sys.driver/driver-vitepress/src/m.Vitepress/-backup.test.ts b/code/sys.driver/driver-vitepress/src/m.Vitepress/-backup.test.ts index d87d9ea96d..62e295e9b3 100644 --- a/code/sys.driver/driver-vitepress/src/m.Vitepress/-backup.test.ts +++ b/code/sys.driver/driver-vitepress/src/m.Vitepress/-backup.test.ts @@ -3,8 +3,8 @@ import { Sample } from '../m.Vitepress/-u.ts'; import { Vitepress } from '../m.Vitepress/mod.ts'; describe('cmd: backup (shapshot)', () => { - const assertExists = async (dir: string, exists = true) => { - expect(await Fs.exists(dir)).to.eql(exists, dir); + const assertExists = async (path: string, exists = true) => { + expect(await Fs.exists(path)).to.eql(exists, `path should exist: ${path}`); }; it('perform backup copy', { sanitizeResources: false, sanitizeOps: false }, async () => { @@ -14,19 +14,20 @@ describe('cmd: backup (shapshot)', () => { * - ā build (dist) * - ā backup (snapshot) */ - await Testing.retry(3, async () => { - const test = async (args: Pick<t.VitepressBackupArgs, 'includeDist'> = {}) => { + const test = async (args: Pick<t.VitepressBackupArgs, 'includeDist'> = {}) => { + await Testing.retry(2, async () => { const { includeDist } = args; const sample = Sample.init({}); + const cwd = sample.path; const inDir = sample.path; const backupDir = Fs.join(inDir, PATHS.backup); const distDir = Fs.join(inDir, PATHS.dist); const silent = true; - await Vitepress.Tmpl.update({ inDir, silent }); + await Vitepress.Tmpl.write({ inDir, silent }); await assertExists(distDir, false); // NB: not yet built. - await Vitepress.build({ inDir, silent }); + const buildResponse = await Vitepress.build({ inDir, silent }); await assertExists(backupDir, false); // NB: not yet backed up. const res = await Vitepress.backup({ dir: inDir, includeDist }); @@ -47,12 +48,12 @@ describe('cmd: backup (shapshot)', () => { await assertTargetExists('docs', true); await assertTargetExists('src', true); await assertTargetExists('deno.json', true); - await assertTargetExists('package.json', true); + await assertTargetExists('imports.json', true); await assertTargetExists('.gitignore', true); - }; + }); + }; - await test({}); // default: excludes the /dist folder. - await test({ includeDist: true }); - }); + await test({}); // default: excludes the /dist folder. + await test({ includeDist: true }); }); }); diff --git a/code/sys.driver/driver-vitepress/src/m.Vitepress/-build.test.ts b/code/sys.driver/driver-vitepress/src/m.Vitepress/-build.test.ts index 36e9729cfb..ceb3f837e2 100644 --- a/code/sys.driver/driver-vitepress/src/m.Vitepress/-build.test.ts +++ b/code/sys.driver/driver-vitepress/src/m.Vitepress/-build.test.ts @@ -24,7 +24,7 @@ describe('Vitepress.build', () => { const inDir = Fs.resolve(sample.path); const outDir = Fs.resolve(sample.path, 'dist'); - await Vitepress.Tmpl.update({ inDir }); + await Vitepress.Tmpl.write({ inDir }); const res = await Vitepress.build({ pkg, inDir, silent: false }); expect(res.ok).to.eql(true); @@ -65,7 +65,7 @@ describe('Vitepress.build', () => { const outDir = Fs.resolve(sample.path, '.vitepress/dist'); expect(await Fs.exists(outDir)).to.eql(false); // NB: clean initial condition. - await Vitepress.Tmpl.update({ inDir }); + await Vitepress.Tmpl.write({ inDir }); const res = await Vitepress.build({ pkg, inDir, outDir, silent: true }); expect(res.ok).to.eql(true); diff --git a/code/sys.driver/driver-vitepress/src/m.Vitepress/-dev.test.ts b/code/sys.driver/driver-vitepress/src/m.Vitepress/-dev.test.ts index 8a60a74f57..188608cd48 100644 --- a/code/sys.driver/driver-vitepress/src/m.Vitepress/-dev.test.ts +++ b/code/sys.driver/driver-vitepress/src/m.Vitepress/-dev.test.ts @@ -11,7 +11,7 @@ describe('VitePress.dev', () => { const open = false; const sample = Sample.init(); const { port, inDir } = sample; - await Vitepress.Tmpl.update({ inDir }); + await Vitepress.Tmpl.write({ inDir }); const server = await Vitepress.dev({ port, inDir, open }); // NB: await returns after Vitepress as completed it's startup. try { @@ -43,7 +43,7 @@ describe('VitePress.dev', () => { const open = false; const sample = Sample.init(); const { port, inDir } = sample; - await Vitepress.Tmpl.update({ inDir }); + await Vitepress.Tmpl.write({ inDir }); const server = await Vitepress.dev({ port, inDir, open }); await server.dispose(); diff --git a/code/sys.driver/driver-vitepress/src/m.Vitepress/-u.ts b/code/sys.driver/driver-vitepress/src/m.Vitepress/-u.ts index 5404691efb..2219d3f1d4 100644 --- a/code/sys.driver/driver-vitepress/src/m.Vitepress/-u.ts +++ b/code/sys.driver/driver-vitepress/src/m.Vitepress/-u.ts @@ -43,8 +43,8 @@ export const assertEnvExists = async (dir: t.StringDir, expected = true) => { await assert('.gitignore'); await assert('.vitepress/config.ts'); await assert('.vitepress/theme/index.ts'); - await assert('.sys/-main.ts'); await assert('deno.json'); + await assert('imports.json'); await assert('package.json'); await assert('docs/index.md'); }; diff --git a/code/sys.driver/driver-vitepress/src/m.Vitepress/u.build.ts b/code/sys.driver/driver-vitepress/src/m.Vitepress/u.build.ts index 81dd199ea7..85a2bf420e 100644 --- a/code/sys.driver/driver-vitepress/src/m.Vitepress/u.build.ts +++ b/code/sys.driver/driver-vitepress/src/m.Vitepress/u.build.ts @@ -11,29 +11,46 @@ export const build: B = async (input = {}) => { const options = wrangle.options(input); const { pkg, srcDir = 'docs', silent = false } = options; - const spinner = Cli.spinner(c.gray('building...'), { start: false }); - if (!silent) spinner.start(); - const dirs = wrangle.dirs(options); const inDir = dirs.in; const outDir = dirs.out; + if (!silent) { + const table = Cli.table([]); + const push = (label: string, ...value: string[]) => table.push([c.gray(label), ...value]); + push('Directory:', c.gray(`${Fs.cwd()}/`)); + push(' in:', Fs.trimCwd(dirs.in)); + push(' out:', Fs.trimCwd(dirs.out)); + + console.info(c.bold(c.brightGreen('Paths'))); + console.info(table.toString().trim()); + console.info(); + } + let params = `--outDir=${outDir}`; if (srcDir) params += ` --srcDir=${srcDir}`; const cmd = `deno run -A --node-modules-dir npm:vitepress build ${inDir} ${params}`; const args = cmd.split(' ').slice(1); + + const spinner = Cli.spinner(c.gray('building...'), { start: false }); + if (!silent) spinner.start(); + const output = await Process.invoke({ args, silent: true }); const ok = output.success; spinner?.clear().stop(); - // Write {pkg} into /dist so it's included within the digest-hash. + /** + * Write {pkg} into /dist so it's included within the digest-hash. + */ if (pkg) { const path = Fs.join(dirs.out, 'assets', '-pkg.json'); await Fs.writeJson(path, pkg); } - // Calculate the `/dist.json` file and digest-hash. + /** + * Calculate the digest-hash and store it in the [dist.json] file. + */ const entry = './index.html'; const dist = (await Pkg.Dist.compute({ dir: dirs.out, pkg, entry, save: true })).dist; const elapsed = timer.elapsed.msec; diff --git a/code/sys.driver/driver-vitepress/src/pkg.ts b/code/sys.driver/driver-vitepress/src/pkg.ts index 79cb3f9c80..fc97160599 100644 --- a/code/sys.driver/driver-vitepress/src/pkg.ts +++ b/code/sys.driver/driver-vitepress/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/driver-vitepress', version: '0.0.302' }; diff --git a/code/sys.driver/driver-vitepress/src/types.ts b/code/sys.driver/driver-vitepress/src/types.ts index 41c005ea17..aafcbde0b6 100644 --- a/code/sys.driver/driver-vitepress/src/types.ts +++ b/code/sys.driver/driver-vitepress/src/types.ts @@ -7,5 +7,3 @@ export type * from './-entry/t.ts'; export type * from './m.Vitepress.Log/t.ts'; export type * from './m.Vitepress.Tmpl/t.ts'; export type * from './m.Vitepress/t.ts'; - -export type * from './ui/components/t.ts'; diff --git a/code/sys.driver/driver-vitepress/src/ui/-.test.ts b/code/sys.driver/driver-vitepress/src/ui/-.test.ts deleted file mode 100644 index 6107677d00..0000000000 --- a/code/sys.driver/driver-vitepress/src/ui/-.test.ts +++ /dev/null @@ -1,8 +0,0 @@ -import { type t, describe, it, expect } from '../-test.ts'; - -describe('UI.Components', () => { - it('import ā API', async () => { - const { FOO } = await import('@sys/driver-vitepress/ui'); - console.log('FOO', FOO); - }); -}); diff --git a/code/sys.driver/driver-vitepress/src/ui/common/mod.ts b/code/sys.driver/driver-vitepress/src/ui/common/mod.ts deleted file mode 100644 index 049ed939e5..0000000000 --- a/code/sys.driver/driver-vitepress/src/ui/common/mod.ts +++ /dev/null @@ -1,8 +0,0 @@ -export type * as t from './t.ts'; - -/** - * Libs - */ -export { rx } from '@sys/std'; -export { Color, css } from '@sys/ui-css'; -export { FC } from '@sys/ui-react'; diff --git a/code/sys.driver/driver-vitepress/src/ui/common/t.ts b/code/sys.driver/driver-vitepress/src/ui/common/t.ts deleted file mode 100644 index 95ae0dbb3a..0000000000 --- a/code/sys.driver/driver-vitepress/src/ui/common/t.ts +++ /dev/null @@ -1,3 +0,0 @@ -export type { CssValue } from '@sys/ui-css/t'; - -export type * from '../../common/t.ts'; diff --git a/code/sys.driver/driver-vitepress/src/ui/components/t.Time.ts b/code/sys.driver/driver-vitepress/src/ui/components/t.Time.ts deleted file mode 100644 index 0e181545ba..0000000000 --- a/code/sys.driver/driver-vitepress/src/ui/components/t.Time.ts +++ /dev/null @@ -1,25 +0,0 @@ -import type { t } from './common.ts'; - -/** - * https://w3c.github.io/webvtt - * https://en.wikipedia.org/wiki/WebVTT - * - * @example - * - * WEBVTT - * - * 1 - * 00:00:00.165 --> 00:00:01.735 - * This is a model for understanding - * - * 2 - * 00:00:01.875 --> 00:00:03.335 - * and working with group scale. - * - */ -export type Timestamps = { [key: StringTime]: Timestamp }; -export type Timestamp = { - image?: t.StringUrl; -}; - -export type StringTime = string; // HH:MM:SS.mmm diff --git a/code/sys.driver/driver-vitepress/src/ui/components/t.ts b/code/sys.driver/driver-vitepress/src/ui/components/t.ts deleted file mode 100644 index 32cc611faa..0000000000 --- a/code/sys.driver/driver-vitepress/src/ui/components/t.ts +++ /dev/null @@ -1,20 +0,0 @@ -import type { t } from './common.ts'; -export type * from './t.Time.ts'; - -export type YamlFences = ConceptPlayerDef; - -export type YamlFenceBase = { - debug?: boolean; -}; - -/** - * A YAML block that defines a concept player. - */ -export type ConceptPlayerDef = YamlFenceBase & { - component: 'ConceptPlayer'; - video?: StringVideoUri; - timestamps?: t.Timestamps; -}; - -export type StringVideoUri = StringVimeoUri; -export type StringVimeoUri = t.StringUri; // "vimeo/<id>" diff --git a/code/sys.driver/driver-vitepress/src/ui/components/u.ConceptPlayer.tsx b/code/sys.driver/driver-vitepress/src/ui/components/u.ConceptPlayer.tsx deleted file mode 100644 index 3cf6eb0862..0000000000 --- a/code/sys.driver/driver-vitepress/src/ui/components/u.ConceptPlayer.tsx +++ /dev/null @@ -1,30 +0,0 @@ -import { useEffect, useRef, useState } from 'react'; -import { type t, Color, css, FC, rx } from './common.ts'; - -export type ConceptPlayerProps = { - theme?: t.CommonTheme; - style?: t.CssInput; -}; - -export const ConceptPlayer: React.FC<ConceptPlayerProps> = (props) => { - const {} = props; - - /** - * Render - */ - const t = (ms: t.Msecs, ...attr: string[]) => attr.map((prop) => `${prop} ${ms}ms ease-in-out`); - const transition = t(50, 'opacity'); - const theme = Color.theme(props.theme); - const styles = { - base: css({ - backgroundColor: 'rgba(255, 0, 0, 0.1)' /* RED */, - color: theme.fg, - }), - }; - - return ( - <div style={css(styles.base, props.style)}> - <div>{`š· ConceptPlayer`}</div> - </div> - ); -}; diff --git a/code/sys.driver/driver-vitepress/src/ui/mod.ts b/code/sys.driver/driver-vitepress/src/ui/mod.ts deleted file mode 100644 index 5471b6a4a3..0000000000 --- a/code/sys.driver/driver-vitepress/src/ui/mod.ts +++ /dev/null @@ -1,5 +0,0 @@ -/** - * @module - * UI Components - */ -export const FOO = 'š·'; diff --git a/code/sys.tmp/scripts/-clean.ts b/code/sys.tmp/-scripts/-clean.ts similarity index 100% rename from code/sys.tmp/scripts/-clean.ts rename to code/sys.tmp/-scripts/-clean.ts diff --git a/code/sys.tmp/-scripts/-tmp.ts b/code/sys.tmp/-scripts/-tmp.ts new file mode 100644 index 0000000000..625639163e --- /dev/null +++ b/code/sys.tmp/-scripts/-tmp.ts @@ -0,0 +1 @@ +console.log('tmp', 'š·'); diff --git a/code/sys.tmp/-tmp/-sample.lib/-.test.tsx b/code/sys.tmp/-tmp/-sample.lib/-.test.tsx deleted file mode 100644 index 408f6eb4c8..0000000000 --- a/code/sys.tmp/-tmp/-sample.lib/-.test.tsx +++ /dev/null @@ -1,23 +0,0 @@ -// import { type t, describe, it, expect } from '../-test.ts'; -// -// describe('TMP', () => { -// it('load', async () => { -// // const m1 = await import('./my.js'); -// const m2 = await import('./dist/entry.lib.mjs'); -// -// console.log(`-------------------------------------------`); -// console.log('m2', m2); -// -// const foo = await m2.fn(); -// console.log(`-------------------------------------------`); -// console.log('foo', foo); -// -// const Component = foo.FooSample; -// -// const el = <Component />; -// -// console.log('el', el); -// -// // console.log('foo.msg', foo.msg); -// }); -// }); diff --git a/code/sys.tmp/.gitignore b/code/sys.tmp/.gitignore new file mode 100644 index 0000000000..409ee189ef --- /dev/null +++ b/code/sys.tmp/.gitignore @@ -0,0 +1 @@ +npm/ diff --git a/code/sys.tmp/deno.json b/code/sys.tmp/deno.json index 45a7b2374a..adb9e31252 100644 --- a/code/sys.tmp/deno.json +++ b/code/sys.tmp/deno.json @@ -1,18 +1,18 @@ { "name": "@sys/tmp", - "version": "0.0.95", + "version": "0.0.110", "license": "MIT", "tasks": { "test": "deno test -RW", "lint": "deno lint", "dry": "deno publish --allow-dirty --dry-run", - "clean": "deno run -RWE ./scripts/-clean.ts" + "clean": "deno run -RWE ./-scripts/-clean.ts", + "tmp": "deno run -A ./-scripts/-tmp.ts" }, "exports": { ".": "./src/mod.ts", "./t": "./src/types.ts", "./types": "./src/types.ts", - "./tmp": "./src/mod.tmp.ts", "./ui": "./src/ui/mod.ts", "./sample-imports": "./src/-sample/-sample-imports.ts" } diff --git a/code/sys.tmp/src/common/libs.ts b/code/sys.tmp/src/common/libs.ts index 2fe37cdfca..c9c4908f65 100644 --- a/code/sys.tmp/src/common/libs.ts +++ b/code/sys.tmp/src/common/libs.ts @@ -1 +1 @@ -export { rx, Time, Pkg } from '@sys/std'; +export { isRecord, Path, Pkg, rx, Time } from '@sys/std'; diff --git a/code/sys.tmp/src/mod.tmp.ts b/code/sys.tmp/src/mod.tmp.ts deleted file mode 100644 index 0bd335e846..0000000000 --- a/code/sys.tmp/src/mod.tmp.ts +++ /dev/null @@ -1,11 +0,0 @@ -/** - * @module - * Temp module export - * - * ```ts - * import { Tmp } from '@sys/tmp/tmp'; - * ``` - */ - -export type TmpLib = { count: number }; -export const Tmp: TmpLib = { count: 0 }; diff --git a/code/sys.tmp/src/pkg.ts b/code/sys.tmp/src/pkg.ts index 79cb3f9c80..838ac10ec9 100644 --- a/code/sys.tmp/src/pkg.ts +++ b/code/sys.tmp/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/tmp', version: '0.0.110' }; diff --git a/code/sys.tmp/src/types.ts b/code/sys.tmp/src/types.ts index 0f2f882807..b1a0424bf3 100644 --- a/code/sys.tmp/src/types.ts +++ b/code/sys.tmp/src/types.ts @@ -2,4 +2,4 @@ * @module * Module types. */ -export {}; +export type * from './ui/Foo/t.ts'; diff --git a/code/sys.tmp/src/ui/Foo/api.Signals.ts b/code/sys.tmp/src/ui/Foo/api.Signals.ts new file mode 100644 index 0000000000..e31e2f4f74 --- /dev/null +++ b/code/sys.tmp/src/ui/Foo/api.Signals.ts @@ -0,0 +1,2 @@ +import { Signal } from '@sys/ui-react'; +export const count = Signal.create(0); diff --git a/code/sys.tmp/src/ui/Foo/common.ts b/code/sys.tmp/src/ui/Foo/common.ts new file mode 100644 index 0000000000..30ca005008 --- /dev/null +++ b/code/sys.tmp/src/ui/Foo/common.ts @@ -0,0 +1,2 @@ +export * from '../common.ts'; +export { Color } from '@sys/color'; diff --git a/code/sys.tmp/src/ui/Foo/mod.ts b/code/sys.tmp/src/ui/Foo/mod.ts new file mode 100644 index 0000000000..ee815bfe34 --- /dev/null +++ b/code/sys.tmp/src/ui/Foo/mod.ts @@ -0,0 +1 @@ +export * from './ui.tsx'; diff --git a/code/sys.tmp/src/ui/Foo/t.ts b/code/sys.tmp/src/ui/Foo/t.ts new file mode 100644 index 0000000000..0486df5f9e --- /dev/null +++ b/code/sys.tmp/src/ui/Foo/t.ts @@ -0,0 +1,6 @@ +import type { t } from './common.ts'; + +export type FooProps = { + enabled?: boolean; + style?: t.CssInput; +}; diff --git a/code/sys.tmp/src/ui/Foo/ui.tsx b/code/sys.tmp/src/ui/Foo/ui.tsx new file mode 100644 index 0000000000..8ef05e6f6a --- /dev/null +++ b/code/sys.tmp/src/ui/Foo/ui.tsx @@ -0,0 +1,50 @@ +import React, { useState } from 'react'; +import { count } from './api.Signals.ts'; +import { type t, css, pkg, Signal } from './common.ts'; + +/** + * Component (UI). + */ +export const Foo: React.FC<t.FooProps> = (props) => { + const { enabled = true } = props; + let text = `import ā ${pkg.name}@${pkg.version}/ui:<Foo> | ā”ļø:signal/count:${count.value}`; + + const [isOver, setOver] = useState(false); + const over = (isOver: boolean) => () => setOver(isOver); + + const [, setRender] = useState(0); + const redraw = () => setRender((n) => n + 1); + + /** + * Lifecycle + */ + Signal.useSignalEffect(() => { + count.value; + redraw(); + }); + + /** + * Render: + */ + const styles = { + base: css({ + display: 'inline-block', + cursor: 'pointer', + backgroundColor: `rgba(255, 0, 0, ${isOver ? 0.1 : 0.03})` /* RED */, + }), + code: css({}), + }; + + if (!enabled) text += ' (disabled)'; + + return ( + <div + className={styles.base.class} + onMouseEnter={over(true)} + onMouseLeave={over(false)} + onClick={() => count.value++} + > + <code className={styles.code.class}>š· {text}</code> + </div> + ); +}; diff --git a/code/sys.tmp/src/ui/common.ts b/code/sys.tmp/src/ui/common.ts index a34108591b..1ae8dab12c 100644 --- a/code/sys.tmp/src/ui/common.ts +++ b/code/sys.tmp/src/ui/common.ts @@ -1,2 +1,16 @@ +/** + * @external + */ +import { signal, useSignal, useSignalEffect, effect } from '@preact/signals-react'; +export const Signal = { + signal, + effect, + useSignal, + useSignalEffect, +} as const; + +/** + * @system + */ export { Color, css, Style } from '@sys/ui-css'; export * from '../common.ts'; diff --git a/code/sys.tmp/src/ui/mod.ts b/code/sys.tmp/src/ui/mod.ts index 533b790110..c165919e89 100644 --- a/code/sys.tmp/src/ui/mod.ts +++ b/code/sys.tmp/src/ui/mod.ts @@ -2,5 +2,4 @@ * @module * Sample UI modules and components. */ -export { Foo } from './ui.Foo.tsx'; -export { VideoPlayer } from './ui.VideoPlayer.tsx'; +export { Foo } from './Foo/mod.ts'; diff --git a/code/sys.tmp/src/ui/ui.Foo.tsx b/code/sys.tmp/src/ui/ui.Foo.tsx deleted file mode 100644 index 3e796001e3..0000000000 --- a/code/sys.tmp/src/ui/ui.Foo.tsx +++ /dev/null @@ -1,35 +0,0 @@ -import React from 'react'; -import { type t, css, pkg } from './common.ts'; - -/** - * Sample properties. - */ -export type FooProps = { - enabled?: boolean; - style?: t.CssInput; -}; - -/** - * Component (UI). - */ -export const Foo: React.FC<FooProps> = (props) => { - const { enabled = true } = props; - let text = `${pkg.name}@${pkg.version}/ui:<Foo>`; - - const [isOver, setOver] = React.useState(false); - const over = (isOver: boolean) => () => setOver(isOver); - - const styles = { - base: css({ - display: 'inline-block', - backgroundColor: `rgba(255, 0, 0, ${isOver ? 0.9 : 0.3})` /* RED */, - }), - }; - - if (!enabled) text += ' (disabled)'; - return ( - <div className={styles.base.class} onMouseEnter={over(true)} onMouseLeave={over(false)}> - <code>š· {text}</code> - </div> - ); -}; diff --git a/code/sys.tmp/src/ui/ui.VideoPlayer.tsx b/code/sys.tmp/src/ui/ui.VideoPlayer.tsx deleted file mode 100644 index 61f264a277..0000000000 --- a/code/sys.tmp/src/ui/ui.VideoPlayer.tsx +++ /dev/null @@ -1,45 +0,0 @@ -// @ts-types="@types/react" -import React from 'react'; -import { type t, css } from './common.ts'; - -import '@vidstack/react/player/styles/base.css'; -import '@vidstack/react/player/styles/plyr/theme.css'; - -import { MediaPlayer, MediaProvider } from '@vidstack/react'; -import { PlyrLayout, plyrLayoutIcons } from '@vidstack/react/player/layouts/plyr'; - -export const DEFAULTS = { - src: 'vimeo/499921561', // Tubes. -} as const; - -/** - * Sample properties. - */ -export type VideoPlayerProps = { - title?: string; - src?: string; - style?: t.CssInput; -}; - -/** - * Component (UI). - */ -export const VideoPlayer: React.FC<VideoPlayerProps> = (props) => { - const src = props.src || DEFAULTS.src; - - const styles = { - base: css({}), - }; - - const elPlayer = ( - <MediaPlayer title={props.title} src={src} playsInline={true}> - <MediaProvider /> - <PlyrLayout - // thumbnails="https://files.vidstack.io/sprite-fight/thumbnails.vtt" - icons={plyrLayoutIcons} - /> - </MediaPlayer> - ); - - return <div className={css(styles.base, props.style).class}>{elPlayer}</div>; -}; diff --git a/code/sys.ui/ui-css/README.md b/code/sys.ui/ui-css/README.md index 0448a7a3d3..905e0824dd 100644 --- a/code/sys.ui/ui-css/README.md +++ b/code/sys.ui/ui-css/README.md @@ -1,5 +1,5 @@ # Style/CSS -Tools for working with Styles/CSS programatically (aka "css-in-js"). +Tools for working with strongly-typed styles/css programatically (aka. "css-in-js"). Note: This approach is a pure JS-to-CSS/DOM approach, with zero-dependencies on any other special bundler plugin, or post-css processing type dependencies, which inevitably cause @@ -12,15 +12,28 @@ Applying within a JSX programming style idiom: ```tsx import { css } from '@sys/ui-css'; -export function Component(props:{}) { - const styles = { - base: css({ padding: 10 }), - }; - return <div className={style.base.class}>{'š Hello World!'}</div> +export function MyComponent() { + const base = css({ padding: 10 }) + return <div className={base.class}>{'š Hello World!'}</div> } ``` +If your component takes styles as an incoming property, these can be merged with +internal component styles like so: + + +```tsx +import { css, type CssInput } from '@sys/ui-css'; + +export function MyComponent(props: { style?: CssInput } = {}) { + const base = css({ padding: 10 }) + return <div className={css(base, props.style).class}>{'š Hello'}</div> +} +``` + + + ### CSSProps Transformation pipeline. @@ -58,3 +71,55 @@ The CSS class-name is generated from the hash of the style ( `\<prefix\>-<hx>` ) and calls to this function are memoized, keyed on the hash, to ensure the function is safe to use in "render heavy" frameworks like React (et al.). +## Scope + +### @container + +The @container contextual rule-block, used for creating media-queries at the component level, can be +applied like so: + +```tsx +const styles = { + base: css({ + containerType: 'size', // š¼ ā NB: enable @container rules (width AND height). + // containerType: 'inline-size', // š¼ ā NB: enable @container rules (width only). + }), + + h2: css({ + color: 'red', + fontSize: 50, + transition: 'font-size 200ms, color 200ms', + }) + .container('min-width: 400px', { fontSize: 90, color: 'blue' }) + .container('min-width: 600px', { fontSize: 150, color: 'salmon' }) + .container('max-height: 470px', { display: 'none' }).done, + +}; + +return ( + <div className={styles.base.class}> + <h2 className={styles.h2.class}>š Hello</h2> + </div> +); +``` + +Arbitrary CSS selectors can be scoped within a base container like so: + +```tsx +const base = css({ position: 'relative' }) + .rule('h2', { color: 'red' }) + .rule('h2 code', { color: 'blue' }) + + +return ( + <div className={base.class}> + <h2> + {`š Hello`} <code>World</code> + </h2> + </div> +); +``` + + +--- +- [cross-ref:cast](https://warpcast.com/pjc/0x59783042): conversational description, [template expansions](https://warpcast.com/pjc/0xa908939e) \ No newline at end of file diff --git a/code/sys.ui/ui-css/deno.json b/code/sys.ui/ui-css/deno.json index 18155c239b..72c94a0331 100644 --- a/code/sys.ui/ui-css/deno.json +++ b/code/sys.ui/ui-css/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/ui-css", - "version": "0.0.70", + "version": "0.0.81", "license": "MIT", "tasks": { "lint": "deno lint", diff --git a/code/sys.ui/ui-css/src/-test/common.ts b/code/sys.ui/ui-css/src/-test/common.ts index 9a23332674..659277240d 100644 --- a/code/sys.ui/ui-css/src/-test/common.ts +++ b/code/sys.ui/ui-css/src/-test/common.ts @@ -1,2 +1,2 @@ -export { c, describe, DomMock, expect, it, Testing } from '@sys/testing/server'; +export { c } from '@sys/color/ansi'; export * from '../common.ts'; diff --git a/code/sys.ui/ui-css/src/-test/mod.ts b/code/sys.ui/ui-css/src/-test/mod.ts index ad7ecebfd7..26777a717b 100644 --- a/code/sys.ui/ui-css/src/-test/mod.ts +++ b/code/sys.ui/ui-css/src/-test/mod.ts @@ -1,3 +1,5 @@ +export { beforeEach, describe, DomMock, expect, it, Testing } from '@sys/testing/server'; + export * from './common.ts'; export * from './u.Find.ts'; export * from './u.Print.ts'; diff --git a/code/sys.ui/ui-css/src/-test/u.Find.ts b/code/sys.ui/ui-css/src/-test/u.Find.ts index 8aee355a52..a2ee464b9a 100644 --- a/code/sys.ui/ui-css/src/-test/u.Find.ts +++ b/code/sys.ui/ui-css/src/-test/u.Find.ts @@ -2,21 +2,53 @@ * Helpers for querying the DOM of CSS rules. */ export const FindCss = { - rule(className: string): CSSStyleRule | undefined { + rule(className: string): CSSStyleRule | CSSGroupingRule | undefined { return FindCss.rules(className)[0]; }, - rules(className: string): CSSStyleRule[] { - const res: CSSStyleRule[] = []; + rules(className: string): (CSSStyleRule | CSSGroupingRule)[] { + const res: (CSSStyleRule | CSSGroupingRule)[] = []; const selector = `.${className.replace(/^\./, '')}`.trim(); - for (const sheet of document.styleSheets) { - try { - for (const rule of sheet.cssRules) { - const text = ((rule as any).selectorText || '').trim(); - if (text.startsWith(selector)) res.push(rule as CSSStyleRule); + + // š³ Recursive function to search through CSSRuleList. + // If a grouping rule (e.g. @container) contains a matching rule, + // then return the grouping rule so that the context details are visible. + const searchRules = (ruleList: CSSRuleList): (CSSStyleRule | CSSGroupingRule)[] => { + const found: (CSSStyleRule | CSSGroupingRule)[] = []; + for (const rule of Array.from(ruleList)) { + // Check if this is a grouping rule with nested rules. + if ( + 'cssRules' in rule && + rule.cssRules && + typeof rule.cssText === 'string' && + rule.cssText.trim().startsWith('@') + ) { + // Recursively search its nested rules. + const nestedFound = searchRules(rule.cssRules as any); + if (nestedFound.length > 0) { + // If any nested rule matches, return the grouping rule itself. + found.push(rule as CSSGroupingRule); + continue; + } + } + // Otherwise, if this is a CSSStyleRule, check its selector. + if ('selectorText' in rule && rule.selectorText) { + const text = (rule as any).selectorText.trim(); + if (text.startsWith(selector)) { + found.push(rule as CSSStyleRule); + } } + } + return found; + }; + + for (const sheet of Array.from(document.styleSheets)) { + try { + const found = searchRules(sheet.cssRules); + res.push(...found); } catch (error) { - continue; // NB: Some styleSheets might not be accessible due to cross-origin restrictions. + // NB: Some styleSheets might not be accessible due to cross-origin restrictions. + continue; } } return res; diff --git a/code/sys.ui/ui-css/src/-test/u.Print.ts b/code/sys.ui/ui-css/src/-test/u.Print.ts index a763766862..28f4bdcbef 100644 --- a/code/sys.ui/ui-css/src/-test/u.Print.ts +++ b/code/sys.ui/ui-css/src/-test/u.Print.ts @@ -1,14 +1,34 @@ -import { type t, c } from './common.ts'; +import { type t, c, Str } from './common.ts'; + +const info = console.info; +const cyan = c.brightCyan; +const y = c.yellow; export const TestPrint = { transformed(m: t.CssTransformed) { - console.info(); - console.info(c.brightCyan(`CssTransform:`)); - console.info(m); - console.info(); - console.info(`ā.${c.brightCyan('style')}:`, m.style); - console.info(`ā.${c.brightCyan('class')}:`, `"${c.yellow(m.class)}"`); - console.info(`ā.${c.brightCyan('toString()')}:`, `"${c.yellow(m.toString())}"`); - console.info(); + info(); + info(cyan(`CssTransformed:`)); + info(m); + info(); + info(`ā.${cyan('style')}:`, m.style); + info(`ā.${cyan('class')}:`, `"${y(m.class)}"`); + info(`ā.${cyan('toString()')}:`, `"${y(m.toString())}" ${c.gray('ā default: CssRule')}`); + info( + `ā.${cyan('toString(CssSelector)')}:`, + `"${y(Str.truncate(m.toString('CssSelector'), 60))}"`, + ); + info(); + }, + + container(m: t.CssDomContainerBlock) { + info(); + info(cyan(`CssDomContainerBlock:`)); + info(m); + info(`ā.${cyan('toString()')}:`, `"${y(m.toString())}" ${c.gray('ā default: QueryCondition')}`); + info( + `ā.${cyan('toString(CssSelector)')}:`, + `"${y(Str.truncate(m.toString('CssSelector'), 50))}"`, + ); + info(); }, }; diff --git a/code/sys.ui/ui-css/src/common/t.ts b/code/sys.ui/ui-css/src/common/t.ts index 5119073c6e..9f3baf0288 100644 --- a/code/sys.ui/ui-css/src/common/t.ts +++ b/code/sys.ui/ui-css/src/common/t.ts @@ -1,3 +1,3 @@ export type { ColorLib } from '@sys/color/t'; -export type { Falsy } from '@sys/types'; +export type * from '@sys/types'; export type * from '../types.ts'; diff --git a/code/sys.ui/ui-css/src/m.Css.Dom/-.test.ts b/code/sys.ui/ui-css/src/m.Css.Dom/-.test.ts index f4addb6954..8ae54c6a84 100644 --- a/code/sys.ui/ui-css/src/m.Css.Dom/-.test.ts +++ b/code/sys.ui/ui-css/src/m.Css.Dom/-.test.ts @@ -1,33 +1,83 @@ -import { type t, describe, DomMock, expect, FindCss, it, pkg } from '../-test.ts'; +import { type t, describe, DomMock, expect, FindCss, it, pkg, slug } from '../-test.ts'; import { css } from '../m.Style/mod.ts'; import { DEFAULT } from './common.ts'; import { CssDom } from './mod.ts'; +import { getStylesheetId } from './u.ts'; + +const toString = CssDom.toString; describe( 'Style: CSS ClassName', - - /** NB: leaked timers left around by the "happy-dom" module. */ - { sanitizeOps: false, sanitizeResources: false }, - + { sanitizeOps: false, sanitizeResources: false }, // ā because: "Happy-Dom" () => { DomMock.polyfill(); - describe('create (instance)', () => { - it('prefix: default', () => { - const a = CssDom.create(''); - const b = CssDom.create(' '); - const c = CssDom.create(); - expect(a.prefix).to.eql(DEFAULT.prefix); - expect(b.prefix).to.eql(DEFAULT.prefix); - expect(c.prefix).to.eql(DEFAULT.prefix); + let _count = 0; + const setup = () => { + _count++; + const sheet = CssDom.stylesheet({ instance: `mysheet-${_count}` }); + const classes = sheet.classes(`foo-${_count}`); + return { sheet, classes } as const; + }; + + describe('factory: create <Stylesheet> instance', () => { + it('instance id: default and custom', () => { + const a = CssDom.stylesheet({}); + const b = CssDom.stylesheet({ instance: ' foo ' }); + expect(a.id).to.eql(pkg.name); + expect(a.rules.list).to.eql([]); + expect(b.id).to.eql(`${pkg.name}:foo`); + }); + + it('instance id (as string param)', () => { + const id = slug(); + const sheet = CssDom.stylesheet(id); + expect(sheet.id).to.eql(`${pkg.name}:${id}`); + }); + + it('singleton pooling (instance reuse on data-id)', () => { + const a = CssDom.stylesheet(); + const b = CssDom.stylesheet({ instance: ' ' }); + const c = CssDom.stylesheet({ instance: 'bar' }); + expect(a).to.equal(b); + expect(a).to.not.equal(c); + }); + + it('insert root <style> into DOM (singleton)', () => { + const test = (instance?: t.StringId, classPrefix?: string) => { + const id = getStylesheetId(instance, classPrefix); + + const find = () => document.querySelector(`style[data-controller="${id}"]`); + CssDom.stylesheet({ instance, classPrefix }); + + expect(find()).to.exist; + CssDom.stylesheet({ instance, classPrefix }); + expect(find()).to.equal(find()); // NB: singleton. + }; + + test(); + test('foobar'); + test('foobar', 'my-class-prefix'); }); + }); - it('custom prefix', () => { + describe('.classes(): prefixed class/style DOM insertion', () => { + it('should create <classes> API with default prefix', () => { + const dom = CssDom.stylesheet(); + const a = dom.classes(); + const b = dom.classes(); + expect(a.prefix).to.eql(DEFAULT.classPrefix); + expect(a).to.equal(b); + }); + + it('should create <classes> API with custom prefix', () => { const test = (prefix: string, expected: string) => { - const ns = CssDom.create(prefix); - expect(ns.prefix).to.eql(expected); + const sample = setup(); + const classes = sample.sheet.classes(prefix); + expect(classes.prefix).to.eql(expected); }; test('foo', 'foo'); + test('.foo', 'foo'); test(' foo ', 'foo'); test(' foo- ', 'foo'); // NB: trimmed. test(' foo-- ', 'foo'); @@ -35,25 +85,19 @@ describe( test('foo-123', 'foo-123'); }); - it('pooling (instance reuse keyed on "prefix")', () => { - const a = CssDom.create(); - const b = CssDom.create(DEFAULT.prefix); - const c = CssDom.create('foo'); - expect(a).to.equal(b); - expect(a).to.not.equal(c); - }); - - it('insert root <style> into DOM (singleton)', () => { - const find = () => document.querySelector(`style[data-controller="${pkg.name}"]`); - CssDom.create(); - expect(find()).to.exist; - CssDom.create(); - expect(find()).to.equal(find()); // Singleton. + it('should pass default "classPrefix" value to .classes() API', () => { + const a = CssDom.stylesheet({ instance: slug() }); + const b = CssDom.stylesheet({ instance: slug(), classPrefix: 'foo' }); + const styleA = a.classes().add({ fontSize: 16 }); + const styleB = b.classes().add({ fontSize: 16 }); + expect(styleA.startsWith('sys-')).to.eql(true); + expect(styleB.startsWith('foo-')).to.eql(true); }); it('throw: invalid prefix', () => { + const { sheet } = setup(); const test = (prefix: string) => { - const fn = () => CssDom.create(prefix); + const fn = () => sheet.classes(prefix); expect(fn).to.throw( /String must start with a letter and can contain letters, digits, and hyphens \(hyphen not allowed at the beginning\)/, ); @@ -64,29 +108,21 @@ describe( test('-'); test('foo*bar'); }); - }); - - describe('class/style DOM insertion', () => { - let count = 0; - const setup = (): t.CssDom => { - count++; - const prefix = `sample${count}`; - return CssDom.create(prefix); - }; - it('simple ("hx" not passed)', () => { - const dom = setup(); + it('add: simple ("hx" hash not passed)', () => { + const { sheet } = setup(); + const classes = sheet.classes(); const m = css({ fontSize: 32, display: 'grid', PaddingX: [5, 10] }); - expect(dom.classes.length).to.eql(0); // NB: no "inserted classes" yet. + expect(classes.names.length).to.eql(0); // NB: no "inserted classes" yet. // Baseline: ensure the rule is not yet within the DOM. - const className = `${dom.prefix}-${m.hx}`; + const className = `${classes.prefix}-${m.hx}`; expect(FindCss.rule(className)).to.eql(undefined); // NB: nothing inserted yet. - const a = dom.class(m.style); - const b = dom.class(m.style); + const a = classes.add(m.style); + const b = classes.add(m.style); - expect(dom.classes.length).to.eql(1); // NB: not added twice. + expect(classes.names.length).to.eql(1); // NB: not added twice. expect(a).to.eql(b); expect(a).to.eql(className); @@ -95,28 +131,264 @@ describe( expect(rule?.cssText).to.eql(`.${className} { ${m.toString()} }`); }); - it('hash passed as parameter', () => { - const dom = setup(); - const m = css({ fontSize: 32, display: 'grid' }); + it('add: CSS template ā { Absolute: 0 }', () => { + const { sheet } = setup(); + const classes = sheet.classes(); + const className = classes.add({ Absolute: 0 }); + expect(FindCss.rule(className)?.cssText).to.include( + '{ position: absolute; top: 0px; right: 0px; bottom: 0px; left: 0px; }', + ); + }); + + it('add: hash passed as parameter', () => { + const { sheet } = setup(); + const classes = sheet.classes(); + const { style, hx } = css({ fontSize: 32, display: 'grid' }); - const className = `${dom.prefix}-${m.hx}`; + const className = `${classes.prefix}-${hx}`; expect(FindCss.rule(className)).to.eql(undefined); // NB: nothing inserted yet. - dom.class(m.style, m.hx); + sheet.classes().add(style, { hx }); const rule = FindCss.rule(className); - expect(rule?.cssText).to.eql(`.${className} { ${m.toString()} }`); + expect(rule?.cssText).to.eql(`.${className} { ${toString(style)} }`); }); - describe('pseudo-class', () => { + it('add: CSS template ā { Absolute: 0 }', () => { + const { sheet } = setup(); + const a = sheet.rules.add('.foo', { Absolute: 0 }); + const b = sheet.rule('.bar', { Absolute: 0 }); + + const expected = { position: 'absolute', top: 0, right: 0, bottom: 0, left: 0 }; + expect(a[0].style).to.eql(expected); + expect(b[0].style).to.eql(expected); + }); + + describe('pseudo-classes', () => { it(':hover', () => { - const dom = setup(); - const m = css({ color: 'red', ':hover': { color: ' salmon ' } }); - const className = dom.class(m.style, m.hx); + const { sheet } = setup(); + const classes = sheet.classes(); + const { style, hx } = css({ color: 'red', ':hover': { color: ' salmon ' } }); + const className = classes.add(style, { hx }); const rules = FindCss.rules(className); expect(rules[0].cssText).to.eql(`.${className} { color: red; }`); expect(rules[1].cssText).to.eql(`.${className}:hover { color: salmon; }`); }); }); }); + + describe('.rule(): arbitrary CSS-selector DOM insertion', () => { + it('should insert a simple rule into the stylesheet', () => { + const { sheet } = setup(); + const selector = '.test-rule'; + const style = { color: 'blue', margin: 10 }; + + // Pre-condition. + expect(FindCss.rule(selector)).to.eql(undefined); + expect(sheet.rules.list).to.eql([]); + expect(sheet.rules.length).to.eql(0); + + // Insert the rule. + const res = sheet.rule(selector, style); + expect(res.length).to.eql(1); + expect(res[0].selector).to.eql(selector); + expect(res[0].style).to.eql(style); + + expect(sheet.rules.length).to.eql(1); + expect(sheet.rules.list.length).to.eql(1); + expect(sheet.rules.list).to.eql(res); + + // Verify that the rule is inserted in the DOM. + const rule = FindCss.rule(selector); + expect(rule).to.exist; + expect(rule?.cssText).to.eql(`${selector} { ${toString(style)} }`); + expect(rule?.cssText).to.eql(`.test-rule { color: blue; margin: 10px; }`); // NB: ā (same/same). + }); + + it('should not insert the same rule twice', () => { + const { sheet } = setup(); + const selector = '.test-rule'; + const style = { margin: 10 }; + + const insert = () => sheet.rule(selector, style); + const res = insert(); + + expect(res.length).to.eql(1); + expect(res[0].selector).to.eql(selector); + expect(res[0].style).to.eql(style); + + // NB: further calls do not add more items. + expect(insert()).to.eql([]); + expect(insert()).to.eql([]); + expect(sheet.rules.list.length).to.eql(1); + }); + + describe('rules within context-blocks', () => { + it('should insert a simple rule within an "@container" context', () => { + const { sheet } = setup(); + const selector = `.test-container-${slug()}`; + const style = { color: 'blue', margin: 10 }; + const context = '@container (min-width: 700px)'; + expect(FindCss.rule(selector)).to.eql(undefined); + + // Insert the rule within an "@container" context. + const res = sheet.rule(selector, style, { context }); + expect(res.length).to.eql(1); + expect(res[0].selector).to.eql(selector); + expect(res[0].style).to.eql(style); + + // Verify that the rule is inserted in the DOM, wrapped in the context block. + const rule = FindCss.rule(selector); + expect(rule).to.exist; + const expectedCssText = `${context} { ${selector} { ${toString(style)} } }`; + expect(rule?.cssText).to.eql(expectedCssText); + }); + + it('should insert multiple rules within an "@container" context', () => { + const { sheet } = setup(); + const selector = `.test-container-${slug()}`; + const context = '@container (min-width: 700px)'; + const styles = [ + { color: 'blue', margin: 10 }, + { backgroundColor: 'yellow', padding: 5 }, + ]; + + // Pre-check: Ensure no rule exists for the selector. + expect(FindCss.rules(selector)).to.eql([]); + + // Insert the rules with an "@container" context. + const res = sheet.rule(selector, styles, { context }); + expect(res.length).to.eql(2); + expect(res[0].style).to.eql(styles[0]); + expect(res[1].style).to.eql(styles[1]); + + // Retrieve all inserted rules for the selector. + const rules = FindCss.rules(selector); + expect(rules).to.have.length(2); + + // Verify that each rule is inserted in the DOM, wrapped in the context block. + const expected1 = `${context} { ${selector} { ${toString(styles[0])} } }`; + const expected2 = `${context} { ${selector} { ${toString(styles[1])} } }`; + expect(rules[0].cssText).to.eql(expected1); + expect(rules[1].cssText).to.eql(expected2); + }); + }); + + describe('pseudo-classes', () => { + it('should insert pseudo-class rules along with the base rule', () => { + const { sheet } = setup(); + const selector = '.test-pseudo'; + const style = { + color: 'red', + ':hover': { color: 'green' }, + }; + + // Insert the rule. + sheet.rule(selector, style); + const rules = FindCss.rules(selector); + + // Expect one base rule and one pseudo-class rule. + expect(rules).to.have.length(2); + expect(rules[0].cssText).to.eql(`${selector} { ${toString({ color: 'red' })} }`); + expect(rules[1].cssText).to.eql(`${selector}:hover { ${toString({ color: 'green' })} }`); + }); + + it('should insert multiple pseudo-class rules', () => { + const { sheet } = setup(); + const selector = '.test-multi'; + const style = { + fontSize: '14px', + ':active': { fontSize: '16px' }, + ':focus': { fontWeight: 'bold' }, + }; + + // Insert the rule. + sheet.rule(selector, style); + const rules = FindCss.rules(selector); + + // Expect 1 base rule and 2 pseudo rules. + expect(rules).to.have.length(3); + expect(rules[0].cssText).to.eql(`${selector} { ${toString({ fontSize: '14px' })} }`); + expect(rules[1].cssText).to.eql( + `${selector}:active { ${toString({ fontSize: '16px' })} }`, + ); + expect(rules[2].cssText).to.eql( + `${selector}:focus { ${toString({ fontWeight: 'bold' })} }`, + ); + }); + + it('should ignore pseudo-class rules if the value is not an object', () => { + const test = (invalidValue: any) => { + const { sheet } = setup(); + const selector = `.test-invalid-value-${slug()}`; + const style = { + color: 'blue', + ':hover': invalidValue, // invalid; value must be a {record/object}. + }; + + sheet.rule(selector, style); + const rules = FindCss.rules(selector); + + // Only the base rule should be inserted. + expect(rules).to.have.length(1); + expect(rules[0].cssText).to.eql(`${selector} { ${toString({ color: 'blue' })} }`); + }; + + const NON = ['not-an-object', 123, true, null, undefined, BigInt(0), Symbol('foo'), []]; + NON.forEach(test); + }); + + it('should ignore keys that are not valid pseudo-classes', () => { + const { sheet } = setup(); + const selector = '.test-non-pseudo'; + const style = { + color: 'blue', + ':nonexistent': { color: 'red' }, // not in our DEFAULT pseudo-class set. + }; + + sheet.rule(selector, style); + const rules = FindCss.rules(selector); + + // Only the base rule should be inserted. + expect(rules).to.have.length(1); + expect(rules[0].cssText).to.eql(`${selector} { ${toString({ color: 'blue' })} }`); + }); + + it('should insert an empty pseu0o-class rule when given an empty style object', () => { + const { sheet } = setup(); + const selector = '.test-empty-pseudo'; + const style = { + color: 'blue', + ':hover': {}, // empty nested style. + }; + + sheet.rule(selector, style); + const rules = FindCss.rules(selector); + + // Expect a base rule and a pseudo-class rule, even if the nested style is empty. + expect(rules).to.have.length(2); + expect(rules[0].cssText).to.eql(`${selector} { ${toString({ color: 'blue' })} }`); + expect(rules[1].cssText).to.eql(`${selector}:hover { ${toString({})} }`); + }); + + it('should prevent duplicate pseudo-class rules when the same style is inserted twice', () => { + const { sheet } = setup(); + const selector = '.test-duplicate-pseudo'; + const style = { + color: 'blue', + ':hover': { color: 'red' }, + }; + + // Call rule() twice with the same selector and style. + sheet.rule(selector, style); + sheet.rule(selector, style); + const rules = FindCss.rules(selector); + + // If duplicate prevention is implemented, there should be only one base rule and one pseudo-class rule. + expect(rules).to.have.length(2); + expect(rules[0].cssText).to.eql(`${selector} { ${toString({ color: 'blue' })} }`); + expect(rules[1].cssText).to.eql(`${selector}:hover { ${toString({ color: 'red' })} }`); + }); + }); + }); }, ); diff --git a/code/sys.ui/ui-css/src/m.Css.Dom/-ctx.container.test.ts b/code/sys.ui/ui-css/src/m.Css.Dom/-ctx.container.test.ts new file mode 100644 index 0000000000..921ccc78ea --- /dev/null +++ b/code/sys.ui/ui-css/src/m.Css.Dom/-ctx.container.test.ts @@ -0,0 +1,210 @@ +import { describe, DomMock, expect, FindCss, it, slug, TestPrint } from '../-test.ts'; +import { CssDom } from './mod.ts'; + +const toString = CssDom.toString; + +describe( + 'Stylesheet.container(): scoped @container context', + { sanitizeOps: false, sanitizeResources: false }, // ā because: "Happy-Dom" + () => { + DomMock.polyfill(); + + let _count = 0; + const setup = () => { + _count++; + const sheet = CssDom.stylesheet({ instance: `mysheet-${_count}` }); + return { sheet } as const; + }; + + describe('create', () => { + it('create', () => { + const { sheet } = setup(); + const container = sheet.container(' min-width: 700px '); + TestPrint.container(container); + expect(container.kind).to.eql('@container'); + expect(container.condition).to.eql('(min-width: 700px)'); + expect(container.name).to.eql(undefined); + expect(container.rules.length).to.eql(0); + expect(container.rules.list).to.eql([]); + expect(container.scoped).to.eql([]); + }); + + it("create: cleans up the context's <condition> input", () => { + const { sheet } = setup(); + const test = (condition: string, expected?: string) => { + const container = sheet.container(condition); + expect(container.condition).to.eql(expected ?? condition.trim()); + }; + test(' min-width: 700px ', '(min-width: 700px)'); // NB: parentheses added. + test(' (max-width: 1200px) and (orientation: landscape)'); + }); + + it('create with name', () => { + const { sheet } = setup(); + const container = sheet.container(' card ', ' min-width: 700px '); + TestPrint.container(container); + expect(container.kind).to.eql('@container'); + expect(container.name).to.eql('card'); + expect(container.condition).to.eql('(min-width: 700px)'); + }); + }); + + describe('toString', () => { + it('(default) ā kind: "QueryCondition"', () => { + const { sheet } = setup(); + const a = sheet.container('min-width: 700px'); + const b = sheet.container(' min-width: 700px '); + const c = sheet.container(' my-name ', ' min-width: 700px '); + expect(a.toString()).to.eql('@container (min-width: 700px)'); + expect(b.toString()).to.eql(a.toString()); + expect(c.toString()).to.eql('@container my-name (min-width: 700px)'); + expect(b.toString()).to.eql(b.toString('QueryCondition')); // NB: default kind. + }); + + it('kind: "CssSelector"', () => { + const { sheet } = setup(); + const container = sheet.container('my-name', 'min-width: 700px'); + + const a = container.toString(); + const b = container.toString('CssSelector'); + + container.rules.add('.foo', { fontSize: 16 }); + const c = container.toString('CssSelector'); + container.rules.add('.bar', { color: 'red' }); + const d = container.toString('CssSelector'); + + expect(a).to.eql('@container my-name (min-width: 700px)'); + expect(b).to.eql('@container my-name (min-width: 700px) {}'); + expect(c).to.eql('@container my-name (min-width: 700px) { .foo { font-size: 16px; } }'); + expect(d.split('\n')[0]).to.eql( + '@container my-name (min-width: 700px) { .foo { font-size: 16px; } }', + ); + expect(d.split('\n')[1]).to.eql( + '@container my-name (min-width: 700px) { .bar { color: red; } }', + ); + + TestPrint.container(container); + }); + }); + + describe('adding rules', () => { + it('adds each rule only once', () => { + const { sheet } = setup(); + const styles = [ + { color: 'blue', margin: 10 }, + { backgroundColor: 'yellow', padding: 5 }, + ]; + + const container = sheet.container('min-width: 700px'); + expect(container.rules.list).to.eql([]); + expect(container.rules.length).to.eql(0); + + const selector = `.test-${slug()}`; + const res1 = container.rules.add(selector, styles); + expect(res1.length).to.eql(2); + expect(container.rules.length).to.eql(2); + + // Additional calls with the same style content is not inserted. + expect(container.rules.add(selector, styles)).to.eql([]); // NB: second time - not repeated + expect(container.rules.add(selector, styles[0])).to.eql([]); + expect(container.rules.add(selector, styles[1])).to.eql([]); + expect(container.rules.add(selector, [])).to.eql([]); + + expect(container.rules.list.length).to.eql(2); + expect(container.rules.list[0].selector).to.eql(selector); + expect(container.rules.list[0].style).to.eql(styles[0]); + expect(container.rules.list[1].selector).to.eql(selector); + expect(container.rules.list[1].style).to.eql(styles[1]); + }); + + it('add: CSS template ā { Absolute: 0 }', () => { + const { sheet } = setup(); + const container = sheet.container('min-width: 700px'); + expect(container.rules.list).to.eql([]); + + const selector = `.test-${slug()}`; + const res = container.rules.add(selector, { Absolute: 0 }); + expect(res[0].style).to.eql({ position: 'absolute', top: 0, right: 0, bottom: 0, left: 0 }); + }); + + it('adds to DOM stylesheet', () => { + const { sheet } = setup(); + const selector = `.test-${slug()}`; + const context = '@container (min-width: 700px)'; + const styles = [ + { color: 'blue', margin: 10 }, + { backgroundColor: 'yellow', padding: 5 }, + ]; + + // Pre-check: Ensure no rule exists for the selector. + expect(FindCss.rules(selector)).to.eql([]); + + const container = sheet.container('min-width: 700px'); + container.rules.add(selector, styles); + + // Retrieve all inserted rules for the selector. + const rules = FindCss.rules(selector); + expect(rules).to.have.length(2); + + // Verify that each rule is inserted in the DOM, wrapped in the context block. + const expected1 = `${context} { ${selector} { ${toString(styles[0])} } }`; + const expected2 = `${context} { ${selector} { ${toString(styles[1])} } }`; + expect(rules[0].cssText).to.eql(expected1); + expect(rules[1].cssText).to.eql(expected2); + }); + + it('scenario: 1', () => { + const { sheet } = setup(); + const container = sheet.container('min-width: 600px'); + sheet.rule('.card h2', { fontSize: 50 }); + container.rules.add('.card h2', { fontSize: 200 }); + expect(container.rules.list[0].rule).to.eql( + `@container (min-width: 600px) { .card h2 { font-size: 200px; } }`, + ); + }); + }); + + describe('scope', () => { + it('nested ".classname" selector', () => { + const { sheet } = setup(); + const a = sheet.container('my-name', 'min-width: 700px'); + const b = a.scope('.foo'); + + expect(b.condition).to.eql(a.condition); + expect(b.name).to.eql(a.name); + expect(b).to.not.equal(a); + + expect(a.scoped).to.eql([]); + expect(b.scoped).to.eql(['.foo']); + + a.rules.add('h1', { color: 'red' }); + b.rules.add('h1', { color: 'red' }); + + expect(a.toString('CssSelector')).to.eql( + '@container my-name (min-width: 700px) { h1 { color: red; } }', + ); + + expect(b.toString('CssSelector')).to.eql( + '@container my-name (min-width: 700px) { .foo h1 { color: red; } }', + ); + }); + + it('multi-level nesting', () => { + const { sheet } = setup(); + const a = sheet.container('min-width: 700px'); + const b = a.scope('.foo'); + const c = b.scope('.bar'); + expect(c.scoped).to.eql(['.foo', '.bar']); + + b.rules.add('h2', { color: 'red' }); + c.rules.add('h2', { color: 'red' }); + + const str1 = b.toString('CssSelector'); + const str2 = c.toString('CssSelector'); + + expect(str1).to.eql('@container (min-width: 700px) { .foo h2 { color: red; } }'); + expect(str2).to.eql('@container (min-width: 700px) { .foo .bar h2 { color: red; } }'); + }); + }); + }, +); diff --git a/code/sys.ui/ui-css/src/m.Style/-toString.test.ts b/code/sys.ui/ui-css/src/m.Css.Dom/-toString.test.ts similarity index 95% rename from code/sys.ui/ui-css/src/m.Style/-toString.test.ts rename to code/sys.ui/ui-css/src/m.Css.Dom/-toString.test.ts index db3db832f9..57a9b63087 100644 --- a/code/sys.ui/ui-css/src/m.Style/-toString.test.ts +++ b/code/sys.ui/ui-css/src/m.Css.Dom/-toString.test.ts @@ -1,5 +1,5 @@ import { type t, describe, expect, it } from '../-test.ts'; -import { toString } from './u.toString.ts'; +import { toString } from './mod.ts'; describe('toString', () => { it('empty', () => { diff --git a/code/sys.ui/ui-css/src/m.Css.Dom/common.ts b/code/sys.ui/ui-css/src/m.Css.Dom/common.ts index b5d899b190..fda62824db 100644 --- a/code/sys.ui/ui-css/src/m.Css.Dom/common.ts +++ b/code/sys.ui/ui-css/src/m.Css.Dom/common.ts @@ -1,17 +1,20 @@ -export * from '../common.ts'; - -export { toHash }; import toHash from 'hash-it'; +import { pixelProps } from './const.pixelProps.ts'; +import { pseudoClasses } from './const.pseudoClasses.ts'; export * from '../common.ts'; export { CssTmpl } from '../m.Css.Tmpl/mod.ts'; -export { toString } from '../m.Style/u.toString.ts'; +export { toHash }; +/** + * Constants. + */ export const DEFAULT = { - prefix: 'sys', + classPrefix: 'sys', get pseudoClasses() { return pseudoClasses; }, + get pixelProps() { + return pixelProps; + }, } as const; - -const pseudoClasses = new Set<string>([':hover']); diff --git a/code/sys.ui/ui-css/src/m.Css.Dom/const.pixelProps.ts b/code/sys.ui/ui-css/src/m.Css.Dom/const.pixelProps.ts new file mode 100644 index 0000000000..3e019bd184 --- /dev/null +++ b/code/sys.ui/ui-css/src/m.Css.Dom/const.pixelProps.ts @@ -0,0 +1,36 @@ +/** + * CSS properties that accept unitless + * numbers (equating to "px" pixels). + */ +export const pixelProps = new Set<string>([ + 'width', + 'height', + 'top', + 'right', + 'bottom', + 'left', + 'margin', + 'marginTop', + 'marginRight', + 'marginBottom', + 'marginLeft', + 'padding', + 'paddingTop', + 'paddingRight', + 'paddingBottom', + 'paddingLeft', + 'borderWidth', + 'borderTopWidth', + 'borderRightWidth', + 'borderBottomWidth', + 'borderLeftWidth', + 'borderRadius', + 'fontSize', + 'minWidth', + 'maxWidth', + 'minHeight', + 'maxHeight', + 'outlineWidth', + 'letterSpacing', + 'wordSpacing', +]); diff --git a/code/sys.ui/ui-css/src/m.Css.Dom/const.pseudoClasses.ts b/code/sys.ui/ui-css/src/m.Css.Dom/const.pseudoClasses.ts new file mode 100644 index 0000000000..0b9e7be587 --- /dev/null +++ b/code/sys.ui/ui-css/src/m.Css.Dom/const.pseudoClasses.ts @@ -0,0 +1,59 @@ +/** + * CSS Selectors Level 3 pseudoāclasses. + * https://www.w3.org/TR/selectors-3 + */ +export const level3PseudoClasses = new Set<string>([ + ':hover', + ':active', + ':focus', + ':visited', + ':link', + ':target', + ':checked', + ':disabled', + ':enabled', + ':first-child', + ':last-child', + ':only-child', + ':nth-child', + ':nth-last-child', + ':first-of-type', + ':last-of-type', + ':only-of-type', + ':empty', + ':root', + ':not', + ':lang', +]); + +/** + * CSS Selectors Level 4 pseudoāclasses (and newer form/structural pseudoāclasses). + * https://www.w3.org/TR/selectors-4 + */ +export const level4PseudoClasses = new Set<string>([ + ':focus-visible', + ':focus-within', + ':any-link', + ':default', + ':indeterminate', + ':in-range', + ':invalid', + ':optional', + ':out-of-range', + ':placeholder-shown', + ':read-only', + ':read-write', + ':required', + ':valid', + ':user-invalid', + ':defined', + ':is', + ':where', + ':has', + ':dir', +]); + +/** + * CSS Pseudo-Classes (Levels 3 & 4). + */ +export const pseudoClasses = new Set<string>([...level3PseudoClasses, ...level4PseudoClasses]); diff --git a/code/sys.ui/ui-css/src/m.Css.Dom/mod.ts b/code/sys.ui/ui-css/src/m.Css.Dom/mod.ts index 263f4f4c71..a8c3453d2b 100644 --- a/code/sys.ui/ui-css/src/m.Css.Dom/mod.ts +++ b/code/sys.ui/ui-css/src/m.Css.Dom/mod.ts @@ -2,9 +2,13 @@ * @module * Tools for programatically managing CSS stylesheets within the browser DOM. */ -import { type t, DEFAULT } from './common.ts'; -import { create } from './u.create.ts'; +import { type t } from './common.ts'; +import { create as stylesheet } from './u.stylesheet.ts'; +import { toString } from './u.toString.ts'; + +export { toString }; export const CssDom: t.CssDomLib = { - create, + stylesheet, + toString, }; diff --git a/code/sys.ui/ui-css/src/m.Css.Dom/t.ctx.ts b/code/sys.ui/ui-css/src/m.Css.Dom/t.ctx.ts new file mode 100644 index 0000000000..12f801b1c6 --- /dev/null +++ b/code/sys.ui/ui-css/src/m.Css.Dom/t.ctx.ts @@ -0,0 +1,37 @@ +import type { t } from './common.ts'; + +/** + * Represents a CSS/DOM context-block that encapsulates a set of CSS rules + * applied within a @container context. + */ +export type CssDomContainerBlock = { + /** The type of the context-block. */ + readonly kind: '@container'; + + /** The conditional rules for the context block, eg "min-width: 700px". */ + readonly condition: string; + + /** The name of the container (empty if unnamed)/ */ + readonly name?: string; + + /** Raw rule API. */ + readonly rules: { + /** The total number of inserted rules. */ + readonly length: number; + /** List of inserted rules wihtin the container. */ + readonly list: Readonly<t.CssDomInsertedRule[]>; + /** Inserts CSS styles with the given selector within a context-block. */ + add(selector: t.StringCssSelector, style: t.CssValue | t.CssValue[]): t.CssDomInsertedRule[]; + }; + + /** String representation of the block. */ + toString(kind?: t.CssDomContainerToStringKind): string; + + /** Creates a scoped sub-block prefixing the child rules with the given selector. */ + scope(selector: t.StringCssSelector): CssDomContainerBlock; + /** The list of CSS selectors that represent the scope this container is within. */ + readonly scoped: Readonly<t.StringCssSelector[]>; +}; + +/** Flags indicating the kind of string to export from the `toString` method. */ +export type CssDomContainerToStringKind = 'QueryCondition' | 'CssSelector'; diff --git a/code/sys.ui/ui-css/src/m.Css.Dom/t.ts b/code/sys.ui/ui-css/src/m.Css.Dom/t.ts index c1de12bac2..28b80a29e2 100644 --- a/code/sys.ui/ui-css/src/m.Css.Dom/t.ts +++ b/code/sys.ui/ui-css/src/m.Css.Dom/t.ts @@ -1,24 +1,121 @@ import type { t } from './common.ts'; +export type * from './t.ctx.ts'; /** * Tools for programatically managing CSS stylesheets within the browser DOM. */ export type CssDomLib = { - /** Generator factory. */ - create(prefix?: string): t.CssDom; + /** Factory for a DOM <style> stylesheet element (singleton instances). */ + stylesheet(options?: CssDomStylesheetOptions | t.StringId): t.CssDomStylesheet; + + /** Convert a {style} props object to a CSS string. */ + toString: t.StyleLib['toString']; }; +/** Options passed to the `Style.Dom.stylesheet` method. */ +export type CssDomStylesheetOptions = { instance?: t.StringId; classPrefix?: string }; + /** * A <style> DOM element used to store and manage CSS-classes * generated from CssProps */ -export type CssDom = { - /** The root prefix applied to generated class-names. */ +export type CssDomStylesheet = { + readonly id: t.StringId; + + /** + * Inserts CSS style rules into the stylesheet. + * Accepts either a single style object or an array of style objects. + * Optionally, a context string can be provided to wrap the rules. + * + * Example: + * + * api.add(".card h2", { fontSize: "2em" }, { context: "@container (min-width: 700px)" }); + * + * // or multiple style objects: + * api.add(".card", [ + * { color: "blue" }, + * { margin: "1rem", ":hover": { color: "red" } } + * ], { context: "@media (min-width: 600px)" }); + */ + rule( + selector: string, + style: t.CssValue | t.CssValue[], + options?: CssDomRuleOptions, + ): CssDomInsertedRule[]; + /** Rules API. */ + readonly rules: t.CssDomRules; + + /** + * Retrieve the singleton instance of the classes API + * with the given classname "prefix". + */ + classes(prefix?: string): t.CssDomClasses; + + /** + * Retrieve the singleton instance of an @container API + * for specifying style rules within a specific size container. + */ + container(condition: string): t.CssDomContainerBlock; + container(name: string, condition: string): t.CssDomContainerBlock; +}; + +/** + * API for inserting CSS class-styes into a DOM's stylesheet. + */ +export type CssDomClasses = { + /** The root prefix applied to generated class-names: "<prefix>-<hash>". */ readonly prefix: string; - /** List of CSS class-names that have been inserted into the DOM. */ - readonly classes: Readonly<string[]>; + /** List of CSS class-names that have been inserted into the DOM. */ + readonly names: Readonly<string[]>; + + /** + * Generates a CSS classname as the selector and inserts the given + * {Style} object as a set of rules into the DOM (with caching). + */ + add(style: t.CssValue, options?: { hx?: number }): string; +}; + +/** + * API for inserting CSS rules into a DOM's stylesheet. + */ +export type CssDomRules = { + /** The total number of inserted rules. */ + readonly length: number; + + /** List of CSS rules that have been inserted into the DOM. */ + readonly list: Readonly<CssDomInsertedRule[]>; + + /** + * Inserts generic CSS style rules into the stylesheet. + * Accepts either a single style object or an array of style objects. + * Optionally, a context string can be provided to wrap the rules. + * + * Example: + * + * api.add(".card h2", { fontSize: "2em" }, { context: "@container (min-width: 700px)" }); + * + * // or multiple style objects: + * api.add(".card", [ + * { color: "blue" }, + * { margin: "1rem", ":hover": { color: "red" } } + * ], { context: "@media (min-width: 600px)" }); + * + * @returns true if inserted, or false if already inserted. + */ + add( + selector: string, + style: t.CssValue | t.CssValue[], + options?: CssDomRuleOptions, + ): CssDomInsertedRule[]; +}; + +/** Options passed to the rule insertion method. */ +export type CssDomRuleOptions = { context?: string }; - /** Generates a CSS class-name and inserts the given {Style} into the DOM. */ - class(style: t.CssProps, hx?: number): string; +/** The receipt of a rule that has been inserted into the DOM. */ +export type CssDomInsertedRule = { + readonly selector: string; + readonly style: t.CssProps; + readonly rule: string; }; diff --git a/code/sys.ui/ui-css/src/m.Css.Dom/u.classes.ts b/code/sys.ui/ui-css/src/m.Css.Dom/u.classes.ts new file mode 100644 index 0000000000..fb47d6fbe2 --- /dev/null +++ b/code/sys.ui/ui-css/src/m.Css.Dom/u.classes.ts @@ -0,0 +1,37 @@ +import { type t, DEFAULT, toHash, V } from './common.ts'; +import { AlphanumericWithHyphens } from './u.ts'; + +export function createClasses(args: { rules: t.CssDomRules; prefix?: string }): t.CssDomClasses { + const { rules } = args; + const inserted = new Set<string>(); + const prefix = wrangleClassPrefix(args.prefix); + V.parse(AlphanumericWithHyphens, prefix); + + const api: t.CssDomClasses = { + prefix, + get names() { + return Array.from(inserted); + }, + add(style, options = {}) { + const hx = options.hx ?? toHash(style); + const className = `${prefix}-${hx}`; + if (inserted.has(className)) { + return className; + } else { + rules.add(`.${className}`, style); + inserted.add(className); + return className; + } + }, + }; + + return api; +} + +/** + * Helpers + */ +export function wrangleClassPrefix(input: string | undefined, defaultPrefix?: string) { + const res = (input ?? '').trim() || (defaultPrefix ?? DEFAULT.classPrefix); + return res.replace(/^\.*/, '').replace(/-*$/, ''); +} diff --git a/code/sys.ui/ui-css/src/m.Css.Dom/u.create.ts b/code/sys.ui/ui-css/src/m.Css.Dom/u.create.ts deleted file mode 100644 index ec9b1acdff..0000000000 --- a/code/sys.ui/ui-css/src/m.Css.Dom/u.create.ts +++ /dev/null @@ -1,67 +0,0 @@ -import { type t, DEFAULT, isRecord, pkg, toHash, toString, V } from './common.ts'; -import { AlphanumericWithHyphens } from './u.ts'; - -type Prefix = string; -const singletons = new Map<Prefix, t.CssDom>(); -let _sheet: CSSStyleSheet | null = null; - -/** - * Generator factory - */ -export const create: t.CssDomLib['create'] = (prefix) => { - prefix = ((prefix ?? '').trim() || DEFAULT.prefix).replace(/-*$/, ''); - V.parse(AlphanumericWithHyphens, prefix); - if (singletons.has(prefix)) return singletons.get(prefix)!; - - const sheet = getOrCreateStylesheet(); - const inserted = new Set<string>(); - const insertRule = (rule: string) => sheet.insertRule?.(rule, sheet.cssRules.length); - - const api: t.CssDom = { - prefix, - get classes() { - return Array.from(inserted); - }, - class(style, hxInput) { - const hx = hxInput ?? toHash(style); - const className = `${prefix}-${hx}`; - if (inserted.has(className)) return className; - - // Initial creation. - inserted.add(className); - insertRule(`.${className} { ${toString(style)} }`); - - Object.entries(style) - .filter(([key]) => DEFAULT.pseudoClasses.has(key)) - .filter(([_, value]) => isRecord(value)) - .forEach(([key, value]) => insertRule(`.${className}${key} { ${toString(value)} }`)); - - return className; - }, - }; - - singletons.set(prefix, api); - return api; -}; - -/** - * Helpers - */ - -/** - * Singleton <style> element management. - * If one doesn't exist, we create one and append it to the <head>. - */ -function getOrCreateStylesheet(): CSSStyleSheet { - if (_sheet) return _sheet; - - if (typeof document === 'undefined') { - return {} as CSSStyleSheet; // Dummy (NB: safe when running on server). - } else { - const el = document.createElement('style'); - el.setAttribute('data-controller', pkg.name); - document.head.appendChild(el); - _sheet = el.sheet as CSSStyleSheet; - return _sheet; - } -} diff --git a/code/sys.ui/ui-css/src/m.Css.Dom/u.ctx.container.ts b/code/sys.ui/ui-css/src/m.Css.Dom/u.ctx.container.ts new file mode 100644 index 0000000000..d7ef4606da --- /dev/null +++ b/code/sys.ui/ui-css/src/m.Css.Dom/u.ctx.container.ts @@ -0,0 +1,85 @@ +import { type t } from './common.ts'; + +/** + * CSS: @container API + */ +export function createContainer(args: { + rules: t.CssDomRules; + condition: string; + name?: string; + scoped?: t.StringCssSelector[]; +}): t.CssDomContainerBlock { + const { rules, name, scoped = [] } = args; + const condition = wrangle.condition(args.condition); + const inserted = new Set<t.CssDomInsertedRule>(); + + const api: t.CssDomContainerBlock = { + kind: '@container', + condition, + name, + toString(kind = 'QueryCondition') { + return toString(api, kind); + }, + rules: { + get length() { + return inserted.size; + }, + get list() { + return Array.from(inserted); + }, + add(selector, style) { + selector = wrangle.selector(selector, scoped); + const context = api.toString(); + const styles = Array.isArray(style) ? style : [style]; + const res = rules.add(selector, styles, { context }); + if (res.length > 0) res.forEach((m) => inserted.add(m)); + return res; + }, + }, + scoped, + scope(selector) { + return createContainer({ rules, name, condition, scoped: [...scoped, selector] }); + }, + }; + + return api; +} + +/** + * Convert a container block to a string. + */ +export function toString( + container: t.CssDomContainerBlock, + kind: t.CssDomContainerToStringKind = 'QueryCondition', +): string { + if (kind === 'QueryCondition') { + let res = container.kind; + if (container.name) res += ` ${container.name}`; + res += ` ${container.condition}`; + return res; + } + + if (kind === 'CssSelector') { + const rules = container.rules.list; + if (rules.length === 0) return `${toString(container, 'QueryCondition')} {}`; + return rules.map(({ rule }) => rule).join('\n'); + } + + throw new Error(`toString kind value "${kind}" not supported`); +} + +/** + * Helpers + */ +const wrangle = { + condition(text: string): string { + text = (text || '').trim(); + if (!text.includes('(')) text = `(${text}`; + if (!text.includes(')')) text = `${text})`; + return text; + }, + + selector(selector: string, scope: t.StringCssSelector[]) { + return `${scope.join(' ')} ${selector}`.trim(); + }, +} as const; diff --git a/code/sys.ui/ui-css/src/m.Css.Dom/u.rules.ts b/code/sys.ui/ui-css/src/m.Css.Dom/u.rules.ts new file mode 100644 index 0000000000..ef7d90506f --- /dev/null +++ b/code/sys.ui/ui-css/src/m.Css.Dom/u.rules.ts @@ -0,0 +1,58 @@ +import { type t, CssTmpl, DEFAULT, isRecord } from './common.ts'; +import { toString } from './u.ts'; +type StringRule = string; + +export function createRules(args: { sheet: CSSStyleSheet }): t.CssDomRules { + const { sheet } = args; + const inserted = new Map<StringRule, t.CssDomInsertedRule>(); + + const insert = ( + selector: string, + style: t.CssProps, + context?: string, + ): t.CssDomInsertedRule | undefined => { + let rule = `${selector.trim()} { ${toString(style)} }`.trim(); + rule = context ? `${context} { ${rule} }` : rule; + if (inserted.has(rule)) return undefined; + + const res: t.CssDomInsertedRule = { selector, rule, style }; + sheet.insertRule?.(rule, sheet.cssRules.length); + inserted.set(rule, res); + return res; + }; + + const addRule = ( + selector: string, + style: t.CssValue, + context?: string, + ): t.CssDomInsertedRule[] => { + const res: (t.CssDomInsertedRule | undefined)[] = []; + res.push(insert(selector, CssTmpl.transform(style), context)); + Object.entries(style) + .filter(([key]) => DEFAULT.pseudoClasses.has(key)) + .filter(([, value]) => isRecord(value)) + .forEach(([key, style]) => res.push(insert(`${selector}${key}`, style, context))); + return res.filter(Boolean) as t.CssDomInsertedRule[]; + }; + + const api: t.CssDomRules = { + get length() { + return inserted.size; + }, + get list() { + return Array.from(inserted.values()); + }, + + add(selector, styles, options = {}) { + const res: t.CssDomInsertedRule[] = []; + const { context } = options; + const list = Array.isArray(styles) ? styles : [styles]; + list.forEach((style) => { + res.push(...addRule(selector, style, context)); + }); + return res; + }, + }; + + return api; +} diff --git a/code/sys.ui/ui-css/src/m.Css.Dom/u.stylesheet.ts b/code/sys.ui/ui-css/src/m.Css.Dom/u.stylesheet.ts new file mode 100644 index 0000000000..35570495ba --- /dev/null +++ b/code/sys.ui/ui-css/src/m.Css.Dom/u.stylesheet.ts @@ -0,0 +1,83 @@ +import { type t } from './common.ts'; +import { createClasses, wrangleClassPrefix } from './u.classes.ts'; +import { createContainer } from './u.ctx.container.ts'; +import { createRules } from './u.rules.ts'; +import { getStylesheetId } from './u.ts'; + +const singletons = new Map<t.StringId, t.CssDomStylesheet>(); + +/** + * Generator factory + */ +export const create: t.CssDomLib['stylesheet'] = (input) => { + const options = wrangle.options(input); + const id = getStylesheetId(options.instance, options.classPrefix); + if (singletons.has(id)) return singletons.get(id)!; + + const sheet = getOrCreateDomStyleSheet(id); + const rules = createRules({ sheet }); + const cache = { + classes: new Map<string, t.CssDomClasses>(), + getOrCreate<T>(key: string, map: Map<string, T>, factory: () => T): T { + if (!map.has(key)) map.set(key, factory()); + return map.get(key)!; + }, + }; + + const api: t.CssDomStylesheet = { + id, + rules, + rule(selector, style, options) { + return rules.add(selector, style, options); + }, + classes(prefix) { + const key = wrangleClassPrefix(prefix, options.classPrefix); + prefix = prefix ?? options.classPrefix; + return cache.getOrCreate(key, cache.classes, () => createClasses({ rules, prefix })); + }, + container(...args: any[]) { + const { name, condition } = wrangle.containerArgs(args); + return createContainer({ rules, condition, name }); + }, + }; + + singletons.set(id, api); + return api; +}; + +/** + * Helpers: + */ +const wrangle = { + options(input: Parameters<t.CssDomLib['stylesheet']>[0]): t.CssDomStylesheetOptions { + if (!input) return {}; + if (typeof input === 'string') return { instance: input }; + return input; + }, + + containerArgs(args: any[]) { + const done = (condition: string, name?: string) => { + name = name ? name.trim() : name; + condition = condition ? condition.trim() : ''; + return { name, condition }; + }; + if (!args || args.length === 0) return done(''); + if (args.length === 1) return done(args[0]); + return done(args[1], args[0]); + }, +} as const; + +/** + * Singleton <style> element management. + * If one doesn't exist, we create one and append it to the <head>. + */ +function getOrCreateDomStyleSheet(id: string): CSSStyleSheet { + if (typeof document === 'undefined') { + return {} as CSSStyleSheet; // Dummy (safe on server). + } else { + const el = document.createElement('style'); + el.setAttribute('data-controller', id); + document.head.appendChild(el); + return el.sheet as CSSStyleSheet; + } +} diff --git a/code/sys.ui/ui-css/src/m.Style/u.toString.ts b/code/sys.ui/ui-css/src/m.Css.Dom/u.toString.ts similarity index 100% rename from code/sys.ui/ui-css/src/m.Style/u.toString.ts rename to code/sys.ui/ui-css/src/m.Css.Dom/u.toString.ts diff --git a/code/sys.ui/ui-css/src/m.Css.Dom/u.ts b/code/sys.ui/ui-css/src/m.Css.Dom/u.ts index c91e4781e5..ad5e463443 100644 --- a/code/sys.ui/ui-css/src/m.Css.Dom/u.ts +++ b/code/sys.ui/ui-css/src/m.Css.Dom/u.ts @@ -1,4 +1,13 @@ -import { V } from './common.ts'; +import { V, pkg } from './common.ts'; +export { toString } from './u.toString.ts'; + +export function getStylesheetId(instance?: string, defaultPrefix?: string) { + instance = (instance || '').trim(); + let id = pkg.name; + if (instance) id += `:${instance}`; + if (defaultPrefix) id += `:${defaultPrefix}`; + return id; +} /** * Validation. diff --git a/code/sys.ui/ui-css/src/m.Css.Tmpl/mod.ts b/code/sys.ui/ui-css/src/m.Css.Tmpl/mod.ts index 2f7fc234b6..27b0ffa3fc 100644 --- a/code/sys.ui/ui-css/src/m.Css.Tmpl/mod.ts +++ b/code/sys.ui/ui-css/src/m.Css.Tmpl/mod.ts @@ -8,6 +8,9 @@ import { formatSize } from './u.formatSize.ts'; import { toEdges, WrangleEdge } from './u.toEdges.ts'; import { formatScroll } from './u.formatScroll.ts'; +/** + * Helpers for working with the template patterns (a DSL for css of sorts). + */ export const CssTmpl: t.CssTmplLib = { toEdges, diff --git a/code/sys.ui/ui-css/src/m.Css.Tmpl/t.ts b/code/sys.ui/ui-css/src/m.Css.Tmpl/t.ts index bf4ac7d292..e34a8f853b 100644 --- a/code/sys.ui/ui-css/src/m.Css.Tmpl/t.ts +++ b/code/sys.ui/ui-css/src/m.Css.Tmpl/t.ts @@ -1,7 +1,7 @@ import type { t } from './common.ts'; /** - * Helpers for working with the [CssTemplates] DSL. + * Helpers for working with the template patterns (a DSL for css of sorts). */ export type CssTmplLib = { /** diff --git a/code/sys.ui/ui-css/src/m.Style/-.test.ts b/code/sys.ui/ui-css/src/m.Style/-.test.ts index dca74928ff..eff29a3173 100644 --- a/code/sys.ui/ui-css/src/m.Style/-.test.ts +++ b/code/sys.ui/ui-css/src/m.Style/-.test.ts @@ -49,12 +49,13 @@ describe('Style', () => { it('Ęn: transformer ā <default> class-name prefix', () => { const css = Style.transformer(); const m = css({ display: 'grid' }); - expect(m.class.startsWith(`${DEFAULT.prefix}-`)).to.be.true; + expect(m.class.startsWith(`${DEFAULT.classPrefix}-`)).to.be.true; TestPrint.transformed(m); }); it('Ęn: transformer ā <custom> class-name prefix', () => { - const css = Style.transformer({ prefix: 'foo' }); + const sheet = Style.Dom.stylesheet({ classPrefix: 'foo' }); + const css = Style.transformer({ sheet }); const m = css({ display: 'grid' }); expect(m.class.startsWith(`foo-`)).to.be.true; TestPrint.transformed(m); diff --git a/code/sys.ui/ui-css/src/m.Style/-css.transform.test.ts b/code/sys.ui/ui-css/src/m.Style/-css.transform.test.ts new file mode 100644 index 0000000000..ed5f2b1401 --- /dev/null +++ b/code/sys.ui/ui-css/src/m.Style/-css.transform.test.ts @@ -0,0 +1,386 @@ +import { type t, DomMock, FindCss, TestPrint, c, describe, expect, it, slug } from '../-test.ts'; +import { toHash } from './common.ts'; +import { Style, css } from './mod.ts'; + +describe( + 'Style.css ā transform', + + /** NB: leaked timers left around by the "happy-dom" module. */ + { sanitizeOps: false, sanitizeResources: false }, + + () => { + DomMock.polyfill(); + + const setup = () => { + const sheet = Style.Dom.stylesheet(slug()); + const css = Style.transformer({ sheet }); + return { sheet, css } as const; + }; + + it('API', () => { + expect(Style.css).to.equal(css); + }); + + describe('css ā { styles }', () => { + it('empty', () => { + const a = css(); + const b = css([]); + const c = css(...[], false); + const d = css(null, undefined, [], false, ...[]); + + expect(a.style).to.eql({}); + expect(b).to.eql(a); + expect(c).to.eql(a); + expect(d).to.eql(a); + + // Cached instances (on hx). + expect(a.style).to.equal(b.style); + expect(a.style).to.equal(c.style); + expect(a.style).to.equal(d.style); + }); + + it('plain CSS fields', () => { + const a = css({ fontSize: 30 }); + const b = css({ fontSize: 30 }); + const c = css({ fontSize: 31 }); + + expect(a.style.fontSize).to.eql(30); + expect(b.style.fontSize).to.eql(30); + expect(c.style.fontSize).to.eql(31); + + expect(a).to.equal(b); + expect(a).to.not.equal(c); + }); + }); + + describe('{ Template } ā {styles} | known templates starting with a capital-letter', () => { + it('Absolute: 0', () => { + const a = css({ Absolute: 0 }); + const b = { position: 'absolute', top: 0, right: 0, bottom: 0, left: 0 }; + expect(a.style).to.eql(b); + expect(a.hx).to.eql(toHash(b)); + expect((a.style as any).Absolute).to.eql(undefined); // NB: clean up on object after transform. + }); + + it('PaddingX: [10, 20]', () => { + const a = css({ PaddingX: [10, 20] }); + const b = { paddingLeft: 10, paddingRight: 20 }; + expect(a.style).to.eql(b); + expect(a.hx).to.eql(toHash(b)); + expect((a.style as any).PaddingX).to.eql(undefined); // NB: clean up on object after transform. + }); + }); + + describe('css ā "class-name" (inserted into DOM)', () => { + it('simple', () => { + const sheet = Style.Dom.stylesheet({ classPrefix: 'foo' }); + const css = Style.transformer({ sheet }); + + const input: t.CssValue = { PaddingX: [10, 30] }; + const m = css({ PaddingX: [10, 30] }); + const className = `foo-${m.hx}`; + expect(FindCss.rule(className)).to.eql(undefined); + + console.info(c.gray('\nInput:'), input); + console.info(c.gray('ā')); + TestPrint.transformed(m); + + expect(m.class).to.eql(className); // NB: insertion into DOM happens here. + expect(FindCss.rule(className)?.cssText).to.include(className); + expect(FindCss.rule(className)?.cssText).to.include(m.toString()); + }); + }); + + describe('toString', () => { + const style = { fontSize: 30, fontFamily: 'sans-serif' }; + + const print = (kind: t.CssTransformToStringKind, value: string) => { + console.info(); + console.info(`${c.brightCyan(kind)}: "${c.yellow(value)}"`); + console.info(); + }; + + it('empty', () => { + expect(css({}).toString()).to.eql(''); + }); + + it('kind: CssRule (default)', () => { + const m = css(style); + const a = m.toString(); + const b = m.toString('CssRule'); + + print('CssRule', m.toString()); + + expect(a).to.eql('font-size: 30px; font-family: sans-serif;'); + expect(a).to.eql(b); + }); + + it('kind: CssSelector', () => { + const sheet = Style.Dom.stylesheet({ classPrefix: 'foo' }); + const css = Style.transformer({ sheet }); + const m = css(style); + const str = m.toString('CssSelector'); + + print('CssSelector', str); + expect(str).to.eql(`.foo-${m.hx} { font-size: 30px; font-family: sans-serif; }`); + }); + }); + + describe('merging', () => { + it('basic merge', () => { + const a = css({ color: 'red' }); + const b = css({ background: 'blue' }); + + const res = css(a, b); + expect(res.style).to.include({ color: 'red' }); + expect(res.style).to.include({ background: 'blue' }); + }); + + it('deep merge', () => { + const assert = (res: t.CssTransformed) => { + expect(res.style).to.include({ color: 'red' }); + expect(res.style).to.include({ background: 'blue' }); + }; + + const props = { style: { color: 'red' } }; + const styles = { base: css({ background: 'blue' }) }; + + assert(css(styles.base, props.style)); + assert(css(styles.base, css(props.style))); + assert(css(css(styles.base), css(css(props.style)))); + assert(css([css(styles.base), css(css(props.style))])); + assert( + css( + css([css(styles.base), css(css(props.style))]), + css([css(styles.base), [css(css([css([[props.style]])]))]]), + ), + ); + }); + + it('deep merge ā {style} object', () => { + const assert = (res: t.CssTransformed) => { + expect(res.style).to.include({ color: 'red' }); + expect(res.style).to.include({ background: 'blue' }); + }; + + const props = { style: { color: 'red' } }; + const styles = { base: css({ background: 'blue' }) }; + + assert(css(styles.base, props.style)); + assert(css(styles.base, css(props.style))); + assert(css(css(styles.base), css(css(props.style)))); + assert(css([css(styles.base), css(css(props.style))])); + assert( + css( + css([css(styles.base), Style.css(css(props.style))]), + css([css(styles.base), [Style.css(css([css([[props.style]])]))]]), + ), + ); + }); + + it('cache name repeats (reused)', () => { + const props = { style: { color: 'red' } }; + const styles = { base: css({ background: 'blue' }) }; + + const a = css(css(styles.base), css(css(props.style))); + const b = css(css(styles.base), css(css(props.style))); + expect(a.hx).to.equal(b.hx); + }); + + it('wrapped ā equality', () => { + const a = css({ color: 'red' }); + const b = css(a); + const c = css([b]); + const d = css([a, c], b); + expect(b).to.eql(a); + expect(c).to.eql(a); + expect(d).to.eql(a); + }); + }); + + describe('.container', () => { + it('scoped with root class-name', () => { + const a = css({ Absolute: 0 }); + const b = a.container('min-width: 500px'); + expect(b.block.kind).to.eql('@container'); + expect(b.block.scoped).to.eql([`.${a.class}`]); + }); + + it('with {style} param', () => { + const condition = 'min-width: 500px'; + const style = { Absolute: 0, fontSize: 42 }; + const base = css({ Absolute: 0 }); + + const a = base.container(condition); + const b = base.container(condition, style); + const c = base.container('my-name', condition); + const d = base.container('my-name', condition, style); + + [a, b, c, d].forEach((m) => { + expect(m.block.kind).to.eql('@container'); + expect(m.block.condition).to.eql(`(${condition})`); + }); + + [a, b].forEach((m) => expect(m.block.name).to.eql(undefined)); + expect(c.block.name).to.eql('my-name'); + expect(d.block.name).to.eql('my-name'); + + [a, c].forEach((m) => expect(m.block.rules.length).to.eql(0)); + [b, d].forEach((m) => { + expect(m.block.rules.length).to.eql(1); + expect(m.block.rules.list[0].style).to.eql({ + position: 'absolute', + top: 0, + right: 0, + bottom: 0, + left: 0, + fontSize: 42, + }); + }); + }); + + it('nest: multi-level descendents', () => { + const a = css({ Absolute: 0 }); + const b = a.container('min-width: 500px'); + const c = b.nest('h2'); + const d = c.nest('code'); + [b, c, d].forEach(({ block }) => expect(block.kind).to.eql('@container')); + expect(c.block.scoped).to.eql([`.${a.class}`, 'h2']); + expect(d.block.scoped).to.eql([`.${a.class}`, 'h2', 'code']); + }); + + it('add custom selector: .rule()', () => { + const container = css({ Absolute: 0 }).container('min-width: 500px'); + const rules = container.block.rules; + expect(rules.length).to.eql(0); + + const styles = [{ color: 'red' }, { fontSize: 32 }, { color: 'blue' }]; + const a = container.rule('h2', styles[0]); + const b = container.rule('h2', [styles[1], styles[2]]); + + expect(a[0].style).to.eql(styles[0]); + expect(b[0].style).to.eql(styles[1]); + expect(b[1].style).to.eql(styles[2]); + + expect(rules.length).to.eql(3); + + const root = container.block.scoped[0]; + rules.list.forEach(({ rule }) => expect(rule).to.include(`{ ${root} h2 {`)); + }); + + it('add custom selector: .rule() ā CSS template', () => { + const container = css({ Absolute: 0 }).container('min-width: 500px'); + const rules = container.block.rules; + expect(rules.length).to.eql(0); + + const res = container.rule('h2', { Absolute: 0 }); + expect(res[0].style).to.eql({ position: 'absolute', top: 0, right: 0, bottom: 0, left: 0 }); + expect(rules.length).to.eql(1); + }); + + it('add rule: .css()', () => { + const container = css({ Absolute: 0 }).container('min-width: 500px'); + const root = container.block.scoped[0]; + const rules = container.block.rules; + expect(rules.length).to.eql(0); + + const a = container.css({ color: 'red' }); + expect(a).to.equal(container); + expect(rules.length).to.eql(1); + expect(rules.list[0].rule).to.include(`{ ${root} { color: red; } }`); + + container.css([{ PaddingX: 10 }, { color: 'blue' }]); // NB: CSS template expansion. + expect(rules.length).to.eql(3); + + expect(rules.list[1].rule).to.include(`${root} { padding-right: 10px; padding-left: 10px`); + expect(rules.list[2].rule).to.include(`{ ${root} { color: blue; } }`); + }); + + it('done (property): end a fluent API chain', () => { + const base = css({ Absolute: 0 }); + const container = base.container('min-width: 500px'); + expect(container.done).to.equal(base); + expect(container.css({ color: 'red' }).done).to.eql(base); + expect(container.nest('h2').css({ color: 'red' }).done).to.eql(base); + }); + + it('sample: fluent chaining', () => { + const { sheet, css } = setup(); + expect(sheet.rules.length).to.eql(0); + + const styles = { + base: css({ containerType: 'size' }), + h2: css({ fontSize: 50 }) + .container('min-width: 400px', { fontSize: 100 }) + .container('min-width: 700px', { PaddingX: 10 }).done, + }; + + // NB: flush rules. + styles.base.class; + styles.h2.class; + + const list = sheet.rules.list.map((m) => m.style); + expect(list).to.eql([ + { fontSize: 50 }, + { fontSize: 100 }, + { paddingLeft: 10, paddingRight: 10 }, // NB: expanded CSS template (PaddingX) + { containerType: 'size' }, + ]); + }); + }); + + describe('.rule (arbitrary sub-selectors)', () => { + it('nests the sub-selector', () => { + const { sheet, css } = setup(); + const base = css({ position: 'relative' }); + expect(sheet.rules.length).to.eql(0); + + const a = base.rule('h2', { color: 'red' }); + const b = base.rule('h2', { color: 'red' }); // NB: not-added (duplicate). + + expect(a).to.equal(base); // NB: enabled API chaining ("fluent"). + expect(a).to.equal(b); + + const rules = sheet.rules.list; + expect(rules.length).to.eql(2); + expect(rules[0].rule).to.include(`.${base.class} { position: relative; }`); + expect(rules[1].rule).to.include(`.${base.class} h2 { color: red; }`); + }); + + it('sample: chaining', () => { + const { sheet, css } = setup(); + const styles = { + base: css({ position: 'relative' }) + .rule('h1', { color: 'red' }) + .rule('h2', { color: 'blue' }) + .rule('h2 code', { color: 'green' }), + }; + + const baseClass = styles.base.class; + const rules = sheet.rules.list; + + expect(rules.length).to.eql(4); + expect(rules[0].rule).to.include(`.${baseClass} { position: relative; }`); + expect(rules[1].rule).to.include(`.${baseClass} h1 { color: red; }`); + expect(rules[2].rule).to.include(`.${baseClass} h2 { color: blue; }`); + expect(rules[3].rule).to.include(`.${baseClass} h2 code { color: green; }`); + }); + + it('empty selector', () => { + const { sheet, css } = setup(); + + const base = css({ position: 'relative' }); + expect(sheet.rules.length).to.eql(0); + + base.rule('', { color: 'red' }); + base.rule(' ', { color: 'red' }); // NB: not-added (trimmed ā duplicate). + base.rule(' ', { color: 'red' }); + + const rules = sheet.rules.list; + expect(rules.length).to.eql(2); + expect(rules[0].rule).to.include(`.${base.class} { position: relative; }`); + expect(rules[1].rule).to.include(`.${base.class} { color: red; }`); + }); + }); + }, +); diff --git a/code/sys.ui/ui-css/src/m.Style/-transform.test.ts b/code/sys.ui/ui-css/src/m.Style/-transform.test.ts deleted file mode 100644 index 0f8d5c0ccf..0000000000 --- a/code/sys.ui/ui-css/src/m.Style/-transform.test.ts +++ /dev/null @@ -1,194 +0,0 @@ -import { type t, DomMock, FindCss, TestPrint, c, describe, expect, it } from '../-test.ts'; -import { toHash } from './common.ts'; -import { Style, css } from './mod.ts'; - -describe( - 'Style.css ā transform', - - /** NB: leaked timers left around by the "happy-dom" module. */ - { sanitizeOps: false, sanitizeResources: false }, - - () => { - DomMock.polyfill(); - - it('API', () => { - expect(Style.css).to.equal(css); - }); - - describe('css ā { styles }', () => { - it('empty', () => { - const a = css(); - const b = css([]); - const c = css(...[], false); - const d = css(null, undefined, [], false, ...[]); - - expect(a.style).to.eql({}); - expect(b).to.eql(a); - expect(c).to.eql(a); - expect(d).to.eql(a); - - // Cached instances (on hx). - expect(a.style).to.equal(b.style); - expect(a.style).to.equal(c.style); - expect(a.style).to.equal(d.style); - }); - - it('plain CSS fields', () => { - const a = css({ fontSize: 30 }); - const b = css({ fontSize: 30 }); - const c = css({ fontSize: 31 }); - - expect(a.style.fontSize).to.eql(30); - expect(b.style.fontSize).to.eql(30); - expect(c.style.fontSize).to.eql(31); - - expect(a).to.equal(b); - expect(a).to.not.equal(c); - }); - }); - - describe('{ Template } ā {styles} | known templates starting with a capital-letter', () => { - it('Absolute: 0', () => { - const a = css({ Absolute: 0 }); - const b = { position: 'absolute', top: 0, right: 0, bottom: 0, left: 0 }; - expect(a.style).to.eql(b); - expect(a.hx).to.eql(toHash(b)); - expect((a.style as any).Absolute).to.eql(undefined); // NB: clean up on object after transform. - }); - - it('PaddingX: [10, 20]', () => { - const a = css({ PaddingX: [10, 20] }); - const b = { paddingLeft: 10, paddingRight: 20 }; - expect(a.style).to.eql(b); - expect(a.hx).to.eql(toHash(b)); - expect((a.style as any).PaddingX).to.eql(undefined); // NB: clean up on object after transform. - }); - }); - - describe('css ā "class-name" (inserted into DOM)', () => { - it('simple', () => { - const prefix = 'foo'; - const css = Style.transformer({ prefix }); - - const input: t.CssValue = { PaddingX: [10, 30] }; - const m = css({ PaddingX: [10, 30] }); - const className = `${prefix}-${m.hx}`; - expect(FindCss.rule(className)).to.eql(undefined); - - console.info(c.gray('\nInput:'), input); - console.info(c.gray('ā')); - TestPrint.transformed(m); - - expect(m.class).to.eql(className); // NB: insertion into DOM happens here. - expect(FindCss.rule(className)?.cssText).to.include(className); - expect(FindCss.rule(className)?.cssText).to.include(m.toString()); - }); - }); - - describe('toString', () => { - const style = { fontSize: 30, fontFamily: 'sans-serif' }; - - const print = (kind: t.CssTransformStringKind, value: string) => { - console.info(); - console.info(`${c.brightCyan(kind)}: "${c.yellow(value)}"`); - console.info(); - }; - - it('empty', () => { - expect(css({}).toString()).to.eql(''); - }); - - it('kind: CssRule (default)', () => { - const m = css(style); - const a = m.toString(); - const b = m.toString('CssRule'); - - print('CssRule', m.toString()); - - expect(a).to.eql('font-size: 30px; font-family: sans-serif;'); - expect(a).to.eql(b); - }); - - it('kind: CssSelector', () => { - const css = Style.transformer('foo'); - const m = css(style); - const str = m.toString('CssSelector'); - - print('CssSelector', str); - expect(str).to.eql(`.foo-${m.hx} { font-size: 30px; font-family: sans-serif; }`); - }); - }); - - describe('merging', () => { - it('basic merge', () => { - const a = css({ color: 'red' }); - const b = css({ background: 'blue' }); - - const res = css(a, b); - expect(res.style).to.include({ color: 'red' }); - expect(res.style).to.include({ background: 'blue' }); - }); - - it('deep merge', () => { - const assert = (res: t.CssTransformed) => { - expect(res.style).to.include({ color: 'red' }); - expect(res.style).to.include({ background: 'blue' }); - }; - - const props = { style: { color: 'red' } }; - const styles = { base: css({ background: 'blue' }) }; - - assert(css(styles.base, props.style)); - assert(css(styles.base, css(props.style))); - assert(css(css(styles.base), css(css(props.style)))); - assert(css([css(styles.base), css(css(props.style))])); - assert( - css( - css([css(styles.base), css(css(props.style))]), - css([css(styles.base), [css(css([css([[props.style]])]))]]), - ), - ); - }); - - it('deep merge ā {style} object', () => { - const assert = (res: t.CssTransformed) => { - expect(res.style).to.include({ color: 'red' }); - expect(res.style).to.include({ background: 'blue' }); - }; - - const props = { style: { color: 'red' } }; - const styles = { base: css({ background: 'blue' }) }; - - assert(css(styles.base, props.style)); - assert(css(styles.base, css(props.style))); - assert(css(css(styles.base), css(css(props.style)))); - assert(css([css(styles.base), css(css(props.style))])); - assert( - css( - css([css(styles.base), Style.css(css(props.style))]), - css([css(styles.base), [Style.css(css([css([[props.style]])]))]]), - ), - ); - }); - - it('cache name repeats (reused)', () => { - const props = { style: { color: 'red' } }; - const styles = { base: css({ background: 'blue' }) }; - - const a = css(css(styles.base), css(css(props.style))); - const b = css(css(styles.base), css(css(props.style))); - expect(a.hx).to.equal(b.hx); - }); - - it('wrapped ā equality', () => { - const a = css({ color: 'red' }); - const b = css(a); - const c = css([b]); - const d = css([a, c], b); - expect(b).to.eql(a); - expect(c).to.eql(a); - expect(d).to.eql(a); - }); - }); - }, -); diff --git a/code/sys.ui/ui-css/src/m.Style/common.ts b/code/sys.ui/ui-css/src/m.Style/common.ts index 5c551f465a..9b844d14bd 100644 --- a/code/sys.ui/ui-css/src/m.Style/common.ts +++ b/code/sys.ui/ui-css/src/m.Style/common.ts @@ -1,7 +1,7 @@ import toHash from 'hash-it'; import { DEFAULT as CssDomDefaults } from '../m.Css.Dom/common.ts'; -export { CssDom } from '../m.Css.Dom/mod.ts'; +export { CssDom, toString } from '../m.Css.Dom/mod.ts'; export { CssEdges } from '../m.Css.Edges/mod.ts'; export { CssTmpl } from '../m.Css.Tmpl/mod.ts'; export { toHash }; @@ -9,50 +9,13 @@ export { toHash }; export * from '../common.ts'; export const DEFAULT = { - get prefix() { - return CssDomDefaults.prefix; + get classPrefix() { + return CssDomDefaults.classPrefix; }, get pixelProps() { - return pixelProps; + return CssDomDefaults.pixelProps; }, get pseudoClasses() { return CssDomDefaults.pseudoClasses; }, } as const; - -/** - * CSS properties that accept unitless - * numbers (equating to "px" pixels). - */ -const pixelProps = new Set<string>([ - 'width', - 'height', - 'top', - 'right', - 'bottom', - 'left', - 'margin', - 'marginTop', - 'marginRight', - 'marginBottom', - 'marginLeft', - 'padding', - 'paddingTop', - 'paddingRight', - 'paddingBottom', - 'paddingLeft', - 'borderWidth', - 'borderTopWidth', - 'borderRightWidth', - 'borderBottomWidth', - 'borderLeftWidth', - 'borderRadius', - 'fontSize', - 'minWidth', - 'maxWidth', - 'minHeight', - 'maxHeight', - 'outlineWidth', - 'letterSpacing', - 'wordSpacing', -]); diff --git a/code/sys.ui/ui-css/src/m.Style/m.Style.ts b/code/sys.ui/ui-css/src/m.Style/m.Style.ts index 2ccab28192..ab27a0bf95 100644 --- a/code/sys.ui/ui-css/src/m.Style/m.Style.ts +++ b/code/sys.ui/ui-css/src/m.Style/m.Style.ts @@ -1,23 +1,21 @@ import { type t, Color } from './common.ts'; -import { DEFAULT, CssDom as Dom, CssEdges as Edges, CssTmpl as Tmpl } from './common.ts'; +import { CssDom as Dom, CssEdges as Edges, CssTmpl as Tmpl, toString } from './common.ts'; import { toShadow } from './u.toShadow.ts'; -import { toString } from './u.toString.ts'; import { transformer } from './u.transform.ts'; const { toPadding, toMargins } = Edges; -const prefix = DEFAULT.prefix; /** Perform a transformation on a loose set of CSS inputs. */ -export const css: t.CssTransform = transformer({ prefix }); +export const css: t.CssTransform = transformer({}); /** * CSS styling tools. */ export const Style: t.StyleLib = { + Dom, Color, Edges, - Dom, Tmpl, css, diff --git a/code/sys.ui/ui-css/src/m.Style/t.transform.ts b/code/sys.ui/ui-css/src/m.Style/t.transform.ts new file mode 100644 index 0000000000..3224849c8b --- /dev/null +++ b/code/sys.ui/ui-css/src/m.Style/t.transform.ts @@ -0,0 +1,64 @@ +import type { t } from './common.ts'; + +/** Flags indicating the kind of string to export from the `toString` method. */ +export type CssTransformToStringKind = 'CssRule' | 'CssSelector'; + +/** + * Function that transforms 1..n CSS inputs into a style + * object that can be applied to a JSX element. + * + * NB: This is the raw transform containing the style along with cache metadata. + */ +export type CssTransform = (...input: t.CssInput[]) => t.CssTransformed; + +/** + * A transformed CSS properties object. + */ +export type CssTransformed = { + /** The hash of the style (used for caching). */ + readonly hx: number; + + /** Style properties. */ + readonly style: t.CssProps; + + /** The CSS class-name. */ + readonly class: t.CssClassname; + + /** Convert the {style} props object to a CSS string. */ + toString(kind?: t.CssTransformToStringKind): string; + + /** Retrieve the @container API scoped to the current css-class. */ + container(condition: string, style?: t.CssValue): t.CssTransformContainerBlock; + container(name: string, condition: string, style?: t.CssValue): t.CssTransformContainerBlock; + + /** Insert a CSS rule within scope with the current `class` name. */ + rule(selector: t.StringCssSelector, style: t.CssValue | t.CssValue[]): CssTransformed; +}; + +/** + * A specialised @container block API with concenience + * methods for the `CssTransform` functional object. + */ +export type CssTransformContainerBlock = { + /** The underlying @container block being used by the convenience API. */ + readonly block: t.CssDomContainerBlock; + + /** Insert a CSS rule within the @container context with the given arbitrary selector. */ + rule(selector: t.StringCssSelector, style: t.CssValue | t.CssValue[]): t.CssDomInsertedRule[]; + + /** Insert a CSS rule within the @container directly under the CSS class-name scope. */ + css(style: t.CssValue | t.CssValue[]): CssTransformContainerBlock; + + /** Creates a new scoped sub-selector. */ + nest(selector: t.StringCssSelector): CssTransformContainerBlock; + + /** + * Generate a new @container block off the root transform, + * used to create fluent chains of containers. + */ + container(condition: string, style?: t.CssValue): t.CssTransformContainerBlock; + container(name: string, condition: string, style?: t.CssValue): t.CssTransformContainerBlock; + + /** Returns the root `CssTransform` used in ending a fluent chain. */ + readonly done: CssTransformed; +}; diff --git a/code/sys.ui/ui-css/src/m.Style/t.ts b/code/sys.ui/ui-css/src/m.Style/t.ts index 0054784f30..d3dbf73e30 100644 --- a/code/sys.ui/ui-css/src/m.Style/t.ts +++ b/code/sys.ui/ui-css/src/m.Style/t.ts @@ -1,5 +1,6 @@ import type { CSSProperties } from 'react'; import type { t } from './common.ts'; +export type * from './t.transform.ts'; /** * CSS-Properties that accept string AND (inferable "unit" numbers) as values. @@ -10,15 +11,28 @@ import type { t } from './common.ts'; */ export type CssProps = CSSProperties; +/** + * Standard CSS properties with CSS-template extensions. + */ +export type CssValue = t.CssProps & t.CssPseudo & t.CssTemplates; +export type CssInput = + | t.CssValue + | undefined + | null + | false + | never + | t.CssTransformed + | CssInput[]; + /** * A CSS class-name. - * (no period, eg "foo" not ".foo") + * (no period, eg. "foo" not ".foo") */ export type CssClassname = string; export type CssClassPrefix = string; /** Options passed to `Style.transformer` factory function. */ -export type StyleTransformerOptions = { prefix?: string }; +export type StyleTransformerOptions = { sheet?: t.CssDomStylesheet }; /** * CSS styling tools. @@ -28,7 +42,7 @@ export type StyleLib = NamespaceLibs & { readonly css: t.CssTransform; /** Factory to produce `transform` function scoped to the given prefix. */ - transformer(options?: t.CssClassPrefix | t.StyleTransformerOptions): t.CssTransform; + transformer(options?: t.StyleTransformerOptions): t.CssTransform; /** Transform margin spacing. */ readonly toMargins: t.CssEdgesLib['toMargins']; @@ -55,40 +69,6 @@ type NamespaceLibs = { readonly Dom: t.CssDomLib; }; -/** - * Standard CSS properties with CSS-template extensions. - */ -export type CssValue = t.CssProps & t.CssPseudo & t.CssTemplates; -export type CssInput = t.CssValue | undefined | null | false | never | CssTransformed | CssInput[]; - -/** - * Function that transforms 1..n CSS inputs into a style - * object that can be applied to a JSX element. - * - * NB: This is the raw transform containing the style along with cache metadata. - */ -export type CssTransform = (...input: t.CssInput[]) => t.CssTransformed; - -/** - * A transformed CSS properties object. - */ -export type CssTransformed = { - /** The hash of the style (used for caching). */ - readonly hx: number; - - /** Style properties. */ - readonly style: t.CssProps; - - /** The CSS class-name. */ - readonly class: t.CssClassname; - - /** Convert the {style} props object to a CSS string. */ - toString(kind?: t.CssTransformStringKind): string; -}; - -/** Flags indicating the kind of string to export from the `toString` method. */ -export type CssTransformStringKind = 'CssRule' | 'CssSelector'; - /** * Shadow */ diff --git a/code/sys.ui/ui-css/src/m.Style/u.transform.container.ts b/code/sys.ui/ui-css/src/m.Style/u.transform.container.ts new file mode 100644 index 0000000000..92e3a21ab3 --- /dev/null +++ b/code/sys.ui/ui-css/src/m.Style/u.transform.container.ts @@ -0,0 +1,31 @@ +import { type t } from './common.ts'; + +/** + * Factory for creating the CSS @container convenience method + * for the `CssTransformer` API. + */ +export function createTransformContainer( + base: t.CssTransformed, + block: t.CssDomContainerBlock, + style?: t.CssProps | undefined, +): t.CssTransformContainerBlock { + const api: t.CssTransformContainerBlock = { + block, + + container: base.container, + rule: (selector, style) => block.rules.add(selector, style), + nest: (selector) => createTransformContainer(base, block.scope(selector)), + + css(style) { + block.rules.add('', style); + return api; + }, + + get done() { + return base; + }, + }; + + if (style) api.css(style); + return api; +} diff --git a/code/sys.ui/ui-css/src/m.Style/u.transform.ts b/code/sys.ui/ui-css/src/m.Style/u.transform.ts index 7856140f52..c7ff8e7cb7 100644 --- a/code/sys.ui/ui-css/src/m.Style/u.transform.ts +++ b/code/sys.ui/ui-css/src/m.Style/u.transform.ts @@ -1,6 +1,6 @@ -import { type t, CssDom, CssTmpl, DEFAULT, toHash } from './common.ts'; +import { type t, CssDom, CssTmpl, toHash, toString } from './common.ts'; import { isTransformed } from './u.is.ts'; -import { toString } from './u.toString.ts'; +import { createTransformContainer } from './u.transform.container.ts'; type M = Map<number, t.CssTransformed>; type O = Record<string, unknown>; @@ -9,20 +9,29 @@ type F = t.StyleLib['transformer']; /** * Generator (factory). */ -export const transformer: F = (input) => { - const options = wrangle.options(input); - const { prefix = DEFAULT.prefix } = options; - const dom = CssDom.create(prefix); +export const transformer: F = (options = {}) => { const cache = new Map<number, t.CssTransformed>(); - const fn: t.CssTransform = (...input) => transform({ dom, cache, input }); + + let _sheet: t.CssDomStylesheet | undefined; + const lazySheet = () => options.sheet ?? _sheet ?? CssDom.stylesheet(/* default config */); + + const fn: t.CssTransform = (...input) => { + const sheet = lazySheet(); + return transform({ sheet, cache, input }); + }; return fn; }; /** * Perform a cacheable transformation on a loose set of CSS inputs. */ -function transform(args: { dom: t.CssDom; cache: M; input: t.CssInput[] }): t.CssTransformed { - const { dom, cache } = args; +function transform(args: { + sheet: t.CssDomStylesheet; + cache: M; + input: t.CssInput[]; +}): t.CssTransformed { + const { sheet, cache } = args; + const style: t.CssProps = CssTmpl.transform(wrangle.input(args.input)); const hx = toHash(style); if (cache.has(hx)) return cache.get(hx)!; @@ -33,7 +42,8 @@ function transform(args: { dom: t.CssDom; cache: M; input: t.CssInput[] }): t.Cs return style; }, get class() { - return dom.class(style, hx); + const classes = sheet.classes(); + return classes.add(style, { hx }); }, toString(kind = 'CssRule') { const rule = toString(style); @@ -41,6 +51,16 @@ function transform(args: { dom: t.CssDom; cache: M; input: t.CssInput[] }): t.Cs if (kind === 'CssSelector') return `.${api.class} { ${rule} }`; throw new Error(`Kind '${kind}' not supported`); }, + container(...args: any[]) { + const { name, condition, style } = wrangle.containerArgs(args); + const container = name ? sheet.container(name, condition) : sheet.container(condition); + const block = container.scope(`.${api.class}`); + return createTransformContainer(api, block, style); + }, + rule(selector, style) { + sheet.rules.add(`.${api.class} ${selector}`.trim(), style); + return api; + }, }; cache.set(hx, api); @@ -48,15 +68,9 @@ function transform(args: { dom: t.CssDom; cache: M; input: t.CssInput[] }): t.Cs } /** - * Helpers + * Helpers: */ const wrangle = { - options(input?: Parameters<F>[0]): t.StyleTransformerOptions { - if (!input) return {}; - if (typeof input === 'string') return { prefix: input }; - return input; - }, - input(input: any): t.CssProps { if (Array.isArray(input)) { return input.reduce((acc, next) => ({ ...acc, ...wrangle.input(next) }), {} as O); @@ -66,4 +80,24 @@ const wrangle = { return input; } }, + + containerArgs(args: any[]) { + const done = (condition: string, name?: string, style?: t.CssProps) => { + name = name ? name.trim() : name; + condition = condition ? condition.trim() : ''; + return { name, condition, style }; + }; + if (!args || args.length === 0) return done(''); + if (args.length === 1) return done(args[0]); + if (args.length === 2) { + const [p1, p2] = args; + if (typeof p2 === 'object') return done(p1, undefined, p2); + if (typeof p2 === 'string') return done(p2, p1); + } + if (args.length === 3) { + const [p1, p2, p3] = args; + return done(p2, p1, p3); + } + throw new Error(`Failed to parse [container.scope] arguments: ${args}`); + }, } as const; diff --git a/code/sys.ui/ui-css/src/pkg.ts b/code/sys.ui/ui-css/src/pkg.ts index 79cb3f9c80..6ee002d709 100644 --- a/code/sys.ui/ui-css/src/pkg.ts +++ b/code/sys.ui/ui-css/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/ui-css', version: '0.0.81' }; diff --git a/code/sys.ui/ui-css/src/types.ts b/code/sys.ui/ui-css/src/types.ts index f1205036d0..585df0d03d 100644 --- a/code/sys.ui/ui-css/src/types.ts +++ b/code/sys.ui/ui-css/src/types.ts @@ -6,3 +6,6 @@ export type * from './m.Css.Dom/t.ts'; export type * from './m.Css.Edges/t.ts'; export type * from './m.Css.Tmpl/t.ts'; export type * from './m.Style/t.ts'; + +/** A string that represents a CSS selector. */ +export type StringCssSelector = string; diff --git a/code/sys.ui/ui-dom/deno.json b/code/sys.ui/ui-dom/deno.json index da5555f21d..3dfa098468 100644 --- a/code/sys.ui/ui-dom/deno.json +++ b/code/sys.ui/ui-dom/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/ui-dom", - "version": "0.0.77", + "version": "0.0.87", "license": "MIT", "tasks": { "test": "deno test -RW", diff --git a/code/sys.ui/ui-dom/src/pkg.ts b/code/sys.ui/ui-dom/src/pkg.ts index 79cb3f9c80..e449bfe858 100644 --- a/code/sys.ui/ui-dom/src/pkg.ts +++ b/code/sys.ui/ui-dom/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/ui-dom', version: '0.0.87' }; diff --git a/code/sys.ui/ui-react-components/-scripts/-clean.ts b/code/sys.ui/ui-react-components/-scripts/-clean.ts deleted file mode 100644 index 30ae5e6470..0000000000 --- a/code/sys.ui/ui-react-components/-scripts/-clean.ts +++ /dev/null @@ -1,4 +0,0 @@ -import { Fs } from '@sys/fs'; - -const removeDir = (path: string) => Fs.remove(Fs.resolve(path), { log: true }); -await removeDir('./.tmp'); diff --git a/code/sys.ui/ui-react-components/-scripts/-main.ts b/code/sys.ui/ui-react-components/-scripts/-main.ts index b0fd86ea46..16e17b93a6 100644 --- a/code/sys.ui/ui-react-components/-scripts/-main.ts +++ b/code/sys.ui/ui-react-components/-scripts/-main.ts @@ -1,3 +1 @@ -import { ViteEntry } from '@sys/driver-vite'; -await ViteEntry.main(Deno.args); -Deno.exit(0); +import '../../../sys.driver/driver-vite/src/-entry/-main.ts'; diff --git a/code/sys.ui/ui-react-components/-scripts/-tmp.ts b/code/sys.ui/ui-react-components/-scripts/-tmp.ts new file mode 100644 index 0000000000..ba662b31d4 --- /dev/null +++ b/code/sys.ui/ui-react-components/-scripts/-tmp.ts @@ -0,0 +1 @@ +console.info(`\nš ${import.meta.url}\n`); diff --git a/code/sys.ui/ui-react-components/README.md b/code/sys.ui/ui-react-components/README.md index b91c224a0b..c2b9509266 100644 --- a/code/sys.ui/ui-react-components/README.md +++ b/code/sys.ui/ui-react-components/README.md @@ -1,3 +1,2 @@ -# UI Components (Common) - - +# UI Components +Library of common `<React>` system components. diff --git a/code/sys.ui/ui-react-components/deno.json b/code/sys.ui/ui-react-components/deno.json index a2bcb0d348..5020c99b6b 100644 --- a/code/sys.ui/ui-react-components/deno.json +++ b/code/sys.ui/ui-react-components/deno.json @@ -1,19 +1,23 @@ { "name": "@sys/ui-react-components", - "version": "0.0.38", - "license": "MIT", + "version": "0.0.41", "tasks": { - "lint": "deno lint", - "dry": "deno publish --allow-dirty --dry-run", - "clean": "deno run -RWE ./-scripts/-clean.ts", - "test": "deno test -RWNE --allow-run --allow-ffi", - "dev": "deno run -A ./-scripts/-main.ts --cmd=dev --in=./src/-test/index.html", - "build": "deno run -A ./-scripts/-main.ts --cmd=build --in=./src/-test/index.html", - "serve": "deno run -A ./-scripts/-main.ts --cmd=serve --in=./src/-test/index.html" + "dev": "deno run -RWNE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=dev --in=./src/-test/index.html", + "build": "deno run -RWE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=build --in=./src/-test/index.html", + "serve": "deno run -RNE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=serve", + "test": "deno test -RWNE --allow-run --allow-ffi --allow-sys", + "dry": "deno publish --allow-dirty --dry-run", + "clean": "deno run -RWE --allow-ffi ./-scripts/-main.ts --cmd=clean", + "upgrade": "deno run -RWNE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=upgrade", + "backup": "deno run -RWE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=backup", + "help": "deno run -RE --allow-ffi ./-scripts/-main.ts --cmd=help", + "tmp": "deno run -A ./-scripts/-tmp.ts" }, "exports": { ".": "./src/mod.ts", "./t": "./src/types.ts", - "./types": "./src/types.ts" - } + "./types": "./src/types.ts", + "./specs": "./src/-test/entry.Specs.ts" + }, + "license": "MIT" } diff --git a/code/sys.ui/ui-react-components/src/-sample/-css-container/-SPEC.Debug.tsx b/code/sys.ui/ui-react-components/src/-sample/-css-container/-SPEC.Debug.tsx new file mode 100644 index 0000000000..31123c5cb2 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/-sample/-css-container/-SPEC.Debug.tsx @@ -0,0 +1,56 @@ +import React from 'react'; +import { type t, Button, Color, css, Signal } from '../-test.ui.ts'; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; +type P = DebugProps; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + const s = Signal.create; + const props = { theme: s<t.CommonTheme>('Light') }; + const api = { props }; + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<P> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => p.theme.value); + + /** + * Render: + */ + const theme = Color.theme(p.theme.value); + const styles = { + base: css({ + // color: theme.fg, + }), + title: css({ fontWeight: 'bold' }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={styles.title.class}>CSS: @container</div> + <hr /> + + <Button + block + label={`theme: ${p.theme}`} + onClick={() => Signal.cycle<t.CommonTheme>(p.theme, ['Light', 'Dark'])} + /> + + <hr /> + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/-sample/-css-container/-SPEC.tsx b/code/sys.ui/ui-react-components/src/-sample/-css-container/-SPEC.tsx new file mode 100644 index 0000000000..79cc8c437f --- /dev/null +++ b/code/sys.ui/ui-react-components/src/-sample/-css-container/-SPEC.tsx @@ -0,0 +1,28 @@ +import { Dev, Spec, Signal } from '../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { Container } from './ui.tsx'; + +export default Spec.describe('css:container-type', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + Dev.Theme.signalEffect(ctx, p.theme, 1); + Signal.effect(() => { + // š· TODO: hook into signals here. + ctx.redraw(); + }); + + ctx.subject + .size('fill') + .display('grid') + .render((e) => <Container theme={p.theme.value} />); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/-sample/-css-container/styles.css b/code/sys.ui/ui-react-components/src/-sample/-css-container/styles.css new file mode 100644 index 0000000000..c456e785d1 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/-sample/-css-container/styles.css @@ -0,0 +1,20 @@ +/* + SAMPLE: + See the equivalent programmatic [css-in-js] in "./ui.tsx". + + @container ā MDN Docs: + https://developer.mozilla.org/en-US/docs/Web/CSS/@container + + Baseline compatible "broadly supported across the major browsers as of <2023>" + https://developer.mozilla.org/en-US/docs/Glossary/Baseline/Compatibility +*/ + +.card h2 { + font-size: 2em; +} + +@container (min-width: 700px) { + .card h2 { + font-size: 5em; + } +} diff --git a/code/sys.ui/ui-react-components/src/-sample/-css-container/ui.tsx b/code/sys.ui/ui-react-components/src/-sample/-css-container/ui.tsx new file mode 100644 index 0000000000..d838841fcc --- /dev/null +++ b/code/sys.ui/ui-react-components/src/-sample/-css-container/ui.tsx @@ -0,0 +1,114 @@ +import React from 'react'; +import { type t, Color, css, Style, useSizeObserver } from '../-test.ui.ts'; + +/** + * š¼ NOTE: import the raw stylesheet + * + * @container ā MDN Docs: + * https://developer.mozilla.org/en-US/docs/Web/CSS/@container + * Compatibility: + * Baseline compatible ā "broadly supported across the major browsers as of <2023>" + * https://developer.mozilla.org/en-US/docs/Glossary/Baseline/Compatibility + * + */ +// import './styles.css'; + +export type ContainerProps = { + theme?: t.CommonTheme; + style?: t.CssInput; +}; + +type P = ContainerProps; + +/** + * Component: + */ +export const Container: React.FC<P> = (props) => { + const {} = props; + + const size = useSizeObserver(); + + /** + * REF: https://chatgpt.com/share/67ddffa7-a668-800b-87f1-4aa855733c7b + */ + React.useEffect(() => { + // return; + const sheet = Style.Dom.stylesheet(); + + // sheet.rule('.card h2', { fontSize: 50 }); + // sheet.rule('.card h2', { fontSize: 100 }, { context: '@container (min-width: 600px)' }); + + // const container = sheet.container('min-width: 600px'); + // container.rules.add('.card h2', { fontSize: 150 }); + + // sheet.rule(`.${styles.base.class} h2`, { fontSize: 50 }); + // const container = sheet.container('min-width: 600px'); + // container.rules.add(`.${styles.base.class} h2`, { fontSize: 150 }); + + // const container = sheet.container('min-width: 600px'); + // const scope = container.scope(`.${styles.h2.class}`); + + // const scope = styles.h2.container('min-width: 600px'); + // scope.rules.add('', { fontSize: 100 }); + }, []); + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const condition = 'min-width: 600px'; + + const styles = { + base: css({ + padding: 30, + color: theme.fg, + letterSpacing: -0.5, + containerType: 'size', // š¼ ā NB: turn this on for the @container rules to take effect (width AND height). + // containerType: 'inline-size', // š¼ ā NB: turn this on for the @container rules to take effect (width only). + overflow: 'hidden', + display: 'grid', + placeItems: 'center', + + // color: 'orange', + }) + .rule('h2', { color: 'red' }) + .rule('h2 code', { color: 'blue' }), // š¼ NB: aribitrary CSS sub-selector rule. + + h1: css({ + color: 'red', + fontSize: 50, + transition: 'font-size 200ms, color 200ms', + whiteSpace: 'nowrap', + }) + .container('min-width: 400px', { fontSize: 90, color: 'blue' }) + .container('min-width: 600px', { fontSize: 150, color: theme.fg, letterSpacing: -4 }) + .container('max-height: 500px', { color: 'orange' }).done, + + size: css({ Absolute: [8, 10, null, null], fontSize: 16 }), + }; + + // styles.base. + + /** + * Equivalent: (sample code) + */ + // const container = styles.base.container('min-width: 600px'); + // container.rule('h2', { + // fontSize: 140, + // color: 'blue', + // letterSpacing: -3, + // transition: 'font-size 200ms', + // }); + + const elSize = <div className={styles.size.class}>{`${size.toString()}`}</div>; + + return ( + <div ref={size.ref} className={css(styles.base, props.style).class}> + <h1 className={styles.h1.class}>{`Hello š`}</h1> + <h2> + {`H2 Heading`} <code>Code</code> + </h2> + {elSize} + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/-sample/-dom-useMouse/-SPEC.Debug.tsx b/code/sys.ui/ui-react-components/src/-sample/-dom-useMouse/-SPEC.Debug.tsx new file mode 100644 index 0000000000..2763b7dd7d --- /dev/null +++ b/code/sys.ui/ui-react-components/src/-sample/-dom-useMouse/-SPEC.Debug.tsx @@ -0,0 +1,55 @@ +import React from 'react'; +import { type t, Button, Color, css, Signal } from '../-test.ui.ts'; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + const s = Signal.create; + const props = {}; + const api = { + props, + listen() { + const p = props; + }, + }; + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => debug.listen()); + + /** + * Render: + */ + const styles = { + base: css({}), + title: css({ fontWeight: 'bold', marginBottom: 10 }), + cols: css({ display: 'grid', gridTemplateColumns: 'auto 1fr auto' }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={css(styles.title, styles.cols).class}> + <div>{'useMouse'}</div> + <div /> + <div>{'Hook'}</div> + </div> + + <hr /> + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/-sample/-dom-useMouse/-SPEC.tsx b/code/sys.ui/ui-react-components/src/-sample/-dom-useMouse/-SPEC.tsx new file mode 100644 index 0000000000..d189ad0954 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/-sample/-dom-useMouse/-SPEC.tsx @@ -0,0 +1,26 @@ +import { Dev, Spec, Signal } from '../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { Sample } from './ui.tsx'; + +export default Spec.describe('MyComponent', (e) => { + const debug = createDebugSignals(); + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + Signal.effect(() => { + debug.listen(); + ctx.redraw(); + }); + + ctx.subject + .size() + .display('grid') + .render((e) => <Sample debug={debug} />); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/-sample/-dom-useMouse/ui.tsx b/code/sys.ui/ui-react-components/src/-sample/-dom-useMouse/ui.tsx new file mode 100644 index 0000000000..cd8ba4b16c --- /dev/null +++ b/code/sys.ui/ui-react-components/src/-sample/-dom-useMouse/ui.tsx @@ -0,0 +1,39 @@ +import { useClickInside, useClickOutside, useMouse } from '@sys/ui-react'; +import React from 'react'; + +import { type t, css } from '../-test.ui.ts'; +import { type DebugSignals } from './-SPEC.Debug.tsx'; + +export type SampleProps = { + debug: DebugSignals; + style?: t.CssInput; +}; + +export const Sample: React.FC<SampleProps> = (props) => { + const { debug } = props; + + const ref = React.useRef<HTMLDivElement>(null); + useClickInside({ ref, callback: (e) => console.info(`ā”ļø click-inside:`, e) }); + useClickOutside({ ref, callback: (e) => console.info(`ā”ļø click-outside:`, e) }); + + const mouse = useMouse(); + console.info('mouse.is', mouse.is); + + /** + * Render: + */ + const styles = { + base: css({ + position: 'relative', + overflow: 'hidden', + userSelect: 'none', + padding: 20, + }), + }; + + return ( + <div ref={ref} className={css(styles.base, props.style).class}> + <div {...mouse.handlers}>{'š· Hello'}</div> + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/-sample/-dom-useSizeObserver/-SPEC.Debug.tsx b/code/sys.ui/ui-react-components/src/-sample/-dom-useSizeObserver/-SPEC.Debug.tsx new file mode 100644 index 0000000000..a96ba679db --- /dev/null +++ b/code/sys.ui/ui-react-components/src/-sample/-dom-useSizeObserver/-SPEC.Debug.tsx @@ -0,0 +1,43 @@ +import React from 'react'; +import { type t, css, ObjectView, Signal } from '../-test.ui.ts'; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + const s = Signal.create; + const api = { rect: s<t.DomRect>() }; + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const rect = debug.rect.value; + + Signal.useRedrawEffect(() => debug.rect.value); + + /** + * Render: + */ + const styles = { + base: css({}), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div>ā”ļø via signal:</div> + <ObjectView name={'rect'} data={rect} margin={[5, 22]} expand={1} /> + <hr /> + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/-sample/-dom-useSizeObserver/-SPEC.tsx b/code/sys.ui/ui-react-components/src/-sample/-dom-useSizeObserver/-SPEC.tsx new file mode 100644 index 0000000000..e27bbb8ba6 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/-sample/-dom-useSizeObserver/-SPEC.tsx @@ -0,0 +1,26 @@ +import { Signal, Spec } from '../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { Sample } from './ui.tsx'; + +export default Spec.describe('useSizeObserver', (e) => { + const debug = createDebugSignals(); + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + Signal.effect(() => { + debug.rect.value; + ctx.redraw(); + }); + + ctx.subject + .size('fill') + .display('grid') + .render((e) => <Sample debug={debug} />); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/-sample/-dom-useSizeObserver/ui.tsx b/code/sys.ui/ui-react-components/src/-sample/-dom-useSizeObserver/ui.tsx new file mode 100644 index 0000000000..d86c091665 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/-sample/-dom-useSizeObserver/ui.tsx @@ -0,0 +1,33 @@ +import React from 'react'; +import { type t, css, ObjectView, useSizeObserver } from '../-test.ui.ts'; +import { type DebugSignals } from './-SPEC.Debug.tsx'; + +export type SampleProps = { + debug: DebugSignals; + style?: t.CssInput; +}; + +/** + * Component: + */ +export const Sample: React.FC<SampleProps> = (props) => { + const { debug } = props; + const size = useSizeObserver((e) => (debug.rect.value = e.toObject())); + + /** + * Render: + */ + const styles = { + base: css({ position: 'relative', overflow: 'hidden' }), + body: css({ boxSizing: 'border-box', margin: 20 }), + }; + + return ( + <div ref={size.ref} className={css(styles.base, props.style).class}> + <div className={styles.body.class}> + <div>{'š useSizeObserver:'}</div> + <ObjectView name={'size.rect'} data={size.toObject()} block margin={[10, 20]} /> + </div> + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/-sample/-test.ui.ts b/code/sys.ui/ui-react-components/src/-sample/-test.ui.ts new file mode 100644 index 0000000000..6077d9257d --- /dev/null +++ b/code/sys.ui/ui-react-components/src/-sample/-test.ui.ts @@ -0,0 +1,3 @@ +export * from '../ui/-test.ui.ts'; +export { Button } from '../ui/Button/mod.ts'; +export { ObjectView } from '../ui/ObjectView/mod.ts'; diff --git a/code/sys.ui/ui-react-components/src/-sample/images/sample.larger.svg b/code/sys.ui/ui-react-components/src/-sample/images/sample.larger.svg new file mode 100644 index 0000000000..67dde8d42e --- /dev/null +++ b/code/sys.ui/ui-react-components/src/-sample/images/sample.larger.svg @@ -0,0 +1,24 @@ +<svg width="233" height="98" viewBox="0 0 233 98" fill="none" xmlns="http://www.w3.org/2000/svg"> +<g clip-path="url(#clip0_711_130)"> +<rect width="233" height="98" fill="white"/> +<path d="M85.82 70.186C85.572 70.186 85.3447 70.1033 85.138 69.938C84.9727 69.7727 84.89 69.5867 84.89 69.38C84.89 68.9253 85.2207 68.6153 85.882 68.45C87.2873 68.0367 88.4653 67.6853 89.416 67.396C90.3667 67.0653 91.0693 66.6107 91.524 66.032C92.02 65.412 92.268 64.544 92.268 63.428V40.178C92.268 39.3513 92.02 38.7107 91.524 38.256C91.028 37.8013 90.3667 37.4913 89.54 37.326C88.7547 37.1193 87.9073 36.9747 86.998 36.892C86.626 36.8507 86.254 36.7473 85.882 36.582C85.5513 36.3753 85.386 36.086 85.386 35.714C85.386 35.342 85.5513 35.0527 85.882 34.846C86.2127 34.598 86.6053 34.4533 87.06 34.412C89.1267 34.164 90.966 33.8127 92.578 33.358C94.2313 32.862 95.864 31.9527 97.476 30.63C97.5173 30.5887 97.5793 30.568 97.662 30.568C97.7447 30.5267 97.8273 30.506 97.91 30.506C98.158 30.506 98.3647 30.568 98.53 30.692C98.7367 30.816 98.84 30.9607 98.84 31.126C98.7573 31.7047 98.6747 32.3453 98.592 33.048C98.5507 33.7507 98.53 34.5773 98.53 35.528V63.986C98.53 64.9367 98.7367 65.6807 99.15 66.218C99.6047 66.714 100.287 67.1273 101.196 67.458C102.147 67.7473 103.387 68.078 104.916 68.45C105.205 68.5327 105.453 68.6567 105.66 68.822C105.867 68.946 105.97 69.1527 105.97 69.442C105.97 69.938 105.681 70.186 105.102 70.186C104.027 70.186 102.932 70.1447 101.816 70.062C100.741 70.0207 99.6667 69.9793 98.592 69.938C97.5587 69.8967 96.5253 69.876 95.492 69.876C94.4587 69.876 93.4047 69.8967 92.33 69.938C91.2553 69.9793 90.16 70.0207 89.044 70.062C87.9693 70.1447 86.8947 70.186 85.82 70.186ZM126 70.93C123.603 70.93 121.433 70.434 119.49 69.442C117.548 68.45 115.998 67.0033 114.84 65.102C113.683 63.1593 113.104 60.8033 113.104 58.034C113.104 55.7193 113.373 53.632 113.91 51.772C114.448 49.8707 115.109 48.1967 115.894 46.75C116.721 45.262 117.548 44.0013 118.374 42.968C119.201 41.9347 119.904 41.108 120.482 40.488C121.474 39.4133 122.756 38.2353 124.326 36.954C125.938 35.6727 127.798 34.4533 129.906 33.296C132.014 32.0973 134.288 31.1467 136.726 30.444C137.181 30.2787 137.594 30.2373 137.966 30.32C138.338 30.4027 138.483 30.5473 138.4 30.754C138.318 31.126 138.09 31.436 137.718 31.684C137.346 31.8907 136.974 32.0767 136.602 32.242C133.998 33.358 131.766 34.66 129.906 36.148C128.088 37.636 126.434 39.4753 124.946 41.666C124.326 42.658 123.582 44.0013 122.714 45.696C121.846 47.3907 121.102 49.292 120.482 51.4C119.862 53.508 119.552 55.6573 119.552 57.848C119.594 60.1213 120.007 62.0433 120.792 63.614C121.578 65.1847 122.57 66.3833 123.768 67.21C124.967 68.0367 126.166 68.45 127.364 68.45C128.894 68.4913 130.216 67.892 131.332 66.652C132.49 65.412 133.068 63.5313 133.068 61.01C133.068 59.7287 132.841 58.5713 132.386 57.538C131.932 56.4633 131.312 55.5953 130.526 54.934C129.741 54.2313 128.852 53.88 127.86 53.88C126.992 53.88 126.228 53.942 125.566 54.066C124.946 54.1487 124.347 54.3347 123.768 54.624C123.272 54.872 122.962 54.872 122.838 54.624C122.756 54.3347 122.859 54.004 123.148 53.632C123.438 53.3427 123.686 53.0947 123.892 52.888C124.14 52.6813 124.43 52.4747 124.76 52.268C125.256 51.8547 126.124 51.3793 127.364 50.842C128.604 50.2633 129.782 49.974 130.898 49.974C132.386 49.974 133.75 50.3253 134.99 51.028C136.272 51.7307 137.305 52.6813 138.09 53.88C138.876 55.0787 139.268 56.4633 139.268 58.034C139.268 59.8527 138.896 61.5473 138.152 63.118C137.45 64.6887 136.478 66.0733 135.238 67.272C133.998 68.4293 132.572 69.3387 130.96 70C129.39 70.62 127.736 70.93 126 70.93Z" fill="#383057" fill-opacity="0.7"/> +<path d="M143.117 70.186C142.207 70.186 141.753 69.938 141.753 69.442C141.753 69.1113 141.897 68.884 142.187 68.76C142.476 68.636 142.807 68.4913 143.179 68.326C143.799 68.1193 144.377 67.8713 144.915 67.582C145.452 67.2927 145.721 66.714 145.721 65.846V35.28C145.721 33.916 145.535 32.8827 145.163 32.18C144.791 31.4773 144.088 31.002 143.055 30.754C142.683 30.6713 142.497 30.382 142.497 29.886C142.497 29.4313 142.662 29.1627 142.993 29.08C144.067 28.8733 145.059 28.6047 145.969 28.274C146.919 27.9433 147.787 27.6333 148.573 27.344C149.399 27.0133 150.123 26.724 150.743 26.476C151.032 26.352 151.259 26.29 151.425 26.29C151.59 26.29 151.693 26.352 151.735 26.476C151.817 26.5587 151.859 26.7033 151.859 26.91C151.859 27.282 151.776 28.0053 151.611 29.08C151.487 30.1547 151.425 31.7253 151.425 33.792V55.12C151.425 55.5333 151.549 55.74 151.797 55.74C152.003 55.74 152.272 55.6573 152.603 55.492C152.933 55.3267 153.264 55.0993 153.595 54.81C155.248 53.4047 156.653 52.082 157.811 50.842C158.968 49.602 159.753 48.7133 160.167 48.176C160.332 47.928 160.456 47.742 160.539 47.618C160.663 47.4527 160.725 47.308 160.725 47.184C160.725 46.9773 160.539 46.7913 160.167 46.626C159.836 46.4193 159.443 46.2747 158.989 46.192C158.493 46.1093 158.245 45.9027 158.245 45.572C158.245 45.324 158.369 45.138 158.617 45.014C158.906 44.8487 159.237 44.766 159.609 44.766H162.647C163.308 44.766 164.073 44.7247 164.941 44.642C165.85 44.518 166.718 44.394 167.545 44.27C168.371 44.146 168.991 44.084 169.405 44.084C169.942 44.084 170.211 44.3113 170.211 44.766C170.211 45.014 170.066 45.2413 169.777 45.448C169.529 45.6547 169.177 45.82 168.723 45.944C167.979 46.1093 167.214 46.3987 166.429 46.812C165.643 47.184 164.899 47.5767 164.197 47.99C163.783 48.238 163.143 48.6927 162.275 49.354C161.407 50.0153 160.477 50.7593 159.485 51.586C158.493 52.4127 157.563 53.2187 156.695 54.004C156.447 54.2107 156.323 54.4173 156.323 54.624C156.323 54.7067 156.343 54.7893 156.385 54.872C156.467 54.9547 156.529 55.0373 156.571 55.12C157.645 56.4427 158.741 57.7653 159.857 59.088C160.973 60.4107 162.027 61.6093 163.019 62.684C164.011 63.7587 164.817 64.6267 165.437 65.288C166.139 65.9907 166.945 66.6107 167.855 67.148C168.805 67.6853 169.591 68.078 170.211 68.326C170.624 68.45 171.017 68.5947 171.389 68.76C171.802 68.884 172.009 69.1113 172.009 69.442C172.009 69.938 171.595 70.186 170.769 70.186C168.826 70.186 166.78 70.1447 164.631 70.062C162.523 70.0207 160.415 70 158.307 70C157.521 70 157.129 69.7727 157.129 69.318C157.129 69.07 157.232 68.8633 157.439 68.698C157.687 68.4913 157.976 68.3467 158.307 68.264C158.72 68.0987 159.051 67.9333 159.299 67.768C159.588 67.5613 159.733 67.2927 159.733 66.962C159.733 66.838 159.691 66.6933 159.609 66.528C159.526 66.3627 159.402 66.1767 159.237 65.97L152.231 57.538C152.065 57.29 151.9 57.166 151.735 57.166C151.528 57.2073 151.425 57.352 151.425 57.6C151.383 58.22 151.363 58.8607 151.363 59.522C151.363 60.1833 151.363 60.8653 151.363 61.568C151.363 62.2293 151.363 62.9527 151.363 63.738C151.363 64.482 151.383 65.2053 151.425 65.908C151.425 66.652 151.611 67.1893 151.983 67.52C152.355 67.8507 152.851 68.1193 153.471 68.326C153.801 68.45 154.111 68.5947 154.401 68.76C154.69 68.884 154.835 69.1113 154.835 69.442C154.835 69.7313 154.669 69.9173 154.339 70C154.008 70.124 153.739 70.186 153.533 70.186C152.747 70.186 152.107 70.1447 151.611 70.062C151.115 70.0207 150.66 69.9793 150.247 69.938C149.833 69.8967 149.255 69.876 148.511 69.876C147.808 69.876 147.209 69.8967 146.713 69.938C146.217 69.9793 145.7 70.0207 145.163 70.062C144.625 70.1447 143.943 70.186 143.117 70.186ZM187.825 70.868C186.791 70.868 185.799 70.7647 184.849 70.558C183.898 70.3927 183.009 70.2067 182.183 70C181.397 69.7933 180.695 69.6073 180.075 69.442C179.455 69.2353 179.021 69.132 178.773 69.132C178.442 69.132 178.132 69.2147 177.843 69.38C177.595 69.5453 177.347 69.7313 177.099 69.938C176.809 70.186 176.582 70.31 176.417 70.31C176.169 70.31 175.941 70.2273 175.735 70.062C175.528 69.8967 175.425 69.7107 175.425 69.504C175.425 69.132 175.507 68.4707 175.673 67.52C175.879 66.5693 175.983 65.5567 175.983 64.482L175.859 35.28C175.859 34.2053 175.631 33.2753 175.177 32.49C174.722 31.6633 174.061 31.0847 173.193 30.754C172.986 30.6713 172.8 30.568 172.635 30.444C172.469 30.2787 172.387 30.072 172.387 29.824C172.387 29.6173 172.449 29.452 172.573 29.328C172.738 29.204 172.924 29.1213 173.131 29.08C173.792 28.956 174.639 28.7287 175.673 28.398C176.706 28.026 177.677 27.654 178.587 27.282C179.537 26.91 180.219 26.6413 180.633 26.476C180.922 26.352 181.129 26.29 181.253 26.29C181.418 26.29 181.542 26.352 181.625 26.476C181.707 26.5587 181.749 26.6827 181.749 26.848C181.749 27.0547 181.707 27.6953 181.625 28.77C181.583 29.8033 181.563 31.0227 181.563 32.428L181.501 45.324C181.501 46.0267 181.542 46.5227 181.625 46.812C181.749 47.1013 181.893 47.246 182.059 47.246C183.34 46.254 184.766 45.4687 186.337 44.89C187.949 44.27 189.416 43.96 190.739 43.96C192.847 43.96 194.748 44.4973 196.443 45.572C198.137 46.6467 199.481 48.1347 200.473 50.036C201.506 51.896 202.023 54.004 202.023 56.36C202.023 59.212 201.423 61.7333 200.225 63.924C199.026 66.1147 197.352 67.83 195.203 69.07C193.095 70.2687 190.635 70.868 187.825 70.868ZM189.127 68.698C190.532 68.698 191.731 68.2227 192.723 67.272C193.756 66.28 194.541 64.9573 195.079 63.304C195.657 61.6507 195.947 59.832 195.947 57.848C195.947 56.1947 195.678 54.5207 195.141 52.826C194.603 51.1313 193.756 49.726 192.599 48.61C191.441 47.4527 189.974 46.874 188.197 46.874C186.874 46.874 185.634 47.0807 184.477 47.494C183.319 47.9073 182.327 48.4653 181.501 49.168V61.63C181.501 63.0353 181.852 64.2753 182.555 65.35C183.299 66.4247 184.249 67.2513 185.407 67.83C186.605 68.4087 187.845 68.698 189.127 68.698Z" fill="#F60000"/> +<path d="M209.953 71.24C208.878 71.24 207.948 70.8473 207.163 70.062C206.377 69.2767 205.985 68.3467 205.985 67.272C205.985 66.1147 206.377 65.164 207.163 64.42C207.948 63.6347 208.878 63.242 209.953 63.242C211.069 63.242 211.999 63.6347 212.743 64.42C213.528 65.164 213.921 66.1147 213.921 67.272C213.921 68.3467 213.528 69.2767 212.743 70.062C211.999 70.8473 211.069 71.24 209.953 71.24Z" fill="#383057"/> +<mask id="mask0_711_130" style="mask-type:luminance" maskUnits="userSpaceOnUse" x="14" y="11" width="57" height="90"> +<path d="M14 11H70.124V100.383H14V11Z" fill="white"/> +</mask> +<g mask="url(#mask0_711_130)"> +<mask id="mask1_711_130" style="mask-type:luminance" maskUnits="userSpaceOnUse" x="14" y="11" width="50" height="79"> +<path d="M14 11H63.116V89.221H14V11Z" fill="white"/> +</mask> +<g mask="url(#mask1_711_130)"> +<path fill-rule="evenodd" clip-rule="evenodd" d="M60.156 23.301C61.7828 24.93 62.7 27.1358 62.708 29.438L62.835 80.21C62.8312 82.5988 61.882 84.8889 60.195 86.58C58.5107 88.2682 56.2247 89.2182 53.84 89.221H49.064C48.536 89.221 48.03 89.011 47.657 88.637C47.2833 88.2625 47.0733 87.7551 47.073 87.226V75.181H22.996C21.8151 75.18 20.6459 74.9463 19.5554 74.4933C18.4648 74.0404 17.4741 73.3769 16.64 72.541C14.9527 70.8496 14.0036 68.5591 14 66.17V19.7C14.0016 17.3943 14.9163 15.183 16.544 13.55C17.3487 12.743 18.3045 12.1026 19.3568 11.6652C20.4092 11.2278 21.5374 11.0021 22.677 11.001H27.74C28.1382 10.9944 28.5291 11.1076 28.8621 11.3261C29.195 11.5445 29.4546 11.8581 29.607 12.226C29.71 12.47 29.762 12.732 29.762 12.996V20.755H54.031C56.328 20.759 58.531 21.675 60.156 23.302V23.301ZM19.358 16.373C18.4772 17.2568 17.9818 18.4532 17.98 19.701L17.988 58.691C19.4662 57.6911 21.2104 57.1575 22.995 57.159H25.782V14.992H22.677C21.4313 14.9941 20.2375 15.4909 19.358 16.373ZM22.996 71.189C22.3375 71.189 21.6856 71.0591 21.0774 70.8067C20.4692 70.5543 19.9169 70.1843 19.452 69.718C18.5112 68.7749 17.9829 67.4971 17.9829 66.165C17.9829 64.8329 18.5112 63.5551 19.452 62.612C19.9169 62.1457 20.4692 61.7757 21.0774 61.5233C21.6856 61.2709 22.3375 61.141 22.996 61.141H27.772C28.3 61.141 28.806 60.931 29.179 60.556C29.552 60.182 29.762 59.674 29.762 59.146V24.736H54.031C55.2753 24.7384 56.4679 25.2343 57.347 26.115C58.2272 26.997 58.7232 28.191 58.727 29.437L58.831 72.706C57.3564 71.7119 55.6184 71.1812 53.84 71.182L22.996 71.19V71.189ZM51.054 85.23H53.84C55.169 85.23 56.444 84.701 57.383 83.759C58.324 82.816 58.8525 81.5382 58.8525 80.206C58.8525 78.8738 58.324 77.596 57.383 76.653C56.9182 76.1869 56.366 75.817 55.758 75.5646C55.1501 75.3122 54.4983 75.1822 53.84 75.182H51.054V85.23Z" fill="#F60000"/> +</g> +</g> +</g> +<defs> +<clipPath id="clip0_711_130"> +<rect width="233" height="98" fill="white"/> +</clipPath> +</defs> +</svg> diff --git a/code/sys.ui/ui-react-components/src/-sample/images/sample.small.svg b/code/sys.ui/ui-react-components/src/-sample/images/sample.small.svg new file mode 100644 index 0000000000..614640ea09 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/-sample/images/sample.small.svg @@ -0,0 +1,22 @@ +<svg width="1059" height="1059" viewBox="0 0 1059 1059" fill="none" xmlns="http://www.w3.org/2000/svg"> +<g id="svg.sample" clip-path="url(#clip0)"> +<rect id="border-outline" x="114.792" y="114.792" width="828.646" height="828.646" rx="149.98" stroke="#383057" stroke-width="26.1768"/> +<g id="inner"> +<g id="tick"> +<path id="Vector" d="M748.501 529.114L699.837 473.469L706.618 399.875L634.62 383.521L596.925 319.699L529.115 348.818L461.305 319.699L423.61 383.321L351.611 399.476L358.392 473.27L309.729 529.114L358.392 584.758L351.611 658.552L423.61 674.906L461.305 738.528L529.115 709.21L596.925 738.328L634.62 674.706L706.618 658.352L699.837 584.758L748.501 529.114ZM491.021 623.25L415.233 547.263L444.751 517.745L491.021 564.215L607.695 447.143L637.213 476.66L491.021 623.25Z" fill="#383057"/> +</g> +<circle id="circle" cx="529.114" cy="529.114" r="313.049" stroke="#383057" stroke-width="2.14564" stroke-dasharray="12.87 12.87"/> +</g> +<g id="corner-pointers" opacity="0.5"> +<line id="Line 3125" x1="0.353553" y1="-0.353553" x2="284.671" y2="283.964" stroke="#383057"/> +<line id="Line 3126" y1="-0.5" x2="402.085" y2="-0.5" transform="matrix(-0.707107 0.707107 0.707107 0.707107 284.316 773.91)" stroke="#383057"/> +<line id="Line 3127" y1="-0.5" x2="402.085" y2="-0.5" transform="matrix(-0.707107 0.707107 0.707107 0.707107 1058.23 0)" stroke="#383057"/> +<line id="Line 3128" x1="774.266" y1="773.557" x2="1058.58" y2="1057.87" stroke="#383057"/> +</g> +</g> +<defs> +<clipPath id="clip0"> +<rect width="1058.23" height="1058.23" fill="white"/> +</clipPath> +</defs> +</svg> diff --git a/deploy/slc.db.team/src/ui/common.ts b/code/sys.ui/ui-react-components/src/-test/common.ts similarity index 100% rename from deploy/slc.db.team/src/ui/common.ts rename to code/sys.ui/ui-react-components/src/-test/common.ts diff --git a/code/sys.ui/ui-react-components/src/-test/entry.Specs.ts b/code/sys.ui/ui-react-components/src/-test/entry.Specs.ts index 18d00d8cb4..c6de160309 100644 --- a/code/sys.ui/ui-react-components/src/-test/entry.Specs.ts +++ b/code/sys.ui/ui-react-components/src/-test/entry.Specs.ts @@ -1,3 +1,41 @@ -import { type t } from '../common.ts'; +/** + * @module + * DevHarness visual specs. + */ +import type { t } from './common.ts'; +const ns = 'sys.ui.react.component'; -export const Specs = {} as t.SpecImports; +/** + * Components: + */ +export const SpecsComponents = { + [`${ns}: Button`]: () => import('../ui/Button/-SPEC.tsx'), + [`${ns}: Cropmarks`]: () => import('../ui/Cropmarks/-SPEC.tsx'), + [`${ns}: Icon`]: () => import('../ui/Icon/-SPEC.tsx'), + [`${ns}: IFrame`]: () => import('../ui/IFrame/-SPEC.tsx'), + [`${ns}: Image.Svg`]: () => import('../ui/Image.Svg/-SPEC.tsx'), + [`${ns}: ObjectView`]: () => import('../ui/ObjectView/-SPEC.tsx'), + [`${ns}: Panel`]: () => import('../ui/Panel/-SPEC.tsx'), + [`${ns}: Preload`]: () => import('../ui/Preload/-SPEC.tsx'), + [`${ns}: Sheet`]: () => import('../ui/Sheet/-SPEC.tsx'), + [`${ns}: Spinners.Bar`]: () => import('../ui/Spinners.Bar/-SPEC.tsx'), + + [`${ns}: Player.Video`]: () => import('../ui/Player.Video/-SPEC.tsx'), + [`${ns}: Player.Concept`]: () => import('../ui/Player.Concept/-SPEC.tsx'), + [`${ns}: Player.Thumbnails`]: () => import('../ui/Player.Thumbnails/-SPEC.tsx'), + [`${ns}: VimeoBackground`]: () => import('../ui/VimeoBackground/-SPEC.tsx'), +} as t.SpecImports; + +/** + * Samples from external libs: + */ +export const SpecsExternal = { + 'sys.ui.css: @container': () => import('../-sample/-css-container/-SPEC.tsx'), + 'sys.ui.react: useMouse': () => import('../-sample/-dom-useMouse/-SPEC.tsx'), + 'sys.ui.react: useSizeObserver': () => import('../-sample/-dom-useSizeObserver/-SPEC.tsx'), +} as t.SpecImports; + +/** + * Specs + */ +export const Specs = { ...SpecsComponents, ...SpecsExternal } as t.SpecImports; diff --git a/code/sys.ui/ui-react-components/src/-test/entry.tsx b/code/sys.ui/ui-react-components/src/-test/entry.tsx index 2fa8a1f04c..2e8e89e8b7 100644 --- a/code/sys.ui/ui-react-components/src/-test/entry.tsx +++ b/code/sys.ui/ui-react-components/src/-test/entry.tsx @@ -1,31 +1,42 @@ -import { StrictMode } from 'react'; +import React, { StrictMode } from 'react'; import { createRoot } from 'react-dom/client'; + import { pkg } from '../common.ts'; /** * Render UI. */ globalThis.document.title = pkg.name; -console.info('Pkg', pkg); +console.info('š· ./entry.tsx ā Pkg:š¦', pkg); -/** - * Main Entry. - */ -async function main() { +export async function main() { const params = new URL(location.href).searchParams; const isDev = params.has('dev') || params.has('d'); + const root = createRoot(document.getElementById('root')!); - console.log('isDev', isDev); + if (isDev) { + const { render, useKeyboard } = await import('@sys/ui-react-devharness'); + const { Specs } = await import('./entry.Specs.ts'); - const { render } = await import('@sys/ui-react-devharness'); - const { Specs } = await import('./entry.Specs.ts'); + const el = await render(pkg, Specs, { hrDepth: 2, style: { Absolute: 0 } }); + function App() { + useKeyboard(); + return el; + } - const el = await render(pkg, Specs, { hrDepth: 2, style: { Absolute: 0 } }); - const root = document.getElementById('root'); - createRoot(root).render(<StrictMode>{el}</StrictMode>); + root.render( + <StrictMode> + <App /> + </StrictMode>, + ); + } else { + const { Splash } = await import('./ui.Splash.tsx'); + root.render( + <StrictMode> + <Splash style={{ Absolute: 0 }} /> + </StrictMode>, + ); + } } -/** - * Run. - */ -main().catch((err) => console.error(`Failed to render`, err)); +main().catch((err) => console.error(`Failed to render DevHarness`, err)); diff --git a/code/sys.ui/ui-react-components/src/-test/mod.ts b/code/sys.ui/ui-react-components/src/-test/mod.ts index a3f41ce278..278cf7f716 100644 --- a/code/sys.ui/ui-react-components/src/-test/mod.ts +++ b/code/sys.ui/ui-react-components/src/-test/mod.ts @@ -1,2 +1,2 @@ -export { describe, expect, expectError, it, Testing } from '@sys/testing/server'; -export * from '../common.ts'; +export { describe, DomMock, expect, expectError, it, Testing } from '@sys/testing/server'; +export * from './common.ts'; diff --git a/code/sys.ui/ui-react-components/src/-test/ui.Splash.tsx b/code/sys.ui/ui-react-components/src/-test/ui.Splash.tsx new file mode 100644 index 0000000000..01b17142cc --- /dev/null +++ b/code/sys.ui/ui-react-components/src/-test/ui.Splash.tsx @@ -0,0 +1,38 @@ +import React from 'react'; +import { type t, Color, css } from './common.ts'; + +import { useKeyboard } from '@sys/ui-react-devharness'; + +export type SplashProps = { + theme?: t.CommonTheme; + style?: t.CssInput; +}; + +export const Splash: React.FC<SplashProps> = (props) => { + const href = '?dev'; + useKeyboard(); + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ + fontFamily: 'sans-serif', + display: 'grid', + placeItems: 'center', + color: theme.fg, + }), + a: css({ + textDecoration: 'none', + fontSize: 30, + color: Color.BLUE, + }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <a href={href} className={styles.a.class}>{`š· ${href}`}</a> + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/.test.ts b/code/sys.ui/ui-react-components/src/.test.ts index 25caefb420..7348c03d0a 100644 --- a/code/sys.ui/ui-react-components/src/.test.ts +++ b/code/sys.ui/ui-react-components/src/.test.ts @@ -1,4 +1,4 @@ -import { describe, expect, it, Pkg, pkg } from './-test.ts'; +import { type t, describe, it, expect, Pkg, pkg } from './-test.ts'; describe(`module: ${Pkg.toString(pkg)}`, () => { it('exists', () => { diff --git a/code/sys.ui/ui-react-components/src/common/libs.ts b/code/sys.ui/ui-react-components/src/common/libs.ts index 6f2db1bc06..7ab8336c72 100644 --- a/code/sys.ui/ui-react-components/src/common/libs.ts +++ b/code/sys.ui/ui-react-components/src/common/libs.ts @@ -1 +1,11 @@ -export { Err, Pkg, rx, Time } from '@sys/std'; +import { motion as Motion } from 'motion/react'; +export { AnimatePresence } from 'motion/react'; +export { Motion as M, Motion }; + +/** + * System + */ +export { Err, Is, isRecord, Path, Pkg, rx, slug, Str, Time, Timestamp } from '@sys/std'; +export { Color, css, Style } from '@sys/ui-css'; +export { Keyboard } from '@sys/ui-dom'; +export { Signal, useSizeObserver, useIsTouchSupported } from '@sys/ui-react'; diff --git a/code/sys.ui/ui-react-components/src/common/t.ts b/code/sys.ui/ui-react-components/src/common/t.ts index b9ff427b14..25fdaf695c 100644 --- a/code/sys.ui/ui-react-components/src/common/t.ts +++ b/code/sys.ui/ui-react-components/src/common/t.ts @@ -1,3 +1,16 @@ -export type { SpecImports } from '@sys/testing/t'; +export type { ReactNode } from 'react'; + export type * from '@sys/types'; + +export type { SpecImports } from '@sys/testing/t'; +export type { + CssEdgesInput, + CssInput, + CssMarginArray, + CssMarginInput, + CssProps, + CssValue, +} from '@sys/ui-css/t'; +export type { ExtractSignalValue, ReadonlySignal, Signal } from '@sys/ui-react/t'; + export type * from '../types.ts'; diff --git a/code/sys.ui/ui-react-components/src/mod.ts b/code/sys.ui/ui-react-components/src/mod.ts index d1cb0441e4..18e4cf6087 100644 --- a/code/sys.ui/ui-react-components/src/mod.ts +++ b/code/sys.ui/ui-react-components/src/mod.ts @@ -1,16 +1,31 @@ /** * @module - * Common UI Components + * Library of common `<React>` components. + * + * @example + * ```ts + * import { Foo } from '@sys/ui-react-components'; * ``` */ export { pkg } from './pkg.ts'; -/** - * Module types. - */ +/** Module types. */ export type * as t from './types.ts'; /** - * Library. + * Components. */ -export { Foo } from './ui/Foo.tsx'; +export { Button } from './ui/Button/mod.ts'; +export { Cropmarks } from './ui/Cropmarks/mod.ts'; +export { Icon } from './ui/Icon/mod.ts'; +export { Svg } from './ui/Image.Svg/mod.ts'; +export { ObjectView } from './ui/ObjectView/mod.ts'; +export { Panel } from './ui/Panel/mod.ts'; +export { Preload } from './ui/Preload/mod.ts'; +export { Sheet } from './ui/Sheet/mod.ts'; +export { Spinners } from './ui/Spinners/mod.ts'; + +export { ConceptPlayer } from './ui/Player.Concept/mod.ts'; +export { VideoPlayer } from './ui/Player.Video/mod.ts'; +export { Player } from './ui/Player/mod.ts'; +export { VimeoBackground } from './ui/VimeoBackground/mod.ts'; diff --git a/code/sys.ui/ui-react-components/src/pkg.ts b/code/sys.ui/ui-react-components/src/pkg.ts index 79cb3f9c80..a072fa9f7b 100644 --- a/code/sys.ui/ui-react-components/src/pkg.ts +++ b/code/sys.ui/ui-react-components/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/ui-react-components', version: '0.0.41' }; diff --git a/code/sys.ui/ui-react-components/src/types.ts b/code/sys.ui/ui-react-components/src/types.ts index 0f2f882807..cf8a421c86 100644 --- a/code/sys.ui/ui-react-components/src/types.ts +++ b/code/sys.ui/ui-react-components/src/types.ts @@ -2,4 +2,19 @@ * @module * Module types. */ -export {}; +export type * from './ui/Button/t.ts'; +export type * from './ui/Cropmarks/t.ts'; +export type * from './ui/Icon/t.ts'; +export type * from './ui/IFrame/t.ts'; +export type * from './ui/Image.Svg/t.ts'; +export type * from './ui/ObjectView/t.ts'; +export type * from './ui/Panel/t.ts'; +export type * from './ui/Player.Concept/t.ts'; +export type * from './ui/Player.Thumbnails/t.ts'; +export type * from './ui/Player.Video/t.ts'; +export type * from './ui/Player/t.ts'; +export type * from './ui/Preload/t.ts'; +export type * from './ui/Sheet/t.ts'; +export type * from './ui/Spinners.Bar/t.ts'; +export type * from './ui/Spinners/t.ts'; +export type * from './ui/VimeoBackground/t.ts'; diff --git a/code/sys.ui/ui-react-components/src/ui/-test.ui.ts b/code/sys.ui/ui-react-components/src/ui/-test.ui.ts new file mode 100644 index 0000000000..cffdc99448 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/-test.ui.ts @@ -0,0 +1,16 @@ +/** + * @module + * Testing tools running in the browser/ui. + */ +export { expect } from '@sys/std/testing'; +export { Dev, Spec } from '@sys/ui-react-devharness'; +export * from '../common.ts'; + +/** + * Constants: + */ +export const VIMEO = { + 'app/tubes': 499921561, + 'stock/running': 287903693, // https://vimeo.com/stock/clip-287903693-silhouette-woman-running-on-beach-waves-splashing-female-athlete-runner-exercising-sprinting-intense-workout-on-rough-ocean-seas + 'public/helvetica': 73809723, +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Button/--SPEC.CopyButton.tsx_ b/code/sys.ui/ui-react-components/src/ui/Button/--SPEC.CopyButton.tsx_ new file mode 100644 index 0000000000..c63e534211 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Button/--SPEC.CopyButton.tsx_ @@ -0,0 +1,95 @@ +import { Button } from '.'; +import { Dev, slug, type t } from '../../test.ui'; + +const DEFAULTS = Button.DEFAULTS; +type T = { props: t.CopyButtonProps }; +const initial: T = { props: {} }; + +/** + * Spec + */ +const name = Button.Copy.displayName ?? ''; + +export default Dev.describe(name, (e) => { + type LocalStore = Pick<t.CopyButtonProps, 'enabled' | 'spinning' | 'theme'>; + const localstore = Dev.LocalStorage<LocalStore>('dev:sys.ui.common.Button.Copy'); + const local = localstore.object({ + enabled: DEFAULTS.enabled, + spinning: DEFAULTS.spinning, + theme: undefined, + }); + + e.it('ui:init', async (e) => { + const ctx = Dev.ctx(e); + const dev = Dev.tools<T>(e, initial); + + const state = await ctx.state<T>(initial); + await state.change((d) => { + d.props.enabled = local.enabled; + d.props.spinning = local.spinning; + d.props.theme = local.theme; + }); + + ctx.debug.width(330); + ctx.subject + .backgroundColor(1) + .display('grid') + .render<T>((e) => { + const { props } = e.state; + Dev.Theme.background(ctx, props.theme, 1); + + return ( + <Button.Copy + {...props} + onClick={(e) => console.info('ā”ļø onClick')} + onCopy={(e) => { + console.info('ā”ļø onCopy', e); + e.copy(`My Text (${slug()})`); + // e.delay(1200); + // e.message('Foobar'); + }} + > + {'My Text Button'} + </Button.Copy> + ); + }); + }); + + e.it('ui:debug', async (e) => { + const dev = Dev.tools<T>(e, initial); + const state = await dev.state(); + + dev.section('Properties', (dev) => { + Dev.Theme.switch(dev, ['props', 'theme'], (next) => (local.theme = next)); + + dev.hr(-1, 5); + + dev.boolean((btn) => { + const value = (state: T) => Boolean(state.props.enabled); + btn + .label((e) => `enabled`) + .value((e) => value(e.state)) + .onClick((e) => e.change((d) => (local.enabled = Dev.toggle(d.props, 'enabled')))); + }); + + dev.boolean((btn) => { + const value = (state: T) => Boolean(state.props.spinning); + btn + .label((e) => `spinning`) + .value((e) => value(e.state)) + .onClick((e) => e.change((d) => (local.spinning = Dev.toggle(d.props, 'spinning')))); + }); + }); + + dev.hr(5, 20); + }); + + e.it('ui:footer', async (e) => { + const dev = Dev.tools<T>(e, initial); + const state = await dev.state(); + dev.footer.border(-0.1).render<T>((e) => { + const data = e.state; + return <Dev.Object name={name} data={data} expand={1} />; + }); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/Button/--SPEC.tsx_ b/code/sys.ui/ui-react-components/src/ui/Button/--SPEC.tsx_ new file mode 100644 index 0000000000..70058c743f --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Button/--SPEC.tsx_ @@ -0,0 +1,195 @@ +import { Button } from '.'; +import { Delete, Dev, Pkg, type t } from '../../test.ui'; + +const DEFAULTS = Button.DEFAULTS; + +type T = { + props: t.ButtonProps; + debug: { useLabel?: boolean; padding?: boolean }; +}; +const initial: T = { + props: {}, + debug: {}, +}; + +const name = Button.displayName ?? 'Unknown'; +export default Dev.describe(name, (e) => { + type LocalStore = T['debug'] & + Pick< + t.ButtonProps, + | 'theme' + | 'enabled' + | 'active' + | 'block' + | 'spinning' + | 'tooltip' + | 'label' + | 'isOver' + | 'isDown' + >; + const localstore = Dev.LocalStorage<LocalStore>(`dev:${Pkg.name}.${name}`); + const local = localstore.object({ + theme: undefined, + useLabel: true, + padding: false, + enabled: DEFAULTS.enabled, + active: DEFAULTS.active, + block: DEFAULTS.block, + spinning: DEFAULTS.spinning, + tooltip: 'My Button', + label: 'Hello-š·', + isOver: undefined, + isDown: undefined, + }); + + e.it('init', async (e) => { + const ctx = Dev.ctx(e); + const state = await ctx.state<T>(initial); + + await state.change((d) => { + d.props.theme = local.theme; + d.props.enabled = local.enabled; + d.props.active = local.active; + d.props.block = local.block; + d.props.spinning = local.spinning; + d.props.tooltip = local.tooltip; + d.props.label = local.label; + d.props.isOver = local.isOver; + d.props.isDown = local.isDown; + d.debug.useLabel = local.useLabel; + d.debug.padding = local.padding; + }); + + ctx.subject.display('grid').render<T>((e) => { + const { debug } = e.state; + Dev.Theme.background(ctx, e.state.props.theme); + + const props = { + ...e.state.props, + padding: debug.padding ? 20 : undefined, + }; + + if (debug.useLabel) { + props.label = 'Label-š·'; + props.children = undefined; + } else { + props.label = undefined; + props.children = <div>{'My Child Element'}</div>; + } + + return ( + <Button + {...props} + onMouse={(e) => console.info(`ā”ļø onMouse`, e)} + onClick={(e) => { + console.info('ā”ļø onClick', e); + // state.change((d) => Dev.toggle(d.props, 'spinning')); + }} + /> + ); + }); + }); + + e.it('ui:debug', async (e) => { + const dev = Dev.tools<T>(e, initial); + + dev.section('Properties', (dev) => { + Dev.Theme.switch(dev, ['props', 'theme'], (next) => (local.theme = next)); + + dev.boolean((btn) => + btn + .label('enabled') + .value((e) => e.state.props.enabled) + .onClick((e) => e.change((d) => (local.enabled = Dev.toggle(d.props, 'enabled')))), + ); + + dev.boolean((btn) => + btn + .label('active') + .value((e) => e.state.props.active) + .onClick((e) => e.change((d) => (local.active = Dev.toggle(d.props, 'active')))), + ); + + dev.boolean((btn) => + btn + .label((e) => 'spinning') + .value((e) => e.state.props.spinning) + .onClick((e) => e.change((d) => (local.spinning = Dev.toggle(d.props, 'spinning')))), + ); + + dev.hr(-1, 5); + + dev.boolean((btn) => { + const value = (state: T) => !!state.props.isOver; + btn + .label((e) => `isOver (force)`) + .value((e) => value(e.state)) + .onClick((e) => e.change((d) => (local.isOver = Dev.toggle(d.props, 'isOver')))); + }); + dev.boolean((btn) => { + const value = (state: T) => !!state.props.isDown; + btn + .label((e) => `isDown (force)`) + .value((e) => value(e.state)) + .onClick((e) => e.change((d) => (local.isDown = Dev.toggle(d.props, 'isDown')))); + }); + + dev.hr(-1, 5); + + dev.boolean((btn) => + btn + .label('block') + .value((e) => e.state.props.block) + .onClick((e) => e.change((d) => (local.block = Dev.toggle(d.props, 'block')))), + ); + + dev.boolean((btn) => { + const value = (state: T) => !!state.props.overlay; + btn + .label((e) => `overlay (Element)`) + .value((e) => value(e.state)) + .onClick((e) => { + e.change((d) => { + const el = <div style={{ fontSize: 11 }}>{`Overlay š·`}</div>; + d.props.overlay = d.props.overlay ? undefined : el; + }); + }); + }); + }); + + dev.hr(5, 20); + + dev.section('Debug', (dev) => { + dev.button((btn) => + btn + .label('content: <Element>') + .right((e) => (!e.state.debug.useLabel ? 'ā current' : '')) + .onClick((e) => e.change((d) => (local.useLabel = d.debug.useLabel = false))), + ); + + dev.button((btn) => + btn + .label('content: "label" property (string)') + .right((e) => (e.state.debug.useLabel ? 'ā current' : '')) + .onClick((e) => e.change((d) => (local.useLabel = d.debug.useLabel = true))), + ); + + dev.hr(-1, 5); + + dev.boolean((btn) => + btn + .label((e) => `padding`) + .value((e) => Boolean(e.state.debug.padding)) + .onClick((e) => e.change((d) => (local.padding = Dev.toggle(d.debug, 'padding')))), + ); + }); + }); + + e.it('ui:footer', async (e) => { + const dev = Dev.tools<T>(e, initial); + dev.footer.border(-0.1).render<T>((e) => { + const data = { props: Delete.undefined(e.state.props) }; + return <Dev.Object name={name} data={data} expand={1} />; + }); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/Button/-SPEC.Debug.tsx b/code/sys.ui/ui-react-components/src/ui/Button/-SPEC.Debug.tsx new file mode 100644 index 0000000000..2f61d5d788 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Button/-SPEC.Debug.tsx @@ -0,0 +1,31 @@ +import React, { useEffect, useRef, useState } from 'react'; +import { type t, Color, css, Signal, DEFAULTS, rx } from './common.ts'; + +export type DebugProps = { + ctx: {}; + theme?: t.CommonTheme; + style?: t.CssInput; +}; + +type P = DebugProps; + +/** + * Component + */ +export const Debug: React.FC<P> = (props) => { + const {} = props; + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ color: theme.fg }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div>{`š· Debug`}</div> + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Button/-SPEC.tsx b/code/sys.ui/ui-react-components/src/ui/Button/-SPEC.tsx new file mode 100644 index 0000000000..db596e62b9 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Button/-SPEC.tsx @@ -0,0 +1,23 @@ +import { Spec } from '../-test.ui.ts'; +import { Debug } from './-SPEC.Debug.tsx'; +import { Button } from './mod.ts'; + +export default Spec.describe('Button', (e) => { + e.it('init', async (e) => { + const ctx = Spec.ctx(e); + ctx.subject.size([224, null]).render((e) => { + return ( + <Button + onClick={(e) => console.info(`ā”ļø onClick:`, e)} + onMouseDown={(e) => console.info(`ā”ļø onMouseDown:`, e)} + onMouseUp={(e) => console.info(`ā”ļø onMouseUp:`, e)} + >{`š Hello Button`}</Button> + ); + }); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug ctx={{}} />); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/Button/common.ts b/code/sys.ui/ui-react-components/src/ui/Button/common.ts new file mode 100644 index 0000000000..0d0a03e9fe --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Button/common.ts @@ -0,0 +1,19 @@ +import { pkg } from '../common.ts'; +export * from '../common.ts'; + +/** + * Constants + */ +const name = 'Button'; +const displayName = `${pkg.name}:${name}`; + +export const DEFAULTS = { + name, + displayName, + enabled: true, + active: true, + block: false, + disabledOpacity: 0.3, + userSelect: false, + pressedOffset: [0, 1] as [number, number], +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Button/mod.ts b/code/sys.ui/ui-react-components/src/ui/Button/mod.ts new file mode 100644 index 0000000000..96209b7d8d --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Button/mod.ts @@ -0,0 +1,5 @@ +/** + * @module + * Clickable buttons. + */ +export { Button } from './ui.tsx'; diff --git a/code/sys.ui/ui-react-components/src/ui/Button/t.ts b/code/sys.ui/ui-react-components/src/ui/Button/t.ts new file mode 100644 index 0000000000..9a80d3d7e9 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Button/t.ts @@ -0,0 +1,62 @@ +import type React from 'react'; +import type { t } from './common.ts'; + +type MouseHandler = React.MouseEventHandler; +type Content = JSX.Element | string | number | false; + +/** + * <Component>: Button (simple clickable element). + */ +export type ButtonProps = { + children?: Content; + label?: string | (() => string); + enabled?: boolean; + active?: boolean; + block?: boolean; + tooltip?: string; + + isOver?: boolean; // force the button into an "is-over" state. + isDown?: boolean; // force the button into an "is-down" state. + + theme?: t.CommonTheme; + style?: t.CssInput; + margin?: t.CssEdgesInput; + padding?: t.CssEdgesInput; + minWidth?: number; + maxWidth?: number; + disabledOpacity?: number; + userSelect?: boolean; + pressedOffset?: [number, number]; + + /** Subscribe to signals that cause the button to redraw. */ + subscribe?: () => void; + + onClick?: MouseHandler; + onMouseDown?: MouseHandler; + onMouseUp?: MouseHandler; + onMouseEnter?: MouseHandler; + onMouseLeave?: MouseHandler; + onDoubleClick?: MouseHandler; + onMouse?: t.ButtonMouseHandler; +}; + +/** + * Events + */ +export type ButtonMouseHandler = (e: ButtonMouseHandlerArgs) => void; +export type ButtonMouseHandlerArgs = { + isDown: boolean; + isOver: boolean; + isEnabled: boolean; + action: 'MouseEnter' | 'MouseLeave' | 'MouseDown' | 'MouseUp'; + event: React.MouseEvent; +}; + +export type ButtonCopyHandler = (e: ButtonCopyHandlerArgs) => void; +export type ButtonCopyHandlerArgs = { + message(value: Content): ButtonCopyHandlerArgs; + fontSize(value: number): ButtonCopyHandlerArgs; + opacity(value: number): ButtonCopyHandlerArgs; + delay(value: t.Msecs): ButtonCopyHandlerArgs; + copy(value?: string | number): Promise<void>; +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Button/u.events.ts b/code/sys.ui/ui-react-components/src/ui/Button/u.events.ts new file mode 100644 index 0000000000..0ffca8aa44 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Button/u.events.ts @@ -0,0 +1,155 @@ +import type React from 'react'; +import { type t, DEFAULTS as D } from './common.ts'; + +type P = t.ButtonProps; +type S = [boolean, (next: boolean) => void]; + +/** + * An object containing the current state of the button + * used to pass into handlers. + */ +export type EventState = { + readonly props: P; + over: boolean; + down: boolean; +}; + +/** + * EventState Factory: + */ +export function toEventState(props: P, over: S, down: S): EventState { + return { + props, + get over() { + return over[0]; + }, + set over(value) { + over[1](value); + }, + get down() { + return down[0]; + }, + set down(value) { + down[1](value); + }, + }; +} + +/** + * Events for desktop devices. + */ +const Desktop = { + over(state: EventState, isOver: boolean): React.MouseEventHandler { + return (e) => { + const props = wrangle.props(state); + if (!props.active) return; + state.over = isOver; + if (!isOver && state.down) state.down = false; + if (props.enabled) { + if (isOver) props.onMouseEnter?.(e); + if (!isOver) props.onMouseLeave?.(e); + } + + const isDown = !isOver ? false : state.down; + const isEnabled = props.enabled; + const action = isOver ? 'MouseEnter' : 'MouseLeave'; + props.onMouse?.({ event: e, isOver, isDown, isEnabled, action }); + }; + }, + + down(state: EventState, isDown: boolean): React.MouseEventHandler { + return (e) => { + const props = wrangle.props(state); + if (!props.active) return; + + state.down = isDown; + if (props.enabled) { + if (isDown) props.onMouseDown?.(e); + if (!isDown) props.onMouseUp?.(e); + if (!isDown) props.onClick?.(e); + } + + const isOver = state.over; + const isEnabled = props.enabled; + const action = isDown ? 'MouseDown' : 'MouseUp'; + props.onMouse?.({ event: e, isOver, isDown, isEnabled, action }); + }; + }, +} as const; + +/** + * Events for mobile devices. + */ +const Mobile = { + down(state: EventState, isDown: boolean): React.TouchEventHandler { + return (e: React.TouchEvent) => { + const props = wrangle.props(state); + if (!props.active) return; + + state.down = isDown; + const event = wrangle.asMouseEvent(e); + if (props.enabled) { + if (isDown && props.onMouseDown) props.onMouseDown(event); + if (!isDown && props.onMouseUp) props.onMouseUp(event); + if (!isDown && props.onClick) props.onClick(event); + } + + const isEnabled = props.enabled; + const action = isDown ? 'MouseDown' : 'MouseUp'; + props.onMouse?.({ event, isOver: isDown, isDown, isEnabled, action }); + }; + }, +} as const; + +/** + * Event tools: + */ +export const Event = { + Desktop, + Mobile, + handlers(state: EventState, isMobile: boolean) { + return isMobile + ? { + onTouchStart: Event.Mobile.down(state, true), + onTouchEnd: Event.Mobile.down(state, false), + onTouchCancel: Event.Mobile.down(state, false), + } + : { + onMouseEnter: Event.Desktop.over(state, true), + onMouseLeave: Event.Desktop.over(state, false), + onMouseDown: Event.Desktop.down(state, true), + onMouseUp: Event.Desktop.down(state, false), + onClick(e: React.MouseEvent) { + // NB: "onClick" is sythetically derived from mouse-down/up events + // so prevent the native onClick event from confusingly propgating + // a different version of "onClick" up in the DOM hierarchy. + e.stopPropagation(); + }, + }; + }, +} as const; + +/** + * Helpers + */ +const wrangle = { + props(state: EventState) { + const { enabled = D.enabled, active = D.active } = state.props; + return { ...state.props, enabled, active }; + }, + + asMouseEvent(e: React.TouchEvent): React.MouseEvent { + const touch = e.changedTouches[0]; + return { + ...e, + button: 0, + buttons: 1, + clientX: touch.clientX, + clientY: touch.clientY, + pageX: touch.pageX, + pageY: touch.pageY, + screenX: touch.screenX, + screenY: touch.screenY, + } as unknown as React.MouseEvent; + }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Button/ui.tsx b/code/sys.ui/ui-react-components/src/ui/Button/ui.tsx new file mode 100644 index 0000000000..2cb8f59be0 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Button/ui.tsx @@ -0,0 +1,119 @@ +import React, { useEffect, useState } from 'react'; +import { type t, Color, css, DEFAULTS, Style, useIsTouchSupported, Signal } from './common.ts'; +import { Event, toEventState } from './u.events.ts'; + +type P = t.ButtonProps; + +/** + * Component + */ +export const Button: React.FC<P> = (props) => { + const { + enabled = DEFAULTS.enabled, + active = DEFAULTS.active, + block = DEFAULTS.block, + disabledOpacity = DEFAULTS.disabledOpacity, + userSelect = DEFAULTS.userSelect, + pressedOffset = DEFAULTS.pressedOffset, + } = props; + const isBlurred = false; + const isEnabled = enabled; + const label = wrangle.label(props); + + const over = useState(false); + const down = useState(false); + const [isOver, setOver] = over; + const [isDown, setDown] = down; + + const isMobile = useIsTouchSupported(); + const eventState = toEventState(props, over, down); + + /** + * Effects: + */ + Signal.useRedrawEffect(() => props.subscribe?.()); + useEffect(() => { + if (!active) { + setDown(false); + setOver(false); + } + }, [active]); + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ + ...Style.toMargins(props.margin), + ...Style.toPadding(props.padding), + position: 'relative', + display: block ? 'block' : 'inline-block', + minWidth: props.minWidth, + maxWidth: props.maxWidth, + opacity: enabled ? 1 : disabledOpacity, + cursor: enabled && active && !isBlurred ? 'pointer' : 'default', + userSelect: userSelect ? 'auto' : 'none', + }), + body: css({ + color: wrangle.color({ + isEnabled, + isOver: !!(isOver || props.isOver), + theme: theme.name, + }), + transform: wrangle.pressedOffset({ + isEnabled, + isOver: !!(isOver || props.isOver), + isDown: !!(isDown || props.isDown), + pressedOffset, + }), + opacity: isBlurred ? 0.15 : 1, + filter: `blur(${isBlurred ? 3 : 0}px) grayscale(${isBlurred ? 100 : 0}%)`, + transition: 'opacity 0.1s ease', + }), + }; + + return ( + <div + role={'button'} + title={props.tooltip} + className={css(styles.base, props.style).class} + onDoubleClick={props.onDoubleClick} + {...Event.handlers(eventState, isMobile)} + > + <div className={styles.body.class}> + {label && <div>{label}</div>} + {props.children} + </div> + </div> + ); +}; + +/** + * Helpers + */ +const wrangle = { + color(args: { isEnabled: boolean; isOver?: boolean; theme?: t.CommonTheme }) { + const color = args.theme === 'Dark' ? Color.WHITE : Color.DARK; + if (!args.isEnabled) return color; + return args.isOver ? Color.BLUE : color; + }, + + pressedOffset(args: { + isEnabled: boolean; + isOver: boolean; + isDown: boolean; + pressedOffset: [number, number]; + }) { + const { isEnabled, isOver, isDown, pressedOffset } = args; + if (!isEnabled) return undefined; + if (!isOver) return undefined; + if (!isDown) return undefined; + return `translateX(${pressedOffset[0]}px) translateY(${pressedOffset[1]}px)`; + }, + + label(props: P) { + const { label } = props; + return typeof label === 'function' ? label() : label; + }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Cropmarks/-SPEC.Debug.tsx b/code/sys.ui/ui-react-components/src/ui/Cropmarks/-SPEC.Debug.tsx new file mode 100644 index 0000000000..5f66070ab9 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Cropmarks/-SPEC.Debug.tsx @@ -0,0 +1,168 @@ +import React from 'react'; +import { Button } from '../Button/mod.ts'; +import { type t, Color, css, Signal } from './common.ts'; + +import { type DebugSignals } from './-SPEC.signals.tsx'; + +/** + * Component: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => { + p.theme.value; + p.size.value; + p.subjectOnly.value; + }); + + /** + * Render: + */ + const theme = Color.theme(p.theme.value); + const styles = { + base: css({ color: theme.fg }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <Button + block + label={`theme: ${p.theme}`} + onClick={() => Signal.cycle<t.CommonTheme>(p.theme, ['Light', 'Dark'])} + /> + <Button + block + label={`subjectOnly: ${p.subjectOnly}`} + onClick={() => Signal.toggle(p.subjectOnly)} + /> + + <hr /> + + <Button block label={`set size: <undefined>`} onClick={() => (p.size.value = undefined)} /> + <Button + block + label={`set size: "fill"`} + onClick={() => { + p.size.value = { mode: 'fill', x: true, y: true, margin: [40, 40, 40, 40] }; + }} + /> + <Button + block + label={`set size: "center"`} + onClick={() => { + p.size.value = { mode: 'center' }; + }} + /> + + <hr /> + + {p.size.value?.mode === 'center' && <DebugCenter debug={debug} />} + {p.size.value?.mode === 'fill' && <DebugFill debug={debug} />} + </div> + ); +}; + +/** + * Component: + */ +export type DebugFillProps = { debug: DebugSignals; style?: t.CssInput }; +export const DebugFill: React.FC<DebugFillProps> = (props) => { + const { debug } = props; + const p = debug.props; + const size = p.size.value; + if (size?.mode !== 'fill') return null; + + /** + * Render: + */ + const styles = { + base: css({}), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <Button + block + label={`size.margin: ${size.margin ?? '<undefined>'}`} + onClick={() => { + const current = size.margin[0]; + let next = [40, 40, 40, 40]; + if (current === 40) next = [80, 60, 30, 10]; + if (current === 80) next = [100, 100, 100, 100]; + if (current === 100) next = [40, 40, 40, 40]; + p.size.value = { ...size, margin: next as t.CssMarginArray }; + }} + /> + + <Button + block + label={`size.x: ${size.x}`} + onClick={() => (p.size.value = { ...size, x: !size.x })} + /> + + <Button + block + label={`size.y: ${size.y}`} + onClick={() => (p.size.value = { ...size, y: !size.y })} + /> + </div> + ); +}; + +/** + * Component: + */ +export type DebugCenterProps = { + debug: DebugSignals; + style?: t.CssInput; +}; +export const DebugCenter: React.FC<DebugCenterProps> = (props) => { + const { debug } = props; + const p = debug.props; + + const size = p.size.value; + if (size?.mode !== 'center') return null; + + /** + * Render: + */ + const styles = { + base: css({}), + }; + + const sizes = [0, 300, 500]; + + return ( + <div className={css(styles.base, props.style).class}> + <Button + block + label={`size.width: ${size.width}`} + onClick={() => { + const next = cycleNumber(size.width ?? 0, sizes); + p.size.value = { ...size, width: next }; + }} + /> + + <Button + block + label={`size.height: ${size.height}`} + onClick={() => { + const next = cycleNumber(size.height ?? 0, sizes); + p.size.value = { ...size, height: next }; + }} + /> + </div> + ); +}; + +/** + * Helpers + */ +export function cycleNumber(current: number, values: number[]): number { + const index = values.indexOf(current); + const nextIndex = index >= 0 ? (index + 1) % values.length : 0; + return values[nextIndex]; +} diff --git a/code/sys.ui/ui-react-components/src/ui/Cropmarks/-SPEC.signals.tsx b/code/sys.ui/ui-react-components/src/ui/Cropmarks/-SPEC.signals.tsx new file mode 100644 index 0000000000..7b76de02f6 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Cropmarks/-SPEC.signals.tsx @@ -0,0 +1,30 @@ +import { type t, Signal } from './common.ts'; + +/** + * Types: + */ +export type DebugSignals = ReturnType<typeof createDebugSignals>; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + type P = t.CropmarksProps; + const s = Signal.create; + const props = { + theme: s<P['theme']>('Light'), + size: s<P['size']>(), + subjectOnly: s<P['subjectOnly']>(false), + }; + const api = { + props, + listen() { + const p = props; + p.theme.value; + p.size.value; + p.subjectOnly.value; + }, + }; + init?.(api); + return api; +} diff --git a/code/sys.ui/ui-react-components/src/ui/Cropmarks/-SPEC.tsx b/code/sys.ui/ui-react-components/src/ui/Cropmarks/-SPEC.tsx new file mode 100644 index 0000000000..d2b5a4f690 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Cropmarks/-SPEC.tsx @@ -0,0 +1,41 @@ +import { Dev, Signal, Spec, css } from '../-test.ui.ts'; +import { Debug } from './-SPEC.Debug.tsx'; +import { createDebugSignals } from './-SPEC.signals.tsx'; +import { Cropmarks } from './mod.ts'; + +export default Spec.describe('Cropmarks', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + Dev.Theme.signalEffect(ctx, p.theme, 1); + Signal.effect(() => { + debug.listen(); + ctx.redraw(); + }); + + const styles = { + subject: css({ + backgroundColor: 'rgba(255, 0, 0, 0.06)' /* RED */, + overflow: 'hidden', + padding: 8, + }), + }; + + ctx.subject + .size('fill') + .display('grid') + .render((e) => ( + <Cropmarks theme={p.theme.value} size={p.size.value} subjectOnly={p.subjectOnly.value}> + <div className={styles.subject.class}>{'š· Hello Cropmarks'}</div> + </Cropmarks> + )); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/Cropmarks/common.ts b/code/sys.ui/ui-react-components/src/ui/Cropmarks/common.ts new file mode 100644 index 0000000000..95e7fd1bf8 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Cropmarks/common.ts @@ -0,0 +1,7 @@ +import { type t } from '../common.ts'; +export * from '../common.ts'; + +/** + * Constants: + */ +export const DEFAULTS = {} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Cropmarks/mod.ts b/code/sys.ui/ui-react-components/src/ui/Cropmarks/mod.ts new file mode 100644 index 0000000000..3f88932779 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Cropmarks/mod.ts @@ -0,0 +1,4 @@ +/** + * @module + */ +export { Cropmarks } from './ui.tsx'; diff --git a/code/sys.ui/ui-react-components/src/ui/Cropmarks/t.ts b/code/sys.ui/ui-react-components/src/ui/Cropmarks/t.ts new file mode 100644 index 0000000000..f5526664ad --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Cropmarks/t.ts @@ -0,0 +1,38 @@ +import type { t } from './common.ts'; + +/** + * <Component>: + */ +export type CropmarksProps = { + children?: t.ReactNode; + subjectOnly?: boolean; + size?: t.CropmarksSize; + borderWidth?: number; + borderOpacity?: number; + theme?: t.CommonTheme; + style?: t.CssInput; +}; + +/** + * The size configuration of the <Cropmarks>. + */ +export type CropmarksSize = CropmarksSizeCenter | CropmarksSizeFill; +/** The display-mode flag for <Cropmarks> size. */ +export type CropmarksSizeMode = CropmarksSize['mode']; + +/** The subject is centered within the <Cropmaks> host. */ +export type CropmarksSizeCenter = { + mode: 'center'; + width?: number; + height?: number; +}; +/** The subject fills the <Cropmaks> host. */ +export type CropmarksSizeFill = { + mode: 'fill'; + /** Pixel margin around the */ + margin: t.CssMarginArray; + /** Fills the X (horizontal) plane. */ + x: boolean; + /** Fills the Y (vertical) plane. */ + y: boolean; +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Cropmarks/u.ts b/code/sys.ui/ui-react-components/src/ui/Cropmarks/u.ts new file mode 100644 index 0000000000..9da365e112 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Cropmarks/u.ts @@ -0,0 +1,26 @@ +import { type t, Style } from './common.ts'; + +export const Wrangle = { + componentSize(value?: t.CropmarksSize) { + let width: number | undefined; + let height: number | undefined; + + if (!value) return { width, height } as const; + if (value.mode === 'fill') return { width, height } as const; + + width = value.width; + height = value.height; + return { width, height } as const; + }, + + fillMargin(value?: t.CropmarksSize) { + const DEFAULT = 40; + if (!value) return Wrangle.asMargin(DEFAULT); + if (value.mode !== 'fill') return Wrangle.asMargin(DEFAULT); + return value.margin; + }, + + asMargin(value: number): t.CssMarginArray { + return Style.Edges.toArray(value ?? 0); + }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Cropmarks/ui.tsx b/code/sys.ui/ui-react-components/src/ui/Cropmarks/ui.tsx new file mode 100644 index 0000000000..02cf21ab3d --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Cropmarks/ui.tsx @@ -0,0 +1,98 @@ +import React from 'react'; +import { type t, Color, css } from './common.ts'; +import { Wrangle } from './u.ts'; + +type P = t.CropmarksProps; + +export const Cropmarks: React.FC<P> = (props) => { + const { size, subjectOnly = false } = props; + + if (subjectOnly) return props.children; + + const fillMargin = Wrangle.fillMargin(size); + const sizeMode: t.CropmarksSizeMode = size?.mode ?? 'center'; + const is = { + x: size?.mode === 'fill' && size.x && !size.y, + y: size?.mode === 'fill' && !size.x && size.y, + } as const; + + const Grid = { + Fill: { + columns: `[left] ${fillMargin[3]}px [body-x] 1fr [right] ${fillMargin[1]}px`, + rows: `[top] ${fillMargin[0]}px [body-y] 1fr [bottom] ${fillMargin[2]}px`, + }, + Center: { + columns: `[left] 1fr [body-x] auto [right] 1fr`, + rows: `[top] 1fr [body-y] auto [bottom] 1fr`, + }, + } as const; + + /** + * Render: + */ + + const fill = { + gridTemplateColumns: is.y ? Grid.Center.columns : Grid.Fill.columns, + gridTemplateRows: is.x ? Grid.Center.rows : Grid.Fill.rows, + }; + const center = { + gridTemplateColumns: Grid.Center.columns, + gridTemplateRows: Grid.Center.rows, + }; + const grid = { + center: sizeMode === 'center' && center, + fill: sizeMode === 'fill' && fill, + }; + + const border = wrangle.border(props); + const borderLeft = border; + const borderRight = border; + const borderTop = border; + const borderBottom = border; + + const styles = { + base: css({ position: 'relative', display: 'grid' }), + block: css({}), + subject: css({ + position: 'relative', + border, + width: size?.mode === 'center' ? size.width : undefined, + height: size?.mode === 'center' ? size.height : undefined, + display: 'grid', + }), + }; + + const className = css( + styles.base, + sizeMode === 'center' ? grid.center : undefined, + sizeMode === 'fill' ? grid.fill : undefined, + props.style, + ).class; + + return ( + <div className={className}> + <div className={styles.block.class} /> + <div className={css(styles.block, { borderLeft, borderRight }).class} /> + <div className={styles.block.class} /> + <div className={css(styles.block, { borderTop, borderBottom }).class} /> + <div className={styles.subject.class}>{props.children}</div> + <div className={css(styles.block, { borderTop, borderBottom }).class} /> + <div className={styles.block.class} /> + <div className={css(styles.block, { borderLeft, borderRight }).class} /> + <div className={styles.block.class} /> + </div> + ); +}; + +/** + * Helpers + */ +const wrangle = { + border(props: P) { + const theme = Color.theme(props.theme); + const { borderWidth = 1, borderOpacity } = props; + const opacity = typeof borderOpacity === 'number' ? borderOpacity : theme.is.dark ? 0.1 : 0.07; + if (borderWidth <= 0 || opacity <= 0) return; + return `solid ${borderWidth}px ${Color.alpha(theme.fg, opacity)}`; + }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Foo.tsx b/code/sys.ui/ui-react-components/src/ui/Foo.tsx deleted file mode 100644 index 4034e7059f..0000000000 --- a/code/sys.ui/ui-react-components/src/ui/Foo.tsx +++ /dev/null @@ -1,18 +0,0 @@ -// @ts-types="@types/react" -import React from 'react'; -import { pkg } from '../pkg.ts'; - -/** - * Sample component properties. - */ -export type FooProps = { enabled?: boolean }; - -/** - * Sample component. - */ -export const Foo: React.FC<FooProps> = (props) => { - const { enabled = true } = props; - let text = `${pkg.name}@${pkg.version}/ui:<Foo>`; - if (!enabled) text += ' (disabled)'; - return <code>{text}</code>; -}; diff --git a/code/sys.ui/ui-react-components/src/ui/IFrame/-SPEC.Debug.tsx b/code/sys.ui/ui-react-components/src/ui/IFrame/-SPEC.Debug.tsx new file mode 100644 index 0000000000..380d0052e3 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/IFrame/-SPEC.Debug.tsx @@ -0,0 +1,121 @@ +import React from 'react'; +import { Button } from '../Button/mod.ts'; +import { type t, Color, css, Signal, Str } from './common.ts'; + +const local = new URL(`${location.origin}${location.pathname}`); + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; +type P = DebugProps; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + const s = Signal.create; + const props = { + showBackground: s<boolean>(true), + + theme: s<t.CommonTheme>('Light'), + src: s<t.IFrameProps['src']>(local.href), + allowFullScreen: s<t.IFrameProps['allowFullScreen']>(), + loading: s<t.IFrameProps['loading']>(), + sandbox: s<t.IFrameSandbox[] | undefined>(), + }; + const api = { + props, + listen() { + const p = props; + p.theme.value; + p.src.value; + p.showBackground.value; + p.allowFullScreen.value; + p.loading.value; + p.sandbox.value; + }, + }; + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<P> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => { + debug.listen(); + }); + + /** + * Render: + */ + const theme = Color.theme(p.theme.value); + const styles = { + base: css({ color: theme.fg }), + title: css({ fontWeight: 'bold' }), + }; + + const srcLoader = (label: string, src?: t.IFrameProps['src']) => { + return <Button block label={label} onClick={() => (p.src.value = src)} />; + }; + + return ( + <div className={css(styles.base, props.style).class}> + <Button + label={`theme: ${p.theme}`} + onClick={() => Signal.cycle<t.CommonTheme>(p.theme, ['Light', 'Dark'])} + /> + <Button + block + label={`show background image: ${p.showBackground}`} + onClick={() => Signal.toggle(p.showBackground)} + /> + + <hr /> + + <Button + block + label={`allowFullScreen: ${p.allowFullScreen}`} + onClick={() => Signal.toggle(p.allowFullScreen)} + /> + + <Button + block + label={`loading: ${p.loading}`} + onClick={() => { + Signal.cycle<t.IFrameProps['loading']>(p.loading, ['eager', 'lazy', undefined]); + }} + /> + + <Button + block + label={`sandbox: ${p.sandbox}`} + onClick={() => { + Signal.cycle<t.IFrameSandbox[] | undefined>(p.sandbox, [ + undefined, + ['allow-popups', 'allow-same-origin'], + ['allow-pointer-lock'], + ]); + }} + /> + + <hr /> + + <div className={styles.title.class}>{`src: ${Str.truncate(String(p.src.value), 30)}`}</div> + {srcLoader(local.host, local.href)} + {srcLoader('Wikipedia: "W3C"', `https://en.wikipedia.org/wiki/World_Wide_Web_Consortium`)} + {srcLoader('Wikipedia: "Foobar" mobile format', `https://en.m.wikipedia.org/wiki/Foobar`)} + <hr /> + {srcLoader('error: google.com (ā blocked)', 'https://google.com')} + <hr /> + {srcLoader('{ html }', { html: '<h1>Hello š<h1>' })} + {srcLoader('<undefined> ā unload', undefined)} + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/ui/IFrame/-SPEC.tsx b/code/sys.ui/ui-react-components/src/ui/IFrame/-SPEC.tsx new file mode 100644 index 0000000000..e2f7512f49 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/IFrame/-SPEC.tsx @@ -0,0 +1,60 @@ +import { Dev, Spec } from '../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { Signal } from './common.ts'; +import { IFrame } from './mod.ts'; + +/** + * Sample Data. + */ +const SAMPLE = { + src: 'https://en.wikipedia.org/wiki/World_Wide_Web_Consortium', + allow: `camera; microphone`, +} as const; + +export default Spec.describe('IFrame', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + Dev.Theme.signalEffect(ctx, p.theme); + + Signal.effect(() => { + debug.listen(); + ctx.redraw(); + }); + + /** + * Effect: Host Backbground Image + */ + Signal.effect(() => { + const showBackground = p.showBackground.value; + const IMG_HREF = `https://images.unsplash.com/photo-1558591710-4b4a1ae0f04d?ixlib=rb-4.0.3&ixid=MnwxMjA3fDB8MHxwaG90by1wYWdlfHx8fGVufDB8fHx8&auto=format&fit=crop&w=1287&q=80`; + const url = showBackground ? IMG_HREF : ''; + ctx.host.backgroundImage({ url, opacity: 0.3 }); + }); + + /** + * Render: + */ + ctx.subject + .size('fill') + .display('grid') + .backgroundColor('') + .render((e) => { + return ( + <IFrame + src={p.src.value} + allow={SAMPLE.allow} + onReady={(e) => console.info('ā”ļø onReady:', e)} + onLoad={(e) => console.info('ā”ļø onLoad:', e)} + /> + ); + }); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/IFrame/common.ts b/code/sys.ui/ui-react-components/src/ui/IFrame/common.ts new file mode 100644 index 0000000000..eddccde9dc --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/IFrame/common.ts @@ -0,0 +1,12 @@ +import { type t } from './common.ts'; +export * from '../common.ts'; + +/** + * Constants: + */ +export const DEFAULTS = { + sandbox: true, + get loading(): t.IFrameLoading { + return 'eager'; + }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/IFrame/mod.ts b/code/sys.ui/ui-react-components/src/ui/IFrame/mod.ts new file mode 100644 index 0000000000..2c4935499b --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/IFrame/mod.ts @@ -0,0 +1,4 @@ +/** + * @module + */ +export { IFrame } from './ui.tsx'; diff --git a/code/sys.ui/ui-react-components/src/ui/IFrame/t.ts b/code/sys.ui/ui-react-components/src/ui/IFrame/t.ts new file mode 100644 index 0000000000..0b2a0f4993 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/IFrame/t.ts @@ -0,0 +1,56 @@ +import type { t } from './common.ts'; +import type { HTMLAttributeReferrerPolicy } from 'react'; + +type HttpPermissionsPolicy = string; // https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Permissions-Policy +type Ref = React.RefObject<HTMLIFrameElement>; + +/** + * Component + */ +export type IFrameProps = { + src?: t.StringUrl | IFrameSrc; + width?: string | number; + height?: string | number; + title?: string; + name?: string; + sandbox?: true | t.IFrameSandbox[]; + allow?: HttpPermissionsPolicy; + allowFullScreen?: boolean; + referrerPolicy?: HTMLAttributeReferrerPolicy | undefined; + loading?: t.IFrameLoading; + style?: t.CssInput; + onReady?: IFrameReadyHandler; + onLoad?: IFrameLoadedEventHandler; +}; + +/** + * Applies extra restrictions to the content in the frame. + * https://developer.mozilla.org/en-US/docs/Web/HTML/Element/iframe#attr-sandbox + */ +export type IFrameSandbox = + | 'allow-downloads-without-user-activation' + | 'allow-downloads' + | 'allow-forms' + | 'allow-modals' + | 'allow-orientation-lock' + | 'allow-pointer-lock' + | 'allow-popups' + | 'allow-popups-to-escape-sandbox' + | 'allow-presentation' + | 'allow-same-origin' + | 'allow-scripts' + | 'allow-storage-access-by-user-activation' + | 'allow-top-navigation' + | 'allow-top-navigation-by-user-activation'; + +export type IFrameLoading = 'eager' | 'lazy'; +export type IFrameSrc = { html?: string; url?: t.StringUrl }; + +/** + * Events + */ +export type IFrameReadyHandler = (e: IFrameReadyHandlerArgs) => void; +export type IFrameReadyHandlerArgs = { ref: Ref }; + +export type IFrameLoadedEventHandler = (e: IFrameLoadedEventHandlerArgs) => void; +export type IFrameLoadedEventHandlerArgs = { href: t.StringUrl; ref: Ref }; diff --git a/code/sys.ui/ui-react-components/src/ui/IFrame/ui.tsx b/code/sys.ui/ui-react-components/src/ui/IFrame/ui.tsx new file mode 100644 index 0000000000..5c0fc92c61 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/IFrame/ui.tsx @@ -0,0 +1,78 @@ +import React, { useEffect, useRef } from 'react'; +import { type t, css, DEFAULTS } from './common.ts'; + +export const IFrame: React.FC<t.IFrameProps> = (props) => { + const { width, height, loading = 'eager' } = props; + const content = wrangle.content(props); + const ref = useRef<HTMLIFrameElement>(null); + + /** + * Handlers: + */ + const handleLoad = () => { + let href = content.src ?? ''; + try { + href = ref.current?.contentWindow?.location.href ?? href; + } catch (error) { + // [Ignore]: This will be a cross-origin block. + // Fire the best guess at what the URL is. + } + props.onLoad?.({ ref, href }); + }; + + /** + * Lifecycle: + */ + useEffect(() => props.onReady?.({ ref }), []); + + /** + * Render: + */ + const styles = { + base: css({ position: 'relative', width, height }), + iframe: css({ + Absolute: 0, + width: width ?? '100%', + height: height ?? '100%', + border: 'none', + background: 'transparent', + }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + {props.src && ( + <iframe + className={styles.iframe.class} + ref={ref} + src={content.src} + srcDoc={content.html} + title={props.title} + name={props.name} + allow={props.allow} + allowFullScreen={props.allowFullScreen} + referrerPolicy={props.referrerPolicy} + loading={loading} + sandbox={wrangle.sandbox(props)} + onLoad={handleLoad} + /> + )} + </div> + ); +}; + +/** + * Helpers + */ +const wrangle = { + sandbox(props: t.IFrameProps) { + const { sandbox = DEFAULTS.sandbox } = props; + return Array.isArray(sandbox) ? sandbox.join(' ') : undefined; // NB: <undefined> === all restrictions applied. + }, + + content(props: t.IFrameProps): { src?: string; html?: string } { + if (!props.src) return { src: undefined, html: undefined }; + if (typeof props.src === 'string') return { src: props.src }; + return { src: props.src.url, html: props.src.html }; + }, +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Icon/-SPEC.Debug.tsx b/code/sys.ui/ui-react-components/src/ui/Icon/-SPEC.Debug.tsx new file mode 100644 index 0000000000..fe40c17bf8 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Icon/-SPEC.Debug.tsx @@ -0,0 +1,29 @@ +import React, { useEffect, useRef, useState } from 'react'; +import { type t, Color, css, Signal, rx } from './common.ts'; + +export type DebugProps = { ctx: {}; theme?: t.CommonTheme; style?: t.CssValue }; +type P = DebugProps; + +/** + * Component + */ +export const Debug: React.FC<P> = (props) => { + const {} = props; + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ + backgroundColor: 'rgba(255, 0, 0, 0.1)' /* RED */, + color: theme.fg, + }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div>{`š· Debug`}</div> + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Icon/-SPEC.sample.ts b/code/sys.ui/ui-react-components/src/ui/Icon/-SPEC.sample.ts new file mode 100644 index 0000000000..edad5f2325 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Icon/-SPEC.sample.ts @@ -0,0 +1,13 @@ +import { MdFace } from 'react-icons/md'; +import { VscSymbolClass } from 'react-icons/vsc'; +import { Icon } from './mod.ts'; + +const icon = Icon.renderer; + +/** + * Icon collection: + */ +export const Icons = { + Face: icon(MdFace), + Object: icon(VscSymbolClass), +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Icon/-SPEC.tsx b/code/sys.ui/ui-react-components/src/ui/Icon/-SPEC.tsx new file mode 100644 index 0000000000..800607e5d5 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Icon/-SPEC.tsx @@ -0,0 +1,122 @@ +import { css, Spec, type t } from '../-test.ui.ts'; +import { Debug } from './-SPEC.Debug.tsx'; +import { Icons } from './-SPEC.sample.ts'; + +type T = { + props: t.IconProps; + debug: { inheritColor: boolean }; +}; +// const initial: T = { +// props: { tooltip: 'Hello' }, +// debug: { inheritColor: true }, +// }; + +export default Spec.describe('Icon', (e) => { + e.it('init', async (e) => { + const ctx = Spec.ctx(e); + // await ctx.state<T>(initial); + + ctx.subject + .backgroundColor(1) + .display('grid') + .render<T>((e) => { + // const debug = e.state.debug; + const styles = { + base: css({ + // color: debug.inheritColor ? COLORS.MAGENTA : undefined, // NB: button should inherit this color. + }), + }; + return ( + <div className={styles.base.class}> + <Icons.Face + {...e.state.props} + size={150} + onClick={(e) => console.info(`ā”ļø onClick:`, e)} + /> + </div> + ); + }); + }); + + e.it('ui:debug', async (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug ctx={{}} />); + + /** + * (prior platform-0.0.2) + */ + + // const dev = Dev.tools<T>(e, initial); + // dev.section('Properties', (dev) => { + // dev.boolean((btn) => { + // const value = (state: T) => Boolean(state.props.flipX); + // btn + // .label((e) => `flipX`) + // .value((e) => value(e.state)) + // .onClick((e) => e.change((d) => Dev.toggle(d.props, 'flipX'))); + // }); + // + // dev.boolean((btn) => { + // const value = (state: T) => Boolean(state.props.flipY); + // btn + // .label((e) => `flipY`) + // .value((e) => value(e.state)) + // .onClick((e) => e.change((d) => Dev.toggle(d.props, 'flipY'))); + // }); + // + // dev.boolean((btn) => + // btn + // .label((e) => { + // const offset = e.state.props.offset; + // return `offset: ${offset ? `[${offset.toString()}]` : '<none>'}`; + // }) + // .value((e) => Boolean(e.state.props.offset)) + // .onClick((e) => { + // e.change((d) => { + // const exists = Boolean(d.props.offset); + // d.props.offset = exists ? undefined : [35, -40]; + // }); + // }), + // ); + // }); + // + // dev.hr(5, 20); + // + // dev.section('Debug', (dev) => { + // dev.boolean((btn) => + // btn + // .label((e) => `inherit color: ${e.state.debug.inheritColor ? 'magenta' : '(none)'}`) + // .value((e) => e.state.debug.inheritColor) + // .onClick((e) => e.change((d) => Dev.toggle(d.debug, 'inheritColor'))), + // ); + // }); + // + // dev.hr(5, 20); + // + // dev.section('Debug', (dev) => { + // const color = (name: string, value?: string) => { + // dev.button(name, (e) => e.change((d) => (d.props.color = value))); + // }; + // color('black (dark)', COLORS.DARK); + // color('cyan', COLORS.CYAN); + // color('blue', COLORS.BLUE); + // color('red', COLORS.RED); + // dev.hr(-1, 5); + // color('<undefined> - inherit'); + // }); + // + // dev.hr(); + // + // dev.section('Tooltip', (dev) => { + // dev.button('"Hello"', (e) => e.change((d) => (d.props.tooltip = 'Hello'))); + // dev.button('none', (e) => e.change((d) => (d.props.tooltip = undefined))); + // }); + }); + + e.it('ui:footer', async (e) => { + // const dev = Dev.tools<T>(e, initial); + // dev.footer + // .border(-0.1) + // .render<T>((e) => <Dev.Object name={'spec'} data={e.state} expand={1} />); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/Icon/common.ts b/code/sys.ui/ui-react-components/src/ui/Icon/common.ts new file mode 100644 index 0000000000..8cae67176a --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Icon/common.ts @@ -0,0 +1 @@ +export * from '../common.ts'; diff --git a/code/sys.ui/ui-react-components/src/ui/Icon/m.Icon.tsx b/code/sys.ui/ui-react-components/src/ui/Icon/m.Icon.tsx new file mode 100644 index 0000000000..b8bffbb59f --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Icon/m.Icon.tsx @@ -0,0 +1,10 @@ +import React from 'react'; +import type { t } from './common.ts'; +import { IconView } from './ui.tsx'; + +/** + * Tools for rendering icons. + */ +export const Icon: t.IconLib = { + renderer: (type) => (props) => <IconView type={type} {...props} />, +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Icon/mod.ts b/code/sys.ui/ui-react-components/src/ui/Icon/mod.ts new file mode 100644 index 0000000000..c987d7ef55 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Icon/mod.ts @@ -0,0 +1,9 @@ +/** + * @module + * SVC Icons Components + * + * References: + * - https://react-icons.github.io/react-icons + * - https://fonts.google.com/icons + */ +export { Icon } from './m.Icon.tsx'; diff --git a/code/sys.ui/ui-react-components/src/ui/Icon/t.ts b/code/sys.ui/ui-react-components/src/ui/Icon/t.ts new file mode 100644 index 0000000000..cc3df1d600 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Icon/t.ts @@ -0,0 +1,47 @@ +import type React from 'react'; +import type { IconType } from 'react-icons'; +import type { t } from '../common.ts'; + +type MouseHandler = React.MouseEventHandler; + +/** + * Tools for rendering icons. + */ +export type IconLib = { + /** Factory that produces an icon-renderer for the given Icon definition. */ + renderer(type: IconType): t.IconRenderer; +}; + +/** + * An <Icon> component function. + */ +export type IconRenderer = (props: IconProps) => JSX.Element; + +/** + * <Component>: Display properties for an icon. + */ +export type IconProps = { + size?: number; + color?: number | string; + opacity?: t.Percent; + tooltip?: string; + offset?: [number, number]; // x,y | (+/-) pixels. + flipX?: boolean; + flipY?: boolean; + margin?: t.CssMarginInput; + style?: t.CssInput; + onClick?: MouseHandler; + onDoubleClick?: MouseHandler; + onMouseDown?: MouseHandler; + onMouseUp?: MouseHandler; + onMouseEnter?: MouseHandler; + onMouseLeave?: MouseHandler; +}; + +/** + * <Component>: The inner renderer of an icon. + */ +export type IconViewProps = t.IconProps & { + type: IconType; + tabIndex?: number; +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Icon/ui.tsx b/code/sys.ui/ui-react-components/src/ui/Icon/ui.tsx new file mode 100644 index 0000000000..52050b8d68 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Icon/ui.tsx @@ -0,0 +1,57 @@ +import React from 'react'; +import { type t, Color, Style, css } from './common.ts'; + +type P = t.IconViewProps; + +/** + * The renderer of the <SVG> icon. + */ +export const IconView: React.FC<P> = (props) => { + const { size = 24, opacity } = props; + const Component = props.type; + + const styles = { + base: css({ + display: 'inline-block', + overflow: 'hidden', + Size: size, + transform: wrangle.transform(props), + opacity: opacity === undefined ? 1 : opacity, + ...Style.toMargins(props.margin), + }), + }; + + return ( + <div + className={css(styles.base, props.style).class} + tabIndex={props.tabIndex} + title={props.tooltip} + onClick={props.onClick} + onDoubleClick={props.onDoubleClick} + onMouseDown={props.onMouseDown} + onMouseUp={props.onMouseUp} + onMouseEnter={props.onMouseEnter} + onMouseLeave={props.onMouseLeave} + > + <Component size={size} color={wrangle.color(props)} /> + </div> + ); +}; + +/** + * Helpers + */ +const wrangle = { + color(props: P) { + return props.color ? Color.format(props.color) : undefined; + }, + + transform(props: P) { + const { offset, flipX, flipY } = props; + let res = ''; + if (offset) res += ` translate(${offset[0]}px, ${offset[1]}px)`; + if (flipX) res += ` scaleX(-1)`; + if (flipY) res += ` scaleY(-1)`; + return res.trim() || undefined; + }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Icons.ts b/code/sys.ui/ui-react-components/src/ui/Icons.ts new file mode 100644 index 0000000000..fed6462ca4 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Icons.ts @@ -0,0 +1,14 @@ +import { MdFace, MdErrorOutline } from 'react-icons/md'; +import { VscSymbolClass } from 'react-icons/vsc'; +import { Icon } from './Icon/mod.ts'; + +const icon = Icon.renderer; + +/** + * Icon collection: + */ +export const Icons = { + Face: icon(MdFace), + Object: icon(VscSymbolClass), + Error: icon(MdErrorOutline), +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Image.Svg/-.test.ts b/code/sys.ui/ui-react-components/src/ui/Image.Svg/-.test.ts new file mode 100644 index 0000000000..0dbc06b146 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Image.Svg/-.test.ts @@ -0,0 +1,18 @@ +import { describe, expect, it } from '../../-test.ts'; +import { Svg } from './mod.ts'; + +describe('Image.Svg', () => { + it('API', async () => { + const { SvgElement } = await import('./common.ts'); + expect(Svg.Element).to.equal(SvgElement); + }); + + it('import: Sample', async () => { + /** + * NOTE: this will cause an error in Deno, as '.svg' imports are not supported. + * Solutions: + * - Explore using the FileMap. + */ + // const { Sample } = await import('./-SPEC.Sample.tsx'); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/Image.Svg/-SPEC.Debug.tsx b/code/sys.ui/ui-react-components/src/ui/Image.Svg/-SPEC.Debug.tsx new file mode 100644 index 0000000000..afeb36f74e --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Image.Svg/-SPEC.Debug.tsx @@ -0,0 +1,92 @@ +import React from 'react'; +import { Button } from '../Button/mod.ts'; +import { type t, Color, css, Signal } from './common.ts'; + +/** + * Types: + */ +export type DebugImportStyle = 'Static' | 'Function ā Promise'; +export type DebugImage = 'Small' | 'Larger'; +export type DebugProps = { debug: DebugSignals; style?: t.CssValue }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; +type P = DebugProps; + +/** + * Signals: + */ +export function createDebugSignals() { + const s = Signal.create; + const props = { + theme: s<t.CommonTheme>('Light'), + width: s<number | undefined>(350), + color: s<'dark' | 'blue'>('dark'), + importStyle: s<DebugImportStyle>('Function ā Promise'), + image: s<DebugImage>('Small'), + }; + const api = { + props, + listen() { + const p = props; + p.theme.value; + p.width.value; + p.image.value; + p.color.value; + p.image.value; + p.importStyle.value; + }, + }; + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<P> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => debug.listen()); + + /** + * Render: + */ + const theme = Color.theme(p.theme.value); + const styles = { + base: css({ color: theme.fg }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <Button + block + label={`theme: ${p.theme}`} + onClick={() => Signal.cycle(p.theme, ['Light', 'Dark'])} + /> + <hr /> + <Button + block + label={`width: ${p.width.value ?? '<undefined>'}`} + onClick={() => Signal.cycle(p.width, [80, 200, 350, undefined])} + /> + <Button + block + label={`import style: ${p.importStyle}`} + onClick={() => { + Signal.cycle<DebugImportStyle>(p.importStyle, ['Static', 'Function ā Promise']); + }} + /> + + <hr /> + + <Button + block={true} + label={`image (src): ${p.image}`} + onClick={() => { + Signal.cycle<DebugImage>(p.image, ['Small', 'Larger']); + }} + /> + + <hr /> + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Image.Svg/-SPEC.Sample.tsx b/code/sys.ui/ui-react-components/src/ui/Image.Svg/-SPEC.Sample.tsx new file mode 100644 index 0000000000..3fc166cde4 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Image.Svg/-SPEC.Sample.tsx @@ -0,0 +1,117 @@ +import React from 'react'; + +import StaticLarger from '../../-sample/images/sample.larger.svg'; +import StaticSmall from '../../-sample/images/sample.small.svg'; + +import type { DebugImage, DebugImportStyle, DebugSignals } from './-SPEC.Debug.tsx'; +import { type t, css, Signal } from './common.ts'; +import { Svg } from './mod.ts'; + +export type SampleProps = { signals: DebugSignals }; + +/** + * SAMPLE: render an SVG within a React component. + */ +export const Sample: React.FC<SampleProps> = (props) => { + const { signals } = props; + const p = signals.props; + const input = wrangle.importInput(p.importStyle.value, p.image.value); + const width = p.width.value; + const isFill = width === undefined; + + type H = HTMLDivElement; + const viewbox = wrangle.size(p.image.value); + // const svg = Svg.useSvg<H>(input, viewbox); + const svg = Svg.useSvg<H>(input, viewbox, (e) => { + /** + * Optional: configuration. + */ + // e.draw.width(1234) + }); + + console.groupCollapsed(`š³ SVG (hook)`); + console.info(svg); + console.info('input:', input); + console.info(`svg.query('#tick'):`, svg.query('#tick')); + console.info(`svg.queryAll('line'):`, svg.queryAll('line')); + console.groupEnd(); + + /** + * Redraw the component on signal changes. + */ + Signal.useRedrawEffect(() => { + p.width.value; + p.theme.value; + p.color.value; + p.image.value; + p.importStyle.value; + }); + + /** + * Dynamically adjust the SVG on signal changes. + */ + Signal.useEffect(() => { + const width = p.width.value; + const color = p.color.value; + + const draw = svg.draw; + if (!draw) return; + + const tick = draw.findOne('#tick'); + const borderOutline = draw.findOne('#border-outline'); + + if (tick) { + tick.attr({ opacity: color === 'dark' ? 1 : 0.2 }); + } + if (borderOutline) { + borderOutline.attr({ stroke: width === 200 ? '#383057' : '#0000FF' }); + } + }); + + /** + * Render: + */ + const styles = { + base: css({ + position: 'relative', + overflow: 'hidden', + lineHeight: 0, // NB: ensure no "baseline" gap below the <MediaPlayer>. + }), + svg: css({ + Absolute: isFill ? [0, null, null, 0] : undefined, + width: isFill ? '100%' : undefined, + height: isFill ? '100%' : undefined, + }), + }; + + return ( + <div className={styles.base.class}> + <div ref={svg.ref} className={styles.svg.class} /> + </div> + ); +}; + +/** + * Helpers + */ +const wrangle = { + importInput(importStyle: DebugImportStyle, image: DebugImage = 'Small'): t.SvgImportInput { + if (importStyle === 'Static') { + return image === 'Small' ? StaticSmall : StaticLarger; + } + + if (importStyle === 'Function ā Promise') { + return image === 'Small' + ? () => import('../../-sample/images/sample.small.svg') + : () => import('../../-sample/images/sample.larger.svg'); + } + + throw new Error(`Import style "${importStyle}" not supported.`); + }, + + size(image: DebugImage) { + if (image === 'Small') return [1059, 1059]; + if (image === 'Larger') return [233, 98]; + throw new Error(`Image "${image}" not supported.`); + }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Image.Svg/-SPEC.tsx b/code/sys.ui/ui-react-components/src/ui/Image.Svg/-SPEC.tsx new file mode 100644 index 0000000000..6e42a04556 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Image.Svg/-SPEC.tsx @@ -0,0 +1,34 @@ +import { Dev, Signal, Spec } from '../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { Sample } from './-SPEC.Sample.tsx'; + +export default Spec.describe('Image.Svg', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + const updateSize = () => { + const width = p.width.value; + if (width === undefined) ctx.subject.size('fill'); + else ctx.subject.size([width, null]); + }; + + Dev.Theme.signalEffect(ctx, p.theme, 1); + Signal.effect(() => { + debug.listen(); + updateSize(); + ctx.redraw(); + }); + + ctx.subject.display('grid').render((e) => { + return <Sample signals={debug} />; + }); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/Image.Svg/common.ts b/code/sys.ui/ui-react-components/src/ui/Image.Svg/common.ts new file mode 100644 index 0000000000..7a5872e88e --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Image.Svg/common.ts @@ -0,0 +1,2 @@ +export { SVG, Element as SvgElement } from '@svgdotjs/svg.js'; +export * from '../common.ts'; diff --git a/code/sys.ui/ui-react-components/src/ui/Image.Svg/m.Svg.ts b/code/sys.ui/ui-react-components/src/ui/Image.Svg/m.Svg.ts new file mode 100644 index 0000000000..e862b14de5 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Image.Svg/m.Svg.ts @@ -0,0 +1,14 @@ +import { SvgElement as Element } from './common.ts'; +import { useSvg } from './use.Svg.ts'; + +/** + * Helpers for working with SVG objects within the given container element. + * + * NOTE: + * This helper assumes <svg> data assembled via the webpack plugin + * (see [cell.compiler]). + */ +export const Svg = { + useSvg, + Element, +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Image.Svg/mod.ts b/code/sys.ui/ui-react-components/src/ui/Image.Svg/mod.ts new file mode 100644 index 0000000000..9655f8e21e --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Image.Svg/mod.ts @@ -0,0 +1,6 @@ +/** + * @module + * SVG Image Renderer (library) + * See: https://svgjs.dev/ + */ +export { Svg } from './m.Svg.ts'; diff --git a/code/sys.ui/ui-react-components/src/ui/Image.Svg/t.ts b/code/sys.ui/ui-react-components/src/ui/Image.Svg/t.ts new file mode 100644 index 0000000000..79a3d01f93 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Image.Svg/t.ts @@ -0,0 +1,57 @@ +import type { Element as SvgElement } from '@svgdotjs/svg.js'; +import type { t } from './common.ts'; + +export { SvgElement }; + +type NumberWidth = t.Pixels; +type NumberHeight = t.Pixels; + +/** + * SVG rendering tools. + */ +export type SvgLib = { + readonly Element: SvgElement; + readonly useSvg: UseSvgFactory; +}; + +/** + * Hook: SVG image import/renderer. + */ +export type UseSvgFactory = { + <T extends HTMLElement>( + svgImport: SvgImportInput, + viewBox: [NumberWidth, NumberHeight], + init?: UseSvgInit<T> | NumberWidth, + ): SvgInstance<T>; + + <T extends HTMLElement>( + svgImport: SvgImportInput, + viewBox: number[], // NB: type-hack so typescript does not complain when number-arrays (rather than a tuple) is passed. + init?: UseSvgInit<T> | NumberWidth, + ): SvgInstance<T>; +}; + +/** Input parameter for the `useSvg` hook. */ +export type SvgImportInput = string | (() => SvgImportPromise); + +/** An dynamic import of an SVG file: eg, `import('path/to/file.svg')`. */ +export type SvgImportPromise = Promise<{ default: string }>; + +/** Callback to initialize the SVG upon creation. */ +export type UseSvgInit<T extends HTMLElement> = (e: UseSvgInitArgs<T>) => void; +export type UseSvgInitArgs<T extends HTMLElement> = Pick<SvgInstance<T>, 'query' | 'queryAll'> & { + /** Drawing API. */ + readonly draw: SvgElement; +}; + +/** + * An instance of an SVG image with API for manipulating + * the it programatically. + */ +export type SvgInstance<T extends HTMLElement> = { + readonly ready: boolean; + readonly ref: React.RefObject<T>; + readonly draw?: SvgElement; + query(selector: string): SvgElement | undefined; + queryAll(selector: string): SvgElement[]; +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Image.Svg/use.Svg.ts b/code/sys.ui/ui-react-components/src/ui/Image.Svg/use.Svg.ts new file mode 100644 index 0000000000..5f4a176d64 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Image.Svg/use.Svg.ts @@ -0,0 +1,115 @@ +import { useEffect, useRef, useState } from 'react'; +import { type t, rx, SVG, SvgElement } from './common.ts'; + +/** + * Hook: SVG image import/renderer. + */ +export function useSvg<T extends HTMLElement>( + svgImport: string | (() => t.SvgImportPromise), + viewBox: [t.Pixels, t.Pixels] | number[], + init?: t.UseSvgInit<T> | number, +): t.SvgInstance<T> { + type R = t.SvgInstance<T>; + + const ref = useRef<T>(null); + const drawRef = useRef<SvgElement>(); + + const [ready, setReady] = useState(false); + const [importString, setImportString] = useState(''); + const [svgMarkup, setSvgMarkup] = useState(''); + + /** + * Methods: + */ + const query: R['query'] = (selector) => { + if (!ref.current) return undefined; + const el = ref.current.querySelector(selector); + return el ? SVG(el) : undefined; + }; + + const queryAll: R['queryAll'] = (selector) => { + if (!ref.current) return []; + const matches = ref.current.querySelectorAll(selector); + return Array.from(matches).map((el) => SVG(el)); + }; + + /** + * Effect: import SVG data. + */ + useEffect(() => { + const life = rx.lifecycle(); + if (typeof svgImport === 'string') setImportString(svgImport); + if (typeof svgImport === 'function') svgImport().then((m) => setImportString(m.default)); + return life.dispose; + }, [svgImport]); + + /** + * Effect: load/parse the import string into <SVG> markup. + */ + useEffect(() => { + if (!importString) return; + + // Load from embedded data-uri: + if (importString.startsWith('data:image/')) { + const svg = decodeURIComponent(importString.split(',')[1]); + setSvgMarkup(svg); + return; + } + + // Fetch SVG data from server: + const life = rx.lifecycle(); + fetch(importString).then(async (res) => { + if (life.disposed) return; + setSvgMarkup(res.ok ? await res.text() : ''); + }); + return life.dispose; + }, [importString]); + + /** + * Effect: Load the SVG data and inject into DOM. + */ + useEffect(() => { + if (!ref.current) return; + if (!importString) return; + + // Create an SVG canvas inside the container element. + const draw: SvgElement = SVG().addTo(ref.current); + + // NB: Set the viewBox to ensure proper scaling of the inner content. + // These values will be the "viewBox" attribute on the <svg> root tag of the [.svg] file. + const [width, height] = viewBox; + draw.attr({ + viewBox: `0 0 ${width} ${height}`, + width: '100%', + height: '100%', + preserveAspectRatio: 'xMidYMid', + }); + draw.svg(svgMarkup); + + // Initialize. + drawRef.current = draw; + if (typeof init === 'function') init({ draw, query, queryAll }); + if (typeof init === 'number') draw.width(init); + + // Finish up. + setReady(true); + return () => { + draw.clear(); + draw.remove(); + }; + }, [svgMarkup]); + + /** + * API: + */ + const api: R = { + ready, + ref, + get draw() { + return drawRef.current; + }, + query, + queryAll, + }; + return api; +} diff --git a/code/sys.ui/ui-react-components/src/ui/ObjectView/-SPEC.Debug.tsx b/code/sys.ui/ui-react-components/src/ui/ObjectView/-SPEC.Debug.tsx new file mode 100644 index 0000000000..70964150ae --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/ObjectView/-SPEC.Debug.tsx @@ -0,0 +1,123 @@ +import React from 'react'; +import { Button } from '../Button/mod.ts'; +import { type t, css, DEFAULTS as D, Signal } from './common.ts'; + +type P = t.ObjectViewProps; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; + +const DATA = { + msg: 'š', + count: 0, + foo: { + yes: true, + list: [1, 2, 3], + fn: () => 'hello', + }, +}; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + const s = Signal.create; + const props = { + theme: s<P['theme']>('Light'), + fontSize: s<P['fontSize']>(), + name: s<P['name']>('my-name'), + data: s<P['data']>({ ...DATA }), + sortKeys: s<P['sortKeys']>(D.sortKeys), + showNonenumerable: s<t.ObjectViewShow['nonenumerable']>(D.show.nonenumerable), + showRootSummary: s<t.ObjectViewShow['rootSummary']>(D.show.rootSummary), + expandPaths: s<string[] | undefined>(), + }; + const p = props; + const api = { + props, + get show(): t.ObjectViewShow { + const nonenumerable = props.showNonenumerable.value; + const rootSummary = props.showRootSummary.value; + return { nonenumerable, rootSummary }; + }, + listen() { + const p = props; + p.theme.value; + p.fontSize.value; + p.data.value; + p.name.value; + p.showNonenumerable.value; + p.showRootSummary.value; + p.sortKeys.value; + p.expandPaths.value; + }, + }; + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => debug.listen()); + + /** + * Render: + */ + const styles = { + base: css({}), + title: css({ fontWeight: 'bold', marginBottom: 10 }), + cols: css({ display: 'grid', gridTemplateColumns: 'auto 1fr auto' }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={css(styles.title, styles.cols).class}>{'ObjectView'}</div> + + <Button + block + label={`theme: "${p.theme}"`} + onClick={() => Signal.cycle<t.CommonTheme>(p.theme, ['Light', 'Dark'])} + /> + <Button + block + label={`fontSize: ${p.fontSize ?? '<undefined>'}`} + onClick={() => Signal.cycle<P['fontSize']>(p.fontSize, [undefined, 14, 18, 32])} + /> + + <hr /> + <Button block label={`sortKeys: ${p.sortKeys}`} onClick={() => Signal.toggle(p.sortKeys)} /> + + <Button + block + label={`showNonenumerable: ${p.showNonenumerable}`} + onClick={() => Signal.toggle(p.showNonenumerable)} + /> + <Button + block + label={`showRootSummary: ${p.showRootSummary}`} + onClick={() => Signal.toggle(p.showRootSummary)} + /> + + <hr /> + <Button + block + label={`expandPaths: ${p.expandPaths.value || '[ ]'}`} + onClick={() => { + const paths = p.expandPaths.value ?? []; + const next = paths.length === 0 ? ['$', '$.foo'] : undefined; + p.expandPaths.value = next; + }} + /> + + <hr /> + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/ui/ObjectView/-SPEC.tsx b/code/sys.ui/ui-react-components/src/ui/ObjectView/-SPEC.tsx new file mode 100644 index 0000000000..3b7bcf73f1 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/ObjectView/-SPEC.tsx @@ -0,0 +1,41 @@ +import { Dev, Signal, Spec } from '../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { ObjectView } from './mod.ts'; + +export default Spec.describe('Obj', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + Dev.Theme.signalEffect(ctx, p.theme, 1); + Signal.effect(() => { + debug.listen(); + ctx.redraw(); + }); + + ctx.subject + .size() + .display('grid') + .render((e) => { + const paths = p.expandPaths.value; + return ( + <ObjectView + theme={p.theme.value} + fontSize={p.fontSize.value} + name={p.name.value} + data={p.data.value} + sortKeys={p.sortKeys.value} + show={debug.show} + expand={{ paths }} + /> + ); + }); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/ObjectView/common.ts b/code/sys.ui/ui-react-components/src/ui/ObjectView/common.ts new file mode 100644 index 0000000000..ba5b1b10d7 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/ObjectView/common.ts @@ -0,0 +1,14 @@ +import type { t } from './common.ts'; +export * from '../common.ts'; + +/** + * Constants: + */ +const theme: t.CommonTheme = 'Light'; +export const DEFAULTS = { + theme, + font: { size: 12 }, + block: true, + sortKeys: false, + show: { rootSummary: false, nonenumerable: false }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/ObjectView/mod.ts b/code/sys.ui/ui-react-components/src/ui/ObjectView/mod.ts new file mode 100644 index 0000000000..5fab976d20 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/ObjectView/mod.ts @@ -0,0 +1,4 @@ +/** + * @module + */ +export { ObjectView } from './ui.tsx'; diff --git a/code/sys.ui/ui-react-components/src/ui/ObjectView/t.ts b/code/sys.ui/ui-react-components/src/ui/ObjectView/t.ts new file mode 100644 index 0000000000..099589e00d --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/ObjectView/t.ts @@ -0,0 +1,32 @@ +import type { t } from './common.ts'; + +/** + * <Component>: + */ +export type ObjectViewProps = { + name?: string; + data?: any; + + show?: Partial<ObjectViewShow>; + expand?: number | ObjectViewExpand; + sortKeys?: boolean; + + block?: boolean; + fontSize?: number; + theme?: t.CommonTheme; + margin?: t.CssMarginInput; + style?: t.CssInput; +}; + +/** + * Object show feature flags. + */ +export type ObjectViewShow = { + nonenumerable: boolean; + rootSummary: boolean; +}; + +export type ObjectViewExpand = { + level?: number; + paths?: string[] | boolean; +}; diff --git a/code/sys.ui/ui-react-components/src/ui/ObjectView/ui.Renderer.tsx b/code/sys.ui/ui-react-components/src/ui/ObjectView/ui.Renderer.tsx new file mode 100644 index 0000000000..71a0bb8bfe --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/ObjectView/ui.Renderer.tsx @@ -0,0 +1,33 @@ +import { ObjectLabel, ObjectPreview, ObjectRootLabel } from 'react-inspector'; + +type NodeRenderer = (args: NodeRendererArgs) => JSX.Element; +type NodeRendererArgs = { + depth: number; + name: string; + data: any; + isNonenumerable: boolean; + expanded: boolean; +}; + +export function renderer(props: { rootSummary: boolean }): NodeRenderer { + return (args) => { + const { depth, isNonenumerable, name, expanded } = args; + + if (depth === 0) { + if ( + args.data === null || + args.data === undefined || + typeof args.data === 'boolean' || + typeof args.data === 'number' || + typeof args.data === 'string' + ) { + return <ObjectPreview data={args.data} />; + } + + const data = expanded || !props.rootSummary ? {} : args.data; + return <ObjectRootLabel name={args.name} data={data} />; + } else { + return <ObjectLabel name={name} data={args.data} isNonenumerable={isNonenumerable} />; + } + }; +} diff --git a/code/sys.ui/ui-react-components/src/ui/ObjectView/ui.tsx b/code/sys.ui/ui-react-components/src/ui/ObjectView/ui.tsx new file mode 100644 index 0000000000..6c566eadda --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/ObjectView/ui.tsx @@ -0,0 +1,100 @@ +import React, { useEffect, useState } from 'react'; +import { chromeDark, chromeLight, ObjectInspector } from 'react-inspector'; +import { type t, css, DEFAULTS, Style } from './common.ts'; +import { renderer } from './ui.Renderer.tsx'; + +type P = t.ObjectViewProps; +const D = DEFAULTS; + +export const ObjectView: React.FC<P> = (props) => { + const { block = D.block, sortKeys = D.sortKeys } = props; + const { expandLevel, expandPaths } = wrangle.expand(props); + const show = wrangle.show(props); + + const [key, setKey] = useState(wrangle.key(props)); + + /** + * Effect: ensure key/render when the expand level changes. + */ + useEffect(() => setKey(wrangle.key(props)), [expandPaths?.join(), expandLevel]); + + /** + * Render: + */ + const styles = { + base: css({ + display: block ? 'block' : 'inline-block', + ...Style.toMargins(props.margin), + }), + }; + + const theme = wrangle.theme(props); + const el = ( + <ObjectInspector + key={key} + data={props.data} + name={props.name} + theme={theme as any} + sortObjectKeys={sortKeys} + showNonenumerable={show?.nonenumerable} + nodeRenderer={renderer({ rootSummary: show?.rootSummary })} + expandLevel={expandLevel} + expandPaths={expandPaths} + /> + ); + return <div className={css(styles.base, props.style).class}>{el}</div>; +}; + +/** + * Helpers + */ +const wrangle = { + key(props: t.ObjectViewProps) { + const { expandLevel, expandPaths } = wrangle.expand(props); + return `obj:${expandLevel ?? 0}:${(expandPaths ?? []).join(',')}`; + }, + + theme(props: t.ObjectViewProps) { + const fontSize = `${props.fontSize ?? DEFAULTS.font.size}px`; + const lineHeight = '1.5em'; + return { + ...wrangle.baseTheme(props.theme), + BASE_BACKGROUND_COLOR: 'transparent', + BASE_FONT_SIZE: fontSize, + TREENODE_FONT_SIZE: fontSize, + BASE_LINE_HEIGHT: lineHeight, + TREENODE_LINE_HEIGHT: lineHeight, + }; + }, + + baseTheme(theme?: t.CommonTheme) { + theme = theme ?? DEFAULTS.theme; + if (theme === 'Light') return chromeLight; + if (theme === 'Dark') return chromeDark; + throw new Error(`Theme '${theme}' not supported.`); + }, + + show(props: P): t.ObjectViewShow { + const D = DEFAULTS.show; + const { show = {} } = props; + const { nonenumerable = D.nonenumerable, rootSummary = D.nonenumerable } = show; + return { nonenumerable, rootSummary }; + }, + + expand(props: P) { + const { expand } = props; + let expandLevel: number | undefined = undefined; + let expandPaths: string[] | undefined; + + if (typeof expand === 'number') { + expandLevel = expand; + } + + if (typeof expand === 'object') { + expandLevel = expand.level; + expandPaths = Array.isArray(expand.paths) ? expand.paths : undefined; + } + + return { expandLevel, expandPaths }; + }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Panel/-SPEC.Debug.tsx b/code/sys.ui/ui-react-components/src/ui/Panel/-SPEC.Debug.tsx new file mode 100644 index 0000000000..b266f8668d --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Panel/-SPEC.Debug.tsx @@ -0,0 +1,57 @@ +import React from 'react'; +import { Button } from '../Button/mod.ts'; +import { type t, Color, css, Signal } from './common.ts'; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssValue }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; +type P = DebugProps; + +/** + * Signals: + */ +export function createDebugSignals() { + const s = Signal.create; + const props = { theme: s<t.CommonTheme>('Light') }; + const api = { + props, + listen() { + const p = props; + p.theme.value; + }, + }; + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<P> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => { + p.theme.value; + }); + + /** + * Render: + */ + const theme = Color.theme(p.theme.value); + const styles = { + base: css({ color: theme.fg }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <Button + label={`theme: ${p.theme}`} + onClick={() => Signal.cycle<t.CommonTheme>(p.theme, ['Light', 'Dark'])} + /> + + <hr /> + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Panel/-SPEC.tsx b/code/sys.ui/ui-react-components/src/ui/Panel/-SPEC.tsx new file mode 100644 index 0000000000..9cc3a0254b --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Panel/-SPEC.tsx @@ -0,0 +1,23 @@ +import { Dev, Spec } from '../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { Panel } from './mod.ts'; + +export default Spec.describe('Panel', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + Dev.Theme.signalEffect(ctx, p.theme, 1); + + ctx.subject + .size([224, null]) + .display('grid') + .render((e) => <Panel theme={p.theme.value} />); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/Panel/common.ts b/code/sys.ui/ui-react-components/src/ui/Panel/common.ts new file mode 100644 index 0000000000..fc72b6f116 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Panel/common.ts @@ -0,0 +1,2 @@ +export * from '../common.ts'; +export const DEFAULTS = {} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Panel/mod.ts b/code/sys.ui/ui-react-components/src/ui/Panel/mod.ts new file mode 100644 index 0000000000..e42bc11c79 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Panel/mod.ts @@ -0,0 +1 @@ +export { Panel } from './ui.tsx'; diff --git a/code/sys.ui/ui-react-components/src/ui/Panel/t.ts b/code/sys.ui/ui-react-components/src/ui/Panel/t.ts new file mode 100644 index 0000000000..a7094b27c2 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Panel/t.ts @@ -0,0 +1,11 @@ +import type { t } from './common.ts'; + +/** + * <Component>: + */ +export type PanelProps = { + text?: string; + debug?: boolean; + theme?: t.CommonTheme; + style?: t.CssInput; +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Panel/ui.tsx b/code/sys.ui/ui-react-components/src/ui/Panel/ui.tsx new file mode 100644 index 0000000000..13857aee14 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Panel/ui.tsx @@ -0,0 +1,26 @@ +import React from 'react'; +import { type t, Color, css } from './common.ts'; + +export const Panel: React.FC<t.PanelProps> = (props) => { + const { text = 'š· Panel' } = props; + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ + backgroundColor: 'rgba(255, 0, 0, 0.1)' /* RED */, + color: theme.fg, + padding: 3, + borderRadius: 5, + fontSize: 14, + }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div>{text}</div> + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Concept/-SPEC.Debug.tsx b/code/sys.ui/ui-react-components/src/ui/Player.Concept/-SPEC.Debug.tsx new file mode 100644 index 0000000000..0577faaf76 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Concept/-SPEC.Debug.tsx @@ -0,0 +1,56 @@ +import React from 'react'; +import { Button } from '../Button/mod.ts'; +import { Color, css, Signal, type t } from './common.ts'; + +export type DebugProps = { + ctx: { signals: t.VideoPlayerSignals }; + theme?: t.CommonTheme; + style?: t.CssInput; +}; + +type P = DebugProps; + +/** + * Component + */ +export const Debug: React.FC<P> = (props) => { + const { ctx } = props; + const s = ctx.signals; + const p = s.props; + + Signal.useRedrawEffect(() => { + p.ready.value; + p.loop.value; + p.playing.value; + }); + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ color: theme.fg }), + title: css({ fontWeight: 'bold', marginBottom: 10 }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={css(styles.title).class}>{'WIP'}</div> + + <Button block={true} label={`action: jumpTo(12, play)`} onClick={() => s.jumpTo(12)} /> + <Button + block={true} + label={`action: jumpTo(12, paused)`} + onClick={() => s.jumpTo(12, { play: false })} + /> + <hr /> + <Button block={true} label={`play: ${p.playing}`} onClick={() => toggle(p.playing)} /> + <Button block={true} label={`loop: ${p.loop}`} onClick={() => toggle(p.loop)} /> + </div> + ); +}; + +/** + * Helpers + */ +const toggle = (signal: t.Signal<boolean>) => (signal.value = !signal.value); diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Concept/-SPEC.tsx b/code/sys.ui/ui-react-components/src/ui/Player.Concept/-SPEC.tsx new file mode 100644 index 0000000000..e174d07198 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Concept/-SPEC.tsx @@ -0,0 +1,28 @@ +import { Spec, expect } from '../-test.ui.ts'; +import { Debug } from './-SPEC.Debug.tsx'; + +import { Player } from '../../mod.ts'; +import { sampleTimestamps } from '../Player.Thumbnails/-SPEC.sample.ts'; +import { ConceptPlayer } from './mod.ts'; + +export default Spec.describe('VideoPlayer', (e) => { + const s = Player.Video.signals(); + + e.it('API', (e) => { + expect(Player.Concept.View).to.equal(ConceptPlayer); + }); + + e.it('init', async (e) => { + const ctx = Spec.ctx(e); + ctx.subject.size([688, null]).render((e) => { + return ( + <Player.Concept.View thumbnails={true} timestamps={sampleTimestamps} videoSignals={s} /> + ); + }); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug ctx={{ signals: s }} />); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Concept/-tmp.tsx b/code/sys.ui/ui-react-components/src/ui/Player.Concept/-tmp.tsx new file mode 100644 index 0000000000..ca83640e0a --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Concept/-tmp.tsx @@ -0,0 +1,55 @@ +import React, { useEffect, useState } from 'react'; +import { createPortal } from 'react-dom'; +import { type t, rx, Color, css, Button } from './common.ts'; + +export type OverlayProps = { + ctx: t.ConceptPlayerProps; + theme?: t.CommonTheme; + style?: t.CssInput; + onClose: () => void; +}; + +type P = OverlayProps; + +/** + * Component. + */ +export const Overlay: React.FC<P> = (props) => { + const {} = props; + const [el, setEl] = useState<React.ReactElement>(); + + useEffect(() => { + const life = rx.disposable(); + import('./ui.tsx').then(({ ConceptPlayer }) => { + setEl(<ConceptPlayer {...props.ctx} />); + }); + return life.dispose; + }, []); + + /** + * Render:. + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ + Absolute: 0, + position: 'fixed', + backgroundColor: theme.bg, + color: theme.fg, + zIndex: 9999999, + }), + body: css({ Absolute: 20 }), + close: css({ + Absolute: [null, 10, 10, null], + }), + }; + + const elBody = ( + <div className={css(styles.base, props.style).class}> + <div className={styles.body.class}>{el}</div> + <Button label={'Close'} onClick={props.onClose} style={styles.close} /> + </div> + ); + + return createPortal(elBody, document.body); +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Concept/common.ts b/code/sys.ui/ui-react-components/src/ui/Player.Concept/common.ts new file mode 100644 index 0000000000..d197b80a38 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Concept/common.ts @@ -0,0 +1,8 @@ +export * from '../common.ts'; + +export { Button } from '../Button/mod.ts'; +export { Icons } from '../Icons.ts'; +export { Timestamp, Thumbnails } from '../Player.Thumbnails/mod.ts'; +export { playerSignalsFactory, VideoPlayer } from '../Player.Video/mod.ts'; + +export const DEFAULTS = {} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Concept/mod.ts b/code/sys.ui/ui-react-components/src/ui/Player.Concept/mod.ts new file mode 100644 index 0000000000..484a1b19aa --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Concept/mod.ts @@ -0,0 +1 @@ +export { ConceptPlayer } from './ui.tsx'; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Concept/t.ts b/code/sys.ui/ui-react-components/src/ui/Player.Concept/t.ts new file mode 100644 index 0000000000..efb6cdf51e --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Concept/t.ts @@ -0,0 +1,17 @@ +import type { t } from './common.ts'; + +/** + * An extended video player. + */ +export type ConceptPlayerProps = { + debug?: boolean; + title?: string; + video?: string; + timestamps?: t.VideoTimestamps; + thumbnails?: boolean; + + theme?: t.CommonTheme; + style?: t.CssInput; + + videoSignals?: t.VideoPlayerSignals; +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Concept/ui.DisplayImage.tsx b/code/sys.ui/ui-react-components/src/ui/Player.Concept/ui.DisplayImage.tsx new file mode 100644 index 0000000000..acca76a74d --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Concept/ui.DisplayImage.tsx @@ -0,0 +1,73 @@ +import React, { useEffect, useState } from 'react'; +import { type t, Color, css, Icons, Signal, Time, Timestamp } from './common.ts'; + +export type DisplayImageProps = { + timestamps: t.VideoTimestamps; + theme?: t.CommonTheme; + style?: t.CssInput; + videoSignals: t.VideoPlayerSignals; +}; + +type P = DisplayImageProps; + +/** + * Component + */ +export const DisplayImage: React.FC<P> = (props) => { + const { videoSignals: s, timestamps } = props; + + const [error, setError] = useState(false); + const [src, setSrc] = useState<string>(); + const updateSrc = (value?: string) => { + setSrc(value); + setError(false); + }; + + /** + * Lifecycle + */ + Signal.useEffect(() => { + const elapsed = s.props.currentTime.value; + const match = Timestamp.find(timestamps, elapsed); + if (match?.image !== src) Time.delay(() => updateSrc(match?.image)); + }); + + useEffect(() => { + if (src) { + const image = new Image(); + image.src = src ?? ''; + image.onerror = () => setError(true); + } + }, [src]); + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ position: 'relative', color: theme.fg }), + image: css({ + Absolute: 10, + backgroundImage: src ? `url(${src})` : undefined, + backgroundSize: 'cover', + backgroundPosition: 'center', + backgroundRepeat: 'no-repeat', + }), + error: css({ Absolute: 0, display: 'grid', placeItems: 'center' }), + }; + + const elError = error && ( + <div className={styles.error.class}> + <Icons.Error color={Color.RED} /> + </div> + ); + + const tooltip = !error ? '' : `Failed to load image: ${src}`; + + return ( + <div className={css(styles.base, props.style).class} title={tooltip}> + {src && <div className={styles.image.class} />} + {elError} + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Concept/ui.tsx b/code/sys.ui/ui-react-components/src/ui/Player.Concept/ui.tsx new file mode 100644 index 0000000000..d6c3031f73 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Concept/ui.tsx @@ -0,0 +1,104 @@ +import React, { useRef, useState } from 'react'; +import { Overlay } from './-tmp.tsx'; +import { + type t, + Button, + Color, + css, + playerSignalsFactory, + Thumbnails, + VideoPlayer, +} from './common.ts'; +import { DisplayImage } from './ui.DisplayImage.tsx'; + +type P = t.ConceptPlayerProps; + +/** + * Component + */ +export const ConceptPlayer: React.FC<P> = (props) => { + const timestamps = wrangle.timestamps(props); + const playerSignalsRef = wrangle.playerSignals(props); + const playerSignals = playerSignalsRef.current; + + const [showOverlay, setShowOverlay] = useState<boolean>(false); + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ color: theme.fg, fontSize: 14 }), + body: css({ display: 'grid', gridTemplateColumns: `1fr 2fr`, columnGap: '5px' }), + videoPlayer: css({}), + thumbnails: css({ marginTop: 30 }), + + tmp: css({ + marginTop: 50, + }), + }; + + const elThumbnails = props.thumbnails && ( + <Thumbnails + style={styles.thumbnails} + timestamps={timestamps} + videoSignals={props.videoSignals} + onTimestampClick={(e) => { + const isPlaying = playerSignals.props.playing.value; + const target = e.total.sec; + if (e.isCurrent) { + playerSignals.toggle(!isPlaying); + } else { + playerSignals.jumpTo(target); + } + }} + /> + ); + + const elPlayer = ( + <VideoPlayer title={props.title} style={styles.videoPlayer} signals={playerSignals} /> + ); + + const elTmp = ( + <div className={styles.tmp.class}> + <Button + label={'š· Full Screen Concept Player'} + onClick={() => { + /** + * TODO š· + */ + setShowOverlay(true); + }} + /> + </div> + ); + + return ( + <div className={css(styles.base, props.style).class}> + <div className={styles.body.class}> + {elPlayer} + <DisplayImage timestamps={timestamps} videoSignals={playerSignals} /> + </div> + {elThumbnails} + + {elTmp} + {showOverlay && <Overlay ctx={props} onClose={() => setShowOverlay(false)} />} + </div> + ); +}; + +/** + * Helpers + */ +const wrangle = { + timestamps(props: P) { + const { timestamps } = props; + if (typeof timestamps !== 'object') return {}; + return timestamps; + }, + + playerSignals(props: P) { + type T = t.VideoPlayerSignals; + return useRef<T>(props.videoSignals ?? playerSignalsFactory()); + }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/-.test.ts b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/-.test.ts new file mode 100644 index 0000000000..98c336bc85 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/-.test.ts @@ -0,0 +1,19 @@ +import { type t, expect, describe, it } from '../../-test.ts'; +import { Timestamp } from './mod.ts'; + +describe('Thumbnails', () => { + describe('Video Timestamps', () => { + it('has image', () => { + const timestamps: t.VideoTimestamps = { + '00:00:00.000': {}, + '00:00:10.000': { image: '/path/to-image.png' }, + }; + + const a = Timestamp.find(timestamps, 1); + const b = Timestamp.find(timestamps, 12); + + expect(a?.image).to.eql(undefined); + expect(b?.image).to.eql('/path/to-image.png'); + }); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/-SPEC.sample.ts b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/-SPEC.sample.ts new file mode 100644 index 0000000000..2f0c94a019 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/-SPEC.sample.ts @@ -0,0 +1,19 @@ +import { type t } from './common.ts'; + +const IMG = { + KITTEN: `https://external-content.duckduckgo.com/iu/?u=https%3A%2F%2Fwww.wallpaperflare.com%2Fstatic%2F732%2F902%2F566%2Fanimal-pet-cute-grey-wallpaper.jpg&f=1&nofb=1&ipt=a69e2c104f05537e10d73517716481f2b1db71244a10fdb108b07fd6e9a01652&ipo=images`, + LIZARD: `https://external-content.duckduckgo.com/iu/?u=https%3A%2F%2Ftse1.mm.bing.net%2Fth%3Fid%3DOIP.uZCiSQquNq1uSARFRpKM8wHaEK%26pid%3DApi&f=1&ipt=ecb28ebe255e465723a38e03dd84a17cbbba3f734802b7ad12266035fabaa71d&ipo=images`, + NOT_FOUND: '/404.png', +} as const; + +export const sampleTimestamps: t.VideoTimestamps = { + '00:00:01.000': { image: IMG.KITTEN }, + '00:00:03.123': { image: IMG.LIZARD }, + '00:00:6.001': { image: IMG.NOT_FOUND }, + '00:01:38.002': { image: IMG.NOT_FOUND }, + '00:01:60.003': { image: IMG.NOT_FOUND }, + '00:01:68.004': { image: IMG.NOT_FOUND }, + '00:02:23.000': { image: IMG.NOT_FOUND }, + '00:03:01.001': { image: IMG.LIZARD }, + '00:01:05.002': { image: IMG.KITTEN }, +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/-SPEC.tsx b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/-SPEC.tsx new file mode 100644 index 0000000000..729a92b120 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/-SPEC.tsx @@ -0,0 +1,26 @@ +import { Spec, expect } from '../-test.ui.ts'; +import { Player } from '../../mod.ts'; + +import { sampleTimestamps } from './-SPEC.sample.ts'; +import { Thumbnails } from './mod.ts'; + +export default Spec.describe('VideoPlayer', (e) => { + const s = Player.Video.signals(); + + e.it('API', (e) => { + expect(Player.Timestamp.Thumbnails.View).to.equal(Thumbnails); + }); + + e.it('init', async (e) => { + const ctx = Spec.ctx(e); + ctx.subject.size([520, null]).render((e) => { + return ( + <Player.Timestamp.Thumbnails.View + timestamps={sampleTimestamps} + onTimestampClick={(e) => console.info(`ā”ļø onTimestampClick:`, e)} + videoSignals={s} + /> + ); + }); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/common.ts b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/common.ts new file mode 100644 index 0000000000..1822a5fd88 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/common.ts @@ -0,0 +1,7 @@ +export * from '../common.ts'; +export { Icons } from '../Icons.ts'; +import { PlayerColors } from '../Player/common.ts'; + +export const DEFAULTS = { + BLUE: PlayerColors.BLUE, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/mod.ts b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/mod.ts new file mode 100644 index 0000000000..a893b68698 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/mod.ts @@ -0,0 +1,6 @@ +/** + * @module + * Timestamp thumbnails edit controls. + */ +export { Timestamp } from './u.ts'; +export { Thumbnails } from './ui.Thumbnails.tsx'; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/t.ts b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/t.ts new file mode 100644 index 0000000000..79e914d320 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/t.ts @@ -0,0 +1,27 @@ +import type { t } from './common.ts'; + +/** + * <Component> + */ +export type ThumbnailsProps = { + timestamps?: t.VideoTimestamps; + + theme?: t.CommonTheme; + style?: t.CssInput; + + videoSignals?: t.VideoPlayerSignals; + onTimestampClick?: t.VideoTimestampHandler; +}; + +/** + * Timestamps. + */ +export type VideoTimestamps = t.Timestamps<t.VideoTimestampData>; +export type VideoTimestamp = t.Timestamp<t.VideoTimestampData>; +export type VideoTimestampData = { image?: t.StringPath }; + +/** + * Events. + */ +export type VideoTimestampHandler = (e: VideoTimestampHandlerArgs) => void; +export type VideoTimestampHandlerArgs = t.VideoTimestamp & { isCurrent: boolean }; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/u.timestamp.ts b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/u.timestamp.ts new file mode 100644 index 0000000000..77c80b9413 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/u.timestamp.ts @@ -0,0 +1,57 @@ +import { type t, Timestamp as Std } from './common.ts'; + +const round = 3; +const unit = 'secs'; + +/** + * Check if a given timestamp is the current one based on the elapsed time. + */ +function isCurrent( + timestamps: t.VideoTimestamps, + timestamp: t.StringTimestamp, + currentTime: t.Secs, +): boolean { + return Std.isCurrent(timestamps, timestamp, currentTime, { unit, round }); +} + +/** + * Lookup a timestamp given the current elapsed time. + */ +function find(timestamps: t.VideoTimestamps, elapsed: t.Secs): t.VideoTimestampData | undefined { + return Std.find(timestamps, elapsed, { unit, round })?.data; +} + +/** + * Convert the set of { "HH:MM:SS:mmm":<value> } timestamp + * strings into list of sorted stuctures. + */ +function parseTimes(timestamps?: t.VideoTimestamps): t.VideoTimestamp[] { + return Std.parse<t.VideoTimestampData>(timestamps, { round }); +} + +/** + * Parse a "HH:MM:DD:mmm" string. + */ +function parseTime(timestamp: string): t.TimeDuration { + return Std.parse(timestamp, { round }); +} + +/** + * Generate a sub-range for a timestamp within a map of timestamps. + */ +function range(timestamps: t.VideoTimestamps | undefined, location: t.Secs | t.StringTimestamp) { + if (!timestamps) return undefined; + return Std.range(timestamps, location, { unit, round }); +} + +export const Timestamp = { + parseTime, + parseTimes, + isCurrent, + find, + range, + toString(input: t.VideoTimestamp | string) { + const d = typeof input === 'string' ? input : input.total; + return Std.toString(d); + }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/u.ts b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/u.ts new file mode 100644 index 0000000000..d5f85b49c2 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/u.ts @@ -0,0 +1 @@ +export * from './u.timestamp.ts'; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/ui.Thumbnail.FooterBar.tsx b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/ui.Thumbnail.FooterBar.tsx new file mode 100644 index 0000000000..85b19e3081 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/ui.Thumbnail.FooterBar.tsx @@ -0,0 +1,53 @@ +import React, { useEffect, useRef, useState } from 'react'; +import { type t, Color, css, Signal, DEFAULTS, rx } from './common.ts'; +import { Timestamp } from './u.ts'; + +export type FooterBarProps = { + timestamp: t.StringTimestamp; + timestamps?: t.VideoTimestamps; + isCurrent?: boolean; + theme?: t.CommonTheme; + style?: t.CssInput; +}; + +type P = FooterBarProps; + +/** + * Component. + */ +export const FooterBar: React.FC<P> = (props) => { + const { isCurrent, timestamp, timestamps } = props; + const time = wrangle.time(timestamp); + + /** + * Render:. + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ + borderTop: `solid 1px ${Color.alpha(theme.fg, 0.1)}`, + fontSize: 12, + PaddingY: 1, + display: 'grid', + placeItems: 'center', + backgroundColor: isCurrent ? DEFAULTS.BLUE : undefined, + color: isCurrent ? Color.WHITE : undefined, + }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div>{time}</div> + </div> + ); +}; + +/** + * Helpers + */ +const wrangle = { + time(ts: t.StringTimestamp) { + ts = Timestamp.toString(ts); + return ts.slice(0, ts.indexOf('.')); + }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/ui.Thumbnail.tsx b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/ui.Thumbnail.tsx new file mode 100644 index 0000000000..a382ee901e --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/ui.Thumbnail.tsx @@ -0,0 +1,140 @@ +import React, { useEffect, useRef, useState } from 'react'; +import { type t, Color, css, Icons, Signal, Time } from './common.ts'; +import { Timestamp } from './u.ts'; +import { FooterBar } from './ui.Thumbnail.FooterBar.tsx'; + +export type ThumbnailProps = { + timestamp: t.StringTimestamp; + timestamps?: t.VideoTimestamps; + data: t.VideoTimestampData; + theme?: t.CommonTheme; + style?: t.CssInput; + onClick?: t.VideoTimestampHandler; + videoSignals?: t.VideoPlayerSignals; +}; + +export const Thumbnail: React.FC<ThumbnailProps> = (props) => { + const { timestamp, timestamps, data, onClick, videoSignals } = props; + const src = data.image; + + const rangeRef = useRef(Timestamp.range(timestamps, timestamp)); + const range = rangeRef.current; + + const [loaded, setLoaded] = useState(false); + const [error, setError] = useState(false); + const [isOver, setOver] = useState(false); + const [isDown, setDown] = useState(false); + const [isCurrent, setIsCurrent] = useState(false); + const over = (isOver: boolean) => () => setOver(isOver); + const down = (isDown: boolean) => () => setDown(isDown); + + /** + * Lifecycle. + */ + useEffect(() => { + const image = new Image(); + image.src = src ?? ''; + image.onload = () => setLoaded(true); + image.onerror = () => setError(true); + }, [src]); + + /** + * Highlight when current thumbnail/timestamp. + */ + Signal.useEffect(() => { + const currentTime = videoSignals?.props.currentTime.value ?? -1; + const update = (isCurrent: boolean) => { + // NB: state updated after a micro-delay to ensure writing happens on the next render frame. + Time.delay(() => setIsCurrent(isCurrent)); + }; + + if (timestamps) { + const isCurrent = Timestamp.isCurrent(timestamps, timestamp, currentTime); + update(isCurrent); + } else { + update(false); + } + }); + + /** + * Handlers. + */ + function handleClick() { + const total = Timestamp.parseTime(timestamp); + props.onClick?.({ timestamp, data, total, isCurrent }); + } + + /** + * Render:. + */ + const theme = Color.theme(props.theme); + const borderRadius = 6; + const styles = { + base: css({ + position: 'relative', + userSelect: 'none', + backgroundColor: Color.alpha(theme.fg, 0.03), + color: theme.fg, + width: 100, + height: 80, + display: 'grid', + gridTemplateRows: `1fr auto`, + borderRadius, + border: `solid 1px ${Color.alpha(theme.fg, isOver && onClick ? 0.15 : 0.1)}`, + boxShadow: + isOver && onClick + ? `0 1px ${isDown ? 4 : 15}px 0 ${Color.format(isDown ? -0.05 : -0.1)}` + : undefined, + transform: `translateY(${isDown ? 0 : -1}px)`, + }), + image: { + base: css({ + position: 'relative', + backgroundColor: theme.bg, + borderRadius: `${borderRadius - 1}px ${borderRadius - 1}px 0 0`, + overflow: 'hidden', + display: 'grid', + placeItems: 'center', + }), + img: css({ + Absolute: 0, + backgroundImage: loaded ? `url(${src})` : undefined, + display: 'grid', + backgroundSize: 'cover', + backgroundPosition: 'center', + backgroundRepeat: 'no-repeat', + }), + }, + }; + + const tooltip = !error ? '' : `Failed to load image: ${src}`; + const elBody = ( + <div className={styles.image.base.class} title={tooltip}> + {loaded && <div className={styles.image.img.class} />} + {error && <Icons.Error color={Color.RED} />} + </div> + ); + + const elFooter = ( + <FooterBar + timestamp={timestamp} + timestamps={timestamps} + isCurrent={isCurrent} + theme={theme.name} + /> + ); + + return ( + <div + className={css(styles.base, props.style).class} + onMouseEnter={over(true)} + onMouseLeave={over(false)} + onMouseDown={down(true)} + onMouseUp={down(false)} + onClick={handleClick} + > + {elBody} + {elFooter} + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/ui.Thumbnails.tsx b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/ui.Thumbnails.tsx new file mode 100644 index 0000000000..89b2142f1f --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Thumbnails/ui.Thumbnails.tsx @@ -0,0 +1,41 @@ +import React from 'react'; + +import { type t, Color, css } from './common.ts'; +import { Timestamp } from './u.ts'; +import { Thumbnail } from './ui.Thumbnail.tsx'; + +type P = t.ThumbnailsProps; +export const Thumbnails: React.FC<P> = (props) => { + const { timestamps = {} } = props; + const times = Timestamp.parseTimes(timestamps); + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ + color: theme.fg, + display: 'flex', + alignContent: 'flex-start', + flexWrap: 'wrap', + gap: '8px', + }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + {times.map(({ timestamp, data }, i) => ( + <Thumbnail + key={`${i}.${timestamp}`} + data={data} + timestamp={timestamp} + timestamps={timestamps} + onClick={(e) => props.onTimestampClick?.(e)} + videoSignals={props.videoSignals} + theme={theme.name} + /> + ))} + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Video/-SPEC.Debug.tsx b/code/sys.ui/ui-react-components/src/ui/Player.Video/-SPEC.Debug.tsx new file mode 100644 index 0000000000..4c71034ff3 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Video/-SPEC.Debug.tsx @@ -0,0 +1,169 @@ +import React from 'react'; +import { Player } from '../../mod.ts'; +import { Button } from '../Button/mod.ts'; +import { type t, css, D, Signal } from './common.ts'; + +/** + * Types: + */ +export type DebugProps = { + debug: DebugSignals; + theme?: t.CommonTheme; + style?: t.CssInput; +}; +export type DebugSignals = ReturnType<typeof createDebugSignals>; + +/** + * Signals: + */ +export function createDebugSignals() { + const s = Signal.create; + + const video = Player.Video.signals({ + // loop: true, + // autoPlay: true, + // showControls: false, + }); + + const props = {}; + const api = { + props, + video, + listen() { + const p = video.props; + p.ready.value; + + // Media: + p.src.value; + p.playing.value; + p.muted.value; + p.autoPlay.value; + p.loop.value; + + // Appearance: + p.showControls.value; + p.showFullscreenButton.value; + p.showVolumeControl.value; + p.background.value; + p.cornerRadius.value; + p.aspectRatio.value; + p.scale.value; + + // Commands: + p.jumpTo.value; + }, + }; + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const video = debug.video; + const p = video.props; + + Signal.useRedrawEffect(() => debug.listen()); + + /** + * Render: + */ + const styles = { + base: css({}), + title: css({ fontWeight: 'bold', marginBottom: 10 }), + cols: css({ display: 'grid', gridTemplateColumns: 'auto 1fr auto' }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={css(styles.title, styles.cols).class}> + <div>{'Player.Video'}</div> + <div /> + <CurrentTime video={video} /> + </div> + + <Button block label={`method: jumpTo(12, play)`} onClick={() => video.jumpTo(12)} /> + <Button + block + label={`method: jumpTo(12, paused)`} + onClick={() => video.jumpTo(12, { play: false })} + /> + <hr /> + <Button block label={`playing: ${p.playing}`} onClick={() => Signal.toggle(p.playing)} /> + <Button block label={`muted: ${p.muted}`} onClick={() => Signal.toggle(p.muted)} /> + <Button block label={`autoplay: ${p.autoPlay}`} onClick={() => Signal.toggle(p.autoPlay)} /> + <Button block label={`loop: ${p.loop}`} onClick={() => Signal.toggle(p.loop)} /> + <Button + block + label={`background: ${p.background} ā ${p.background.value ? 'fill' : 'fixed-size'}`} + onClick={() => Signal.toggle(p.background)} + /> + <hr /> + <Button + block + label={`showControls: ${p.showControls}`} + onClick={() => Signal.toggle(p.showControls)} + /> + <Button + block + label={`showFullscreenButton: ${p.showFullscreenButton}`} + onClick={() => Signal.toggle(p.showFullscreenButton)} + /> + <Button + block + label={`showVolumeControl: ${p.showVolumeControl}`} + onClick={() => Signal.toggle(p.showVolumeControl)} + /> + <Button + block + label={`cornerRadius: ${p.cornerRadius}`} + onClick={() => Signal.cycle(p.cornerRadius, [0, 5, 10, 15])} + /> + <Button + block + label={`aspectRatio: ${p.aspectRatio}`} + onClick={() => Signal.cycle(p.aspectRatio, [D.aspectRatio, '4/3', '2.39/1', '1/1'])} + /> + <Button + block + label={() => { + const current = p.scale.value; + return `scale: ${typeof current === 'function' ? 'Ęn' : current}`; + }} + onClick={() => { + const fn: t.VideoPlayerScale = (e) => { + const pixels = 1; + const res = e.enlargeBy(pixels); + console.info(`ā”ļø scale (callback):`, e); + console.info(` increment (${pixels}px):`, res); + return res; + }; + Signal.cycle(p.scale, [undefined, 1, fn, 2]); + }} + /> + + <hr /> + + <Button + block + label={`src: ${p.src}`} + onClick={() => + Signal.cycle(p.src, [ + D.video, // Default: "tubes" + 'vimeo/727951677', // Rowan: "group scale" + ]) + } + /> + + <hr /> + </div> + ); +}; + +function CurrentTime(props: { video: t.VideoPlayerSignals; prefix?: string }) { + const { video, prefix = 'elapsed' } = props; + const p = video.props; + Signal.useRedrawEffect(() => p.currentTime.value); + return <div>{`(${prefix}: ${p.currentTime.value.toFixed(2)}s)`}</div>; +} diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Video/-SPEC.tsx b/code/sys.ui/ui-react-components/src/ui/Player.Video/-SPEC.tsx new file mode 100644 index 0000000000..8508cdec27 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Video/-SPEC.tsx @@ -0,0 +1,40 @@ +import React from 'react'; + +import { Signal, Spec, expect } from '../-test.ui.ts'; +import { Player } from '../../mod.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { VideoPlayer } from './mod.ts'; + +export default Spec.describe('VideoPlayer', (e) => { + const debug = createDebugSignals(); + const video = debug.video; + + e.it('API', (e) => { + expect(Player.Video.View).to.equal(VideoPlayer); + }); + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + const updateSize = () => { + const fill = video.props.background.value; + if (fill) ctx.subject.size('fill'); + else ctx.subject.size([520, null]); + }; + + Signal.effect(() => { + updateSize(); + debug.listen(); + }); + updateSize(); + + ctx.subject.display('grid').render((e) => { + return <VideoPlayer signals={video} />; + }); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Video/-Signals.test.ts b/code/sys.ui/ui-react-components/src/ui/Player.Video/-Signals.test.ts new file mode 100644 index 0000000000..b79a707882 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Video/-Signals.test.ts @@ -0,0 +1,125 @@ +import { type t, describe, expect, it } from '../../-test.ts'; +import { Player } from '../Player/mod.ts'; +import { D } from './common.ts'; +import { playerSignalsFactory } from './mod.ts'; + +describe('VideoPlayer: Signals API', () => { + describe('props', () => { + it('initial values (defaults)', () => { + const s = playerSignalsFactory(); + const p = s.props; + + expect(p.ready.value).to.eql(false); + expect(p.src.value).to.eql(D.video); + + expect(p.playing.value).to.eql(false); + expect(p.currentTime.value).to.eql(0); + expect(p.loop.value).to.eql(D.loop); + expect(p.autoPlay.value).to.eql(D.autoPlay); + expect(p.muted.value).to.eql(D.muted); + + expect(p.showControls.value).to.eql(true); + expect(p.showFullscreenButton.value).to.eql(D.showFullscreenButton); + expect(p.showVolumeControl.value).to.eql(D.showVolumeControl); + expect(p.cornerRadius.value).to.eql(D.cornerRadius); + expect(p.aspectRatio.value).to.eql(D.aspectRatio); + expect(p.background.value).to.eql(D.background); + expect(p.scale.value).to.eql(D.scale); + + expect(p.jumpTo.value).to.eql(undefined); + + p.playing.value = true; + expect(p.playing.value).to.eql(true); + }); + + it('param: custom { defaults }', () => { + const scale: t.VideoPlayerScale = (e) => e.enlargeBy(1); + const s = Player.Video.signals({ + src: 'vimeo/foobar', + loop: true, + showControls: false, + showFullscreenButton: true, + showVolumeControl: false, + cornerRadius: 0, + aspectRatio: '2.39/1', + autoPlay: true, + muted: true, + background: true, + scale, + }); + + const p = s.props; + expect(p.src.value).to.eql('vimeo/foobar'); + expect(p.loop.value).to.eql(true); + expect(p.aspectRatio.value).to.eql('2.39/1'); + expect(p.autoPlay.value).to.eql(true); + expect(p.muted.value).to.eql(true); + + expect(p.showControls.value).to.eql(false); + expect(p.showFullscreenButton.value).to.eql(true); + expect(p.showVolumeControl.value).to.eql(false); + expect(p.cornerRadius.value).to.eql(0); + expect(p.background.value).to.eql(true); + expect(p.scale.value).to.equal(scale); + }); + + it('param: src param (string)', () => { + const s = Player.Video.signals('vimeo/foobar'); + expect(s.props.src.value).to.eql('vimeo/foobar'); + }); + }); + + describe('methods', () => { + it('jumpTo() method ā props.jumpTo', () => { + const s = playerSignalsFactory(); + expect(s.props.jumpTo.value).to.eql(undefined); + + const res = s.jumpTo(10); + expect(res).to.equal(s); + expect(s.props.jumpTo.value).to.eql({ second: 10, play: true }); + + s.jumpTo(15, { play: false }); + expect(s.props.jumpTo.value).to.eql({ second: 15, play: false }); + }); + + it('play', () => { + const s = playerSignalsFactory(); + const assertPlaying = (value: boolean) => expect(s.props.playing.value).to.eql(value); + assertPlaying(false); + + const res = s.play(); + expect(res).to.equal(s); + assertPlaying(true); + }); + + it('pause', () => { + const s = playerSignalsFactory(); + const assertPlaying = (value: boolean) => expect(s.props.playing.value).to.eql(value); + assertPlaying(false); + + s.play(); + assertPlaying(true); + + const res = s.play().pause(); + expect(res).to.equal(s); + assertPlaying(false); + }); + + it('toggle', () => { + const s = playerSignalsFactory(); + const assertPlaying = (value: boolean) => expect(s.props.playing.value).to.eql(value); + assertPlaying(false); + + const res = s.toggle(); + expect(res).to.equal(s); + assertPlaying(true); + + s.toggle(); + assertPlaying(false); + s.toggle(false); + assertPlaying(false); + s.toggle(true); + assertPlaying(true); + }); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Video/common.ts b/code/sys.ui/ui-react-components/src/ui/Player.Video/common.ts new file mode 100644 index 0000000000..def6418293 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Video/common.ts @@ -0,0 +1,20 @@ +export * from '../common.ts'; + +/** + * Constants + */ +export const DEFAULTS = { + video: 'vimeo/499921561', // Tubes. + + loop: false, + showFullscreenButton: false, + showVolumeControl: true, + showControls: true, + cornerRadius: 0, + aspectRatio: '16/9', + autoPlay: false, + muted: false, + background: false, + scale: 1, +} as const; +export const D = DEFAULTS; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Video/m.Signals.ts b/code/sys.ui/ui-react-components/src/ui/Player.Video/m.Signals.ts new file mode 100644 index 0000000000..56639151b6 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Video/m.Signals.ts @@ -0,0 +1,74 @@ +import { type t, DEFAULTS, Signal } from './common.ts'; + +type T = t.VideoPlayerSignals; +type P = t.VideoPlayerSignalProps; +const D = DEFAULTS; + +/** + * Factory: create a new instance of signals + */ +export const playerSignalsFactory: t.PlayerSignalsFactory = (input = {}) => { + const defaults = wrangle.defaults(input); + const s = Signal.create; + + const props: T['props'] = { + ready: s(false), + + // Media: + src: s<t.StringVideoAddress>(defaults.src ?? D.video), + playing: s<boolean>(false), + currentTime: s<t.Secs>(0), + loop: s<boolean>(defaults.loop ?? D.loop), + autoPlay: s<boolean>(defaults.autoPlay ?? D.autoPlay), + muted: s<boolean>(defaults.muted ?? D.muted), + + // Appearance: + showControls: s<boolean>(defaults.showControls ?? D.showControls), + showFullscreenButton: s<boolean>(defaults.showFullscreenButton ?? D.showFullscreenButton), + showVolumeControl: s<boolean>(defaults.showVolumeControl ?? D.showVolumeControl), + background: s<boolean>(defaults.background ?? D.background), + cornerRadius: s<number>(defaults.cornerRadius ?? D.cornerRadius), + aspectRatio: s<string>(defaults.aspectRatio ?? D.aspectRatio), + scale: s<number | t.VideoPlayerScale>(defaults.scale ?? D.scale), + + // Commands: + jumpTo: s<t.VideoPlayerJumpTo | undefined>(), + }; + + const api: T = { + get props() { + return props; + }, + + /** + * Methods: + */ + jumpTo(second, options = {}) { + const { play = true } = options; + props.jumpTo.value = { second, play }; + return api; + }, + play: () => api.toggle(true), + pause: () => api.toggle(false), + toggle(playing) { + const next = typeof playing === 'boolean' ? playing : !props.playing.value; + props.playing.value = next; + return api; + }, + }; + + return api; +}; + +/** + * Helpers + */ +const wrangle = { + defaults( + input?: t.PlayerSignalsFactoryDefaults | t.StringVideoAddress, + ): t.PlayerSignalsFactoryDefaults { + if (!input) return {}; + if (typeof input === 'string') return { src: input }; + return input; + }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Video/mod.ts b/code/sys.ui/ui-react-components/src/ui/Player.Video/mod.ts new file mode 100644 index 0000000000..b86065fca4 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Video/mod.ts @@ -0,0 +1,6 @@ +/** + * @module + * A signal's controllable flexible `<VideoPlayer>` UI component. + */ +export { playerSignalsFactory } from './m.Signals.ts'; +export { VideoPlayer } from './ui.tsx'; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Video/t.Signals.ts b/code/sys.ui/ui-react-components/src/ui/Player.Video/t.Signals.ts new file mode 100644 index 0000000000..92ff936bbb --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Video/t.Signals.ts @@ -0,0 +1,81 @@ +import type { t } from './common.ts'; + +/** + * Signals: + */ +export type PlayerSignalsFactory = ( + defaults?: PlayerSignalsFactoryDefaults | t.StringVideoAddress, +) => t.VideoPlayerSignals; + +/** Defaults passed to the signals API factory. */ +export type PlayerSignalsFactoryDefaults = { + src?: t.StringVideoAddress; + loop?: boolean; + showControls?: boolean; + showFullscreenButton?: boolean; + showVolumeControl?: boolean; + cornerRadius?: number; + aspectRatio?: string; + scale?: number | t.VideoPlayerScale; + autoPlay?: boolean; + muted?: boolean; + background?: boolean; +}; + +/** + * Signals API for dynamic control of the <VideoPlayer>. + */ +export type VideoPlayerSignals = { + readonly props: VideoPlayerSignalProps; + jumpTo(second: t.Secs, options?: { play?: boolean }): VideoPlayerSignals; + play(): VideoPlayerSignals; + pause(): VideoPlayerSignals; + toggle(playing?: boolean): VideoPlayerSignals; +}; + +/** The raw signal properties that make up the VideoPlayer signals API/. */ +export type VideoPlayerSignalProps = { + readonly ready: t.Signal<boolean>; + readonly src: t.Signal<t.StringVideoAddress | undefined>; + + /** + * Media: + */ + readonly playing: t.Signal<boolean>; + readonly currentTime: t.Signal<t.Secs>; + readonly loop: t.Signal<boolean>; + readonly autoPlay: t.Signal<boolean>; + readonly muted: t.Signal<boolean>; + + /** + * Appearance: + */ + readonly showControls: t.Signal<boolean>; + readonly showFullscreenButton: t.Signal<boolean>; + readonly showVolumeControl: t.Signal<boolean>; + /** A background video, covers the container running silently (and generally auto-plays). */ + readonly background: t.Signal<boolean>; + readonly aspectRatio: t.Signal<string>; + readonly cornerRadius: t.Signal<number>; + readonly scale: t.Signal<number | t.VideoPlayerScale>; + + /** + * Commands: + */ + readonly jumpTo: t.Signal<t.VideoPlayerJumpTo | undefined>; +}; + +/** Structure representing a jump-to ("seek") location */ +export type VideoPlayerJumpTo = { second: t.Secs; play: boolean }; + +/** + * A function that calculates the scale transform to apply to the + * video-player based on + * + */ +export type VideoPlayerScale = (e: VideoPlayerScaleArgs) => t.Percent; +export type VideoPlayerScaleArgs = { + readonly width: t.Pixels; + readonly height: t.Pixels; + enlargeBy(increment: t.Pixels): t.Percent; +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Video/t.ts b/code/sys.ui/ui-react-components/src/ui/Player.Video/t.ts new file mode 100644 index 0000000000..d4306edb8d --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Video/t.ts @@ -0,0 +1,32 @@ +import type { MediaPlayerInstance, MediaPlayerProps } from '@vidstack/react'; +import type { RefObject } from 'react'; +import type { t } from './common.ts'; + +export type * from './t.Signals.ts'; + +/** The address of a video (eg. "vimeo/499921561"). */ +export type StringVideoAddress = string; + +/** A React reference to the MediaPlayer instance. */ +export type VideoPlayerRef = RefObject<MediaPlayerInstance>; + +/** Structure representing a jump-to ("seek") location */ +export type VideoPlayerJumpTo = { second: t.Secs; play: boolean }; + +/** + * Component: Video Player. + */ +export type VideoPlayerProps = { + debug?: boolean; + title?: string; + style?: t.CssInput; + + // Events: + onPlay?: MediaPlayerProps['onPlay']; + onPlaying?: MediaPlayerProps['onPlaying']; + onPause?: MediaPlayerProps['onPause']; + onEnded?: MediaPlayerProps['onEnded']; + + // State: + signals?: t.VideoPlayerSignals; +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Video/ui.tsx b/code/sys.ui/ui-react-components/src/ui/Player.Video/ui.tsx new file mode 100644 index 0000000000..e7d717c334 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Video/ui.tsx @@ -0,0 +1,168 @@ +import type { MediaPlayerInstance } from '@vidstack/react'; +import { MediaPlayer, MediaProvider } from '@vidstack/react'; +import { PlyrLayout, plyrLayoutIcons } from '@vidstack/react/player/layouts/plyr'; +import React, { useEffect, useRef, useState } from 'react'; + +import { type t, css, DEFAULTS, Signal, Style, useSizeObserver } from './common.ts'; +import { useSignalBinding } from './use.SignalBinding.ts'; +import { useThemeStyles } from './use.ThemeStyles.ts'; + +const D = DEFAULTS; + +/** + * Component: + */ +export const VideoPlayer: React.FC<t.VideoPlayerProps> = (props) => { + const { signals } = props; + const p = signals?.props; + + const src = p?.src.value ?? D.video; + const showControls = p?.showControls.value ?? D.showControls; + const showFullscreenButton = p?.showFullscreenButton.value ?? D.showFullscreenButton; + const showVolumeControl = p?.showVolumeControl?.value ?? D.showVolumeControl; + const autoPlay = p?.autoPlay.value ?? D.autoPlay; + const muted = autoPlay ? true : p?.muted.value ?? D.muted; + const aspectRatio = p?.aspectRatio.value ?? D.aspectRatio; + const cornerRadius = p?.cornerRadius.value ?? D.cornerRadius; + const scale = p?.scale.value ?? D.scale; + const loop = p?.loop.value ?? D.loop; + + const size = useSizeObserver(); + const [calcScale, setCalcScale] = useState<number>(); + + const [playerKey, setPlayerKey] = useState(0); + const playerRef = useRef<MediaPlayerInstance>(null); + useSignalBinding({ signals, playerRef }); + + /** + * Effect: ensure redraw on signal changes. + */ + Signal.useRedrawEffect(() => { + if (!p) return; + p.ready.value; + p.src.value; + + p.muted.value; + p.autoPlay.value; + p.loop.value; + + p.showControls.value; + p.showFullscreenButton.value; + p.showVolumeControl.value; + p.cornerRadius.value; + p.aspectRatio.value; + p.scale.value; + }); + + /** + * Effect: monitor size differene + */ + useEffect(() => { + const { width, height } = size; + const fn = p?.scale.value; + if (width === undefined || height === undefined) return; + if (typeof fn !== 'function') { + setCalcScale(undefined); + } else { + const enlargeBy = (increment: t.Pixels) => wrangle.scale(width, height, increment); + setCalcScale(fn({ width, height, enlargeBy })); + } + }, [size.width, size.height, scale]); + + /** + * Effect (HACK): ensure player style-sheets work consistently when bundled and deployed. + */ + useEffect(() => { + const sheet = Style.Dom.stylesheet(); + sheet.rule('[data-media-provider]', { width: '100%', height: '100%' }); + sheet.rule('[data-media-provider] iframe', { + width: '100%', + height: '100%', + objectFit: 'cover', + objectPosition: 'center', + }); + }, []); + + /** + * Render: + */ + const themeStyles = useThemeStyles('Plyr'); + const isReady = Boolean(themeStyles.loaded && !!p?.ready.value && size.ready); + const styles = { + base: css({ + overflow: 'hidden', + display: 'grid', + visibility: isReady ? 'visible' : 'hidden', // NB: avoid a FOUC ("Flash Of Unstyled Content"). + lineHeight: 0, // NB: ensure no "baseline" gap below the <MediaPlayer>. + }), + }; + + const elPlyrLayout = showControls && ( + <PlyrLayout + // thumbnails="https://files.vidstack.io/sprite-fight/thumbnails.vtt" + icons={plyrLayoutIcons} + slots={{ + settings: false, + fullscreenButton: showFullscreenButton ? undefined : false, + volumeSlider: showVolumeControl ? undefined : false, + }} + /> + ); + + const elPlayer = ( + <MediaPlayer + key={playerKey} + ref={playerRef} + style={{ + transform: `scale(${calcScale ?? scale ?? 1})`, + '--plyr-border-radius': `${cornerRadius}px`, + '--plyr-aspect-ratio': aspectRatio, // e.g. '4/3', '2.39/1', '1/1', etc... + }} + /** + * Props: + */ + title={props.title} + src={src} + playsInline={true} + aspectRatio={aspectRatio} + autoPlay={autoPlay} + muted={muted} + loop={loop} + /** + * Handlers: + */ + onPlay={props.onPlay} + onPlaying={props.onPlaying} + onPause={props.onPause} + onCanPlay={() => { + if (p) p.ready.value = true; + }} + onEnded={(e) => { + // Hack: force the Player back to the first-frame. + if (!loop) setPlayerKey((n) => n + 1); + props.onEnded?.(e); + }} + > + <MediaProvider /> + {elPlyrLayout} + </MediaPlayer> + ); + + return ( + <div ref={size.ref} className={css(styles.base, props.style).class}> + {elPlayer} + </div> + ); +}; + +/** + * Helpers + */ +const wrangle = { + scale(width: t.Pixels, height: t.Pixels, increment: t.Pixels) { + if (width === 0 || height === 0) return 1; + const scaleX = (width + increment) / width; + const scaleY = (height + increment) / height; + return Math.max(scaleX, scaleY); // NB: Return the greater scale factor to ensure both dimensions are increased by at least increment. + }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Video/use.SignalBinding.ts b/code/sys.ui/ui-react-components/src/ui/Player.Video/use.SignalBinding.ts new file mode 100644 index 0000000000..e03bb027b4 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Video/use.SignalBinding.ts @@ -0,0 +1,64 @@ +import { type MediaPlayerInstance, useMediaState } from '@vidstack/react'; +import React from 'react'; +import { type t, Signal } from './common.ts'; + +/** + * Manages keeping the <VideoPlayer> component in sync with + * the current state of the Signals API. + */ +export function useSignalBinding(args: { + playerRef: React.RefObject<MediaPlayerInstance>; + signals?: t.VideoPlayerSignals; +}) { + const { playerRef, signals } = args; + const props = signals?.props; + const player = playerRef.current || undefined; + + const currentTime = useMediaState('currentTime', playerRef); + const isPlaying = useMediaState('playing', playerRef); + + /** + * Effect: Keep the signal updated with the current "is-playing" state. + */ + React.useEffect(() => { + if (!props) return; + if (props.playing.value !== isPlaying) props.playing.value = isPlaying; + }, [isPlaying]); + + /** + * Effect: Sync the signals with the <Player>'s current state. + */ + React.useEffect(() => { + if (!props) return; + if (props.currentTime.value !== currentTime) props.currentTime.value = currentTime; + }, [currentTime, props]); + + /** + * Handle: jumpTo (aka. "seek"). + */ + Signal.useEffect(() => { + props?.ready.value; + const jumpTo = props?.jumpTo.value; + + if (player && props) { + if (typeof jumpTo?.second === 'number') { + player.currentTime = jumpTo.second; + if (jumpTo.play) player.play(); + if (!jumpTo.play) player.pause(); + props.jumpTo.value = undefined; // NB: reset after for next call. + } + } + }); + + /** + * Handle: play/pause. + */ + Signal.useEffect(() => { + props?.ready.value; + const isPlaying = props?.playing.value ?? false; + if (player) { + if (isPlaying && player.paused) player.play(); + if (!isPlaying && !player.paused) player.pause(); + } + }); +} diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Video/use.ThemeStyles.plyr.ts b/code/sys.ui/ui-react-components/src/ui/Player.Video/use.ThemeStyles.plyr.ts new file mode 100644 index 0000000000..fc677cd720 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Video/use.ThemeStyles.plyr.ts @@ -0,0 +1,2 @@ +import '@vidstack/react/player/styles/base.css'; +import '@vidstack/react/player/styles/plyr/theme.css'; diff --git a/code/sys.ui/ui-react-components/src/ui/Player.Video/use.ThemeStyles.ts b/code/sys.ui/ui-react-components/src/ui/Player.Video/use.ThemeStyles.ts new file mode 100644 index 0000000000..eda62f5cff --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player.Video/use.ThemeStyles.ts @@ -0,0 +1,45 @@ +import React from 'react'; +import { Is } from './common.ts'; + +export type ThemeKind = 'Plyr'; + +/** + * NOTE: Conditionally load styles only when running within browser. + * + * Why? Prevents throwing an import error on the server (Deno) + * which occurs when running unit-tests. + * + * 𫵠CONSIDER: + * Handling the .css imports within the Vite bundle (?) + * which would prevent a FOUC ("Flash Of Unstyled Content") + * that does not happen, rather the component just renders + * nothing until the `Styles.loaded` is [true]. + */ +export function useThemeStyles(kind: ThemeKind) { + const [loaded, setLoaded] = React.useState(false); + + React.useEffect(() => { + if (!Is.browser()) { + setLoaded(true); + return; + } + + let max = 0; + let count = 0; + const onLoaded = () => { + count++; + if (count >= max) setLoaded(true); + }; + + if (kind === 'Plyr') { + max = 1; + import('./use.ThemeStyles.plyr.ts').then(onLoaded); + return; + } + + throw new Error(`Theme "${kind}" not supported.`); + }, []); + + // API. + return { loaded } as const; +} diff --git a/code/sys.ui/ui-react-components/src/ui/Player/-.test.ts b/code/sys.ui/ui-react-components/src/ui/Player/-.test.ts new file mode 100644 index 0000000000..3abf8c20f5 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player/-.test.ts @@ -0,0 +1,12 @@ +import { describe, expect, it } from '../../-test.ts'; +import { ConceptPlayer, VideoPlayer } from '../../mod.ts'; +import { Thumbnails } from '../Player.Thumbnails/mod.ts'; +import { Player } from './mod.ts'; + +describe('Player', () => { + it('API', () => { + expect(Player.Concept.View).to.equal(ConceptPlayer); + expect(Player.Video.View).to.equal(VideoPlayer); + expect(Player.Timestamp.Thumbnails.View).to.equal(Thumbnails); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/Player/common.ts b/code/sys.ui/ui-react-components/src/ui/Player/common.ts new file mode 100644 index 0000000000..57e74207ff --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player/common.ts @@ -0,0 +1,3 @@ +export const PlayerColors = { + BLUE: '#4BA3E7', +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Player/m.Player.ts b/code/sys.ui/ui-react-components/src/ui/Player/m.Player.ts new file mode 100644 index 0000000000..4b4cf7b176 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player/m.Player.ts @@ -0,0 +1,18 @@ +import { type t } from '../common.ts'; + +import { ConceptPlayer } from '../Player.Concept/mod.ts'; +import { Thumbnails } from '../Player.Thumbnails/mod.ts'; +import { VideoPlayer, playerSignalsFactory } from '../Player.Video/mod.ts'; + +export const Player: t.PlayerLib = { + Concept: { + View: ConceptPlayer, + }, + Video: { + View: VideoPlayer, + signals: playerSignalsFactory, + }, + Timestamp: { + Thumbnails: { View: Thumbnails }, + }, +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Player/mod.ts b/code/sys.ui/ui-react-components/src/ui/Player/mod.ts new file mode 100644 index 0000000000..a65930b1fc --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player/mod.ts @@ -0,0 +1,8 @@ +/** + * @module + * "Player" namespace + * - Video + * - Concept + * - Timestamps / Thumbnails + */ +export { Player } from './m.Player.ts'; diff --git a/code/sys.ui/ui-react-components/src/ui/Player/t.ts b/code/sys.ui/ui-react-components/src/ui/Player/t.ts new file mode 100644 index 0000000000..1b2dcc6484 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Player/t.ts @@ -0,0 +1,18 @@ +import type React from 'react'; +import type { t } from '../common.ts'; + +/** + * Library: Players + */ +export type PlayerLib = { + Concept: { + View: React.FC<t.ConceptPlayerProps>; + }; + Video: { + View: React.FC<t.VideoPlayerProps>; + signals: t.PlayerSignalsFactory; + }; + Timestamp: { + Thumbnails: { View: React.FC<t.ThumbnailsProps> }; + }; +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Preload/-SPEC.Debug.tsx b/code/sys.ui/ui-react-components/src/ui/Preload/-SPEC.Debug.tsx new file mode 100644 index 0000000000..3d3b10bf22 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Preload/-SPEC.Debug.tsx @@ -0,0 +1,81 @@ +import React from 'react'; +import { Button } from '../Button/mod.ts'; + +import { type t, css, Signal } from './common.ts'; +import { Preload } from './mod.ts'; + +type O = t.PreloadOptions; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + const s = Signal.create; + const props = { + disposeDelay: s<O['disposeDelay']>(500), + }; + const api = { + props, + listen() { + const p = props; + p.disposeDelay.value; + }, + }; + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => debug.listen()); + + /** + * Render: + */ + const styles = { + base: css({}), + title: css({ fontWeight: 'bold', marginBottom: 10 }), + cols: css({ display: 'grid', gridTemplateColumns: 'auto 1fr auto' }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={css(styles.title, styles.cols).class}> + <div>{'PreloadPortal'}</div> + </div> + + <Button + block + label={() => { + const msecs = p.disposeDelay.value; + return `disposeAfter: ${msecs !== undefined ? `${msecs}ms` : '<undefined>'}`; + }} + onClick={() => Signal.cycle<O['disposeDelay']>(p.disposeDelay, [undefined, 0, 500, 2000])} + /> + + <hr /> + + <Button + block + label={() => 'Preload.render'} + onClick={async () => { + const disposeAfter = p.disposeDelay.value; + const el = <div>Hello</div>; + const res = await Preload.render(el, { disposeDelay: disposeAfter }); + console.info(res); + }} + /> + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Preload/-SPEC.tsx b/code/sys.ui/ui-react-components/src/ui/Preload/-SPEC.tsx new file mode 100644 index 0000000000..2cbb775c3a --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Preload/-SPEC.tsx @@ -0,0 +1,23 @@ +import { Signal, Spec } from '../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; + +export default Spec.describe('PreloadPortal', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + Signal.effect(() => { + debug.listen(); + ctx.redraw(); + }); + + ctx.subject.size().display('grid'); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/Preload/common.ts b/code/sys.ui/ui-react-components/src/ui/Preload/common.ts new file mode 100644 index 0000000000..a45006948c --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Preload/common.ts @@ -0,0 +1,7 @@ +export * from '../common.ts'; + +/** + * Constants: + */ +export const DEFAULTS = {} as const; +export const D = DEFAULTS; diff --git a/code/sys.ui/ui-react-components/src/ui/Preload/m.Preload.ts b/code/sys.ui/ui-react-components/src/ui/Preload/m.Preload.ts new file mode 100644 index 0000000000..55a602e148 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Preload/m.Preload.ts @@ -0,0 +1,8 @@ +import type { t } from './common.ts'; +import { render } from './u.render.tsx'; +import { PreloadPortal as Portal } from './ui.tsx'; + +export const Preload: t.PreloadLib = { + Portal, + render, +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Preload/mod.ts b/code/sys.ui/ui-react-components/src/ui/Preload/mod.ts new file mode 100644 index 0000000000..ec4100c734 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Preload/mod.ts @@ -0,0 +1,4 @@ +/** + * @module + */ +export { Preload } from './m.Preload.ts'; diff --git a/code/sys.ui/ui-react-components/src/ui/Preload/t.ts b/code/sys.ui/ui-react-components/src/ui/Preload/t.ts new file mode 100644 index 0000000000..d3124a5d2c --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Preload/t.ts @@ -0,0 +1,30 @@ +import type { t } from './common.ts'; + +/** + * Preload rendering helpers. + */ +export type PreloadLib = { + render: Preload; + Portal: React.FC<PreloadPortalProps>; +}; + +/** + * <Component>: + * A portal container used for pre-loading component content off screen. + */ +export type PreloadPortalProps = { + children: React.ReactNode; + size?: t.SizeTuple; +}; + +/** + * Preload function that performs all DOM insertion and clean up. + */ +export type Preload = ( + children: React.ReactNode, + options?: t.PreloadOptions | t.Msecs, +) => Promise<t.Lifecycle>; +export type PreloadOptions = { + disposeDelay?: t.Msecs; + size?: t.SizeTuple; +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Preload/u.render.tsx b/code/sys.ui/ui-react-components/src/ui/Preload/u.render.tsx new file mode 100644 index 0000000000..52a7d0e3ba --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Preload/u.render.tsx @@ -0,0 +1,42 @@ +import { createRoot } from 'react-dom/client'; + +import { type t, Time, rx } from './common.ts'; +import { PreloadPortal } from './ui.tsx'; + +export const render: t.Preload = async (children, op) => { + const life = rx.lifecycle(); + const { size, disposeDelay } = wrangle.options(op); + + if (typeof document === 'undefined') { + life.dispose(); + return life; + } + + life.dispose$.subscribe(() => { + root.unmount(); + document.body.removeChild(div); + }); + + /** + * Render Portal: + */ + const div = document.createElement('div'); + document.body.appendChild(div); + const root = createRoot(div); + root.render(<PreloadPortal size={size}>{children}</PreloadPortal>); + + // Finish up. + if (typeof disposeDelay === 'number') await Time.delay(disposeDelay, life.dispose); + return life; +}; + +/** + * Helpers + */ +const wrangle = { + options(input?: t.PreloadOptions | t.Msecs): t.PreloadOptions { + if (!input) return {}; + if (typeof input === 'number') return { disposeDelay: input }; + return input; + }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Preload/ui.tsx b/code/sys.ui/ui-react-components/src/ui/Preload/ui.tsx new file mode 100644 index 0000000000..a0d00e8709 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Preload/ui.tsx @@ -0,0 +1,45 @@ +import React, { useEffect, useState } from 'react'; +import { createPortal } from 'react-dom'; +import { type t } from './common.ts'; + +const OFFSCREEN = '-99999px'; + +export const PreloadPortal: React.FC<t.PreloadPortalProps> = (props) => { + const { size = [0, 0] } = props; + const [container, setContainer] = useState<HTMLDivElement | null>(null); + + /** + * Effect: + */ + useEffect(() => { + const el = document.createElement('div'); + el.setAttribute('data-component', 'sys.preload'); + + el.style.position = 'absolute'; + el.style.left = OFFSCREEN; + el.style.top = OFFSCREEN; + el.style.width = `${size[0]}px`; + el.style.height = `${size[1]}px`; + el.style.overflow = 'hidden'; + el.style.opacity = '0'; + + document.body.appendChild(el); + setContainer(el); + + const dispose = () => { + if (el.parentElement) { + document.body.removeChild(el); + setContainer(null); + } + }; + + // Finish up. + return dispose; + }, [size.join()]); + + /** + * Render: + */ + if (!container) return null; + return createPortal(props.children, container); +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Sheet/-.test.ts b/code/sys.ui/ui-react-components/src/ui/Sheet/-.test.ts new file mode 100644 index 0000000000..a48fa45e04 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Sheet/-.test.ts @@ -0,0 +1,186 @@ +import { describe, expect, it, Signal } from '../../-test.ts'; +// import { VIDEO } from '../App.Content/mod.ts'; +import { Sheet } from './mod.ts'; + +type T = { id: string }; + +describe('Sheet', () => { + describe('Sheet.Signals:stack', () => { + const a: T = { id: 'a' }; + const b: T = { id: 'b' }; + const c: T = { id: 'c' }; + + describe('factory', () => { + it('passed in signal (param)', () => { + const signal = Signal.create<T[]>([]); + const state = Sheet.Signals.stack(signal); + + expect(state.length).to.eql(0); + expect(state.items).to.eql([]); + expect(state.items).to.not.equal(signal.value); // NB: cloned array - protect from mutation. + expect(state.toSignal()).to.equal(signal); + + signal.value = [{ id: 'foo' }]; + expect(state.items).to.eql([{ id: 'foo' }]); + }); + + it('generate signal (no param)', () => { + const state = Sheet.Signals.stack(); + + expect(state.length).to.eql(0); + state.toSignal().value = [{ id: 'foo' }]; + expect(state.items).to.eql([{ id: 'foo' }]); + }); + }); + + describe('method: push', () => { + it('push (1) - triggers signal effect', () => { + const signal = Signal.create<T[]>([]); + const stack = Sheet.Signals.stack(signal); + const fired: number[] = []; + Signal.effect(() => { + fired.push(signal.value.length); + }); + + expect(stack.length).to.eql(0); + stack.push(a); + expect(stack.length).to.eql(1); + expect(signal.value).to.eql([a]); + expect(fired).to.eql([0, 1]); + + stack.push(b); + expect(stack.length).to.eql(2); + expect(signal.value).to.eql([a, b]); + expect(fired).to.eql([0, 1, 2]); + }); + + it('push (many)', () => { + const signal = Signal.create<T[]>([]); + const stack = Sheet.Signals.stack(signal); + const fired: number[] = []; + Signal.effect(() => { + fired.push(signal.value.length); + }); + + stack.push(a); + expect(stack.length).to.eql(1); + expect(fired).to.eql([0, 1]); + + stack.push(b, c); + expect(stack.length).to.eql(3); + expect(signal.value).to.eql([a, b, c]); + expect(fired).to.eql([0, 1, 3]); + }); + + it('push <undefined>', () => { + const signal = Signal.create<T[]>([]); + const stack = Sheet.Signals.stack(signal); + stack.push(); + stack.push(undefined, a, undefined, b); + expect(stack.length).to.eql(2); + expect(signal.value).to.eql([a, b]); + }); + }); + + describe('method: clear', () => { + it('clear all', () => { + const signal = Signal.create<T[]>([]); + const stack = Sheet.Signals.stack(signal); + const fired: number[] = []; + Signal.effect(() => { + fired.push(signal.value.length); + }); + + stack.push(a, b); + expect(signal.value).to.eql([a, b]); + expect(fired).to.eql([0, 2]); + expect(stack.length).to.eql(2); + + stack.clear(); + expect(fired).to.eql([0, 2, 0]); + expect(stack.length).to.eql(0); + }); + + it('clear( leave )', () => { + const test = (leave: number) => { + const signal = Signal.create<T[]>([]); + const stack = Sheet.Signals.stack(signal); + stack.push(a, b, c); + expect(stack.length).to.eql(3); + + stack.clear(leave); + expect(stack.length).to.eql(leave); + }; + test(0); + test(1); + test(2); + }); + }); + + it('method: exists', () => { + const signal = Signal.create<T[]>([]); + const stack = Sheet.Signals.stack(signal); + stack.push(a, b); + + expect(stack.exists((m) => m.id === a.id)).to.eql(true); + expect(stack.exists((m) => m.id === b.id)).to.eql(true); + expect(stack.exists((m) => m.id === c.id)).to.eql(false); + }); + + describe('method: pop', () => { + it('pop: removes top most layer', () => { + const signal = Signal.create<T[]>([]); + const stack = Sheet.Signals.stack(signal); + const fired: number[] = []; + Signal.effect(() => { + fired.push(signal.value.length); + }); + + stack.push(a, b, c); + expect(fired).to.eql([0, 3]); + + stack.pop(); + expect(fired).to.eql([0, 3, 2]); + expect(signal.value).to.eql([a, b]); + + stack.pop(); + stack.pop(); + expect(fired).to.eql([0, 3, 2, 1, 0]); + expect(signal.value).to.eql([]); + + stack.pop(); // NB: ā already empty at this point. + expect(signal.value).to.eql([]); + + expect(fired.length).to.eql(5); + stack.pop(); + stack.pop(); + expect(fired.length).to.eql(5); // NB: no change (already empty). + }); + + it('pop( leave ) ā retain minimum level', () => { + const signal = Signal.create<T[]>([]); + const stack = Sheet.Signals.stack(signal); + stack.push(a, b, c); + expect(stack.length).to.eql(3); + + stack.pop(3); + stack.pop(3); + expect(stack.length).to.eql(3); + + stack.pop(); + expect(stack.length).to.eql(2); + + stack.pop(1); + stack.pop(1); + stack.pop(1); + expect(stack.length).to.eql(1); + + stack.pop(-1); + expect(stack.length).to.eql(0); + stack.pop(-1); + stack.pop(-1); + expect(stack.length).to.eql(0); + }); + }); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/Sheet/-SPEC.Debug.tsx b/code/sys.ui/ui-react-components/src/ui/Sheet/-SPEC.Debug.tsx new file mode 100644 index 0000000000..5c6e6c810f --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Sheet/-SPEC.Debug.tsx @@ -0,0 +1,89 @@ +import React from 'react'; +import { Button } from '../Button/mod.ts'; +import { type t, css, DEFAULTS, Signal } from './common.ts'; + +type P = t.SheetProps; +const D = DEFAULTS; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + const s = Signal.create; + + const props = { + showing: s(true), + theme: s<P['theme']>('Dark'), + orientation: s<P['orientation']>(D.orientation.default), + edgeMargin: s<P['edgeMargin']>(0), + }; + const api = { + props, + listen() { + const p = props; + p.theme.value; + p.showing.value; + p.orientation.value; + p.edgeMargin.value; + }, + }; + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const d = debug.props; + + Signal.useRedrawEffect(() => debug.listen()); + + /** + * Render: + */ + const styles = { + base: css({}), + title: css({ fontWeight: 'bold', marginBottom: 10 }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={styles.title.class}>{'Mobile Sheet'}</div> + <Button + block + label={`debug.showing: ${d.showing}`} + onClick={() => Signal.toggle(d.showing)} + /> + <hr /> + <Button + block + label={`theme: ${d.theme}`} + onClick={() => Signal.cycle<P['theme']>(d.theme, ['Light', 'Dark'])} + /> + <Button + block + label={`orientation: ${d.orientation.value ?? '<undefined>'}`} + onClick={() => { + Signal.cycle<P['orientation']>(d.orientation, [...D.orientation.all, undefined]); + }} + /> + <Button + block + label={`edgeMargin: ${d.edgeMargin.value ?? '<undefined>'}`} + onClick={() => { + type T = t.SheetMarginInput; + const values: (T | undefined)[] = [undefined, 0, 10, [50, 15], ['1fr', 200, '2em']]; + Signal.cycle<P['edgeMargin']>(d.edgeMargin, values); + }} + /> + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Sheet/-SPEC.tsx b/code/sys.ui/ui-react-components/src/ui/Sheet/-SPEC.tsx new file mode 100644 index 0000000000..5a5d641fe8 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Sheet/-SPEC.tsx @@ -0,0 +1,60 @@ +import { Dev, Signal, Spec } from '../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { AnimatePresence, Color, css } from './common.ts'; +import { Sheet } from './mod.ts'; + +export default Spec.describe('Sheet', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + Dev.Theme.signalEffect(ctx, p.theme, 1); + Signal.effect(() => { + debug.listen(); + ctx.redraw(); + }); + + ctx.subject + .size([390, 844]) + .display('grid') + + .render((e) => { + const orientation = p.orientation.value; + const isShowing = p.showing.value; + const styles = { + base: css({ overflow: 'hidden', display: 'grid' }), + sheet: css({ padding: 15, userSelect: 'none' }), + dim: css({ opacity: 0.3 }), + }; + + const elSheet = isShowing && ( + <Sheet.View + theme={Color.Theme.invert(p.theme.value)} + orientation={orientation} + edgeMargin={p.edgeMargin.value} + onMouseDown={(e) => { + e.stopPropagation(); + p.showing.value = false; + }} + > + <div className={styles.sheet.class}> + {'š MySheet'} ā <span className={styles.dim.class}>{'(click to hide)'}</span> + </div> + </Sheet.View> + ); + + return ( + <div className={styles.base.class} onMouseDown={() => (p.showing.value = true)}> + <AnimatePresence>{elSheet}</AnimatePresence> + </div> + ); + }); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/Sheet/common.ts b/code/sys.ui/ui-react-components/src/ui/Sheet/common.ts new file mode 100644 index 0000000000..4df62bf8a2 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Sheet/common.ts @@ -0,0 +1,20 @@ +import { type t } from './common.ts'; +export * from '../common.ts'; + +/** + * Constants: + */ + +export const DEFAULTS = { + radius: 4, + duration: 0.25, + bounce: 0.2, + shadowColor: -0.15, + + get orientation(): { all: t.SheetOrientation[]; default: t.SheetOrientation } { + return { + default: 'Bottom:Up', + all: ['Bottom:Up', 'Top:Down', 'Left:Right', 'Right:Left'], + }; + }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Sheet/m.Signals.Stack.ts b/code/sys.ui/ui-react-components/src/ui/Sheet/m.Signals.Stack.ts new file mode 100644 index 0000000000..3f5a3fbf70 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Sheet/m.Signals.Stack.ts @@ -0,0 +1,39 @@ +import { type t, Signal } from './common.ts'; + +/** + * Wrap the given signal with the [Stack] API. + */ +export const createStack: t.SheetSignalsLib['stack'] = <T>(input?: t.Signal<T[]>) => { + const signal = input ?? Signal.create<T[]>([]); + + const api: t.SheetSignalStack<T> = { + get length() { + return signal.value.length; + }, + get items() { + return [...signal.value]; + }, + + push(...content) { + const next = [...signal.value, ...content].filter(Boolean) as T[]; + signal.value = next; + }, + pop(leave = 0) { + const stack = signal; + if (stack.value.length > leave) stack.value = stack.value.slice(0, -1); + }, + clear(leave = 0) { + signal.value = signal.value.slice(0, leave); + }, + + exists(fn) { + return api.items.some(fn); + }, + + toSignal() { + return signal; + }, + }; + + return api; +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Sheet/m.Signals.ts b/code/sys.ui/ui-react-components/src/ui/Sheet/m.Signals.ts new file mode 100644 index 0000000000..e140971d1b --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Sheet/m.Signals.ts @@ -0,0 +1,9 @@ +import { type t } from './common.ts'; +import { createStack } from './m.Signals.Stack.ts'; + +/** + * Library: Sheet Signals (State). + */ +export const Signals: t.SheetSignalsLib = { + stack: createStack, +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Sheet/mod.ts b/code/sys.ui/ui-react-components/src/ui/Sheet/mod.ts new file mode 100644 index 0000000000..c84d6ac66e --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Sheet/mod.ts @@ -0,0 +1,12 @@ +/** + * @module + * An animated sliding "sheet" content container. + */ +import type { t } from './common.ts'; +import { Sheet as View } from './ui.tsx'; +import { Signals } from './m.Signals.ts'; + +export const Sheet: t.SheetLib = { + View, + Signals, +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Sheet/t.Signals.ts b/code/sys.ui/ui-react-components/src/ui/Sheet/t.Signals.ts new file mode 100644 index 0000000000..22719779f0 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Sheet/t.Signals.ts @@ -0,0 +1,35 @@ +import type { t } from './common.ts'; + +/** + * Library: Sheet Signals (State). + */ +export type SheetSignalsLib = { + /** Factory: create a new stack API. */ + stack<T>(signal?: t.Signal<T[]>): t.SheetSignalStack<T>; +}; + +/** + * API for managing the stack of sheets. + */ +export type SheetSignalStack<T> = { + /** The number of items in the stack. */ + readonly length: number; + + /** The list of screens. */ + readonly items: T[]; + + /** Determine if the item exists within the stack. */ + exists(fn: (e: T) => boolean): boolean; + + /** Add a screen to the top of the stack. */ + push(...content: (T | undefined)[]): void; + + /** Remove the highest screen from the stack. */ + pop(leave?: number): void; + + /** Removes all screens from the stack. */ + clear(leave?: number): void; + + /** Extract the underlying signal. */ + toSignal(): t.Signal<T[]>; +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Sheet/t.ts b/code/sys.ui/ui-react-components/src/ui/Sheet/t.ts new file mode 100644 index 0000000000..ef57ad5d3c --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Sheet/t.ts @@ -0,0 +1,58 @@ +import type React from 'react'; +import type { t } from './common.ts'; + +export type * from './t.Signals.ts'; +type M = React.MouseEventHandler<HTMLDivElement>; + +/** + * Library: Sheet + */ +export type SheetLib = { + /** The UI component of the sheet. */ + readonly View: React.FC<SheetProps>; + + /** Library: Sheet Signals (State) */ + readonly Signals: t.SheetSignalsLib; +}; + +/** + * The orientation and slide direction of the sheet. + */ +export type SheetOrientation = SheetOrientationX | SheetOrientationY; +export type SheetOrientationX = 'Left:Right' | 'Right:Left'; +export type SheetOrientationY = 'Bottom:Up' | 'Top:Down'; + +/** + * <Component>: + */ +export type SheetProps = { + children?: t.ReactNode; + radius?: t.Pixels; + + orientation?: t.SheetOrientation; + duration?: t.Secs; + bounce?: number; + + edgeMargin?: t.SheetMarginInput; + shadowOpacity?: t.Percent; + pointerEvents?: t.CssProps['pointerEvents']; + theme?: t.CommonTheme; + style?: t.CssInput; + + onClick?: M; + onDoubleClick?: M; + onMouseDown?: M; + onMouseUp?: M; + onMouseEnter?: M; + onMouseLeave?: M; +}; + +/** + * Margins for the sheet edges based on orientation. + * A pixel value or a CSS-grid template value (eg. 'auto' or '1fr' etc). + */ +export type SheetMargin = t.Pixels | string; +export type SheetMarginInput = + | [SheetMargin, SheetMargin, SheetMargin] // ā near | middle | far + | [SheetMargin, SheetMargin] // ā near | (default: 1fr) | far + | SheetMargin; // ā near | (default: 1fr) | far diff --git a/code/sys.ui/ui-react-components/src/ui/Sheet/ui.tsx b/code/sys.ui/ui-react-components/src/ui/Sheet/ui.tsx new file mode 100644 index 0000000000..89448b6844 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Sheet/ui.tsx @@ -0,0 +1,177 @@ +import React from 'react'; +import { type t, Color, css, DEFAULTS as D, M } from './common.ts'; + +type P = t.SheetProps; + +export const Sheet: React.FC<P> = (props) => { + const { duration = D.duration, bounce = D.bounce, pointerEvents = 'auto' } = props; + const is = wrangle.is(props); + const animation = wrangle.animation(props); + const { borderRadius, boxShadow, gridTemplateColumns, gridTemplateRows } = wrangle.styles(props); + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const backgroundColor = theme.bg; + const styles = { + base: css({ position: 'relative', display: 'grid', gridTemplateColumns, gridTemplateRows }), + spacer: css({ pointerEvents: 'none' }), + body: css({ + position: 'relative', + pointerEvents, + color: theme.fg, + backgroundColor, + borderRadius, + boxShadow, + display: 'grid', + }), + mask: css({ + // NB: Extends the sheet to ensure the physics bounce does not show a flash. + // Ensure this component is within a container with { overflow: 'hidden' }. + ...wrangle.maskOrientation(props), + backgroundColor, + }), + }; + + const elBody = ( + <div className={styles.body.class}> + <div className={styles.mask.class} /> + {props.children} + </div> + ); + + return ( + <M.div + data-component={`sys.ui.sheet`} + className={css(styles.base, props.style).class} + /** + * Animation: + */ + initial={animation.initial} + animate={is.vertical ? { y: '0%' } : { x: '0%' }} + exit={animation.exit} + transition={{ duration, type: 'spring', bounce }} + /** + * Handlers: + */ + onClick={props.onClick} + onDoubleClick={props.onDoubleClick} + onMouseDown={props.onMouseDown} + onMouseUp={props.onMouseUp} + onMouseEnter={props.onMouseEnter} + onMouseLeave={props.onMouseLeave} + > + <div className={styles.spacer.class} /> + {elBody} + <div className={styles.spacer.class} /> + </M.div> + ); +}; + +/** + * Helpers + */ +const wrangle = { + is(props: P) { + const { orientation = D.orientation.default } = props; + return { + vertical: orientation === 'Top:Down' || orientation === 'Bottom:Up', + topDown: orientation === 'Top:Down', + leftToRight: orientation === 'Left:Right', + } as const; + }, + + styles(props: P) { + const { radius = D.radius } = props; + const is = wrangle.is(props); + const shadowColor = Color.format(props.shadowOpacity ?? D.shadowColor); + const edgeMargin = wrangle.edgeMarginTemplate(props); + + let borderRadius: string; + let boxShadow: string; + let gridTemplateColumns: string | undefined; + let gridTemplateRows: string | undefined; + + if (is.vertical) { + gridTemplateColumns = edgeMargin; + if (is.topDown) { + borderRadius = `0 0 ${radius}px ${radius}px`; + boxShadow = `0 5px 6px 0 ${shadowColor}`; + } else { + borderRadius = `${radius}px ${radius}px 0 0`; + boxShadow = `0 -5px 6px 0 ${shadowColor}`; + } + } else { + gridTemplateRows = edgeMargin; + if (is.leftToRight) { + borderRadius = `0 ${radius}px ${radius}px 0`; + boxShadow = `5px 0 6px 0 ${shadowColor}`; + } else { + borderRadius = `${radius}px 0 0 ${radius}px`; + boxShadow = `-5px 0 6px 0 ${shadowColor}`; + } + } + + return { borderRadius, boxShadow, gridTemplateColumns, gridTemplateRows } as const; + }, + + edgeMargin(props: P): [t.SheetMargin, t.SheetMargin, t.SheetMargin] { + const v = props.edgeMargin; + if (typeof v === 'number') return [v, '1fr', v]; + if (Array.isArray(v)) { + if (v.length === 2) return [v[0], '1fr', v[1]]; + if (v.length === 3) return v; + } + return [0, '1fr', 0]; + }, + + edgeMarginTemplate(props: P): string { + const margin = wrangle.edgeMargin(props); + return margin.map((v) => (typeof v === 'number' ? `${v}px` : v)).join(' '); + }, + + maskOrientation(props: P): t.CssValue { + const is = wrangle.is(props); + const buffer = 30; + if (is.vertical) { + return { + Absolute: is.topDown ? [-buffer, 0, null, 0] : [null, 0, -buffer, 0], + height: buffer, + }; + } else { + return { + Absolute: is.leftToRight ? [0, null, 0, -buffer] : [0, -buffer, 0, null], + width: buffer, + }; + } + }, + + animation(props: P) { + const is = wrangle.is(props); + const buffer = '10px'; + let initial: Record<string, string>, exit: Record<string, string>; + if (is.vertical) { + if (is.topDown) { + // 'Top:Down' + initial = { y: `calc(-100% - ${buffer})` }; + exit = { y: `calc(-100% - ${buffer})` }; + } else { + // 'Bottom:Up' + initial = { y: `calc(100% + ${buffer})` }; + exit = { y: `calc(100% + ${buffer})` }; + } + } else { + if (is.leftToRight) { + // 'Left:Right' + initial = { x: `calc(-100% - ${buffer})` }; + exit = { x: `calc(-100% - ${buffer})` }; + } else { + // 'Right:Left' + initial = { x: `calc(100% + ${buffer})` }; + exit = { x: `calc(100% + ${buffer})` }; + } + } + return { initial, exit } as const; + }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/Spinners.Bar/-SPEC.Debug.tsx b/code/sys.ui/ui-react-components/src/ui/Spinners.Bar/-SPEC.Debug.tsx new file mode 100644 index 0000000000..4901beca46 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Spinners.Bar/-SPEC.Debug.tsx @@ -0,0 +1,75 @@ +import React from 'react'; +import { Button } from '../Button/mod.ts'; +import { type t, css, D, Signal } from './common.ts'; + +type P = t.BarSpinnerProps; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + const s = Signal.create; + const props = { + theme: s<P['theme']>('Light'), + width: s<P['width']>(), + }; + const api = { + props, + listen() { + const p = props; + p.theme.value; + p.width.value; + }, + }; + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => debug.listen()); + + /** + * Render: + */ + const styles = { + base: css({}), + title: css({ fontWeight: 'bold', marginBottom: 10 }), + cols: css({ display: 'grid', gridTemplateColumns: 'auto 1fr auto' }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={css(styles.title, styles.cols).class}> + <div>{'Spinners.Bar'}</div> + <div /> + <div></div> + </div> + + <Button + block + label={() => `theme: ${p.theme.value ?? '<undefined>'}`} + onClick={() => Signal.cycle<P['theme']>(p.theme, ['Light', 'Dark'])} + /> + + <Button + block + label={() => `width: ${p.width.value ?? '<undefined>'}`} + onClick={() => Signal.cycle<P['width']>(p.width, [undefined, 30, D.width, 300])} + /> + + <hr /> + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Spinners.Bar/-SPEC.tsx b/code/sys.ui/ui-react-components/src/ui/Spinners.Bar/-SPEC.tsx new file mode 100644 index 0000000000..8ff0e73b55 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Spinners.Bar/-SPEC.tsx @@ -0,0 +1,34 @@ +import { css, Dev, Signal, Spec } from '../-test.ui.ts'; +import { Spinners } from '../Spinners/mod.ts'; +import { createDebugSignals, Debug } from './-SPEC.Debug.tsx'; + +export default Spec.describe('Spinners', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + Dev.Theme.signalEffect(ctx, p.theme, 1); + Signal.effect(() => { + debug.listen(); + ctx.redraw(); + }); + + ctx.subject + .size('fill') + .display('grid') + .render((e) => { + return ( + <div className={css({ display: 'grid', placeItems: 'center' }).class}> + <Spinners.Bar theme={p.theme.value} width={p.width.value} /> + </div> + ); + }); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/Spinners.Bar/common.ts b/code/sys.ui/ui-react-components/src/ui/Spinners.Bar/common.ts new file mode 100644 index 0000000000..192e496bbd --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Spinners.Bar/common.ts @@ -0,0 +1,7 @@ +export * from '../common.ts'; + +/** + * Constants: + */ +export const DEFAULTS = { width: 80 } as const; +export const D = DEFAULTS; diff --git a/code/sys.ui/ui-react-components/src/ui/Spinners.Bar/mod.ts b/code/sys.ui/ui-react-components/src/ui/Spinners.Bar/mod.ts new file mode 100644 index 0000000000..f744b5e57d --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Spinners.Bar/mod.ts @@ -0,0 +1,4 @@ +/** + * @module + */ +export { BarSpinner } from './ui.tsx'; diff --git a/code/sys.ui/ui-react-components/src/ui/Spinners.Bar/t.ts b/code/sys.ui/ui-react-components/src/ui/Spinners.Bar/t.ts new file mode 100644 index 0000000000..67ef8e9f89 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Spinners.Bar/t.ts @@ -0,0 +1,10 @@ +import type { t } from './common.ts'; + +/** + * <Component>: + */ +export type BarSpinnerProps = { + width?: number; + theme?: t.CommonTheme; + style?: t.CssInput; +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Spinners.Bar/ui.tsx b/code/sys.ui/ui-react-components/src/ui/Spinners.Bar/ui.tsx new file mode 100644 index 0000000000..70c1a65b13 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Spinners.Bar/ui.tsx @@ -0,0 +1,44 @@ +import React, { useEffect } from 'react'; +import { type t, Color, css, D } from './common.ts'; + +type BarLoaderProps = { color?: string; width?: number }; +let BarLoader: React.ComponentType<BarLoaderProps> | undefined; + +export const BarSpinner: React.FC<t.BarSpinnerProps> = (props) => { + const { width = D.width } = props; + + /** + * Effect: + */ + useEffect(() => { + /** + * HACK: Errors when this file is parsed within Deno on the server + * (because CJS not ESM (??)) + * Only import the component when within a browser. + */ + if (!globalThis.window) return; + import('react-spinners').then((e) => (BarLoader = e.BarLoader)); + }, []); + + if (!BarLoader) return null; + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ + position: 'relative', + overflow: 'hidden', + borderRadius: 10, + opacity: 0.5, + width, + }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <BarLoader color={theme.fg} width={width} /> + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Spinners/common.ts b/code/sys.ui/ui-react-components/src/ui/Spinners/common.ts new file mode 100644 index 0000000000..8cae67176a --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Spinners/common.ts @@ -0,0 +1 @@ +export * from '../common.ts'; diff --git a/code/sys.ui/ui-react-components/src/ui/Spinners/mod.ts b/code/sys.ui/ui-react-components/src/ui/Spinners/mod.ts new file mode 100644 index 0000000000..2974b16429 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Spinners/mod.ts @@ -0,0 +1,10 @@ +/** + * @module + * UI load spinners. + */ +import type { t } from './common.ts'; +import { BarSpinner as Bar } from '../Spinners.Bar/mod.ts'; + +export const Spinners: t.SpinnerLib = { + Bar, +}; diff --git a/code/sys.ui/ui-react-components/src/ui/Spinners/t.ts b/code/sys.ui/ui-react-components/src/ui/Spinners/t.ts new file mode 100644 index 0000000000..ed22a7e408 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/Spinners/t.ts @@ -0,0 +1,8 @@ +import type { t } from './common.ts'; + +/** + * UI load spinners. + */ +export type SpinnerLib = { + Bar: React.FC<t.BarSpinnerProps>; +}; diff --git a/code/sys.ui/ui-react-components/src/ui/VimeoBackground/-SPEC.Debug.tsx b/code/sys.ui/ui-react-components/src/ui/VimeoBackground/-SPEC.Debug.tsx new file mode 100644 index 0000000000..5707243917 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/VimeoBackground/-SPEC.Debug.tsx @@ -0,0 +1,147 @@ +import React from 'react'; +import { VIMEO } from '../-test.ui.ts'; +import { Button } from '../Button/mod.ts'; +import { type t, Color, css, Signal, DEFAULTS, D } from './common.ts'; + +type P = t.VimeoBackgroundProps; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + const s = Signal.create; + const props = { + theme: s<P['theme']>('Light'), + video: s<P['video']>(VIMEO['app/tubes']), + blur: s<P['blur']>(), + opacity: s<P['opacity']>(), + playing: s<P['playing']>(DEFAULTS.playing), + playingTransition: s<P['playingTransition']>(), + jumpTo: s<P['jumpTo']>(), + vimeo: s<t.VimeoIFrame>(), + }; + const api = { + props, + listen() { + const p = props; + p.theme.value; + p.video.value; + p.opacity.value; + p.blur.value; + p.playing.value; + p.playingTransition.value; + p.jumpTo.value; + p.vimeo.value; + }, + }; + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => { + debug.listen(); + }); + + /** + * Render: + */ + const theme = Color.theme(p.theme.value); + const styles = { + base: css({ color: theme.fg }), + title: css({ fontWeight: 'bold' }), + }; + + const srcVideo = (key?: keyof typeof VIMEO) => { + const id = key ? VIMEO[key] : undefined; + return ( + <Button + block + label={key ?? '<undefined>'} + onClick={() => { + p.video.value = id; // NB: both valid input. + p.video.value = `vimeo/${id}`; + }} + /> + ); + }; + + return ( + <div className={css(styles.base, props.style).class}> + <Button + block + label={`theme: "${p.theme}"`} + onClick={() => Signal.cycle<t.CommonTheme>(p.theme, ['Light', 'Dark'])} + /> + <Button + block + label={`opacity: ${p.opacity.value ?? '<undefined>'}`} + onClick={() => { + Signal.cycle<P['opacity']>(p.opacity, [undefined, 0, 0.5, 1]); + }} + /> + <Button + block + label={`blur: ${p.blur.value ?? '<undefined>'}`} + onClick={() => { + Signal.cycle<P['blur']>(p.blur, [undefined, 5, 15, 60]); + }} + /> + + <hr /> + <Button + block + label={`playing: ${p.playing.value}`} + onClick={() => Signal.toggle(p.playing)} + /> + <Button + block + label={() => { + const v = p.playingTransition.value; + return `playingTransition (ms): ${v ?? `<undefined> (${D.playingTransition})`}`; + }} + onClick={() => { + Signal.cycle<P['playingTransition']>(p.playingTransition, [1000, 2000, undefined]); + }} + /> + <Button + block + label={`jumpTo: ${p.jumpTo.value ?? '<undefined>'}`} + onClick={() => { + Signal.cycle<P['jumpTo']>(p.jumpTo, [undefined, 0, 10]); + }} + /> + + <hr /> + <div className={styles.title.class}>{`video: ${p.video.value}`}</div> + {srcVideo('app/tubes')} + {srcVideo('stock/running')} + {srcVideo('public/helvetica')} + + <hr /> + + <div className={styles.title.class}>API</div> + + <Button + block + label={`get: current time (seconds)`} + onClick={async () => { + const api = p.vimeo.value; + console.info('get.time:', await api?.get.time()); + }} + /> + </div> + ); +}; diff --git a/code/sys.ui/ui-react-components/src/ui/VimeoBackground/-SPEC.tsx b/code/sys.ui/ui-react-components/src/ui/VimeoBackground/-SPEC.tsx new file mode 100644 index 0000000000..700f01fc1a --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/VimeoBackground/-SPEC.tsx @@ -0,0 +1,42 @@ +import { Dev, Signal, Spec } from '../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { VimeoBackground } from './mod.ts'; + +export default Spec.describe('VimeoBackground', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + Dev.Theme.signalEffect(ctx, p.theme, 1); + Signal.effect(() => { + debug.listen(); + ctx.redraw(); + }); + + ctx.subject + .size('fill') + .display('grid') + .render((e) => ( + <VimeoBackground + theme={p.theme.value} + video={p.video.value} + opacity={p.opacity.value} + blur={p.blur.value} + playing={p.playing.value} + playingTransition={p.playingTransition.value} + jumpTo={p.jumpTo.value} + onReady={(api) => { + console.info(`ā”ļø onReady`, api); + debug.props.vimeo.value = api; + }} + /> + )); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/code/sys.ui/ui-react-components/src/ui/VimeoBackground/api.Vimeo.ts b/code/sys.ui/ui-react-components/src/ui/VimeoBackground/api.Vimeo.ts new file mode 100644 index 0000000000..1f7e1ede2e --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/VimeoBackground/api.Vimeo.ts @@ -0,0 +1,59 @@ +import { type t, Time } from './common.ts'; + +/** + * Vimeo simple IFrame/message API. + */ +export const Vimeo = { + create(iframe: HTMLIFrameElement, options: { timeout?: t.Msecs } = {}): t.VimeoIFrame { + const { timeout = 3_000 } = options; + + const api: t.VimeoIFrame = { + post(method: string, value?: number | string) { + const targetOrigin = '*'; // NB: Using "*" as targetOrigin is acceptable when not validating the origin. + iframe?.contentWindow?.postMessage({ method, value }, targetOrigin); + }, + get: { + method<T>(method: string, dispose$: t.UntilInput) { + return new Promise<T>((resolve, reject) => { + const failWith = (message: string) => reject(new Error(message)); + const listener = (event: MessageEvent) => { + if (event.origin !== 'https://player.vimeo.com') return; // Ignore: not the Vimeo player iframe. + + const data = event.data || {}; + if (data.method === method) { + cleanup(); + resolve(data.value as T); + } + }; + + window.addEventListener('message', listener); + const cleanup = () => window.removeEventListener('message', listener); + + api.post(method); + Time.until(dispose$).delay(timeout, () => { + cleanup(); + failWith(`Timeout waiting for method: ${method}`); + }); + }); + }, + + currentTime(dispose$) { + return api.get.method<number>('getCurrentTime', dispose$); + }, + + duration(dispose$) { + return api.get.method<number>('getDuration', dispose$); + }, + + async time(dispose$) { + const [current, duration] = await Promise.all([ + api.get.currentTime(dispose$), + api.get.duration(dispose$), + ]); + return { current, duration }; + }, + }, + }; + return api; + }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/VimeoBackground/common.ts b/code/sys.ui/ui-react-components/src/ui/VimeoBackground/common.ts new file mode 100644 index 0000000000..a6128b1031 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/VimeoBackground/common.ts @@ -0,0 +1,12 @@ +export * from '../common.ts'; + +/** + * Constants: + */ +export const DEFAULTS = { + blur: 0, + playing: true, + playingTransition: 0, + opacityTransition: 300, +} as const; +export const D = DEFAULTS; diff --git a/code/sys.ui/ui-react-components/src/ui/VimeoBackground/mod.ts b/code/sys.ui/ui-react-components/src/ui/VimeoBackground/mod.ts new file mode 100644 index 0000000000..fba2439bc6 --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/VimeoBackground/mod.ts @@ -0,0 +1,5 @@ +/** + * @module + */ +export { VimeoBackground } from './ui.tsx'; +export { Vimeo } from './api.Vimeo.ts'; diff --git a/code/sys.ui/ui-react-components/src/ui/VimeoBackground/t.api.ts b/code/sys.ui/ui-react-components/src/ui/VimeoBackground/t.api.ts new file mode 100644 index 0000000000..1ece4da53e --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/VimeoBackground/t.api.ts @@ -0,0 +1,14 @@ +import type { t } from './common.ts'; + +/** + * Vimeo simple IFrame/message API. + */ +export type VimeoIFrame = { + post(method: string, value?: number | string): void; + readonly get: { + method<T>(method: string, dispose$?: t.UntilInput): Promise<T>; + time(dispose$?: t.UntilInput): Promise<{ current: t.Secs; duration: t.Secs }>; + currentTime(dispose$?: t.UntilInput): Promise<t.Secs>; + duration(dispose$?: t.UntilInput): Promise<t.Secs>; + }; +}; diff --git a/code/sys.ui/ui-react-components/src/ui/VimeoBackground/t.ts b/code/sys.ui/ui-react-components/src/ui/VimeoBackground/t.ts new file mode 100644 index 0000000000..3d5ffc46cc --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/VimeoBackground/t.ts @@ -0,0 +1,32 @@ +import type { t } from './common.ts'; +export type * from './t.api.ts'; + +/** + * <Component>: + */ +export type VimeoBackgroundProps = { + video?: number | t.StringVideoAddress; + + playing?: boolean; + /** + * The delay before the play/stop change takes effect. + * Useful when composing into bigger transitioning baesd compositions. + */ + playingTransition?: t.Msecs; + jumpTo?: t.Secs; + + opacity?: number; + opacityTransition?: t.Msecs; + + blur?: number; + theme?: t.CommonTheme; + style?: t.CssInput; + + onReady?: VimeoReadyHandler; +}; + +/** + * Events + */ +export type VimeoReadyHandler = (e: VimeoReadyHandlerArgs) => void; +export type VimeoReadyHandlerArgs = t.VimeoIFrame; diff --git a/code/sys.ui/ui-react-components/src/ui/VimeoBackground/ui.tsx b/code/sys.ui/ui-react-components/src/ui/VimeoBackground/ui.tsx new file mode 100644 index 0000000000..f5b1432d9d --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/VimeoBackground/ui.tsx @@ -0,0 +1,139 @@ +import React, { useEffect, useRef, useState } from 'react'; +import { IFrame } from '../IFrame/mod.ts'; +import { Vimeo } from './api.Vimeo.ts'; +import { type t, Color, DEFAULTS, Time, css, rx } from './common.ts'; + +const D = DEFAULTS; +type P = t.VimeoBackgroundProps; + +export const VimeoBackground: React.FC<P> = (props) => { + const { + opacity, + jumpTo, + blur = D.blur, + opacityTransition = D.opacityTransition, + playingTransition = D.playingTransition, + } = props; + + const [vimeo, setVimeo] = useState<t.VimeoIFrame>(); + const [playing, setPlaying] = useState(props.playing ?? D.playing); + + const [iframe, setIframe] = useState<HTMLIFrameElement>(); + const initialPlaying = useRef(playing); + const src = wrangle.src(props.video, initialPlaying.current); + + /** + * Effect: store API reference when IFrame is ready. + */ + React.useEffect(() => { + if (iframe) { + const vimeo = Vimeo.create(iframe); + setVimeo(vimeo); + props.onReady?.(vimeo); + } + }, [iframe]); + + /** + * Effect: Play/Pause the video via the Vimeo bridge/API. + */ + useEffect(() => { + vimeo?.post(playing ? 'play' : 'pause'); + }, [iframe, playing]); + + /** + * Effect: playing transition. + */ + React.useEffect(() => { + const life = rx.lifecycle(); + const next = props.playing ?? D.playing; + + if (!playingTransition) { + setPlaying(next); // NB: 0 or undefined - immediate change. + } else { + // Make the change after the specified delay. + const time = Time.until(life); + time.delay(playingTransition, () => setPlaying(next)); + } + + return life.dispose; + }, [iframe, props.playing, playing, playingTransition]); + + /** + * Effect: jumpTo (timestamp). + */ + useEffect(() => { + if (jumpTo !== undefined) vimeo?.post('setCurrentTime', jumpTo); + }, [vimeo, iframe, jumpTo]); + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ + position: 'relative', + color: theme.fg, + display: 'grid', + overflow: 'hidden', + opacity: opacity === undefined ? 1 : opacity, + transition: opacityTransition ? `opacity ${opacityTransition}ms` : undefined, + pointerEvents: 'none', + }), + iframe: css({ + userSelect: 'none', + boxSizing: 'border-box', + height: '56.25vw', + left: '50%', + minHeight: '100%', + minWidth: '100%', + transform: 'translate(-50%, -50%)', + position: 'absolute', + top: '50%', + width: '177.77777778vh', + overflow: 'hidden', + border: 'none', + }), + blurMask: css({ + Absolute: 0, + backdropFilter: `blur(${blur}px)`, + opacity: 0.9, + transition: opacityTransition + ? `opacity ${opacityTransition}ms, backdrop-filter ${opacityTransition}ms` + : undefined, + }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <IFrame + style={styles.iframe} + src={src} + allow={'autoplay; fullscreen'} + allowFullScreen={true} + onReady={(e) => setIframe(e.ref.current ?? undefined)} + /> + <div className={styles.blurMask.class} /> + </div> + ); +}; + +/** + * Helpers + */ +const wrangle = { + src(video: P['video'], playing: P['playing']) { + const id = wrangle.id(video); + const autoplay = playing ? '1' : '0'; + const base = 'https://player.vimeo.com/video'; + return `${base}/${id}?background=1&dnt=true&autoplay=${autoplay}`; + }, + + id(video: P['video']) { + if (typeof video === 'number') return video; + if (typeof video === 'string' && video.startsWith('vimeo/')) { + const [, id] = video.split('/'); + return id; + } + throw new Error(`Failed to parse 'video' prop as a Vimeo ID: ${video}`); + }, +} as const; diff --git a/code/sys.ui/ui-react-components/src/ui/common.ts b/code/sys.ui/ui-react-components/src/ui/common.ts new file mode 100644 index 0000000000..8cae67176a --- /dev/null +++ b/code/sys.ui/ui-react-components/src/ui/common.ts @@ -0,0 +1 @@ +export * from '../common.ts'; diff --git a/code/sys.ui/ui-react-components/vite.config.ts b/code/sys.ui/ui-react-components/vite.config.ts index 25af1be683..dd19226f00 100644 --- a/code/sys.ui/ui-react-components/vite.config.ts +++ b/code/sys.ui/ui-react-components/vite.config.ts @@ -1,14 +1,16 @@ -import { Vite } from '@sys/driver-vite'; -import { defineConfig } from 'vite'; -import { pkg } from './src/pkg.ts'; +import { Vite } from 'jsr:@sys/driver-vite'; +import { defineConfig } from 'npm:vite'; export default defineConfig(() => { - return Vite.Plugin.common({ - pkg, + const entry = './src/-test/index.html'; + const paths = Vite.Config.paths({ app: { entry } }); + return Vite.Config.app({ + paths, chunks(e) { e.chunk('react', 'react'); e.chunk('react.dom', 'react-dom'); e.chunk('sys', ['@sys/std']); + e.chunk('css', ['@sys/ui-css']); }, }); }); diff --git a/code/sys.ui/ui-react-devharness/-scripts/-main.ts b/code/sys.ui/ui-react-devharness/-scripts/-main.ts new file mode 100644 index 0000000000..16e17b93a6 --- /dev/null +++ b/code/sys.ui/ui-react-devharness/-scripts/-main.ts @@ -0,0 +1 @@ +import '../../../sys.driver/driver-vite/src/-entry/-main.ts'; diff --git a/code/sys.ui/ui-react-devharness/-scripts/-tmp.ts b/code/sys.ui/ui-react-devharness/-scripts/-tmp.ts new file mode 100644 index 0000000000..b922a505ad --- /dev/null +++ b/code/sys.ui/ui-react-devharness/-scripts/-tmp.ts @@ -0,0 +1 @@ +console.log('hello temp š'); diff --git a/code/sys.ui/ui-react-devharness/deno.json b/code/sys.ui/ui-react-devharness/deno.json index a0e079c131..d726732068 100644 --- a/code/sys.ui/ui-react-devharness/deno.json +++ b/code/sys.ui/ui-react-devharness/deno.json @@ -1,15 +1,20 @@ { "name": "@sys/ui-react-devharness", - "version": "0.0.82", + "version": "0.0.92", "license": "MIT", "tasks": { + "dev": "deno run -RWNE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=dev --in=./src/-test/index.html", + "build": "deno run -RWE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=build --in=./src/-test/index.html", + "serve": "deno run -RNE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=serve", + "test": "deno test -RWNE --allow-run --allow-ffi", - "lint": "deno lint", - "dry": "deno publish --allow-dirty --dry-run", - "clean": "deno run -RWE ./scripts/-clean.ts", - "dev": "deno run -RWNE --allow-run --allow-ffi ./scripts/-dev.ts", - "build": "deno run -RWNE --allow-run --allow-ffi ./scripts/-build.ts", - "serve": "deno run -RNE --allow-run jsr:@sys/http/server/start" + "dry": "deno publish --allow-dirty --dry-run", + "clean": "deno run -RWE --allow-ffi ./-scripts/-main.ts --cmd=clean", + "upgrade": "deno run -RWNE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=upgrade", + "backup": "deno run -RWE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=backup", + + "help": "deno run -RE --allow-ffi ./-scripts/-main.ts --cmd=help", + "tmp": "deno run -A ./-scripts/-tmp.ts" }, "exports": { ".": "./src/mod.ts", diff --git a/code/sys.ui/ui-react-devharness/scripts/-build.ts b/code/sys.ui/ui-react-devharness/scripts/-build.ts deleted file mode 100644 index b73d7e663c..0000000000 --- a/code/sys.ui/ui-react-devharness/scripts/-build.ts +++ /dev/null @@ -1,6 +0,0 @@ -import { Vite } from '@sys/driver-vite'; -import { pkg } from '@sys/ui-react-devharness'; - -const input = './src/-test/index.html'; -const bundle = await Vite.build({ pkg, input }); -console.info(bundle.toString({ pad: true })); diff --git a/code/sys.ui/ui-react-devharness/scripts/-clean.ts b/code/sys.ui/ui-react-devharness/scripts/-clean.ts deleted file mode 100644 index 68bf3b0c1a..0000000000 --- a/code/sys.ui/ui-react-devharness/scripts/-clean.ts +++ /dev/null @@ -1,6 +0,0 @@ -import { Fs } from '@sys/fs'; - -const removeDir = (path: string) => Fs.remove(Fs.resolve(path), { log: true }); -await removeDir('./.tmp'); -await removeDir('./dist'); -await removeDir('./.swc'); diff --git a/code/sys.ui/ui-react-devharness/scripts/-dev.ts b/code/sys.ui/ui-react-devharness/scripts/-dev.ts deleted file mode 100644 index d3ecd700da..0000000000 --- a/code/sys.ui/ui-react-devharness/scripts/-dev.ts +++ /dev/null @@ -1,9 +0,0 @@ -/** - * Run in a child-process (hence the `-allow-run` requirement). - */ -import { Vite } from '@sys/driver-vite'; -import { pkg } from '@sys/ui-react-devharness'; - -const input = './src/-test/index.html'; -const server = await Vite.dev({ pkg, input }); -await server.listen(); diff --git a/code/sys.ui/ui-react-devharness/src/-test/entry.tsx b/code/sys.ui/ui-react-devharness/src/-test/entry.tsx index 2bdcb06472..8790499952 100644 --- a/code/sys.ui/ui-react-devharness/src/-test/entry.tsx +++ b/code/sys.ui/ui-react-devharness/src/-test/entry.tsx @@ -1,7 +1,4 @@ -// @ts-types="@types/react" -import React from 'react'; - -import { StrictMode } from 'react'; +import React, { StrictMode } from 'react'; import { createRoot } from 'react-dom/client'; import { pkg } from '../pkg.ts'; @@ -18,25 +15,37 @@ export async function main() { /** * Sample entry logic. */ - const rootElement = document.getElementById('root'); - const root = createRoot(rootElement); + const root = createRoot(document.getElementById('root')!); if (isDev) { /** * NOTE: * The import of the [Dev] module happens dynamically here AFTER * the URL query-string has been interpreted. This allows the base - * module entry to by code-split in such a way that the [Dev Harness] - * never gets sent in the normal useage payload. + * module entry to be code-split in such a way that the [Dev Harness] + * never gets sent in the normal usage payload. */ - const { render } = await import('../mod.ts'); + const { render, useKeyboard } = await import('../mod.ts'); const { ModuleSpecs, SampleSpecs } = await import('./entry.Specs.ts'); + + // Dynamic "runnable specifications" Library: const Specs = { ...SampleSpecs, ...ModuleSpecs, }; + const el = await render(pkg, Specs, { hrDepth: 2, style: { Absolute: 0 } }); - root.render(<StrictMode>{el as React.ReactNode}</StrictMode>); + function App() { + // NB: Any other environment setup here. + useKeyboard(); + return el; + } + + root.render( + <StrictMode> + <App /> + </StrictMode>, + ); } else { const { MySample } = await import('./sample.specs/MySample.tsx'); const el = <MySample style={{ Absolute: 0 }} />; @@ -44,4 +53,4 @@ export async function main() { } } -main().catch((err) => console.error(`Failed to render`, err)); +main().catch((err) => console.error(`Failed to render DevHarness`, err)); diff --git a/code/sys.ui/ui-react-devharness/src/-test/sample.DevTools/-SPEC.tsx b/code/sys.ui/ui-react-devharness/src/-test/sample.DevTools/-SPEC.tsx index a8163d7a7d..443bfd51da 100644 --- a/code/sys.ui/ui-react-devharness/src/-test/sample.DevTools/-SPEC.tsx +++ b/code/sys.ui/ui-react-devharness/src/-test/sample.DevTools/-SPEC.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { Spec } from '../common.ts'; import { DevTools } from './DevTools.tsx'; diff --git a/code/sys.ui/ui-react-devharness/src/-test/sample.DevTools/DevTools.tsx b/code/sys.ui/ui-react-devharness/src/-test/sample.DevTools/DevTools.tsx index 188fc96ffe..52ee1c75df 100644 --- a/code/sys.ui/ui-react-devharness/src/-test/sample.DevTools/DevTools.tsx +++ b/code/sys.ui/ui-react-devharness/src/-test/sample.DevTools/DevTools.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { Spec, type t } from '../common.ts'; import { ButtonSample } from './ui.Button.tsx'; diff --git a/code/sys.ui/ui-react-devharness/src/-test/sample.DevTools/ui.Button.tsx b/code/sys.ui/ui-react-devharness/src/-test/sample.DevTools/ui.Button.tsx index 71084da91a..d2f847768a 100644 --- a/code/sys.ui/ui-react-devharness/src/-test/sample.DevTools/ui.Button.tsx +++ b/code/sys.ui/ui-react-devharness/src/-test/sample.DevTools/ui.Button.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { RenderCount } from '../../ui/RenderCount/mod.ts'; import { COLORS, Color, css, useMouse, type t } from '../common.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/-test/sample.DevTools/ui.Hr.tsx b/code/sys.ui/ui-react-devharness/src/-test/sample.DevTools/ui.Hr.tsx index 31c2592e90..1857d57060 100644 --- a/code/sys.ui/ui-react-devharness/src/-test/sample.DevTools/ui.Hr.tsx +++ b/code/sys.ui/ui-react-devharness/src/-test/sample.DevTools/ui.Hr.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { Color, css } from '../common.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/-test/sample.specs.unit-test/-Sample-1.SPEC.tsx b/code/sys.ui/ui-react-devharness/src/-test/sample.specs.unit-test/-Sample-1.SPEC.tsx index f394b2e6aa..bceaeddbb5 100644 --- a/code/sys.ui/ui-react-devharness/src/-test/sample.specs.unit-test/-Sample-1.SPEC.tsx +++ b/code/sys.ui/ui-react-devharness/src/-test/sample.specs.unit-test/-Sample-1.SPEC.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { Spec } from '../common.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/-test/sample.specs.unit-test/-Sample-2.SPEC.tsx b/code/sys.ui/ui-react-devharness/src/-test/sample.specs.unit-test/-Sample-2.SPEC.tsx index cab339541c..35b3376454 100644 --- a/code/sys.ui/ui-react-devharness/src/-test/sample.specs.unit-test/-Sample-2.SPEC.tsx +++ b/code/sys.ui/ui-react-devharness/src/-test/sample.specs.unit-test/-Sample-2.SPEC.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { Spec } from '../common.ts'; import { TestLog } from '../TestLog.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/-test/sample.specs/-SPEC.Error.tsx b/code/sys.ui/ui-react-devharness/src/-test/sample.specs/-SPEC.Error.tsx index b36f8f0416..c55015ac1e 100644 --- a/code/sys.ui/ui-react-devharness/src/-test/sample.specs/-SPEC.Error.tsx +++ b/code/sys.ui/ui-react-devharness/src/-test/sample.specs/-SPEC.Error.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { Spec } from '../common.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/-test/sample.specs/-SPEC.MySample.tsx b/code/sys.ui/ui-react-devharness/src/-test/sample.specs/-SPEC.MySample.tsx index 58b5f959c8..f6444d249d 100644 --- a/code/sys.ui/ui-react-devharness/src/-test/sample.specs/-SPEC.MySample.tsx +++ b/code/sys.ui/ui-react-devharness/src/-test/sample.specs/-SPEC.MySample.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React, { useState } from 'react'; import { Keyboard } from '@sys/ui-dom'; diff --git a/code/sys.ui/ui-react-devharness/src/-test/sample.specs/-SPEC.Size.tsx b/code/sys.ui/ui-react-devharness/src/-test/sample.specs/-SPEC.Size.tsx index 1c9bc55509..b48c10c4e9 100644 --- a/code/sys.ui/ui-react-devharness/src/-test/sample.specs/-SPEC.Size.tsx +++ b/code/sys.ui/ui-react-devharness/src/-test/sample.specs/-SPEC.Size.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { type t, css, Spec } from '../common.ts'; import { DevTools } from '../sample.DevTools/mod.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/-test/sample.specs/MySample.tsx b/code/sys.ui/ui-react-devharness/src/-test/sample.specs/MySample.tsx index 18ad539c9e..2d504d84f1 100644 --- a/code/sys.ui/ui-react-devharness/src/-test/sample.specs/MySample.tsx +++ b/code/sys.ui/ui-react-devharness/src/-test/sample.specs/MySample.tsx @@ -1,6 +1,6 @@ -// @ts-types="@types/react" -import React, { useEffect } from 'react'; -import { Color, css, DEFAULTS, Keyboard, type t } from '../common.ts'; +import React from 'react'; +import { useKeyboard } from '../../ui.use/mod.ts'; +import { type t, Color, css } from '../common.ts'; export type MySampleProps = { text?: string; @@ -14,25 +14,13 @@ let _count = 0; export const MySample: React.FC<MySampleProps> = (props) => { if (props.throwError) { - throw new Error('MySample: Intentional error'); + throw new Error('MySample: š· Intentional error'); } - /** - * [Effects] - */ - useEffect(() => { - const keyboard = Keyboard.on({ - Enter(e) { - const url = new URL(globalThis.location.href); - url.searchParams.set(DEFAULTS.qs.dev, 'true'); - globalThis.location.href = url.href; - }, - }); - return () => keyboard.dispose(); - }, []); + useKeyboard(); /** - * [Render] + * Render: */ const styles = { base: css({ diff --git a/code/sys.ui/ui-react-devharness/src/-test/sample.specs/newFile.tsx b/code/sys.ui/ui-react-devharness/src/-test/sample.specs/newFile.tsx index 63087fc55b..f6b77429a8 100644 --- a/code/sys.ui/ui-react-devharness/src/-test/sample.specs/newFile.tsx +++ b/code/sys.ui/ui-react-devharness/src/-test/sample.specs/newFile.tsx @@ -1,4 +1,4 @@ -import { Spec } from '../../u/Spec/m.Spec'; +import { Spec } from '../mod.ts'; export default Spec.describe('Error on initialize', (e) => { e.it('init', (e) => { diff --git a/code/sys.ui/ui-react-devharness/src/common/libs.ts b/code/sys.ui/ui-react-devharness/src/common/libs.ts index df9b99c59d..567ae326fb 100644 --- a/code/sys.ui/ui-react-devharness/src/common/libs.ts +++ b/code/sys.ui/ui-react-devharness/src/common/libs.ts @@ -3,9 +3,9 @@ export { ErrorBoundary } from 'react-error-boundary'; export { Test } from '@sys/testing/spec'; export { Fuzzy } from '@sys/text/fuzzy'; -export { asArray, Is, maybeWait, Path, R, rx, slug, Time } from '@sys/std'; -export { Str, Value } from '@sys/std/value'; +export { asArray, Is, maybeWait, Obj, Path, R, rx, slug, Time } from '@sys/std'; +export { Lorem, Str, Value } from '@sys/std/value'; export { Color, css, Style } from '@sys/ui-css'; export { Keyboard } from '@sys/ui-dom'; -export { FC, useMouse } from '@sys/ui-react'; +export { FC, Signal, useDist, useMouse } from '@sys/ui-react'; diff --git a/code/sys.ui/ui-react-devharness/src/common/t.ts b/code/sys.ui/ui-react-devharness/src/common/t.ts index 7e75411568..c182fe34c4 100644 --- a/code/sys.ui/ui-react-devharness/src/common/t.ts +++ b/code/sys.ui/ui-react-devharness/src/common/t.ts @@ -7,8 +7,8 @@ export type { IconType } from 'react-icons'; /** * @system */ -export type { ColorConstants } from '@sys/color/t'; -export type { ModuleImport, ModuleImporter, ModuleImports } from '@sys/std/t'; +export type { ColorConstants, ColorTheme } from '@sys/color/t'; +export type { ModuleImport, ModuleImporter, ModuleImports, Signal } from '@sys/std/t'; export type * from '@sys/types/t'; export type { @@ -25,11 +25,11 @@ export type { export type { CssEdgesArray, + CssInput, CssMarginArray, CssMarginInput, CssPaddingArray, CssValue, - CssInput, } from '@sys/ui-css/t'; export type { KeyboardEventsUntil, KeyboardModifierFlags } from '@sys/ui-dom/t'; diff --git a/code/sys.ui/ui-react-devharness/src/m.Dev/-.test.ts b/code/sys.ui/ui-react-devharness/src/m.Dev/-.test.ts new file mode 100644 index 0000000000..a19c8b9a60 --- /dev/null +++ b/code/sys.ui/ui-react-devharness/src/m.Dev/-.test.ts @@ -0,0 +1,9 @@ +import { type t, describe, it, expect } from '../-test.ts'; +import { Dev } from './mod.ts'; +import { Theme } from '../u/mod.ts'; + +describe('Dev', () => { + it('API', () => { + expect(Dev.Theme).to.equal(Theme); + }); +}); diff --git a/code/sys.ui/ui-react-devharness/src/m.Dev/m.Dev.ts b/code/sys.ui/ui-react-devharness/src/m.Dev/m.Dev.ts index bd60b388d0..2b8252194e 100644 --- a/code/sys.ui/ui-react-devharness/src/m.Dev/m.Dev.ts +++ b/code/sys.ui/ui-react-devharness/src/m.Dev/m.Dev.ts @@ -2,6 +2,7 @@ import { headless } from '../-test/headless/mod.ts'; import { DevBus as Bus } from '../u/m.Bus/mod.ts'; import { Context } from '../u/m.Ctx/mod.ts'; import { Spec } from '../u/m.Spec/mod.ts'; +import { Theme } from '../u/m.Theme/mod.ts'; import { ValueHandler } from '../u/m.Tools/mod.ts'; import { Harness } from '../ui/Harness/mod.ts'; import { ModuleList } from '../ui/ModuleList/mod.ts'; @@ -15,6 +16,7 @@ export const Dev = { Spec, ModuleList, Harness, + Theme, ValueHandler, headless, } as const; diff --git a/code/sys.ui/ui-react-devharness/src/m.Dev/u.render.tsx b/code/sys.ui/ui-react-devharness/src/m.Dev/u.render.tsx index c13e70c4ba..5f0934c425 100644 --- a/code/sys.ui/ui-react-devharness/src/m.Dev/u.render.tsx +++ b/code/sys.ui/ui-react-devharness/src/m.Dev/u.render.tsx @@ -1,5 +1,5 @@ -// @ts-types="@types/react" import React from 'react'; + import { DevArgs, DevKeyboard } from '../u/mod.ts'; import { Harness } from '../ui/Harness/mod.ts'; import { ModuleList } from '../ui/ModuleList/mod.ts'; @@ -24,7 +24,7 @@ export type Render = ( pkg: { name: string; version: string }, specs: t.SpecImports, options?: RenderOptions, -) => Promise<t.JSXElement>; +) => Promise<React.ReactElement>; /** * Render a harness with the selected `dev=<namespace>` diff --git a/code/sys.ui/ui-react-devharness/src/mod.ts b/code/sys.ui/ui-react-devharness/src/mod.ts index 00b156365e..1a38218d75 100644 --- a/code/sys.ui/ui-react-devharness/src/mod.ts +++ b/code/sys.ui/ui-react-devharness/src/mod.ts @@ -40,8 +40,8 @@ export type * as t from './types.ts'; export { Dev, render } from './m.Dev/mod.ts'; -export { useRubberband } from './ui.use/useRubberband.ts'; +export { useKeyboard, useRubberband } from './ui.use/mod.ts'; export { ModuleList } from './ui/ModuleList/mod.ts'; -export { Badges, COLORS } from './common.ts'; +export { Badges, COLORS, Lorem } from './common.ts'; export { DevArgs, DevKeyboard, Is, Spec, ValueHandler } from './u/mod.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/pkg.ts b/code/sys.ui/ui-react-devharness/src/pkg.ts index 79cb3f9c80..1ba7df8f34 100644 --- a/code/sys.ui/ui-react-devharness/src/pkg.ts +++ b/code/sys.ui/ui-react-devharness/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/ui-react-devharness', version: '0.0.92' }; diff --git a/code/sys.ui/ui-react-devharness/src/t/t.ctx.ts b/code/sys.ui/ui-react-devharness/src/t/t.ctx.ts index 459bbfac57..413267cc8e 100644 --- a/code/sys.ui/ui-react-devharness/src/t/t.ctx.ts +++ b/code/sys.ui/ui-react-devharness/src/t/t.ctx.ts @@ -80,7 +80,7 @@ export type DevCtxSubject = { color(value?: Color | null): DevCtxSubject; backgroundColor(value?: Color | null): DevCtxSubject; size(value: DevSubjectSize): DevCtxSubject; - size(mode: DevFillMode, margin?: t.CssMarginInput): DevCtxSubject; + size(mode?: DevFillMode, margin?: t.CssMarginInput): DevCtxSubject; render<T extends O = O>(fn: t.DevRenderer<T>): DevCtxSubject; }; diff --git a/code/sys.ui/ui-react-devharness/src/t/t.render.props.ts b/code/sys.ui/ui-react-devharness/src/t/t.render.props.ts index 311f31def2..b300a401d3 100644 --- a/code/sys.ui/ui-react-devharness/src/t/t.render.props.ts +++ b/code/sys.ui/ui-react-devharness/src/t/t.render.props.ts @@ -120,7 +120,7 @@ export type DevRenderSizeCenter = { }; /** - * Render size foa an element filling the container. + * Render size for an element filling the container. */ export type DevRenderSizeFill = { mode: 'fill'; diff --git a/code/sys.ui/ui-react-devharness/src/types.ts b/code/sys.ui/ui-react-devharness/src/types.ts index a61795f296..d284faddb1 100644 --- a/code/sys.ui/ui-react-devharness/src/types.ts +++ b/code/sys.ui/ui-react-devharness/src/types.ts @@ -5,5 +5,6 @@ export type * from './t/mod.ts'; export type * from './u/types.ts'; -export type * from './ui.use/types.ts'; + +export type * from './ui.use/t.ts'; export type * from './ui/types.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/u/m.Bus/Bus.Controller.ts b/code/sys.ui/ui-react-devharness/src/u/m.Bus/Bus.Controller.ts index c814cd8651..c2c070c4a3 100644 --- a/code/sys.ui/ui-react-devharness/src/u/m.Bus/Bus.Controller.ts +++ b/code/sys.ui/ui-react-devharness/src/u/m.Bus/Bus.Controller.ts @@ -1,7 +1,7 @@ import { Context } from '../m.Ctx/mod.ts'; import { BusEvents } from './Bus.Events.ts'; import { BusMemoryState } from './Bus.MemoryState.ts'; -import { DEFAULTS, Id, Is, R, Test, rx, type t } from './common.ts'; +import { type t, Obj, DEFAULTS, Id, Is, R, Test, rx } from './common.ts'; /** * Start the controller and return an event API. @@ -172,7 +172,7 @@ export function BusController(args: { } if (typeof e.mutate === 'object') { - draft.render.state = R.clone(e.mutate); // š· TEMP | SLOW (potentially too slow). Not needed when using immutability plugin. + draft.render.state = Obj.clone(e.mutate); // š· TEMP | SLOW (potentially too slow). Not needed when using immutability plugin. } draft.render.revision.state += 1; diff --git a/code/sys.ui/ui-react-devharness/src/u/m.Bus/Bus.MemoryState.ts b/code/sys.ui/ui-react-devharness/src/u/m.Bus/Bus.MemoryState.ts index 9bdf299467..169ebd9536 100644 --- a/code/sys.ui/ui-react-devharness/src/u/m.Bus/Bus.MemoryState.ts +++ b/code/sys.ui/ui-react-devharness/src/u/m.Bus/Bus.MemoryState.ts @@ -1,4 +1,4 @@ -import { DEFAULTS, Id, Is, R, rx, type t } from './common.ts'; +import { type t, DEFAULTS, Id, Is, Obj, R, rx } from './common.ts'; type Revision = { number: number; message: string }; @@ -40,7 +40,7 @@ export function BusMemoryState(args: { * Make these options available as an injected plugin (IoC). */ const before = api.revision; - const clone = R.clone(_current); // TEMP | SLOW (potentially too slow) š· + const clone = Obj.clone(_current); // TEMP | SLOW (potentially too slow) š· if (typeof change === 'function') { const res = change(clone); diff --git a/code/sys.ui/ui-react-devharness/src/u/m.Ctx/-.test.tsx b/code/sys.ui/ui-react-devharness/src/u/m.Ctx/-.test.tsx index 02ce84cbd8..2829c0478b 100644 --- a/code/sys.ui/ui-react-devharness/src/u/m.Ctx/-.test.tsx +++ b/code/sys.ui/ui-react-devharness/src/u/m.Ctx/-.test.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { TestSample, describe, expect, it, type t } from '../../-test.ts'; import { DEFAULTS, Id } from './common.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/u/m.Ctx/Ctx.Props.Subject.ts b/code/sys.ui/ui-react-devharness/src/u/m.Ctx/Ctx.Props.Subject.ts index 860bee92f5..f275a90752 100644 --- a/code/sys.ui/ui-react-devharness/src/u/m.Ctx/Ctx.Props.Subject.ts +++ b/code/sys.ui/ui-react-devharness/src/u/m.Ctx/Ctx.Props.Subject.ts @@ -17,6 +17,8 @@ export function CtxPropsSubject(props: PropArgs) { const current = props.current().subject; current.size = undefined; + if (args[0] === undefined) return api; + if (Array.isArray(args[0])) { const [width, height] = args[0]; if (Is.nil(width) && Is.nil(height)) { diff --git a/code/sys.ui/ui-react-devharness/src/u/m.Theme/-.test.ts b/code/sys.ui/ui-react-devharness/src/u/m.Theme/-.test.ts new file mode 100644 index 0000000000..e97b81d40f --- /dev/null +++ b/code/sys.ui/ui-react-devharness/src/u/m.Theme/-.test.ts @@ -0,0 +1,3 @@ +import { type t, describe, it, expect, Testing } from '../../-test.ts'; + +describe('Dev.Theme', () => {}); diff --git a/code/sys.ui/ui-react-devharness/src/u/m.Theme/common.ts b/code/sys.ui/ui-react-devharness/src/u/m.Theme/common.ts new file mode 100644 index 0000000000..8cae67176a --- /dev/null +++ b/code/sys.ui/ui-react-devharness/src/u/m.Theme/common.ts @@ -0,0 +1 @@ +export * from '../common.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/u/m.Theme/mod.ts b/code/sys.ui/ui-react-devharness/src/u/m.Theme/mod.ts new file mode 100644 index 0000000000..1cf29b6ded --- /dev/null +++ b/code/sys.ui/ui-react-devharness/src/u/m.Theme/mod.ts @@ -0,0 +1,39 @@ +/** + * @module + */ +import { type t, Color, Signal } from './common.ts'; + +type Color = string | number; +const DEFAULT_THEME: t.CommonTheme = 'Light'; + +/** + * Helpers for working with common themes within the harness. + */ +export const Theme: t.DevThemeLib = { + /** + * Adjust the theme of the DevHarness. + */ + background(ctx, theme, subjectLight, subjectDark) { + const { host, subject } = ctx; + const is = Color.theme(theme ?? DEFAULT_THEME).is; + if (is.light) host.color(null).backgroundColor(null).tracelineColor(null); + if (is.dark) host.color(Color.WHITE).backgroundColor(Color.DARK).tracelineColor(0.08); + + if (!!subjectLight || !!subjectDark) { + if (!!subjectLight) subject.backgroundColor(is.light ? subjectLight : 0); + if (!!subjectDark) subject.backgroundColor(is.dark ? subjectDark : 0); + } + + return Color.theme(theme); + }, + + /** + * Hook into a theme Signal and keep the DevHarness sycned with it. + */ + signalEffect(ctx, theme, subjectLight, subjectDark) { + Signal.effect(() => { + Theme.background(ctx, theme.value, subjectLight, subjectDark); + ctx.redraw(); + }); + }, +}; diff --git a/code/sys.ui/ui-react-devharness/src/u/m.Theme/t.ts b/code/sys.ui/ui-react-devharness/src/u/m.Theme/t.ts new file mode 100644 index 0000000000..f64eda4f42 --- /dev/null +++ b/code/sys.ui/ui-react-devharness/src/u/m.Theme/t.ts @@ -0,0 +1,25 @@ +import type { t } from './common.ts'; + +type Color = string | number; +type ThemeInput = t.ColorTheme | t.CommonTheme | null; + +/** + * Helpers for working with common themes within the harness. + */ +export type DevThemeLib = { + /** Adjust the theme of the DevHarness. */ + background( + ctx: t.DevCtx, + theme?: ThemeInput, + subjectLight?: Color | null, + subjectDark?: Color | null, + ): void; + + /** Hook into a theme Signal and keep the DevHarness sycned with it. */ + signalEffect( + ctx: t.DevCtx, + signal: t.Signal, + subjectLight?: Color | null, + subjectDark?: Color | null, + ): void; +}; diff --git a/code/sys.ui/ui-react-devharness/src/u/mod.ts b/code/sys.ui/ui-react-devharness/src/u/mod.ts index c6a579419d..80d939ac7d 100644 --- a/code/sys.ui/ui-react-devharness/src/u/mod.ts +++ b/code/sys.ui/ui-react-devharness/src/u/mod.ts @@ -4,4 +4,5 @@ export { Context } from './m.Ctx/mod.ts'; export { Is } from './m.Is/mod.ts'; export { DevKeyboard } from './m.Keyboard/mod.ts'; export { Spec } from './m.Spec/mod.ts'; +export { Theme } from './m.Theme/mod.ts'; export { ValueHandler } from './m.Tools/mod.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/u/types.ts b/code/sys.ui/ui-react-devharness/src/u/types.ts index f9b4d7273a..c600a344a8 100644 --- a/code/sys.ui/ui-react-devharness/src/u/types.ts +++ b/code/sys.ui/ui-react-devharness/src/u/types.ts @@ -4,4 +4,5 @@ export type * from './m.Ctx/t.ts'; export type * from './m.Is/t.ts'; export type * from './m.Keyboard/t.ts'; export type * from './m.Spec/t.ts'; +export type * from './m.Theme/t.ts'; export type * from './m.Tools/t.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/ui.use/mod.ts b/code/sys.ui/ui-react-devharness/src/ui.use/mod.ts index 2cc60961f6..fff1a0bdef 100644 --- a/code/sys.ui/ui-react-devharness/src/ui.use/mod.ts +++ b/code/sys.ui/ui-react-devharness/src/ui.use/mod.ts @@ -4,3 +4,4 @@ export * from './useRedrawEvent.ts'; export * from './useRenderer.ts'; export * from './useRubberband.ts'; export * from './useSizeObserver.ts'; +export * from './use.Keyboard.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/ui.use/t.ts b/code/sys.ui/ui-react-devharness/src/ui.use/t.ts new file mode 100644 index 0000000000..3cb8dca31a --- /dev/null +++ b/code/sys.ui/ui-react-devharness/src/ui.use/t.ts @@ -0,0 +1,4 @@ +/** + * Hook: DevHarness Keyboard controller. + */ +export type UseKeyboardFactory = () => void; diff --git a/code/sys.ui/ui-react-devharness/src/ui.use/types.ts b/code/sys.ui/ui-react-devharness/src/ui.use/types.ts deleted file mode 100644 index cb0ff5c3b5..0000000000 --- a/code/sys.ui/ui-react-devharness/src/ui.use/types.ts +++ /dev/null @@ -1 +0,0 @@ -export {}; diff --git a/code/sys.ui/ui-react-devharness/src/ui.use/use.Keyboard.ts b/code/sys.ui/ui-react-devharness/src/ui.use/use.Keyboard.ts new file mode 100644 index 0000000000..70aa758e74 --- /dev/null +++ b/code/sys.ui/ui-react-devharness/src/ui.use/use.Keyboard.ts @@ -0,0 +1,50 @@ +import { useEffect } from 'react'; +import { type t, Keyboard } from './common.ts'; + +/** + * Hook: Keyboard controller. + */ +export const useKeyboard: t.UseKeyboardFactory = () => { + useEffect(() => { + const keyboard = Keyboard.until(); + + const getUrl = () => { + const url = new URL(window.location.href); + const query = url.searchParams; + return { url, query }; + }; + + const is = { + get dev() { + return getUrl().query.has('dev'); + }, + } as const; + + /** + * GOTO: DevHarness. + */ + keyboard.on('CMD + Enter', () => { + if (!is.dev) { + const { url, query } = getUrl(); + query.set('dev', 'true'); + window.location.href = url.href; + } + }); + + /** + * GOTO: Root + */ + keyboard.on('CMD + Escape', () => { + const { url, query } = getUrl(); + + if (is.dev) { + const current = query.get('dev'); + if (current === 'true') query.delete('dev'); // ā goto Root screen. + else query.set('dev', 'true'); // ā goto DevHarness index. + window.location.href = url.href; + } + }); + + return keyboard.dispose; + }, []); +}; diff --git a/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Background.tsx b/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Background.tsx index d3c2b5c781..89fde2b1f2 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Background.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Background.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { css, type t } from '../common.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Component.tsx b/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Component.tsx index 1082a410a5..c750d87292 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Component.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Component.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { Color, css, useRenderer, type t } from '../common.ts'; import { Wrangle } from './u.ts'; @@ -41,7 +40,7 @@ export const HostComponent: React.FC<HostComponentProps> = (props) => { <div ref={props.subjectRef} className={styles.body.class} - data-component={'dev.harness:ComponentHost'} + data-component={'sys.ui.dev.harness:ComponentHost'} > {element as t.ReactNode} </div> diff --git a/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Grid.tsx b/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Grid.tsx index f7503215a1..94f09a7adb 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Grid.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Grid.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { css, type t } from '../common.ts'; import { Wrangle } from './u.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Layer.tsx b/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Layer.tsx index 53b6511ab3..dbf472bb64 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Layer.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Layer.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { css, useRenderer, type t } from '../common.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Layers.tsx b/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Layers.tsx index 848632b91f..7b5be8c46b 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Layers.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.Layers.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { css, type t } from '../common.ts'; import { HostLayer } from './Host.Layer.tsx'; diff --git a/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.tsx b/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.tsx index c9e1d1229c..7d3f400ae0 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/Harness.Host/Host.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React, { useEffect, useState } from 'react'; import { Color, css, DEFAULTS, R, Time, useCurrentState, type t } from '../common.ts'; @@ -41,7 +40,7 @@ export const HarnessHost: React.FC<HarnessHostProps> = (props) => { }, [isEmpty]); /** - * Render + * Render: */ const cropmark = wrangle.cropmark(renderProps); const backgroundColor = @@ -57,7 +56,7 @@ export const HarnessHost: React.FC<HarnessHostProps> = (props) => { const styles = { base: css({ position: 'relative', backgroundColor, color }), body: css({ Absolute: 0, display: 'grid', gridTemplateRows: 'auto 1fr auto' }), - main: css({ position: 'relative', display: 'grid' }), + main: css({ position: 'relative', display: 'grid', overflow: 'hidden' }), empty: css({ Absolute: 0, display: 'grid', placeContent: 'center', userSelect: 'none' }), }; diff --git a/code/sys.ui/ui-react-devharness/src/ui/Harness.Panel.Debug/Panel.Body.Row.tsx b/code/sys.ui/ui-react-devharness/src/ui/Harness.Panel.Debug/Panel.Body.Row.tsx index 38d60fc39f..916abbc8cc 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/Harness.Panel.Debug/Panel.Body.Row.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/Harness.Panel.Debug/Panel.Body.Row.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { css, useRenderer, type t } from '../common.ts'; @@ -15,6 +14,7 @@ export const DebugPanelBodyRow: React.FC<DebugPanelBodyRow> = (props) => { base: css({ position: 'relative', display: 'grid', + lineHeight: 1.6, }), }; diff --git a/code/sys.ui/ui-react-devharness/src/ui/Harness.Panel.Debug/Panel.Body.tsx b/code/sys.ui/ui-react-devharness/src/ui/Harness.Panel.Debug/Panel.Body.tsx index fd58e1401e..5072fd950c 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/Harness.Panel.Debug/Panel.Body.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/Harness.Panel.Debug/Panel.Body.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { css, type t } from '../common.ts'; import { DebugPanelBodyRow as Row } from './Panel.Body.Row.tsx'; @@ -13,7 +12,9 @@ export const DebugPanelBody: React.FC<DebugPanelBodyrops> = (props) => { const { instance, current } = props; const renderers = current?.render?.props?.debug.body.renderers ?? []; - const styles = { base: css({ position: 'relative' }) }; + const styles = { + base: css({ position: 'relative' }), + }; const elements = renderers.filter(Boolean).map((renderer) => { return <Row key={renderer.id} instance={instance} renderer={renderer} />; diff --git a/code/sys.ui/ui-react-devharness/src/ui/Harness.Panel.Debug/Panel.tsx b/code/sys.ui/ui-react-devharness/src/ui/Harness.Panel.Debug/Panel.tsx index d6a3bb56bd..951ff308b1 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/Harness.Panel.Debug/Panel.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/Harness.Panel.Debug/Panel.tsx @@ -1,17 +1,20 @@ -// @ts-types="@types/react" import React from 'react'; -import { Color, css, R, useCurrentState, type t } from '../common.ts'; +import { type t, Style, Color, css, R, useCurrentState } from '../common.ts'; import { PanelFooter, PanelHeader } from '../Harness.Panel.Edge/mod.ts'; import { DebugPanelBody as Body } from './Panel.Body.tsx'; export type DebugPanelProps = { instance: t.DevInstance; baseRef?: React.RefObject<HTMLDivElement>; + theme?: t.CommonTheme; style?: t.CssInput; }; +const componentAttr = 'sys.ui.dev.harness:Panel.Debug'; + export const DebugPanel: React.FC<DebugPanelProps> = (props) => { const { instance } = props; + const theme = Color.theme(props.theme); const current = useCurrentState(instance, { distinctUntil }); const debug = current.info?.render.props?.debug; @@ -20,6 +23,18 @@ export const DebugPanel: React.FC<DebugPanelProps> = (props) => { if (width === null) return null; // NB: configured to not render. const isHidden = width <= 0; + /** + * Common styling for children. + */ + React.useEffect(() => { + const sheet = Style.Dom.stylesheet(); + sheet.rule(`[data-component="${componentAttr}"] hr`, { + border: 'none', + borderTop: '1px solid', + borderColor: Color.alpha(theme.fg, 0.2), + }); + }, []); + /** * [Render] */ @@ -27,8 +42,8 @@ export const DebugPanel: React.FC<DebugPanelProps> = (props) => { base: css({ overflow: 'hidden', justifySelf: 'stretch', - borderLeft: `solid 1px ${Color.format(-0.1)}`, width, + borderLeft: `solid 1px ${Color.format(-0.1)}`, display: isHidden ? 'none' : 'grid', gridTemplateRows: 'auto 1fr auto', pointerEvents: isHidden ? 'none' : undefined, @@ -42,7 +57,7 @@ export const DebugPanel: React.FC<DebugPanelProps> = (props) => { return ( <div ref={props.baseRef} - data-component={'dev.harness:Panel.Debug'} + data-component={componentAttr} className={css(styles.base, props.style).class} > <PanelHeader instance={instance} current={debug?.header} /> diff --git a/code/sys.ui/ui-react-devharness/src/ui/Harness.Panel.Edge/Panel.Edge.tsx b/code/sys.ui/ui-react-devharness/src/ui/Harness.Panel.Edge/Panel.Edge.tsx index 9f6e4667ee..1bb2616b9f 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/Harness.Panel.Edge/Panel.Edge.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/Harness.Panel.Edge/Panel.Edge.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { css, useRenderer, type t } from '../common.ts'; import { Wrangle } from './u.ts'; @@ -18,7 +17,7 @@ export const PanelEdge: React.FC<PanelBarProps> = (props) => { if (!renderer || !element) return <div />; /** - * Render + * Render: */ const border = (hasBorder: boolean) => (hasBorder ? Wrangle.borderStyle(current) : undefined); const styles = { diff --git a/code/sys.ui/ui-react-devharness/src/ui/Harness/-SPEC.tsx b/code/sys.ui/ui-react-devharness/src/ui/Harness/-SPEC.tsx index 1db1a9bfee..f7a2cc367f 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/Harness/-SPEC.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/Harness/-SPEC.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { Spec } from '../../-test.ts'; import { Harness } from './mod.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/ui/Harness/ErrorFallback.tsx b/code/sys.ui/ui-react-devharness/src/ui/Harness/ErrorFallback.tsx index 47fd65138c..16a8636293 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/Harness/ErrorFallback.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/Harness/ErrorFallback.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import type { FallbackProps } from 'react-error-boundary'; import { COLORS, css, type t } from '../common.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/ui/Harness/Harness.tsx b/code/sys.ui/ui-react-devharness/src/ui/Harness/Harness.tsx index 0f55e13ace..387d939c51 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/Harness/Harness.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/Harness/Harness.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React, { useEffect, useRef, type FC } from 'react'; import { HarnessHost } from '../Harness.Host/mod.ts'; import { DebugPanel } from '../Harness.Panel.Debug/mod.ts'; @@ -73,7 +72,7 @@ export const Harness: FC<t.HarnessProps> = (props: t.HarnessProps) => { return ( <div - data-component={'dev.harness'} + data-component={'sys.ui.dev.harness'} ref={baseRef} className={css(styles.reset, styles.base, props.style).class} > diff --git a/code/sys.ui/ui-react-devharness/src/ui/ModuleList/-SPEC.tsx b/code/sys.ui/ui-react-devharness/src/ui/ModuleList/-SPEC.tsx index fb0e7c0213..babea86c00 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/ModuleList/-SPEC.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/ModuleList/-SPEC.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { Badges, COLORS, pkg, Spec, type t } from '../../-test.ts'; import { ModuleList } from './mod.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.Footer.tsx b/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.Footer.tsx index 414f59e88e..a0f0096275 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.Footer.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.Footer.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { Color, css, type t } from '../common.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.List.Item.tsx b/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.List.Item.tsx index 8f2a0b3cd9..d6e3b830c5 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.List.Item.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.List.Item.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React, { useEffect, useRef } from 'react'; import { VscSymbolClass } from 'react-icons/vsc'; import { COLORS, Calc, Color, DEFAULTS, css, type t } from './common.ts'; @@ -68,7 +67,7 @@ export const ListItem: React.FC<ListItemProps> = (props) => { }; /** - * Render + * Render: */ const { WHITE, BLUE } = COLORS; const color = Color.theme(props.theme).fg; diff --git a/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.List.tsx b/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.List.tsx index 3c486388ef..e9c754b296 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.List.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.List.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { VscSymbolClass } from 'react-icons/vsc'; import { Color, css, DEFAULTS, type t } from './common.ts'; @@ -29,7 +28,7 @@ export const List: React.FC<ListProps> = (props) => { const hasDevParam = url.searchParams.has(DEFAULTS.qs.dev); /** - * Render + * Render: */ const color = Color.theme(theme).fg; const styles = { diff --git a/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.Title.tsx b/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.Title.tsx index 2548855529..05cb3e766f 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.Title.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.Title.tsx @@ -1,44 +1,67 @@ import React from 'react'; -import { Color, css, type t } from '../common.ts'; +import { type t, Color, css, Str } from '../common.ts'; export type TitleProps = { enabled?: boolean; title?: string; version?: string; + dist?: t.DistPkg; badge?: t.ImageBadge; theme?: t.CommonTheme; style?: t.CssInput; }; export const Title: React.FC<TitleProps> = (props) => { + const { dist } = props; const title = props.title?.trim(); const badge = props.badge; if (!(title || badge)) return null; /** - * Render + * Render: */ - const color = Color.theme(props.theme).fg; + const theme = Color.theme(props.theme); + const color = theme.fg; const styles = { base: css({ display: 'grid', gridTemplateColumns: `1fr auto`, color }), left: css({ fontWeight: 'bold' }), right: css({ display: 'grid', alignContent: 'center' }), block: css({ display: 'block' }), + + title: css({ display: 'grid', gridTemplateColumns: `auto 1fr auto` }), version: css({ color: Color.alpha(color, 0.3), marginLeft: 3 }), + hash: css({ marginLeft: 10, fontWeight: 'normal' }), + link: css({ + color: Color.BLUE, + textDecoration: 'none', + ':hover': { textDecoration: 'underline' }, + }), }; + const linkProps = { target: '_blank', rel: 'noopener noreferrer' }; + const elBadge = badge && ( - <a href={badge?.href} target={'_blank'} rel={'noopener noreferrer'}> + <a href={badge?.href} {...linkProps}> <img className={styles.block.class} src={badge?.image} /> </a> ); + const elDist = dist && ( + <a className={css(styles.hash, styles.link).class} href={'./dist.json'} {...linkProps}> + {`${Str.bytes(dist.size.bytes)} ā dist.pkg:#${dist.hash.digest.slice(-5)}`} + </a> + ); + const elTitle = title && ( - <> - <span>{title}</span> - {props.version && <span className={styles.version.class}>{`@${props.version}`}</span>} - </> + <div className={styles.title.class}> + <div> + <span>{title}</span> + {props.version && <span className={styles.version.class}>{`@${props.version}`}</span>} + </div> + <div /> + <div>{elDist}</div> + </div> ); return ( diff --git a/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.tsx b/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.tsx index 7dca1b19a7..0efb35d66a 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/ModuleList/ui.tsx @@ -1,8 +1,7 @@ -// @ts-types="@types/react" import React from 'react'; import { useEffect, useRef } from 'react'; -import { Color, DEFAULTS, css, useRubberband, type t } from './common.ts'; +import { type t, useDist, Color, DEFAULTS, css, useRubberband } from './common.ts'; import { Footer } from './ui.Footer.tsx'; import { List } from './ui.List.tsx'; import { Title } from './ui.Title.tsx'; @@ -19,6 +18,7 @@ export const View: React.FC<t.ModuleListProps> = (props) => { const baseRef = useRef<HTMLDivElement>(null); const itemRefs = useRef<LiMap>(new Map<number, HTMLLIElement>()); + const dist = useDist({ useSampleFallback: true }); useRubberband(props.allowRubberband ?? false); useScrollObserver(baseRef, itemRefs.current, props.onItemVisibility); useScrollController(baseRef, itemRefs.current, props.scrollTo$); @@ -43,7 +43,7 @@ export const View: React.FC<t.ModuleListProps> = (props) => { }; /** - * Render + * Render: */ const color = Color.theme(theme).fg; const styles = { @@ -95,6 +95,7 @@ export const View: React.FC<t.ModuleListProps> = (props) => { <div className={styles.body.class}> <Title enabled={enabled} + dist={dist?.json} title={props.title} version={props.version} badge={props.badge} diff --git a/code/sys.ui/ui-react-devharness/src/ui/RenderCount/RenderCount.tsx b/code/sys.ui/ui-react-devharness/src/ui/RenderCount/RenderCount.tsx index 8f82fe7823..39adf6e878 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/RenderCount/RenderCount.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/RenderCount/RenderCount.tsx @@ -1,4 +1,3 @@ -// @ts-types="@types/react" import React from 'react'; import { COLORS, css, type t } from '../common.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/ui/Spinners/BarSpinner.tsx b/code/sys.ui/ui-react-devharness/src/ui/Spinners/BarSpinner.tsx index 965b3140e2..6af071dda5 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/Spinners/BarSpinner.tsx +++ b/code/sys.ui/ui-react-devharness/src/ui/Spinners/BarSpinner.tsx @@ -1,8 +1,8 @@ -// @ts-types="@types/react" import React from 'react'; import { COLORS, css, type t } from '../common.ts'; export type BarSpinnerProps = { style?: t.CssInput }; + type BarLoaderProps = { color?: string; width?: number }; let BarLoader: React.ComponentType<BarLoaderProps> | undefined; diff --git a/code/sys.ui/ui-react-devharness/src/ui/Spinners/mod.ts b/code/sys.ui/ui-react-devharness/src/ui/Spinners/mod.ts index 2c1dd04e4d..8e47f3b493 100644 --- a/code/sys.ui/ui-react-devharness/src/ui/Spinners/mod.ts +++ b/code/sys.ui/ui-react-devharness/src/ui/Spinners/mod.ts @@ -1 +1,5 @@ +/** + * @module + * Spinners + */ export * from './BarSpinner.tsx'; diff --git a/code/sys.ui/ui-react/deno.json b/code/sys.ui/ui-react/deno.json index 4349b3c91b..47a22958c0 100644 --- a/code/sys.ui/ui-react/deno.json +++ b/code/sys.ui/ui-react/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/ui-react", - "version": "0.0.84", + "version": "0.0.94", "license": "MIT", "tasks": { "test": "deno test -RW", @@ -13,6 +13,7 @@ "./t": "./src/types.ts", "./types": "./src/types.ts", "./fc": "./src/m.FC/mod.ts", + "./testing/server": "./src/m.Testing.Server/mod.ts", "./use": "./src/m.use/mod.ts" } } diff --git a/code/sys.ui/ui-react/src/-test.ts b/code/sys.ui/ui-react/src/-test.ts index f9f98a3ab9..70890c4c87 100644 --- a/code/sys.ui/ui-react/src/-test.ts +++ b/code/sys.ui/ui-react/src/-test.ts @@ -1,2 +1,2 @@ -export { Testing, describe, expect, it } from '@sys/testing/server'; +export { describe, DomMock, expect, it, Testing } from '@sys/testing/server'; export * from './common.ts'; diff --git a/code/sys.ui/ui-react/src/common/libs.ts b/code/sys.ui/ui-react/src/common/libs.ts index 8f225eff9f..3abb87f746 100644 --- a/code/sys.ui/ui-react/src/common/libs.ts +++ b/code/sys.ui/ui-react/src/common/libs.ts @@ -1 +1,2 @@ -export { Time } from '@sys/std'; +export { Err, rx, Time } from '@sys/std'; +export { css } from '@sys/ui-css'; diff --git a/code/sys.ui/ui-react/src/common/t.ts b/code/sys.ui/ui-react/src/common/t.ts index 27116ce081..4162e85e3b 100644 --- a/code/sys.ui/ui-react/src/common/t.ts +++ b/code/sys.ui/ui-react/src/common/t.ts @@ -1,4 +1,7 @@ -export type { Disposable, Lifecycle, Point, UntilObservable } from '@sys/types'; +export type { ReactNode } from 'react'; + +export type { CssEdgesArray, CssEdgesInput, CssInput } from '@sys/ui-css/t'; export type { KeyboardModifierFlags } from '@sys/ui-dom/t'; +export type * from '@sys/types'; export type * from '../types.ts'; diff --git a/code/sys.ui/ui-react/src/m.Signal/-.test.tsx b/code/sys.ui/ui-react/src/m.Signal/-.test.tsx new file mode 100644 index 0000000000..4d65b41321 --- /dev/null +++ b/code/sys.ui/ui-react/src/m.Signal/-.test.tsx @@ -0,0 +1,81 @@ +import React, { useState } from 'react'; + +import { describe, expect, it, Testing, Time } from '../-test.ts'; +import { TestReact } from '../m.Testing.Server/mod.ts'; +import { Signal } from './mod.ts'; + +import * as Preact from '@preact/signals-react'; + +/** + * See [@sys/std/signal] unit tests for basic API + * usage sceanrios: + * + * ⢠signal ā (update) + * ⢠effect ā (listen and react to change) + * ⢠computed ā (compound signal value) + * ⢠batch ā (multiple changes, single fire of effect listeners) + * + */ +describe('Signals', { sanitizeOps: false, sanitizeResources: false }, () => { + it('API', () => { + expect(Signal.create).to.equal(Preact.signal); + expect(Signal.effect).to.equal(Preact.effect); + expect(Signal.useSignal).to.equal(Preact.useSignal); + expect(Signal.useEffect).to.equal(Preact.useSignalEffect); + }); + + describe('React hooks', () => { + it('useSignal | useSignalEffect', async () => { + let fired: number[] = []; + + /** + * Render: into a DOM container. + */ + function TestComponent() { + const count = Signal.useSignal(0); + + const [, setRender] = useState(0); + const redraw = () => setRender((n) => n + 1); + + // Runs whenever `count.value` changes. + Signal.useEffect(() => { + fired.push(count.value); + redraw(); + }); + + const increment = () => (count.value += 1); + return ( + <div> + <button onClick={increment}>Increment</button> + <span>{count.value}</span> + </div> + ); + } + + const dom = await TestReact.render(<TestComponent />); + await Time.wait(10); + + /** + * Verify initial state. + */ + const button = dom.container.querySelector('button')!; + const span = dom.container.querySelector('span')!; + expect(span.textContent).to.eql('0'); + expect(fired).to.eql([0, 0]); // NB: initial render(s). + + /** + * Simulate a user click, then wait a microtask so React re-renders. + */ + fired = []; // ā (clear before test). + button.click(); + button.click(); + button.click(); + await Testing.wait(); + + expect(fired).to.eql([1, 2, 3]); + expect(span.textContent).to.equal('3'); + + dom.dispose(); + }); + }); +}); diff --git a/code/sys.ui/ui-react/src/m.Signal/common.ts b/code/sys.ui/ui-react/src/m.Signal/common.ts new file mode 100644 index 0000000000..4feaf9291e --- /dev/null +++ b/code/sys.ui/ui-react/src/m.Signal/common.ts @@ -0,0 +1,3 @@ +export { useSignal, useSignalEffect } from '@preact/signals-react'; +export { Signal as Std } from '@sys/std/signal'; +export * from '../common.ts'; diff --git a/code/sys.ui/ui-react/src/m.Signal/m.Signal.ts b/code/sys.ui/ui-react/src/m.Signal/m.Signal.ts new file mode 100644 index 0000000000..1b06a6a884 --- /dev/null +++ b/code/sys.ui/ui-react/src/m.Signal/m.Signal.ts @@ -0,0 +1,16 @@ +import { type t, Std, useSignalEffect as useEffect, useSignal } from './common.ts'; +import { useRedrawEffect } from './u.useRedrawEffect.ts'; + +/** + * Reactive Signals. + * See: + * https://github.com/tc39/proposal-signals + * https://preactjs.com/blog/introducing-signals/ + * https://preactjs.com/guide/v10/signals + */ +export const Signal: t.SignalReactLib = { + ...Std, + useSignal, + useEffect, + useRedrawEffect, +} as const; diff --git a/code/sys.ui/ui-react/src/m.Signal/mod.ts b/code/sys.ui/ui-react/src/m.Signal/mod.ts new file mode 100644 index 0000000000..679b40ce75 --- /dev/null +++ b/code/sys.ui/ui-react/src/m.Signal/mod.ts @@ -0,0 +1,5 @@ +/** + * @module + * Reactive Signals (via "preact/signals") + */ +export { Signal } from './m.Signal.ts'; diff --git a/code/sys.ui/ui-react/src/m.Signal/t.ts b/code/sys.ui/ui-react/src/m.Signal/t.ts new file mode 100644 index 0000000000..c41abbc420 --- /dev/null +++ b/code/sys.ui/ui-react/src/m.Signal/t.ts @@ -0,0 +1,26 @@ +import type Preact from '@preact/signals-react'; +import type { SignalLib } from '@sys/std/t'; + +export type { ExtractSignalValue, ReadonlySignal, Signal } from '@sys/std/t'; + +/** + * Reactive Signals. + * See: + * https://github.com/tc39/proposal-signals + * https://preactjs.com/blog/introducing-signals/ + * https://preactjs.com/guide/v10/signals + */ +export type SignalReactLib = SignalLib & { + useSignal: typeof Preact.useSignal; + useEffect: typeof Preact.useSignalEffect; + + /** + * Safely causes a redraw (via a useState counter incrementing) + * when any of the signals that are hooked into within the + * callback change value. + * + * Safe: == stops effect listeners on tear-down. + * + */ + useRedrawEffect(cb: () => void): void; +}; diff --git a/code/sys.ui/ui-react/src/m.Signal/u.useRedrawEffect.ts b/code/sys.ui/ui-react/src/m.Signal/u.useRedrawEffect.ts new file mode 100644 index 0000000000..1f6be3f960 --- /dev/null +++ b/code/sys.ui/ui-react/src/m.Signal/u.useRedrawEffect.ts @@ -0,0 +1,11 @@ +import { useState } from 'react'; +import { type t, Time, useSignalEffect } from './common.ts'; + +export const useRedrawEffect: t.SignalReactLib['useRedrawEffect'] = (cb) => { + const [, setRender] = useState(0); + const redraw = () => setRender((n) => n + 1); + useSignalEffect(() => { + cb(); + Time.delay(redraw); + }); +}; diff --git a/code/sys.ui/ui-react/src/m.Testing.Server/-.test.tsx b/code/sys.ui/ui-react/src/m.Testing.Server/-.test.tsx new file mode 100644 index 0000000000..c5c8cfd214 --- /dev/null +++ b/code/sys.ui/ui-react/src/m.Testing.Server/-.test.tsx @@ -0,0 +1,38 @@ +import { describe, expect, it } from '../-test.ts'; +import { TestReact } from './mod.ts'; + +describe('TestReact (Server)', { sanitizeOps: false, sanitizeResources: false }, () => { + describe('render', () => { + it('renders into DOM', async () => { + const el = ( + <div> + <span>Hello</span> + </div> + ); + + const res = await TestReact.render(el); + const span = res.container.querySelector('span')!; + expect(span.innerText).to.eql('Hello'); + }); + + it('lifecycle', async () => { + const res = await TestReact.render( + <div> + <span>Hello</span> + </div>, + ); + let count = 0; + res.dispose$.subscribe(() => count++); + + expect(res.disposed).to.eql(false); + expect(res.container.querySelector('span')!.innerText).to.eql('Hello'); + + res.dispose(); + expect(res.disposed).to.eql(true); + expect(count).to.eql(1); + + // NB: should not find. + expect(res.container.querySelector('span')).to.eql(null); + }); + }); +}); diff --git a/code/sys.ui/ui-react/src/m.Testing.Server/common.ts b/code/sys.ui/ui-react/src/m.Testing.Server/common.ts new file mode 100644 index 0000000000..26a269aec8 --- /dev/null +++ b/code/sys.ui/ui-react/src/m.Testing.Server/common.ts @@ -0,0 +1,2 @@ +export { DomMock, Testing } from '@sys/testing/server'; +export * from '../common.ts'; diff --git a/code/sys.ui/ui-react/src/m.Testing.Server/m.TestReact.ts b/code/sys.ui/ui-react/src/m.Testing.Server/m.TestReact.ts new file mode 100644 index 0000000000..f1ae475076 --- /dev/null +++ b/code/sys.ui/ui-react/src/m.Testing.Server/m.TestReact.ts @@ -0,0 +1,9 @@ +import { type t } from './common.ts'; +import { render } from './u.render.tsx'; + +/** + * Renders an element into the service-side test DOM. + */ +export const TestReact: t.TestReactServerLib = { + render, +}; diff --git a/code/sys.ui/ui-react/src/m.Testing.Server/mod.ts b/code/sys.ui/ui-react/src/m.Testing.Server/mod.ts new file mode 100644 index 0000000000..7d69c716ae --- /dev/null +++ b/code/sys.ui/ui-react/src/m.Testing.Server/mod.ts @@ -0,0 +1,6 @@ +/** + * @module + * Tools for testing React on the server. + */ +export { DomMock, Testing } from './common.ts'; +export { TestReact } from './m.TestReact.ts'; diff --git a/code/sys.ui/ui-react/src/m.Testing.Server/t.ts b/code/sys.ui/ui-react/src/m.Testing.Server/t.ts new file mode 100644 index 0000000000..a72b9258a9 --- /dev/null +++ b/code/sys.ui/ui-react/src/m.Testing.Server/t.ts @@ -0,0 +1,23 @@ +import type React from 'react'; +import type { t } from './common.ts'; + +/** + * Tools for testing React on the server. + */ +export type TestReactServerLib = { + /** Renders an element into the service-side test DOM. */ + render(el: React.ReactNode, options?: TestReactRenderOptions): Promise<t.TestReactRendered>; +}; + +/** Options passed to the `TestReact.render` method. */ +export type TestReactRenderOptions = { + /** Flag indicating if the element should be contained within the <StrictMode> render container. */ + strict?: boolean; +}; + +/** + * A server-rendered react component. + */ +export type TestReactRendered = t.Lifecycle & { + readonly container: HTMLDivElement; +}; diff --git a/code/sys.ui/ui-react/src/m.Testing.Server/u.render.tsx b/code/sys.ui/ui-react/src/m.Testing.Server/u.render.tsx new file mode 100644 index 0000000000..61211d6bdc --- /dev/null +++ b/code/sys.ui/ui-react/src/m.Testing.Server/u.render.tsx @@ -0,0 +1,40 @@ +import React, { StrictMode } from 'react'; +import { createRoot } from 'react-dom/client'; + +import { type t, DomMock, rx, Time } from './common.ts'; + +/** + * Renders an element into the service-side test DOM. + */ +export const render: t.TestReactServerLib['render'] = async (el, options = {}) => { + const { strict = true } = options; + DomMock.polyfill(); + + const life = rx.lifecycle(); + life.dispose$.subscribe(() => root.unmount()); + + // Construct the server-side DOM. + const container = document.createElement('div'); + document.body.appendChild(container); + const root = createRoot(container); + + // Render into DOM. + if (strict) el = <StrictMode>{el}</StrictMode>; + root.render(el); + await Time.wait(0); // NB: ensures the render completes before returning. + + /** + * API + */ + const api: t.TestReactRendered = { + container, + + // Lifecycle. + dispose: life.dispose, + dispose$: life.dispose$, + get disposed() { + return life.disposed; + }, + }; + return api; +}; diff --git a/code/sys.ui/ui-react/src/m.use/common.ts b/code/sys.ui/ui-react/src/m.use/common.ts new file mode 100644 index 0000000000..8cae67176a --- /dev/null +++ b/code/sys.ui/ui-react/src/m.use/common.ts @@ -0,0 +1 @@ +export * from '../common.ts'; diff --git a/code/sys.ui/ui-react/src/m.use/mod.ts b/code/sys.ui/ui-react/src/m.use/mod.ts index 066fc8fac4..0a30d25897 100644 --- a/code/sys.ui/ui-react/src/m.use/mod.ts +++ b/code/sys.ui/ui-react/src/m.use/mod.ts @@ -10,11 +10,13 @@ * useMouseDrag, * useClickInside, * useClickOutside, + * useSizeObserver, * } from '@sys/ui-react/use'; * ``` - * */ - -export * from './useClick.ts'; -export * from './useMouse.Drag.ts'; -export * from './useMouse.ts'; +export * from './use.Click.ts'; +export * from './use.Dist.ts'; +export * from './use.Is.TouchSupported.ts'; +export * from './use.Mouse.Drag.ts'; +export * from './use.Mouse.ts'; +export * from './use.SizeObserver.tsx'; diff --git a/code/sys.ui/ui-react/src/m.use/t.ts b/code/sys.ui/ui-react/src/m.use/t.ts index dba04b77ce..849f92b713 100644 --- a/code/sys.ui/ui-react/src/m.use/t.ts +++ b/code/sys.ui/ui-react/src/m.use/t.ts @@ -1,126 +1,6 @@ -import type { RefObject, MouseEventHandler } from 'react'; -import type { t } from '../common.ts'; - -type E = HTMLElement; -type Div = HTMLDivElement; -type M = MouseEventHandler; -type MouseCallback = (e: MouseEvent) => void; - -/** - * Properties passed to the `useMouse` hook. - */ -export type UseMouseProps = { - onDown?: M; - onUp?: M; - onEnter?: M; - onLeave?: M; - onDrag?: UseMouseDragHandler; -}; - -/** - * Hook: keep track of mouse events for an HTML element - */ -export type UseMouseHook = (props?: t.UseMouseProps) => t.UseMouse; - -/** - * Hook: keep track of mouse events for an HTML element. - * Usage: - * - * const mouse = useMouse(); - * <div {...mouse.handlers} /> - */ -export type UseMouse = { - readonly is: { readonly over: boolean; readonly down: boolean; readonly dragging: boolean }; - readonly handlers: { onMouseDown: M; onMouseUp: M; onMouseEnter: M; onMouseLeave: M }; - readonly drag?: t.MouseMovement; - reset(): void; -}; - -/** - * Hook: keep track of a mouse drag operation. - */ -export type UseMouseDragHook = (props?: t.UseMouseDragProps) => t.UseMouseDrag; - -/** - * Properties passed to the `useMouseDrag` hook. - */ -export type UseMouseDragProps = { onDrag?: t.UseMouseDragHandler }; - -/** - * Hook: information about a mouse drag operation. - */ -export type UseMouseDrag = { - readonly is: { readonly dragging: boolean }; - readonly enabled: boolean; - readonly movement?: t.MouseMovement; - start(): void; - cancel(): void; -}; - -/** - * Hook: information about the movement of a mouse cursor. - */ -export type MouseMovement = { - readonly x: number; - readonly y: number; - readonly movement: t.Point; - readonly client: t.Point; - readonly page: t.Point; - readonly offset: t.Point; - readonly screen: t.Point; - readonly button: number; - readonly modifiers: t.KeyboardModifierFlags; -}; - -/** - * Event handler for a mouse-drag operation. - */ -export type UseMouseDragHandler = (e: UseMouseDragHandlerArgs) => void; - -/** - * Agrument supplied to the mouse-drag handler. - */ -export type UseMouseDragHandlerArgs = MouseMovement & { cancel(): void }; - -/** - * Hook: information about a mouse click operations - */ -export type UseClickHook<T extends E = Div> = (input: t.UseClickInput<T>) => UseClick<T>; - -/** - * Hook: Monitors for click events outside the given element. - * Usage: - * Useful for clicking away from modal dialogs or popups. - */ -export type UseClickOutsideHook<T extends E = Div> = UseClickHook<T>; - -/** - * Hook: Monitors for click events within the given element. - * Usage: - * Useful for clicking away from modal dialogs or popups. - */ -export type UseClickInsideHook<T extends E = Div> = UseClickHook<T>; - -/** - * Input passed to the UseClick hook. - */ -export type UseClickInput<T extends E> = t.UseClickProps<T> | MouseCallback; - -/** - * Hook: information about a mouse click operations - */ -export type UseClick<T extends E> = { - readonly ref: RefObject<T>; - readonly stage: t.UseClickStage; -}; - -/** - * Properties passed to the `UseClick` hook. - */ -export type UseClickProps<T extends E> = { - stage?: t.UseClickStage; - ref?: RefObject<T>; - callback?: MouseCallback; -}; - -export type UseClickStage = 'down' | 'up'; +export type * from './t.use.Click.ts'; +export type * from './t.use.Dist.ts'; +export type * from './t.use.Is.ts'; +export type * from './t.use.Mouse.Drag.ts'; +export type * from './t.use.Mouse.ts'; +export type * from './t.use.SizeObserver.ts'; diff --git a/code/sys.ui/ui-react/src/m.use/t.use.Click.ts b/code/sys.ui/ui-react/src/m.use/t.use.Click.ts new file mode 100644 index 0000000000..c88005346a --- /dev/null +++ b/code/sys.ui/ui-react/src/m.use/t.use.Click.ts @@ -0,0 +1,31 @@ +import type { RefObject } from 'react'; +import type { t } from './common.ts'; + +type E = HTMLElement; +type Div = HTMLDivElement; +type MouseCallback = (e: MouseEvent) => void; + +/** + * Hook: information about a mouse click operations + */ +export type UseClickHook<T extends E = Div> = (input: t.UseClickInput<T>) => ClickHook<T>; +/** Input passed to the UseClick hook. */ +export type UseClickInput<T extends E> = t.ClickHookProps<T> | MouseCallback; + +/** + * Hook: information about a mouse click operations + */ +export type ClickHook<T extends E> = { + readonly ref: RefObject<T>; + readonly stage: t.UseClickStage; +}; + +/** + * Properties passed to the `UseClick` hook. + */ +export type ClickHookProps<T extends E> = { + stage?: t.UseClickStage; + ref?: RefObject<T>; + callback?: MouseCallback; +}; +export type UseClickStage = 'down' | 'up'; diff --git a/code/sys.ui/ui-react/src/m.use/t.use.Dist.ts b/code/sys.ui/ui-react/src/m.use/t.use.Dist.ts new file mode 100644 index 0000000000..b5ce92e475 --- /dev/null +++ b/code/sys.ui/ui-react/src/m.use/t.use.Dist.ts @@ -0,0 +1,12 @@ +import type { t } from './common.ts'; + +/** + * Hook: Load the `dist.json` file from the server (if avilable). + */ +export type UseDistFactory = (options?: { useSampleFallback?: boolean }) => UseDist; +export type UseDist = { + readonly count: number; + readonly is: { readonly sample: boolean }; + readonly json?: t.DistPkg; + readonly error?: t.StdError; +}; diff --git a/code/sys.ui/ui-react/src/m.use/t.use.Is.ts b/code/sys.ui/ui-react/src/m.use/t.use.Is.ts new file mode 100644 index 0000000000..c5aa7e08af --- /dev/null +++ b/code/sys.ui/ui-react/src/m.use/t.use.Is.ts @@ -0,0 +1,4 @@ +/** + * Hook: detect if the device supports touch events. + */ +export type UseIsTouchSupported = () => boolean; diff --git a/code/sys.ui/ui-react/src/m.use/t.use.Mouse.Drag.ts b/code/sys.ui/ui-react/src/m.use/t.use.Mouse.Drag.ts new file mode 100644 index 0000000000..d82b29adbb --- /dev/null +++ b/code/sys.ui/ui-react/src/m.use/t.use.Mouse.Drag.ts @@ -0,0 +1,47 @@ +import type { t } from './common.ts'; + +/** + * Hook Factory: keep track of a mouse drag operation. + */ +export type UseMouseDrag = (props?: t.MouseDragHookProps) => t.MouseDragHook; + +/** + * Properties passed to the `useMouseDrag` hook. + */ +export type MouseDragHookProps = { onDrag?: t.UseMouseDragHandler }; + +/** + * Hook: information about a mouse drag operation. + */ +export type MouseDragHook = { + readonly is: { readonly dragging: boolean }; + readonly enabled: boolean; + readonly movement?: t.MouseMovement; + start(): void; + cancel(): void; +}; + +/** + * Hook: information about the movement of a mouse cursor. + */ +export type MouseMovement = { + readonly x: number; + readonly y: number; + readonly movement: t.Point; + readonly client: t.Point; + readonly page: t.Point; + readonly offset: t.Point; + readonly screen: t.Point; + readonly button: number; + readonly modifiers: t.KeyboardModifierFlags; +}; + +/** + * Event handler for a mouse-drag operation. + */ +export type UseMouseDragHandler = (e: UseMouseDragHandlerArgs) => void; + +/** + * Agrument supplied to the mouse-drag handler. + */ +export type UseMouseDragHandlerArgs = MouseMovement & { cancel(): void }; diff --git a/code/sys.ui/ui-react/src/m.use/t.use.Mouse.ts b/code/sys.ui/ui-react/src/m.use/t.use.Mouse.ts new file mode 100644 index 0000000000..0cd90ab3e8 --- /dev/null +++ b/code/sys.ui/ui-react/src/m.use/t.use.Mouse.ts @@ -0,0 +1,34 @@ +import type { MouseEventHandler } from 'react'; +import type { t } from './common.ts'; + +type M = MouseEventHandler; + +/** + * Hook Factory: keep track of mouse events for an HTML element + */ +export type UseMouse = (props?: t.MouseHookProps) => t.MouseHook; + +/** + * Hook: keep track of mouse events for an HTML element. + * Usage: + * + * const mouse = useMouse(); + * <div {...mouse.handlers} /> + */ +export type MouseHook = { + readonly is: { readonly over: boolean; readonly down: boolean; readonly dragging: boolean }; + readonly handlers: { onMouseDown: M; onMouseUp: M; onMouseEnter: M; onMouseLeave: M }; + readonly drag?: t.MouseMovement; + reset(): void; +}; + +/** + * Properties passed to the `useMouse` hook. + */ +export type MouseHookProps = { + onDown?: M; + onUp?: M; + onEnter?: M; + onLeave?: M; + onDrag?: t.UseMouseDragHandler; +}; diff --git a/code/sys.ui/ui-react/src/m.use/t.use.SizeObserver.ts b/code/sys.ui/ui-react/src/m.use/t.use.SizeObserver.ts new file mode 100644 index 0000000000..23e9a7ee8b --- /dev/null +++ b/code/sys.ui/ui-react/src/m.use/t.use.SizeObserver.ts @@ -0,0 +1,60 @@ +import type { RefCallback } from 'react'; +import type { t } from './common.ts'; + +/** + * Hook Factory: monitor size changes to a DOM element using [ResizeObserver]. + */ +export type UseSizeObserver = <T extends HTMLElement>( + onChange?: t.SizeObserverChangeHandler, +) => t.SizeObserverHook<T>; + +/** + * Hook: monitor size changes to a DOM element using [ResizeObserver]. + */ +export type SizeObserverHook<T extends HTMLElement> = { + /** Callback ref to be assigned to the element to observe. */ + readonly ref: RefCallback<T>; + + /** Flag indicating if the first render-pass has allowed a measurement. */ + readonly ready: boolean; + + /** The latest dimensions of the element (or null if not measured yet). */ + readonly rect?: DOMRectReadOnly; + + /** The `rect.width` value. */ + readonly width?: t.Pixels; + + /** The `rect.height` value. */ + readonly height: t.Pixels; + + /** Convert to a simple object. */ + toObject(): t.DomRect; + + /** Create a string representation of the size. */ + toString(): string; + + /** Renders a react element "<width> x <height>". */ + toElement(props?: SizeObserverElementProps | t.CssEdgesArray): t.ReactNode; +}; + +/** + * Fires when the size of an observed DOM element changes. + */ +export type SizeObserverChangeHandler = (e: SizeObserverChangeHandlerArgs) => void; +export type SizeObserverChangeHandlerArgs = { + readonly rect: DOMRectReadOnly; + toObject(): t.DomRect; +}; + +/** + * Properties for the size rendered as an element ("<width> x <height>"). + */ +export type SizeObserverElementProps = { + /** Display: "block" (false, default) or inline-block (true). */ + inline?: boolean; + visible?: boolean; + fontSize?: number; + opacity?: number; + style?: t.CssInput; + Absolute?: t.CssEdgesInput; +}; diff --git a/code/sys.ui/ui-react/src/m.use/useClick.ts b/code/sys.ui/ui-react/src/m.use/use.Click.ts similarity index 63% rename from code/sys.ui/ui-react/src/m.use/useClick.ts rename to code/sys.ui/ui-react/src/m.use/use.Click.ts index 831d61e9ad..724ffff595 100644 --- a/code/sys.ui/ui-react/src/m.use/useClick.ts +++ b/code/sys.ui/ui-react/src/m.use/use.Click.ts @@ -1,44 +1,48 @@ import { useEffect, useRef } from 'react'; -import type { t } from '../common.ts'; +import type { t } from './common.ts'; type E = HTMLElement; type Div = HTMLDivElement; type MouseEventName = 'mousedown' | 'mouseup'; - const DEFAULT_STAGE: t.UseClickStage = 'down'; +/** + * Hook: Monitors for click events outside the given element. + * Usage: + * Useful for clicking away from modal dialogs or popups. + */ export const useClickOutside: t.UseClickHook = <T extends E = Div>(input: t.UseClickInput<T>) => { - const { callback, stage, ref, event } = wrangle.args<T>(input); - - useEffect(() => { - const handler = (e: MouseEvent) => { - if (!ref.current?.contains(e.target as E)) callback?.(e); - }; - document?.addEventListener(event, handler, true); - return () => document?.removeEventListener(event, handler, true); - }, [event, ref, callback]); - - const api: t.UseClick<T> = { ref, stage }; - return api; + return useHandler<T>(input, (el, e) => !el.contains(e.target as E)); }; +/** + * Hook: Monitors for click events within the given element. + * Usage: + * Useful for clicking away from modal dialogs or popups. + */ export const useClickInside: t.UseClickHook = <T extends E = Div>(input: t.UseClickInput<T>) => { - const { callback, stage, ref, event } = wrangle.args<T>(input); + return useHandler<T>(input, (el, e) => el.contains(e.target as E)); +}; +/** + * Helpers + */ +function useHandler<T extends E = Div>( + input: t.UseClickInput<T>, + shouldInvoke: (el: E, e: MouseEvent) => boolean, +): t.ClickHook<T> { + const { callback, stage, ref, event } = wrangle.args<T>(input); useEffect(() => { const handler = (e: MouseEvent) => { - if (ref.current?.contains(e.target as E)) callback?.(e); + const el = ref.current; + if (el && shouldInvoke(el, e)) callback?.(e); }; document?.addEventListener(event, handler, true); return () => document?.removeEventListener(event, handler, true); }, [event, ref, callback]); + return { ref, stage }; +} - return { ref, stage } as const; -}; - -/** - * Helpers - */ const wrangle = { args<T extends E>(input: t.UseClickInput<T>) { const { callback, stage = DEFAULT_STAGE, ref = useRef<T>(null) } = wrangle.input<T>(input); @@ -46,7 +50,7 @@ const wrangle = { return { callback, event, ref, stage }; }, - input<T extends E>(input: t.UseClickInput<T>): t.UseClickProps<T> { + input<T extends E>(input: t.UseClickInput<T>): t.ClickHookProps<T> { if (typeof input === 'object') return input; if (typeof input === 'function') return { callback: input }; throw new Error('Unable to parse parameter input'); diff --git a/code/sys.ui/ui-react/src/m.use/use.Dist.sample.ts b/code/sys.ui/ui-react/src/m.use/use.Dist.sample.ts new file mode 100644 index 0000000000..5333b1ae6f --- /dev/null +++ b/code/sys.ui/ui-react/src/m.use/use.Dist.sample.ts @@ -0,0 +1,70 @@ +import { type t } from './common.ts'; + +export const sample: t.DistPkg = { + '-type:': 'jsr:@sys/types:DistPkg', + pkg: { + name: 'sample', + version: '0.0.0-sample.0', + }, + size: { + bytes: 1119584, + }, + entry: 'pkg/-entry.CsmS4pX8.js', + hash: { + digest: `sha256-0000000000000000000000000000000000000000000000000000000000000000`, + parts: { + 'index.html': `sha256-73118ead01ff731394195389afe6b3fef53f366c92013f90d4098523ef783dc1`, + 'pkg/-entry.CsmS4pX8.js': `sha256-d7c1df2d1d7ccadd611582d340245b446930d9c552b9a6e390e573fcb8069281`, + 'pkg/-pkg.json': `sha256-5cb551c509271d47374b378e56b137f2c30b4ad629fa12836be67f2c896e4565`, + 'pkg/a.CzfbkkOd.svg': `sha256-03dc57becc131b98626bf145a3827966fec98f60e6c284bce7d73a0f137904b7`, + 'pkg/a.DytWQd0P.svg': `sha256-548482b24965277eadcafc4cb299c9b37f7a6583f7ec624dff76b7a4501022f1`, + 'pkg/a.ccVBcAen.css': `sha256-1d6d1a552eabe1df220ee93e445c4f5eb150fdfae998996cdf1f9c60d662c35b`, + 'pkg/m.2f5gzOW6.js': `sha256-5f08d0a5ed787b51ff0e3c8a140668b95fe77cfbabb9fa1b8df085029479bcf5`, + 'pkg/m.6enj-n6N.js': `sha256-3c1100938d2bb3674f3cb696df5b5096151115c024752f7401e84af9327c01d1`, + 'pkg/m.6tNfJC1I.js': `sha256-ae42722c296c1f201a48042a6eaeed858de98585c6d7e44114124b63d1dd5708`, + 'pkg/m.B07hVUpE.js': `sha256-ac6c000836d8e949047f0f367987c38ea81ef14e4963d1e722b291b0bd6ebda3`, + 'pkg/m.B0X5Tzpk.js': `sha256-c4131f76f53fdb76802f60bf7679ccd79a0e390c77b15e7f91b0508a99f954cb`, + 'pkg/m.B7Cwm47C.js': `sha256-3a6e56f9f3e1a717e767f39b296065b868498bdbb7dd6e2fbec75c5c1cc75baf`, + 'pkg/m.B9s02YtB.js': `sha256-5dbfbcbdfd1f50af03605883b36550a847179f6a9f778024ab614537c610a545`, + 'pkg/m.BP6klHZP.js': `sha256-8a7cda4c9f20ade3b84b838c1c1e67cf92a4a93eaebf5744e0fc91b7a0f01608`, + 'pkg/m.BVKZFYvx.js': `sha256-e179d583ae484267be2adda81a83ae0b5c4c521013d4ba4d5291eed7a46414f8`, + 'pkg/m.Ba70fsEO.js': `sha256-af588a67a5779dd492ed3fc6726d6a35d0003bbabba92e45959303912823a899`, + 'pkg/m.Bb98Nh4K.js': `sha256-06e575191386b05b18d7b56f5d5815a0d32fde5908e6043fb50e988397f9516f`, + 'pkg/m.BbGMuf27.js': `sha256-15618e12d24c5ba50d643815909a3001d531a0934ad5375bf5bd425992722742`, + 'pkg/m.BdteEkmZ.js': `sha256-cbd24c58fa9a3608d58919c463d4b83452452b7f998ccbdfefa0a4d629bb92c2`, + 'pkg/m.BsznwMR0.js': `sha256-0e029fcf1770c1e3eca2190a51629683765a744a4c843bdf9973451eae8a0158`, + 'pkg/m.By5VB92O.js': `sha256-02f2948e9425f86b36acc791d162a8fba92e63f71752f1925b467a1df72e470d`, + 'pkg/m.C2DGtZV2.js': `sha256-b50212068d7556c7c3a270826987451a40b32580cbd8dd005bac94b9a0f109c3`, + 'pkg/m.CDGHaSNw.js': `sha256-bf76f8f2ca47251fda3ce17238c6dd3267b115ea20c92b5883a13477ca3045fa`, + 'pkg/m.CFTGxaG4.js': `sha256-66fb099e30d23f8b59f85e6c37528146c9feb73cb5b3961c83f24e6a590858b1`, + 'pkg/m.CHDC_kiv.js': `sha256-ca030dabf34384d92bc2eb4cc750ecc500e175f4835012425616bdef583e7aa7`, + 'pkg/m.CKXg8R_g.js': `sha256-6a2069d6fb2a8cac27f1295d63214574d72950e0bda2da6cf49db6464cae43dd`, + 'pkg/m.CbXFlvb7.js': `sha256-ce20c4bd834cb948ba8868a404f315fecc4870c3672ee4bffd6f924e28f8cca3`, + 'pkg/m.CfKhKnmU.js': `sha256-f704ea57f6887e969b23dd68a81e6b1b2cf87180cb10d1cd731b9e58863135a6`, + 'pkg/m.Ct43jJ6a.js': `sha256-25e470e34075596c9b16366110b3d5bb7b1fc53df5c840cc2d79bb1fcd77de28`, + 'pkg/m.D0Y5-x_C.js': `sha256-05b222be0f2534df761e1debafeae232c01a4d751052a322ca5a685797a54773`, + 'pkg/m.DHs9BToK.js': `sha256-74f9c41114fafa8802c9ce463701a81b9d20d6946c264fa9926e14ca79481ae8`, + 'pkg/m.DKz1sTmA.js': `sha256-c03856d8265824d1aa72ce4e79e43fe03a65ff006b4d9abd41844b3d324a7488`, + 'pkg/m.DM43RIpC.js': `sha256-aacf9d80efe2de10ca655096ea827ab800bc3cbac057812f3cd9e6cf569767f1`, + 'pkg/m.DQgAKyyW.js': `sha256-10716acd89135dc8a996248c715984ae43dbb27d4ddb46177e1e87262d997c2f`, + 'pkg/m.DQt_l2DB.js': `sha256-d21dfba34b433bc9094a08781f42e3def01a757f676b6c61d9883a2f4b14a549`, + 'pkg/m.DXE146RK.js': `sha256-6587c734ae87115dc4bf32f6bf69138b4cba932216f5ee77000ea4f23139a3d7`, + 'pkg/m.DXPUWGVB.js': `sha256-3871a837d066052dc182e679a4d2be70811288c4339d2d777388072d2a7e8cea`, + 'pkg/m.D_9-kWoO.js': `sha256-dd7e221ca2fc58d59037a993b665812b4f0bb66b2254809faceb2a8c5f1584d7`, + 'pkg/m.Ddt9-e-a.js': `sha256-e827694e986a42511675c90a792219523dfb99032487bbd5b637cd507ff784cb`, + 'pkg/m.Djn_Vfvz.js': `sha256-70e2ee39f26a86fac3ed0b5bf74b8bbc4de933742d1c73ede33de10032848610`, + 'pkg/m.DrA-il_j.js': `sha256-0037ace80faf4f1a853aaf9fdcbb4705d15d8abeca787084d52974917e84867d`, + 'pkg/m.DrIctJra.js': `sha256-7b836d03e688109126ffd162b6c559a42d26bf448de166ae7b165a8199b33be4`, + 'pkg/m.DsmzfuLH.js': `sha256-9d4d8e9bdb88c1e83aaaf11471de9017a7cc7df854f40541825e8dab4aeeea00`, + 'pkg/m.DtyQKGjb.js': `sha256-371e390c42e41ae478585eb8304d9cb27407a584921811d7980c3e1e9a83dc29`, + 'pkg/m.DzC4PHOS.js': `sha256-04ae4f600c2b51bb4201b2112afbc376f7791d8b44745ab8e67e5e73ae4a93eb`, + 'pkg/m.FxkjFhCh.js': `sha256-c724be1906d07ecc719c89c5043408f4ad4e5dfa516010d0d50e41758ce0fb35`, + 'pkg/m.JsoYGwwg.js': `sha256-5bdaf5acfd33691ef753bac1732c5564ba04bde4679d259f7a43dad12d7b0429`, + 'pkg/m.Vh11PLjH.js': `sha256-9cd18978f95f944ead11d863a90b483e1658a562c397b7251bb882451061f1d1`, + 'pkg/m.hGJfSRVd.js': `sha256-6aa41025318cfb8f947a12d09952ce080e6e378a4ca4ac0f345b5e02555cc9e6`, + 'pkg/m.ha9d0YpI.js': `sha256-01d0fc7de0feb7b1e291605122020cc18e613455dee99170bd4840d49e71ea19`, + 'pkg/m.kNRDy6Ji.js': `sha256-957cb7554527f2aa004eb576e7738903a4869a00ae3388b26fb05aa61f203607`, + 'pkg/m.txGTKLjD.js': `sha256-00306b58d1f2a9f767df5ee8f36ee497a5858b8209dd52cef7162314b2b55d59`, + }, + }, +}; diff --git a/code/sys.ui/ui-react/src/m.use/use.Dist.ts b/code/sys.ui/ui-react/src/m.use/use.Dist.ts new file mode 100644 index 0000000000..4a92b31004 --- /dev/null +++ b/code/sys.ui/ui-react/src/m.use/use.Dist.ts @@ -0,0 +1,65 @@ +import { Http } from '@sys/http'; +import { useEffect, useRef, useState } from 'react'; +import { type t, Err } from './common.ts'; + +/** + * Hook: Load the `dist.json` file from the server (if avilable). + */ +export const useDist: t.UseDistFactory = (options = {}) => { + const { useSampleFallback = false } = options; + const is: t.UseDist['is'] = { sample: useSampleFallback }; + + const [count, setRender] = useState(0); + const redraw = () => setRender((n) => n + 1); + + const jsonRef = useRef<t.DistPkg>(); + const [error, setError] = useState<t.StdError>(); + + /** + * Effect: Fetch JSON (or optionally load sample data). + */ + useEffect(() => { + const fetch = Http.fetch(); + + const update = (dist?: t.DistPkg) => { + jsonRef.current = dist; + redraw(); + }; + + const loadJson = async () => { + jsonRef.current = undefined; + const res = await fetch.json<t.DistPkg>('./dist.json'); + if (fetch.disposed) return; + if (res.ok) update(res.data); + else { + setError(Err.std(res.error)); + } + }; + + const loadSample = async () => { + jsonRef.current = undefined; + const m = await import('./use.Dist.sample.ts'); + update(m?.sample); + }; + + const finish = async () => { + if (!jsonRef.current && is.sample) loadSample(); + }; + + loadJson().then(finish).catch(finish); + return fetch.dispose; + }, [is.sample]); + + /** + * API + */ + const api: t.UseDist = { + count, + is, + error, + get json() { + return jsonRef.current; + }, + }; + return api; +}; diff --git a/code/sys.ui/ui-react/src/m.use/use.Is.TouchSupported.ts b/code/sys.ui/ui-react/src/m.use/use.Is.TouchSupported.ts new file mode 100644 index 0000000000..65f5363901 --- /dev/null +++ b/code/sys.ui/ui-react/src/m.use/use.Is.TouchSupported.ts @@ -0,0 +1,8 @@ +import type { t } from './common.ts'; + +/** + * Hook: detect if the device supports touch events. + */ +export const useIsTouchSupported: t.UseIsTouchSupported = () => { + return typeof window !== 'undefined' && 'ontouchstart' in window; +}; diff --git a/code/sys.ui/ui-react/src/m.use/useMouse.Drag.ts b/code/sys.ui/ui-react/src/m.use/use.Mouse.Drag.ts similarity index 93% rename from code/sys.ui/ui-react/src/m.use/useMouse.Drag.ts rename to code/sys.ui/ui-react/src/m.use/use.Mouse.Drag.ts index d3c18f8494..91719deb94 100644 --- a/code/sys.ui/ui-react/src/m.use/useMouse.Drag.ts +++ b/code/sys.ui/ui-react/src/m.use/use.Mouse.Drag.ts @@ -1,10 +1,10 @@ import { useEffect, useState } from 'react'; -import type { t } from '../common.ts'; +import type { t } from './common.ts'; /** * Internal hook that trackes mouse movement events (drag). */ -export const useMouseDrag: t.UseMouseDragHook = (props = {}) => { +export const useMouseDrag: t.UseMouseDrag = (props = {}) => { const enabled = Boolean(props.onDrag); const [dragging, setDragging] = useState(false); const [movement, setMovement] = useState<t.MouseMovement>(); @@ -47,7 +47,7 @@ export const useMouseDrag: t.UseMouseDragHook = (props = {}) => { /** * API */ - const api: t.UseMouseDrag = { + const api: t.MouseDragHook = { is: { dragging }, enabled, movement, diff --git a/code/sys.ui/ui-react/src/m.use/useMouse.ts b/code/sys.ui/ui-react/src/m.use/use.Mouse.ts similarity index 87% rename from code/sys.ui/ui-react/src/m.use/useMouse.ts rename to code/sys.ui/ui-react/src/m.use/use.Mouse.ts index b6808fe61e..431fdab3ce 100644 --- a/code/sys.ui/ui-react/src/m.use/useMouse.ts +++ b/code/sys.ui/ui-react/src/m.use/use.Mouse.ts @@ -1,6 +1,6 @@ import { useState } from 'react'; -import type { t } from '../common.ts'; -import { useMouseDrag } from './useMouse.Drag.ts'; +import type { t } from './common.ts'; +import { useMouseDrag } from './use.Mouse.Drag.ts'; /** * Hook: keep track of mouse events for an HTML element @@ -8,9 +8,8 @@ import { useMouseDrag } from './useMouse.Drag.ts'; * * const mouse = useMouse(); * <div {...mouse.handlers} /> - * */ -export const useMouse: t.UseMouseHook = (props = {}) => { +export const useMouse: t.UseMouse = (props = {}) => { const { onDrag } = props; const [isDown, setDown] = useState(false); const [isOver, setOver] = useState(false); @@ -38,7 +37,7 @@ export const useMouse: t.UseMouseHook = (props = {}) => { /** * API */ - const api: t.UseMouse = { + const api: t.MouseHook = { is: { over: isOver, down: isDown, dragging: drag.is.dragging }, handlers: { onMouseDown, onMouseUp, onMouseEnter, onMouseLeave }, drag: drag.movement, diff --git a/code/sys.ui/ui-react/src/m.use/use.SizeObserver.tsx b/code/sys.ui/ui-react/src/m.use/use.SizeObserver.tsx new file mode 100644 index 0000000000..ca541f15a5 --- /dev/null +++ b/code/sys.ui/ui-react/src/m.use/use.SizeObserver.tsx @@ -0,0 +1,92 @@ +import { useCallback, useEffect, useState } from 'react'; +import { type t, css } from './common.ts'; + +/** + * Hook Factory: monitor size changes to a DOM element using [ResizeObserver]. + */ +export const useSizeObserver: t.UseSizeObserver = <T extends HTMLElement>( + onChange?: t.SizeObserverChangeHandler, +) => { + const ref = useCallback((el: T | null) => setElement(el), []); + + const [element, setElement] = useState<T | null>(null); + const [rect, setRect] = useState<DOMRectReadOnly | undefined>(); + + /** + * Effect: monitor DOM element size. + */ + useEffect(() => { + if (!element) return; + if (typeof ResizeObserver !== 'function') return; + + const observer = new ResizeObserver((entries) => { + entries + .filter((entry) => entry.target === element) + .forEach((entry) => setRect(entry.contentRect)); + }); + + observer.observe(element); + return () => observer.disconnect(); + }, [element]); + + /** + * Effect: alert listeners on change. + */ + useEffect(() => { + if (rect) onChange?.({ rect, toObject }); + }, [rect]); + + /** + * API + */ + const toObject = () => wrangle.asObject(rect); + const api: t.SizeObserverHook<T> = { + ref, + get ready() { + return rect !== undefined; + }, + get width() { + return rect?.width ?? 0; + }, + get height() { + return rect?.height ?? 0; + }, + rect, + toObject, + toString() { + const width = wrangle.sizeString(rect?.width); + const height = wrangle.sizeString(rect?.height); + return `${width} x ${height}`; + }, + toElement(input) { + const props = wrangle.elementProps(input); + const { opacity = 0.3, fontSize = 12, Absolute } = props; + let display = props.visible === false ? 'none' : props.inline ? 'inline-block' : 'block'; + const base = css({ Absolute, display, fontSize, opacity }); + return <div className={css(base, props.style).class}>{`${api.toString()}`}</div>; + }, + }; + return api; +}; + +/** + * Helpers + */ +const wrangle = { + asObject(rect?: DOMRect): t.DomRect { + if (!rect) return { x: 0, y: 0, width: 0, height: 0, top: 0, right: 0, bottom: 0, left: 0 }; + const { x, y, width, height, top, right, bottom, left } = rect; + return { x, y, width, height, top, right, bottom, left }; + }, + + sizeString(input?: number) { + if (input === undefined) return '-'; + return input.toFixed(0); + }, + + elementProps(input?: t.SizeObserverElementProps | t.CssEdgesArray): t.SizeObserverElementProps { + if (!input) return {}; + if (Array.isArray(input)) return { Absolute: input }; + return input; + }, +} as const; diff --git a/code/sys.ui/ui-react/src/mod.ts b/code/sys.ui/ui-react/src/mod.ts index fd0a07a0a1..2bce50147e 100644 --- a/code/sys.ui/ui-react/src/mod.ts +++ b/code/sys.ui/ui-react/src/mod.ts @@ -31,6 +31,19 @@ export { pkg } from './pkg.ts'; export type * as t from './types.ts'; +/** + * Library + */ export { FC } from './m.FC/mod.ts'; -export { useClickInside, useClickOutside, useMouse, useMouseDrag } from './m.use/mod.ts'; -export { ReactEvent } from './u/mod.ts'; +export { Signal } from './m.Signal/mod.ts'; +export { ReactEvent, ReactString } from './u/mod.ts'; + +export { + useClickInside, + useClickOutside, + useDist, + useIsTouchSupported, + useMouse, + useMouseDrag, + useSizeObserver, +} from './m.use/mod.ts'; diff --git a/code/sys.ui/ui-react/src/pkg.ts b/code/sys.ui/ui-react/src/pkg.ts index 79cb3f9c80..054c375c03 100644 --- a/code/sys.ui/ui-react/src/pkg.ts +++ b/code/sys.ui/ui-react/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/ui-react', version: '0.0.94' }; diff --git a/code/sys.ui/ui-react/src/types.ts b/code/sys.ui/ui-react/src/types.ts index 2e49dadcb4..69214ae96b 100644 --- a/code/sys.ui/ui-react/src/types.ts +++ b/code/sys.ui/ui-react/src/types.ts @@ -6,6 +6,8 @@ import type { JSX } from 'react'; export type { FC } from 'react'; export type * from './m.FC/t.ts'; +export type * from './m.Signal/t.ts'; +export type * from './m.Testing.Server/t.ts'; export type * from './m.use/t.ts'; export type * from './u/t.ts'; diff --git a/code/sys.ui/ui-react/src/u/m.ReactString.tsx b/code/sys.ui/ui-react/src/u/m.ReactString.tsx new file mode 100644 index 0000000000..99508d6a1f --- /dev/null +++ b/code/sys.ui/ui-react/src/u/m.ReactString.tsx @@ -0,0 +1,20 @@ +import { Fragment } from 'react'; +import { type t } from './common.ts'; + +export const ReactString: t.ReactStringLib = { + break(input: string | t.ReactNode) { + if (typeof input !== 'string') return input; + return input + .trim() + .split('\n') + .map((line, index, array) => { + const isLast = index === array.length - 1; + return ( + <Fragment key={index}> + <span>{line.trim()}</span> + {!isLast && <br />} + </Fragment> + ); + }); + }, +}; diff --git a/code/sys.ui/ui-react/src/u/mod.ts b/code/sys.ui/ui-react/src/u/mod.ts index a779f58aae..971265cb37 100644 --- a/code/sys.ui/ui-react/src/u/mod.ts +++ b/code/sys.ui/ui-react/src/u/mod.ts @@ -1 +1,2 @@ export { ReactEvent } from './m.ReactEvent.ts'; +export { ReactString } from './m.ReactString.tsx'; diff --git a/code/sys.ui/ui-react/src/u/t.ts b/code/sys.ui/ui-react/src/u/t.ts index 9ed65279b6..e45b4f2b73 100644 --- a/code/sys.ui/ui-react/src/u/t.ts +++ b/code/sys.ui/ui-react/src/u/t.ts @@ -8,3 +8,14 @@ export type ReactEventLib = { /** Convert react mouse events into keyboard modifier info object. */ modifiers(e: MouseEvent): t.KeyboardModifierFlags; }; + +/** + * Helpers for working with strings in react. + */ +export type ReactStringLib = { + /** + * Breaks a string with newline ("\n") characters into a fragment + * of <span>'s and <br> elements. + */ + break(text: string | t.ReactNode): t.ReactNode; +}; diff --git a/code/sys/cli/deno.json b/code/sys/cli/deno.json index d5737f30f4..2306c3a81a 100644 --- a/code/sys/cli/deno.json +++ b/code/sys/cli/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/cli", - "version": "0.0.64", + "version": "0.0.74", "license": "MIT", "tasks": { "lint": "deno lint", diff --git a/code/sys/cli/src/pkg.ts b/code/sys/cli/src/pkg.ts index 79cb3f9c80..75dbdc51c3 100644 --- a/code/sys/cli/src/pkg.ts +++ b/code/sys/cli/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/cli', version: '0.0.74' }; diff --git a/code/sys/cmd/deno.json b/code/sys/cmd/deno.json index 13a25a19ca..94c062b9b9 100644 --- a/code/sys/cmd/deno.json +++ b/code/sys/cmd/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/cmd", - "version": "0.0.80", + "version": "0.0.90", "license": "MIT", "tasks": { "test": "deno test -RW", diff --git a/code/sys/cmd/src/common/libs.ts b/code/sys/cmd/src/common/libs.ts index 54b38639df..47ea2f7bd0 100644 --- a/code/sys/cmd/src/common/libs.ts +++ b/code/sys/cmd/src/common/libs.ts @@ -1 +1 @@ -export { Immutable, ObjectPath, R, Time, Value, rx, slug, isObject } from '@sys/std'; +export { Dispose, Immutable, isObject, ObjectPath, R, rx, slug, Time, Value } from '@sys/std'; diff --git a/code/sys/cmd/src/m.Cmd/Cmd.ts b/code/sys/cmd/src/m.Cmd/Cmd.ts deleted file mode 100644 index a2d22310ae..0000000000 --- a/code/sys/cmd/src/m.Cmd/Cmd.ts +++ /dev/null @@ -1,32 +0,0 @@ -import { create } from './Cmd.impl.ts'; -import { Events, Is, Patch, Path, Queue, toPaths, toTransport, toIssuer } from './u.ts'; -import type { t } from './common.ts'; - -export type CmdLib = { - readonly Is: t.CmdIsLib; -}; - -/** - * Command event structure on an observable/syncing CRDT. - * Primitive for building up an actor model ("message passing computer"). - */ -export const Cmd = { - Is, - Path, - Patch, - Events, - Queue, - - /** - * Factory. - */ - create, - - /** - * Helpers. - */ - autopurge: Queue.autopurge, - toTransport, - toPaths, - toIssuer, -} as const; diff --git a/code/sys/cmd/src/m.Cmd/u.Events.ts b/code/sys/cmd/src/m.Cmd/u.Events.ts index 7fa502e95a..59e3d9ba8f 100644 --- a/code/sys/cmd/src/m.Cmd/u.Events.ts +++ b/code/sys/cmd/src/m.Cmd/u.Events.ts @@ -1,4 +1,4 @@ -import { R, rx, type t, type u } from './common.ts'; +import { type t, type u, Dispose, R, rx } from './common.ts'; import { Patch } from './u.Patch.ts'; import { Path } from './u.Path.ts'; @@ -73,9 +73,9 @@ export const Events: t.CmdEventsLib = { } /** - * API + * API: */ - const api: t.CmdEvents<C> = { + const api = rx.toLifecycle<t.CmdEvents<C>>(life, { $, tx$, error$, @@ -92,17 +92,10 @@ export const Events: t.CmdEventsLib = { type R = ReturnType<t.CmdEvents<C>['issuer']>; const issuer = [...issuers, ...wrangle.issuers(input)]; const res = Events.create(doc, { ...options, issuer, dispose$ }); - delete (res as any).dispose; - return res as unknown as R; + return Dispose.omitDispose(res) as R; }, + }); - // Lifecycle. - dispose, - dispose$, - get disposed() { - return life.disposed; - }, - }; return api; }, diff --git a/code/sys/cmd/src/m.Cmd/u.Listener.ts b/code/sys/cmd/src/m.Cmd/u.Listener.ts index 889a6bfa36..3e79377fa5 100644 --- a/code/sys/cmd/src/m.Cmd/u.Listener.ts +++ b/code/sys/cmd/src/m.Cmd/u.Listener.ts @@ -91,7 +91,7 @@ function create<Req extends t.CmdType, Res extends t.CmdType>( /** * API */ - const api: L = { + const api: L = rx.toLifecycle<L>(life, { $, tx, issuer, @@ -136,14 +136,7 @@ function create<Req extends t.CmdType, Res extends t.CmdType>( handlers.timeout.add(fn); return api; }, - - // Lifecycle. - dispose, - dispose$, - get disposed() { - return life.disposed; - }, - }; + }); return api; } diff --git a/code/sys/cmd/src/m.Cmd/u.Queue.ts b/code/sys/cmd/src/m.Cmd/u.Queue.ts index 432238da6c..3d9ee860ed 100644 --- a/code/sys/cmd/src/m.Cmd/u.Queue.ts +++ b/code/sys/cmd/src/m.Cmd/u.Queue.ts @@ -68,19 +68,12 @@ export const Queue: t.CmdQueueLib = { /** * API */ - const api: t.CmdQueueMonitor = { + const api = rx.toLifecycle<t.CmdQueueMonitor>(events, { bounds: { min, max }, get total() { return Queue.totals(cmd); }, - - // Lifecycle. - dispose, - dispose$, - get disposed() { - return events.disposed; - }, - }; + }); return api; }, } as const; diff --git a/code/sys/cmd/src/pkg.ts b/code/sys/cmd/src/pkg.ts index 79cb3f9c80..57e18f061d 100644 --- a/code/sys/cmd/src/pkg.ts +++ b/code/sys/cmd/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/cmd', version: '0.0.90' }; diff --git a/code/sys/color/deno.json b/code/sys/color/deno.json index d29872c11c..093b6cc6d6 100644 --- a/code/sys/color/deno.json +++ b/code/sys/color/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/color", - "version": "0.0.35", + "version": "0.0.45", "license": "MIT", "tasks": { "lint": "deno lint", diff --git a/code/sys/color/src/m.Rgb/-.test.ts b/code/sys/color/src/m.Rgb/-.test.ts index 8c3a5dfd3d..c68c37f863 100644 --- a/code/sys/color/src/m.Rgb/-.test.ts +++ b/code/sys/color/src/m.Rgb/-.test.ts @@ -1,7 +1,11 @@ import { describe, expect, it } from '../-test.ts'; -import { Color } from './m.Color.ts'; +import { Color, Theme } from './mod.ts'; describe('Color', () => { + it('API', () => { + expect(Color.Theme).to.equal(Theme); + }); + describe('Color.format', () => { const test = (value: string | number | boolean | undefined, output?: string) => { expect(Color.format(value)).to.eql(output); @@ -65,11 +69,24 @@ describe('Color', () => { }); describe('Color.theme', () => { + it('create from root API', () => { + const a = Color.Theme.create(); + const b = Color.theme(); + expect(a.name).to.eql(b.name); + }); + + it('toString', () => { + const a = Color.theme(); + const b = Color.theme('Dark'); + expect(a.toString()).to.eql('Light'); + expect(b.toString()).to.eql('Dark'); + }); + it('name: Light (default)', () => { const res1 = Color.theme(); const res2 = Color.theme('Light'); const res3 = Color.theme('Light', 'red', 'salmon'); - expect(res1.name).to.eql('Light'); + expect(res1.toString()).to.eql(res1.name); expect(res1.is.light).to.eql(true); expect(res1.is.dark).to.eql(false); expect(res1.fg).to.eql(Color.DARK); @@ -137,6 +154,15 @@ describe('Color', () => { expect(light.invert()).to.not.equal(light); // NB: monad. }); + it('Theme.invert (static method)', () => { + const a = Color.theme(); + expect(a.toString()).to.eql('Light'); + + const b = Theme.invert(a.name); + expect(b).to.eql('Dark'); + expect(Theme.invert()).to.eql('Dark'); + }); + it('invert: custom colors', () => { const theme = Color.theme('Light', 'red', 'salmon'); const inverted = theme.invert(); diff --git a/code/sys/color/src/m.Rgb/m.Color.ts b/code/sys/color/src/m.Rgb/m.Color.ts index 128b2547dc..f8fae685d5 100644 --- a/code/sys/color/src/m.Rgb/m.Color.ts +++ b/code/sys/color/src/m.Rgb/m.Color.ts @@ -1,18 +1,18 @@ export * from './u.const.ts'; export * from './u.format.ts'; -export * from './u.theme.ts'; import type { t } from './common.ts'; import { COLORS } from './u.const.ts'; import { alpha, darken, format, lighten } from './u.format.ts'; -import { theme } from './u.theme.ts'; +import { Theme } from './m.Theme.ts'; /** * Library: Helpers for working with colors. */ export const Color: t.ColorLib = { ...COLORS, - theme, + Theme, + theme: Theme.create, format, alpha, lighten, diff --git a/code/sys/color/src/m.Rgb/u.theme.ts b/code/sys/color/src/m.Rgb/m.Theme.ts similarity index 75% rename from code/sys/color/src/m.Rgb/u.theme.ts rename to code/sys/color/src/m.Rgb/m.Theme.ts index f6753120b9..862f674c2a 100644 --- a/code/sys/color/src/m.Rgb/u.theme.ts +++ b/code/sys/color/src/m.Rgb/m.Theme.ts @@ -9,7 +9,7 @@ const defaultTheme: t.CommonTheme = 'Light'; /** * A color theme helper object. */ -export function theme( +export function create( input?: t.CommonTheme | t.ColorTheme | null, // NB: loose input. defaultLight?: ColorInput, defaultDark?: ColorInput, @@ -28,8 +28,8 @@ function factory( defaultDark?: ColorInput, ): t.ColorTheme { const fg = wrangle.color(name, defaultLight, defaultDark); - const bg = wrangle.color(wrangle.invert(name), defaultLight, defaultDark); - return { + const bg = wrangle.color(invert(name), defaultLight, defaultDark); + const theme: t.ColorTheme = { name, fg, bg, @@ -46,10 +46,27 @@ function factory( }, }; }, - invert: () => theme(wrangle.invert(name), defaultLight, defaultDark), - } as const; + invert: () => create(invert(name), defaultLight, defaultDark), + toString: () => name, + }; + return theme; } +/** + * Invert a color scheme. + */ +export function invert(theme: t.CommonTheme = defaultTheme): t.CommonTheme { + return theme === 'Dark' ? 'Light' : 'Dark'; +} + +/** + * API + */ +export const Theme: t.ColorThemeLib = { + create, + invert, +}; + /** * Helpers */ @@ -59,8 +76,4 @@ const wrangle = { const dark = defaultDark ?? WHITE; return theme === 'Dark' ? dark : light; }, - - invert(theme: t.CommonTheme = defaultTheme): t.CommonTheme { - return theme === 'Dark' ? 'Light' : 'Dark'; - }, } as const; diff --git a/code/sys/color/src/m.Rgb/mod.ts b/code/sys/color/src/m.Rgb/mod.ts index 4b22e3350d..6ff8afd72e 100644 --- a/code/sys/color/src/m.Rgb/mod.ts +++ b/code/sys/color/src/m.Rgb/mod.ts @@ -11,4 +11,5 @@ * const myColor = Color.alpha(theme.fg, 0.3); * ``` */ -export { COLORS, Color } from './m.Color.ts'; +export { Color, COLORS } from './m.Color.ts'; +export { Theme } from './m.Theme.ts'; diff --git a/code/sys/color/src/m.Rgb/t.ts b/code/sys/color/src/m.Rgb/t.ts index 86aaca344d..0a006dd60a 100644 --- a/code/sys/color/src/m.Rgb/t.ts +++ b/code/sys/color/src/m.Rgb/t.ts @@ -30,14 +30,29 @@ export type ColorLib = t.ColorConstants & { */ darken(color: string, amount: number): string; + /** ColorThemeLib */ + readonly Theme: ColorThemeLib; + /** Create a color theme instance. */ + theme: ColorThemeLib['create']; +}; + +/** + * Tools for working with the basic color theme ("Light" / "Dark"). + */ +export type ColorThemeLib = { /** * Create a color theme instance. */ - theme( + create( input?: t.CommonTheme | t.ColorTheme | null, // NB: loose input. defaultLight?: ColorInput, defaultDark?: ColorInput, ): t.ColorTheme; + + /** + * Invert a color scheme. + */ + invert(theme?: t.CommonTheme): t.CommonTheme; }; /** @@ -61,6 +76,9 @@ export type ColorTheme = ColorThemeColors & { /** Retrieve a new theme inverted (eg. "Dark" ā "Light") */ invert(): ColorTheme; + + /** Convert to string. */ + toString(): string; }; /** diff --git a/code/sys/color/src/pkg.ts b/code/sys/color/src/pkg.ts index 79cb3f9c80..cf2b90315f 100644 --- a/code/sys/color/src/pkg.ts +++ b/code/sys/color/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/color', version: '0.0.45' }; diff --git a/code/sys/crypto/deno.json b/code/sys/crypto/deno.json index 78365de310..ee81912b4f 100644 --- a/code/sys/crypto/deno.json +++ b/code/sys/crypto/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/crypto", - "version": "0.0.64", + "version": "0.0.74", "license": "MIT", "tasks": { "lint": "deno lint", diff --git a/code/sys/crypto/src/pkg.ts b/code/sys/crypto/src/pkg.ts index 79cb3f9c80..8cf55a17c5 100644 --- a/code/sys/crypto/src/pkg.ts +++ b/code/sys/crypto/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/crypto', version: '0.0.74' }; diff --git a/code/sys/fs/deno.json b/code/sys/fs/deno.json index 226f44cd2b..df8dd09e46 100644 --- a/code/sys/fs/deno.json +++ b/code/sys/fs/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/fs", - "version": "0.0.79", + "version": "0.0.89", "license": "MIT", "tasks": { "lint": "deno lint", diff --git a/code/sys/fs/src/-test/-sample-files/volcano.jpg b/code/sys/fs/src/-test/-sample-files/volcano.jpg new file mode 100644 index 0000000000..ab0cef116f Binary files /dev/null and b/code/sys/fs/src/-test/-sample-files/volcano.jpg differ diff --git a/code/sys/fs/src/common/libs.ts b/code/sys/fs/src/common/libs.ts index 35ae282c51..93432962c7 100644 --- a/code/sys/fs/src/common/libs.ts +++ b/code/sys/fs/src/common/libs.ts @@ -1,4 +1,4 @@ -export { ensureDir, exists } from '@std/fs'; +export { ensureDir, ensureSymlink, exists } from '@std/fs'; export { Date, Delete, R, rx, slug, Time } from '@sys/std'; export { Err } from '@sys/std/error'; diff --git a/code/sys/fs/src/common/mod.ts b/code/sys/fs/src/common/mod.ts index 2f2b241bea..fa98d7391f 100644 --- a/code/sys/fs/src/common/mod.ts +++ b/code/sys/fs/src/common/mod.ts @@ -2,4 +2,4 @@ export type * as t from './t.ts'; export { pkg } from '../pkg.ts'; export * from './libs.ts'; -export * from './u.ts'; +export * from './u.is.ts'; diff --git a/code/sys/fs/src/common/u.is.ts b/code/sys/fs/src/common/u.is.ts new file mode 100644 index 0000000000..fd645aa5e8 --- /dev/null +++ b/code/sys/fs/src/common/u.is.ts @@ -0,0 +1,74 @@ +import type * as t from './t.ts'; +import { exists, StdPath } from './libs.ts'; + +/** + * Determine if the given path points to a directory. + */ +export async function isDir(path: t.StringPath | URL) { + try { + path = wrangle.path(path); + if (!(await exists(path))) return false; + return (await Deno.stat(path)).isDirectory; + } catch (_error: any) { + return false; + } +} + +/** + * Determine if the given path points to a file (not a directory). + */ +export async function isFile(path: t.StringPath | URL) { + try { + path = wrangle.path(path); + if (!(await exists(path))) return false; + return (await Deno.stat(path)).isFile; + } catch (_error: any) { + return false; + } +} + +/** + * Checks whether a file is binary. + * It reads up to 8KB of the file and looks for <null? bytes + * and a high percentage of non-printable characters. + */ +export async function isBinary(path: t.StringPath | URL): Promise<boolean> { + path = wrangle.path(path); + if (!(await isFile(path))) return false; + + const data = await Deno.readFile(path); + + // Use a chunk size of 8KB (or the entire file if smaller). + const chunkSize = 8192; + const chunk = data.slice(0, Math.min(data.length, chunkSize)); + + // Check for a null byte. + if (chunk.includes(0)) { + return true; + } + + // Count non-printable characters (excluding common whitespace). + let nonPrintableCount = 0; + for (const byte of chunk) { + // Accept common control chars: tab (9), newline (10), and carriage return (13). + if (byte < 32 && byte !== 9 && byte !== 10 && byte !== 13) { + nonPrintableCount++; + } + } + + // If more than 30% of the bytes are non-printable, assume it's binary. + if (nonPrintableCount / chunk.length > 0.3) { + return true; + } + + return false; +} + +/** + * Helpers + */ +const wrangle = { + path(path: string | URL) { + return typeof path === 'string' ? StdPath.resolve(path) : path; + }, +} as const; diff --git a/code/sys/fs/src/common/u.ts b/code/sys/fs/src/common/u.ts deleted file mode 100644 index 60526a4119..0000000000 --- a/code/sys/fs/src/common/u.ts +++ /dev/null @@ -1,26 +0,0 @@ -import type * as t from './t.ts'; -import { exists } from './libs.ts'; - -/** - * Determine if the given path points to a directory. - */ -export async function isDir(path: t.StringPath | URL) { - try { - if (!(await exists(path))) return false; - return (await Deno.stat(path)).isDirectory; - } catch (_error: any) { - return false; - } -} - -/** - * Determine if the given path points to a file (not a directory). - */ -export async function isFile(path: t.StringPath | URL) { - try { - if (!(await exists(path))) return false; - return (await Deno.stat(path)).isFile; - } catch (_error: any) { - return false; - } -} diff --git a/code/sys/fs/src/m.Fs/-.test.ts b/code/sys/fs/src/m.Fs/-.test.ts index 30ff6efeec..843688d397 100644 --- a/code/sys/fs/src/m.Fs/-.test.ts +++ b/code/sys/fs/src/m.Fs/-.test.ts @@ -1,3 +1,4 @@ +import * as StdFs from '@std/fs'; import { Path as StdPath } from '@sys/std'; import { describe, expect, it } from '../-test.ts'; import { Glob } from '../m.Glob/mod.ts'; @@ -9,6 +10,9 @@ describe('Fs: filesystem', () => { expect(Fs.glob).to.equal(Glob.create); expect(Fs.ls).to.equal(Glob.ls); expect(Fs.trimCwd).to.equal(Path.trimCwd); + + expect(Fs.ensureDir).to.eql(StdFs.ensureDir); + expect(Fs.ensureSymlink).to.eql(StdFs.ensureSymlink); }); describe('Fs.Path', () => { @@ -36,4 +40,15 @@ describe('Fs: filesystem', () => { expect(res3).to.eql(path3); // NB: not-found, no change. }); }); + + describe('Fs.cwd', () => { + it('returns the CWD', () => { + expect(Fs.cwd()).to.eql(Deno.cwd()); + }); + + it('returns the initiating CWD', () => { + const dir = Fs.cwd('init'); + expect(dir).to.eql(Deno.env.get('INIT_CWD')); + }); + }); }); diff --git a/code/sys/fs/src/m.Fs/-fs.meta.is.test.ts b/code/sys/fs/src/m.Fs/-fs.meta.is.test.ts new file mode 100644 index 0000000000..fdb7572ff1 --- /dev/null +++ b/code/sys/fs/src/m.Fs/-fs.meta.is.test.ts @@ -0,0 +1,33 @@ +import { describe, expect, it } from '../-test.ts'; +import { Path } from './common.ts'; +import { Fs } from './mod.ts'; + +describe('Fs.Is (flags)', () => { + const Is = Fs.Is; + + it('has mapped Path methods', () => { + // NB: mapped helpers (convenience). + expect(Is.absolute).to.equal(Fs.Path.Is.absolute); + expect(Is.glob).to.equal(Fs.Path.Is.glob); + }); + + it('Is.dir', async () => { + expect(await Is.dir('.')).to.eql(true); + expect(await Is.dir(Path.resolve('.'))).to.eql(true); + expect(await Is.dir('./deno.json')).to.eql(false); + expect(await Is.dir('./404.json')).to.eql(false); // NB: target does not exist. + }); + + it('Is.file', async () => { + expect(await Is.file('.')).to.eql(false); + expect(await Is.file('./deno.json')).to.eql(true); + expect(await Is.file(Path.resolve('./deno.json'))).to.eql(true); + expect(await Is.file('./404.json')).to.eql(false); // NB: target does not exist. + }); + + it('Is.binary', async () => { + expect(await Is.binary('.')).to.eql(false); + expect(await Is.binary('./deno.json')).to.eql(false); + expect(await Is.binary('./src/-test/-sample-files/volcano.jpg')).to.eql(true); + }); +}); diff --git a/code/sys/fs/src/m.Fs/-fs.meta.test.ts b/code/sys/fs/src/m.Fs/-fs.meta.test.ts index 3cc2b4e4ce..ae1eed083d 100644 --- a/code/sys/fs/src/m.Fs/-fs.meta.test.ts +++ b/code/sys/fs/src/m.Fs/-fs.meta.test.ts @@ -1,27 +1,21 @@ import { describe, expect, it } from '../-test.ts'; -import { Path } from './common.ts'; import { Fs } from './mod.ts'; describe('Fs: info/meta-data operations on the file-system', () => { - describe('Fs.Is (flags)', () => { - const Is = Fs.Is; - - it('has mapped Path methods', () => { - // NB: mapped helpers (convenience). - expect(Is.absolute).to.equal(Fs.Path.Is.absolute); - expect(Is.glob).to.equal(Fs.Path.Is.glob); - }); - - it('Is.dir', async () => { - expect(await Is.dir(Path.resolve('.'))).to.eql(true); - expect(await Is.dir(Path.resolve('./deno.json'))).to.eql(false); - expect(await Is.dir(Path.resolve('./404.json'))).to.eql(false); // NB: target does not exist. + describe('Fs.stat', () => { + it('file does not exist ā <undefined>', async () => { + const path = Fs.resolve('./404.json'); + const res = await Fs.stat(path); + expect(res).to.eql(undefined); }); - it('Is.file', async () => { - expect(await Is.file(Path.resolve('.'))).to.eql(false); - expect(await Is.file(Path.resolve('./deno.json'))).to.eql(true); - expect(await Is.file(Path.resolve('./404.json'))).to.eql(false); // NB: target does not exist. + it('file exists', async () => { + const path = './src/-test/-sample-1/foo.txt'; + const a = await Fs.stat(Fs.resolve(path)); + const b = await Fs.stat(path); + expect(a?.isFile).to.eql(true); + expect(a?.size).to.be.greaterThan(10); + expect(a).to.eql(b); // NB: auto-resolves path internally. }); }); diff --git a/code/sys/fs/src/m.Fs/m.Fs.ts b/code/sys/fs/src/m.Fs/m.Fs.ts index 34ddc3ae72..451a2fb0ca 100644 --- a/code/sys/fs/src/m.Fs/m.Fs.ts +++ b/code/sys/fs/src/m.Fs/m.Fs.ts @@ -1,4 +1,4 @@ -import { type t, ensureDir, exists, ls, Path } from './common.ts'; +import { type t, ensureDir, ensureSymlink, exists, ls, Path } from './common.ts'; import { create as glob } from '../m.Glob/u.create.ts'; import { Watch } from '../m.Watch/mod.ts'; @@ -7,10 +7,12 @@ import { Size } from './m.Size.ts'; import { copy, copyDir, copyFile } from './u.copy.ts'; import { read, readJson, readText, readYaml } from './u.read.ts'; import { remove } from './u.remove.ts'; +import { stat } from './u.stat.ts'; import { toDir } from './u.toDir.ts'; import { toFile } from './u.toFile.ts'; import { walk, walkUp } from './u.walk.ts'; import { write, writeJson } from './u.write.ts'; +import { cwd } from './u.cwd.ts'; export { Path }; const { join, resolve, basename, dirname, extname } = Path; @@ -23,8 +25,9 @@ export const Fs: t.FsLib = { Path, Size, Watch, - stat: Deno.stat, - cwd: Deno.cwd, + stat, + + cwd, trimCwd: Path.trimCwd, join, @@ -41,6 +44,7 @@ export const Fs: t.FsLib = { exists, ensureDir, + ensureSymlink, remove, read, diff --git a/code/sys/fs/src/m.Fs/m.Is.ts b/code/sys/fs/src/m.Fs/m.Is.ts index 557d39e374..1c5522dae1 100644 --- a/code/sys/fs/src/m.Fs/m.Is.ts +++ b/code/sys/fs/src/m.Fs/m.Is.ts @@ -1,4 +1,4 @@ -import { type t, isDir, isFile, StdPath } from './common.ts'; +import { type t, isBinary, isDir, isFile, StdPath } from './common.ts'; /** * Filesystem/Path type verification flags. @@ -7,4 +7,5 @@ export const Is: t.FsIsLib = { ...StdPath.Is, dir: isDir, file: isFile, + binary: isBinary, } as const; diff --git a/code/sys/fs/src/m.Fs/t.ts b/code/sys/fs/src/m.Fs/t.ts index dc0db550f5..c2f2697915 100644 --- a/code/sys/fs/src/m.Fs/t.ts +++ b/code/sys/fs/src/m.Fs/t.ts @@ -50,11 +50,11 @@ export type FsLib = Methods & { /** Recursively walk up a directory tree (visitor pattern). */ readonly walkUp: t.FsWalkUp; - /** Start a file-system watcher */ + /** Start a file-system watcher. */ readonly watch: t.FsWatchLib['start']; /** Current working directory. */ - cwd(): t.StringDir; + cwd(kind?: 'init'): t.StringDir; /** Removes the CWD (current-working-directory) from the given path if it exists. */ trimCwd: t.FsPathLib['trimCwd']; @@ -116,6 +116,9 @@ type StdMethods = { /** Asynchronously ensures that the directory exists, like `mkdir -p.` */ readonly ensureDir: typeof StdFs.ensureDir; + /** Asynchronously ensures that the link exists, and points to a valid file. */ + readonly ensureSymlink: typeof StdFs.ensureSymlink; + /** Recursively walks through a directory and yields information about each file and directory encountered. */ readonly walk: typeof StdFs.walk; }; @@ -129,12 +132,16 @@ export type FsIsLib = t.PathLib['Is'] & { /** Determine if the given path points to a file (not a directory). */ file(path: t.StringPath | URL): Promise<boolean>; + + /** Determine if the given path points to a binary (non-string) file. */ + binary(path: t.StringPath | URL): Promise<boolean>; }; /** * Retrieve information about the given path. */ -export type FsGetStat = (path: t.StringPath | URL) => Promise<Deno.FileInfo>; +export type FsGetStat = (path: t.StringPath | URL) => Promise<FsFileInfo | undefined>; +export type FsFileInfo = Deno.FileInfo; /** * Copy a file or directory. diff --git a/code/sys/fs/src/m.Fs/u.cwd.ts b/code/sys/fs/src/m.Fs/u.cwd.ts new file mode 100644 index 0000000000..3f2d4f5859 --- /dev/null +++ b/code/sys/fs/src/m.Fs/u.cwd.ts @@ -0,0 +1,10 @@ +import { type t, exists, Path, Err } from './common.ts'; + +/** + * Current Working Directory: + * https://docs.deno.com/runtime/reference/cli/task/#specifying-the-current-working-directory + */ +export const cwd: t.FsLib['cwd'] = (kind) => { + if (kind === 'init') return Deno.env.get('INIT_CWD') ?? Deno.cwd(); + return Deno.cwd(); +}; diff --git a/code/sys/fs/src/m.Fs/u.stat.ts b/code/sys/fs/src/m.Fs/u.stat.ts new file mode 100644 index 0000000000..c138209074 --- /dev/null +++ b/code/sys/fs/src/m.Fs/u.stat.ts @@ -0,0 +1,11 @@ +import { type t, Path, exists } from './common.ts'; + +/** + * Resolves to a Deno.FileInfo for the specified path. + * Will always follow symlinks. + */ +export const stat: t.FsLib['stat'] = async (path: t.StringPath | URL) => { + if (!(await exists(path))) return undefined; + path = typeof path === 'string' ? Path.resolve(path) : path; + return Deno.stat(path); +}; diff --git a/code/sys/fs/src/m.Fs/u.walk.ts b/code/sys/fs/src/m.Fs/u.walk.ts index 9dd4817f66..b832d25de5 100644 --- a/code/sys/fs/src/m.Fs/u.walk.ts +++ b/code/sys/fs/src/m.Fs/u.walk.ts @@ -1,6 +1,5 @@ import { walk, type WalkEntry } from '@std/fs'; - -import { type t, Path } from './common.ts'; +import { Path, type t } from './common.ts'; export { walk }; diff --git a/code/sys/fs/src/m.Pkg/-Pkg.Dist.test.ts b/code/sys/fs/src/m.Pkg/-Pkg.Dist.test.ts index 07199c41f4..b22a49611c 100644 --- a/code/sys/fs/src/m.Pkg/-Pkg.Dist.test.ts +++ b/code/sys/fs/src/m.Pkg/-Pkg.Dist.test.ts @@ -43,6 +43,7 @@ describe('Pkg.Dist', () => { expect(res.exists).to.eql(true); expect(res.error).to.eql(undefined); + expect(res.dist['-type:'] === 'jsr:@sys/types:DistPkg').to.eql(true); expect(res.dir).to.eql(Fs.resolve(dir)); expect(res.dist.pkg).to.eql(pkg); expect(res.dist.entry).to.eql(Path.normalize(entry)); diff --git a/code/sys/fs/src/m.Pkg/m.Dist.ts b/code/sys/fs/src/m.Pkg/m.Dist.ts index 81be24fdea..f7816cd4e4 100644 --- a/code/sys/fs/src/m.Pkg/m.Dist.ts +++ b/code/sys/fs/src/m.Pkg/m.Dist.ts @@ -35,7 +35,13 @@ export const Dist: t.PkgDistFsLib = { const hash = exists ? await wrangle.hashes(dir) : { digest: '', parts: {} }; const bytes = await wrangle.bytes(dir, Object.keys(hash.parts)); const size: t.DistPkg['size'] = { bytes }; - const dist: t.DistPkg = { pkg, size, entry: wrangle.entry(entry), hash }; + const dist: t.DistPkg = { + '-type:': 'jsr:@sys/types:DistPkg', + pkg, + size, + entry: wrangle.entry(entry), + hash, + }; /** * Prepare response. @@ -132,7 +138,7 @@ const wrangle = { let count = 0; for (const file of files) { const stat = await Fs.stat(Fs.join(dir, file)); - count += stat.size; + count += stat?.size ?? 0; } return count; }, diff --git a/code/sys/fs/src/m.Watch/m.Watch.ts b/code/sys/fs/src/m.Watch/m.Watch.ts index d96a19d664..89456c80ce 100644 --- a/code/sys/fs/src/m.Watch/m.Watch.ts +++ b/code/sys/fs/src/m.Watch/m.Watch.ts @@ -12,13 +12,12 @@ export const Watch: t.FsWatchLib = { const paths = asArray(pathInput); const errors = Err.errors(); const life = rx.lifecycle(options.dispose$); - const { dispose, dispose$ } = life; const $$ = rx.subject<t.FsWatchEvent>(); - const $ = $$.pipe(rx.takeUntil(dispose$)); + const $ = $$.pipe(rx.takeUntil(life.dispose$)); let _watcher: Deno.FsWatcher | undefined; - dispose$.subscribe(() => _watcher?.close()); + life.dispose$.subscribe(() => _watcher?.close()); const exists = await wrangle.exists(paths); if (!exists.ok) { @@ -46,7 +45,7 @@ export const Watch: t.FsWatchLib = { /** * API */ - const api: t.FsWatcher = { + const api = rx.toLifecycle<t.FsWatcher>(life, { get $() { return $; }, @@ -62,20 +61,7 @@ export const Watch: t.FsWatchLib = { get error() { return errors.toError('Several errors occured while watching file-system paths.'); }, - - /** Lifecycle: disposes of the watches. */ - dispose, - - /** Lifecycle: observable that fires when the watcher disposes. */ - get dispose$() { - return dispose$; - }, - - /** Lifecycle: flag indicating if the watcher has been disposed. */ - get disposed() { - return life.disposed; - }, - }; + }); return api; }, }; diff --git a/code/sys/fs/src/pkg.ts b/code/sys/fs/src/pkg.ts index 79cb3f9c80..1eaf6ee545 100644 --- a/code/sys/fs/src/pkg.ts +++ b/code/sys/fs/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/fs', version: '0.0.89' }; diff --git a/code/sys/http/deno.json b/code/sys/http/deno.json index a961d4ce98..ba8321ad7e 100644 --- a/code/sys/http/deno.json +++ b/code/sys/http/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/http", - "version": "0.0.46", + "version": "0.0.56", "license": "MIT", "tasks": { "lint": "deno lint", diff --git a/code/sys/http/src/ns.client/m.Http.Fetch/u.create.ts b/code/sys/http/src/ns.client/m.Http.Fetch/u.create.ts index 0c1e2843b0..1aa4eb5fbd 100644 --- a/code/sys/http/src/ns.client/m.Http.Fetch/u.create.ts +++ b/code/sys/http/src/ns.client/m.Http.Fetch/u.create.ts @@ -105,13 +105,12 @@ export const create: F = (input: Parameters<F>[0]) => { } as t.FetchResponse<T>; }; - const api: t.HttpFetch = { + const api: t.HttpFetch = rx.toLifecycle<t.HttpFetch>(life, { + header: (name) => (api.headers as any)[name], get headers() { return wrangle.headers(options); }, - header: (name) => (api.headers as any)[name], - async json<T>(input: RequestInput, init: RequestInit = {}, options = {}) { return invokeFetch<T>('application/json', input, init, options, (res) => res.json()); }, @@ -119,18 +118,7 @@ export const create: F = (input: Parameters<F>[0]) => { async text(input: RequestInput, init: RequestInit = {}, options = {}) { return invokeFetch<string>('text/plain', input, init, options, (res) => res.text()); }, - - /** - * Lifecycle. - */ - dispose: life.dispose, - get dispose$() { - return life.dispose$; - }, - get disposed() { - return life.disposed; - }, - }; + }); return api; }; diff --git a/code/sys/http/src/pkg.ts b/code/sys/http/src/pkg.ts index 79cb3f9c80..83eb7de192 100644 --- a/code/sys/http/src/pkg.ts +++ b/code/sys/http/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/http', version: '0.0.56' }; diff --git a/code/sys/jsr/deno.json b/code/sys/jsr/deno.json index 142751e714..5db51b22d8 100644 --- a/code/sys/jsr/deno.json +++ b/code/sys/jsr/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/jsr", - "version": "0.0.45", + "version": "0.0.55", "license": "MIT", "tasks": { "lint": "deno lint", diff --git a/code/sys/jsr/src/pkg.ts b/code/sys/jsr/src/pkg.ts index 79cb3f9c80..3a6869bc00 100644 --- a/code/sys/jsr/src/pkg.ts +++ b/code/sys/jsr/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/jsr', version: '0.0.55' }; diff --git a/code/sys/main/deno.json b/code/sys/main/deno.json index 17b60f0ddb..be85924086 100644 --- a/code/sys/main/deno.json +++ b/code/sys/main/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/main", - "version": "0.0.57", + "version": "0.0.67", "license": "MIT", "tasks": { "lint": "deno lint", diff --git a/code/sys/main/src/pkg.ts b/code/sys/main/src/pkg.ts index 79cb3f9c80..d01b3ac424 100644 --- a/code/sys/main/src/pkg.ts +++ b/code/sys/main/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/main', version: '0.0.67' }; diff --git a/code/sys/process/deno.json b/code/sys/process/deno.json index 09e58b0c21..b17558603d 100644 --- a/code/sys/process/deno.json +++ b/code/sys/process/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/process", - "version": "0.0.65", + "version": "0.0.75", "license": "MIT", "tasks": { "lint": "deno lint", diff --git a/code/sys/process/src/pkg.ts b/code/sys/process/src/pkg.ts index 79cb3f9c80..c5aba01c0a 100644 --- a/code/sys/process/src/pkg.ts +++ b/code/sys/process/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/process', version: '0.0.75' }; diff --git a/code/sys/std/deno.json b/code/sys/std/deno.json index 4ff47723e0..891672992b 100644 --- a/code/sys/std/deno.json +++ b/code/sys/std/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/std", - "version": "0.0.131", + "version": "0.0.141", "license": "MIT", "tasks": { "test": "deno test -RWN", @@ -25,6 +25,7 @@ "./rx": "./src/m.Rx/mod.ts", "./semver": "./src/m.Semver/mod.ts", "./semver/server": "./src/m.Semver.Server/mod.ts", + "./signal": "./src/m.Signal/mod.ts", "./testing": "./src/m.Testing/mod.ts", "./testing/server": "./src/m.Testing.Server/mod.ts", "./value": "./src/m.Value/mod.ts" diff --git a/code/sys/std/src/common/u.ts b/code/sys/std/src/common/u.ts index dbcb3e341d..e9c0073998 100644 --- a/code/sys/std/src/common/u.ts +++ b/code/sys/std/src/common/u.ts @@ -10,13 +10,13 @@ export function isObject(input: any): input is object { /** * Determine if the given input is a simple {key:value} record object. */ -export function isRecord(input: any): input is O { +export function isRecord<T extends O>(input: any): input is T { return isObject(input) && !Array.isArray(input); } /** * Determine if the given object is empty of all fields. */ -export function isEmptyRecord(input: any): input is object { +export function isEmptyRecord<T extends O>(input: any): input is T { return isRecord(input) && Object.keys(input).length === 0; } diff --git a/code/sys/std/src/m.DateTime/-Time.test.ts b/code/sys/std/src/m.DateTime/-Time.test.ts index 0431f3ba81..8dfca34071 100644 --- a/code/sys/std/src/m.DateTime/-Time.test.ts +++ b/code/sys/std/src/m.DateTime/-Time.test.ts @@ -44,7 +44,7 @@ describe('Time', () => { expect(res.timeout).to.eql(0); }); - it('()=> | ā handler only, defaults to 0:msecs', async () => { + it('()=> | ā handler only, defaults to micro-task ("tick")', async () => { let fired = 0; const startedAt = new Date(); const res = Time.delay(() => fired++); @@ -68,7 +68,7 @@ describe('Time', () => { }); describe('Time.wait', () => { - it('Time.wait( n ) ā number/msecs param', async () => { + it('Time.wait( n ) ā number/msecs param (macro-task queue)', async () => { const startedAt = new Date(); expect(calcDiff(startedAt)).to.be.closeTo(0, 10); @@ -80,15 +80,29 @@ describe('Time', () => { expect(res.is).to.eql({ completed: true, cancelled: false, done: true }); }); - it('Time.wait() ā no param', async () => { - const startedAt = new Date(); - expect(calcDiff(startedAt)).to.be.closeTo(0, 10); + it('Time.wait() ā no param (micro-task, aka. "tick")', async () => { + await Testing.retry(3, async () => { + const timer = Time.timer(); + let microtaskResolved = false; + let macrotaskResolved = false; - const res = Time.wait(); - await res; + const stop = setTimeout(() => (macrotaskResolved = true), 0); // ā Schedule a macro-task for comparison. + const res = Time.wait(); // ā the micro-task delay (no param). + await res; - expect(res.timeout).to.eql(0); - expect(calcDiff(startedAt)).to.be.lessThan(10); + const elapsed = timer.elapsed.msec; + microtaskResolved = true; + + expect(res.is.done).to.eql(true); + expect(res.is.completed).to.eql(true); + expect(res.is.cancelled).to.eql(false); + + expect(microtaskResolved).to.be.true; + expect(macrotaskResolved).to.be.false; // Microtasks should run before macrotasks. + expect(elapsed).to.be.closeTo(0, 2); // Allow up to 2ms tolerance. + + clearTimeout(stop); + }); }); it('Time.wait ā cancel', () => { diff --git a/code/sys/std/src/m.DateTime/-Time.until.test.ts b/code/sys/std/src/m.DateTime/-Time.until.test.ts index 9e93d8892f..bda4c09986 100644 --- a/code/sys/std/src/m.DateTime/-Time.until.test.ts +++ b/code/sys/std/src/m.DateTime/-Time.until.test.ts @@ -31,7 +31,6 @@ describe('Time.until', () => { let count = 0; const time = Time.until(dispose$); time.dispose$.subscribe(() => disposeFired++); - expect(time.disposed).to.eql(false); const res = time.delay(10, () => (count += 1)); @@ -43,5 +42,22 @@ describe('Time.until', () => { expect(time.disposed).to.eql(true); expect(disposeFired).to.eql(1); }); + + it('is a disposable', async () => { + let disposeFired = 0; + let count = 0; + const time = Time.until(); + time.dispose$.subscribe(() => disposeFired++); + expect(time.disposed).to.eql(false); + + const res = time.delay(10, () => (count += 1)); + time.dispose(); + + await res; + await Time.wait(20); + expect(count).to.eql(0); + expect(time.disposed).to.eql(true); + expect(disposeFired).to.eql(1); + }); }); }); diff --git a/code/sys/std/src/m.DateTime/-Timer.test.ts b/code/sys/std/src/m.DateTime/-Timer.test.ts index fc7240ed00..27f7b7f0a7 100644 --- a/code/sys/std/src/m.DateTime/-Timer.test.ts +++ b/code/sys/std/src/m.DateTime/-Timer.test.ts @@ -1,15 +1,17 @@ import { add, startOfDay, sub } from 'date-fns'; -import { describe, expect, it } from '../-test.ts'; +import { describe, expect, it, Testing } from '../-test.ts'; import { Time } from './mod.ts'; const FORMAT = 'yyyy-MM-dd hh:mm:ss'; const format = (date: Date) => Time.utc(date).format(FORMAT); describe('timer', () => { - it('starts with current date', () => { - const now = Time.now.format(FORMAT); - const timer = Time.timer(); - expect(format(timer.startedAt)).to.eql(now); + it('starts with current date', async () => { + await Testing.retry(3, async () => { + const now = Time.now.format(FORMAT); + const timer = Time.timer(); + expect(format(timer.startedAt)).to.eql(now); + }); }); it('starts with given date', () => { diff --git a/code/sys/std/src/m.DateTime/-Timestamp.test.ts b/code/sys/std/src/m.DateTime/-Timestamp.test.ts new file mode 100644 index 0000000000..3f0e22fdc9 --- /dev/null +++ b/code/sys/std/src/m.DateTime/-Timestamp.test.ts @@ -0,0 +1,370 @@ +import { type t, describe, expect, it } from '../-test.ts'; +import { Num } from '../m.Value.Num/mod.ts'; +import { Time } from './m.Time.ts'; +import { Timestamp } from './mod.ts'; + +describe('Timestamp', () => { + type T = { image: string }; + type MyTimestamps = t.Timestamps<T>; + + describe('parse: "HH:MM:SS.mmm"', () => { + it('should parse "00:00:00.000" as 0 msecs and 0 secs', () => { + const res = Timestamp.parse('00:00:00.000'); + expect(res.msec).to.eql(0); + expect(res.sec).to.eql(0); + expect(res.hour).to.eql(0); + expect(res.min).to.eql(0); + }); + + it('should parse "01:02:03.004" correctly', () => { + // Expected values: + // hours: 1, minutes: 2, seconds: 3, milliseconds: 4 + const expectedMsecs = 1 * 3600000 + 2 * 60000 + 3 * 1000 + 4; // 3,723,004 msecs + const expectedSecs = expectedMsecs / 1000; // 3,723.004 secs + const expectedMins = expectedMsecs / 60000; // 62.05006666666667 mins + const expectedHours = expectedMsecs / 3600000; // 1.0341677777777778 hours + + const res = Timestamp.parse('01:02:03.004'); + expect(res.msec).to.eql(expectedMsecs); + expect(res.sec).to.eql(Num.round(expectedSecs, 1)); + expect(res.min).to.eql(Num.round(expectedMins, 1)); + expect(res.hour).to.eql(Num.round(expectedHours, 1)); + }); + + it('should parse "12:34:56.789" correctly', () => { + // Expected values: + // hours: 12, minutes: 34, seconds: 56, milliseconds: 789 + const expectedMsecs = 12 * 3600000 + 34 * 60000 + 56 * 1000 + 789; + const expectedSecs = expectedMsecs / 1000; + const expectedMins = expectedMsecs / 60000; + const expectedHours = expectedMsecs / 3600000; + + const res = Timestamp.parse('12:34:56.789'); + expect(res.msec).to.eql(expectedMsecs); + expect(res.sec).to.eql(Num.round(expectedSecs, 1)); + expect(res.min).to.eql(Num.round(expectedMins, 1)); + expect(res.hour).to.eql(Num.round(expectedHours, 1)); + }); + + it('should fix single-digit unit values', () => { + const res = Timestamp.parse('1:2:3.4'); + expect(res.hour).to.eql(1); + expect(res.min).to.eql(62.1); + expect(res.sec).to.eql(3723); + expect(res.msec).to.eql(3723004); + expect(res.toString()).to.eql('1h'); + }); + + describe('errors', () => { + it('should throw an error for invalid time format with missing parts', () => { + expect(() => Timestamp.parse('12:34')).to.throw('Invalid time format'); + }); + + it('should throw an error for invalid seconds/milliseconds format', () => { + expect(() => Timestamp.parse('12:34:56')).to.throw('Invalid seconds/milliseconds format'); + + // This test expects that an empty millisecond part will eventually result in an invalid number error. + expect(() => Timestamp.parse('12:34:56.')).to.throw('Invalid number in timestamp'); + }); + + it('should throw an error for non-numeric values', () => { + expect(() => Timestamp.parse('aa:bb:cc.ddd')).to.throw('Invalid number in timestamp'); + }); + }); + }); + + describe('parse: timestamp map ā { "HH:MM:SS.mmm": T }', () => { + it('should return an empty [array] when given an empty object', () => { + const test = (input?: any) => { + expect(Timestamp.parse(input)).to.eql([]); + }; + test({}); + test(undefined); + test(null); + }); + + it('should ensure "00:00:00.000" entry', () => { + const a = Timestamp.parse({}); + const b = Timestamp.parse({}, { ensureZero: true }); + expect(a).to.eql([]); + + expect(b.length).to.eql(1); + expect(b[0].timestamp).to.eql('00:00:00.000'); + expect(b[0].data).to.eql({}); + expect(b[0].total.msec).to.eql(0); + }); + + it('should correctly parse a single timestamp', () => { + const input: MyTimestamps = { + '00:01:02.003': { image: 'single' }, + }; + const result = Timestamp.parse(input); + expect(result).to.have.lengthOf(1); + const parsed = result[0]; + + // The original timestamp should be preserved. + expect(parsed.timestamp).to.eql('00:01:02.003'); + + // Calculate expected parsed values. + // For "00:01:02.003": + // hours: 0, minutes: 1, seconds: 2, milliseconds: 3 + const expectedMsecs = 0 * 3600000 + 1 * 60000 + 2 * 1000 + 3; // 62003 msecs + const expectedSecs = expectedMsecs / 1000; + const expectedMins = expectedMsecs / 60000; + const expectedHours = expectedMsecs / 3600000; + + expect(parsed.total.msec).to.eql(expectedMsecs); + expect(parsed.total.sec).to.eql(Num.round(expectedSecs, 1)); + expect(parsed.total.min).to.eql(Num.round(expectedMins, 1)); + expect(parsed.total.hour).to.eql(Num.round(expectedHours, 1)); + + // The associated data should be preserved. + expect(parsed.data).to.eql({ image: 'single' }); + }); + + it('should correctly parse and sort multiple timestamps', () => { + const input: MyTimestamps = { + '00:00:10.000': { image: 'second' }, + '00:00:15.000': { image: 'third' }, + '00:00:05.000': { image: 'first' }, + }; + const result = Timestamp.parse(input); + expect(result).to.have.lengthOf(3); + + // Expect the results to be sorted by time in ascending order. + expect(result[0].timestamp).to.eql('00:00:05.000'); + expect(result[1].timestamp).to.eql('00:00:10.000'); + expect(result[2].timestamp).to.eql('00:00:15.000'); + + // Validate the parsed milliseconds for the first timestamp. + expect(result[0].total.msec).to.eql(5000); + expect(result[0].data).to.eql({ image: 'first' }); + }); + + it('should throw an error if one of the timestamps has an invalid format', () => { + const input: MyTimestamps = { + '00:00:05.000': { image: 'first' }, + invalid: { image: 'broken' }, + }; + expect(() => Timestamp.parse(input)).to.throw('Invalid time format'); + }); + }); + + describe('find', () => { + const timestamps: MyTimestamps = { + '00:00:10.000': { image: 'second' }, + '00:00:15.000': { image: 'third' }, + '00:00:05.000': { image: 'first' }, + }; + + it('should return [undefined] if elapsed time is before the first timestamp', () => { + // Elapsed time of 2 seconds is less than the first timestamp (5 seconds) + const res = Timestamp.find(timestamps, 2_000); + expect(res).to.be.undefined; + }); + + it('should return the first timestamp when elapsed equals its time', () => { + // Elapsed time exactly 5 seconds. + const res = Timestamp.find(timestamps, 5_000); + expect(res?.data).to.eql({ image: 'first' }); + }); + + it('should return the correct timestamp when elapsed falls between two timestamps', () => { + // Elapsed time 7 seconds: the only timestamp <= 7 seconds is the first (5 seconds). + const a = Timestamp.find(timestamps, 7_000); + expect(a?.data).to.eql({ image: 'first' }); + + // Elapsed time 12 seconds: the latest timestamp with time <= 12 seconds is the second (10 seconds). + const b = Timestamp.find(timestamps, 12_000); + expect(b?.data).to.eql({ image: 'second' }); + }); + + it('should return the last timestamp if elapsed time exceeds all timestamps', () => { + // Elapsed time 20 seconds: all timestamps are <= 20 seconds, so it returns the last one. + const res = Timestamp.find(timestamps, 20_000); + expect(res?.data).to.eql({ image: 'third' }); + }); + + it('option: should use seconds (instead of milliseconds) as time comparison unit', () => { + const a = Timestamp.find(timestamps, 12_000); + const b = Timestamp.find(timestamps, 12, { unit: 'secs' }); + const c = Timestamp.find(timestamps, 12); // expect failure. + expect(a?.data).to.eql({ image: 'second' }); + expect(a?.data).to.eql(b?.data); + expect(c).to.not.eql(a); + }); + }); + + describe('isCurrent', () => { + const timestamps: MyTimestamps = { + '00:00:00.000': { image: 'image-0.png' }, + '00:00:05.000': { image: 'image-5.png' }, + '00:00:10.000': { image: 'image-10.png' }, + }; + + it('should return [false] when current time is before the first timestamp', () => { + // When current time is less than the first timestamp, no candidate is found. + expect(Timestamp.isCurrent(timestamps, '00:00:00.000', -1)).to.be.false; + }); + + it('should return [true] when current time equals the first timestamp', () => { + // currentTime is exactly at the first timestamp. + expect(Timestamp.isCurrent(timestamps, '00:00:00.000', 0)).to.be.true; + }); + + it('should return [false] if the candidate timestamp does not match the provided timestamp', () => { + // For currentTime 3 sec, the candidate should be '00:00:00.000', not '00:00:05.000' + expect(Timestamp.isCurrent(timestamps, '00:00:05.000', 3_000)).to.be.false; + }); + + it('should return [true] when current time exactly matches a later timestamp', () => { + // For currentTime 5 sec, the candidate should be '00:00:05.000' + expect(Timestamp.isCurrent(timestamps, '00:00:05.000', 5_000)).to.be.true; + }); + + it('should return [true] when current time is between two timestamps and candidate remains unchanged', () => { + // For currentTime 7 sec, candidate remains '00:00:05.000' + expect(Timestamp.isCurrent(timestamps, '00:00:05.000', 7_000)).to.be.true; + }); + + it('should return [false] when the provided timestamp is not present in the timestamps', () => { + // Provided timestamp does not exist in the timestamps object. + expect(Timestamp.isCurrent(timestamps, '00:00:20.000', 7_000)).to.be.false; + }); + + it('option: should use seconds (instead of milliseconds) as time comparison unit', () => { + const a = Timestamp.isCurrent(timestamps, '00:00:05.000', 5_000); + const b = Timestamp.isCurrent(timestamps, '00:00:05.000', 5, { unit: 'secs' }); + const c = Timestamp.isCurrent(timestamps, '00:00:05.000', 5); // expect failure. + expect(a).to.be.true; + expect(b).to.be.true; + expect(c).to.not.eql(a); + }); + }); + + describe('toString', () => { + it('should format 0 milliseconds as "00:00:00.000"', () => { + const ts = Timestamp.parse('00:00:00.000'); + expect(Timestamp.toString(ts)).to.eql('00:00:00.000'); + expect(Timestamp.toString('0:0:0.0')).to.eql('00:00:00.000'); + }); + + it('should format 123 milliseconds as "00:00:00.123"', () => { + const ts = '00:00:00.123'; + expect(Timestamp.toString(ts)).to.eql(ts); + }); + + it('should format 1000 milliseconds as "00:00:01.000"', () => { + const ts = '00:00:01.000'; + expect(Timestamp.toString(ts)).to.eql(ts); + }); + + it('should format 61234 milliseconds as "00:01:01.234"', () => { + // 61234 ms = 0 hours, 1 minute, 1 second, 234 ms + const duration = Time.Duration.create(61234); + expect(Timestamp.toString(duration)).to.eql('00:01:01.234'); + }); + + it('should format a duration with days and hours correctly', () => { + // Example: 1 day, 7 hours, 6 minutes, 45 seconds, and 123 milliseconds. + // 1 day = 86,400,000 ms + // 7 hours = 7 * 3,600,000 = 25,200,000 ms + // 6 minutes = 6 * 60,000 = 360,000 ms + // 45 seconds = 45 * 1,000 = 45,000 ms + // Total = 86,400,000 + 25,200,000 + 360,000 + 45,000 + 123 = 112,005,123 ms. + const totalMsec = 112005123; + const duration = Time.Duration.create(totalMsec); + expect(Timestamp.toString(duration)).to.eql('31:06:45.123'); + }); + }); + + describe('range', () => { + const timestamps: MyTimestamps = { + '00:00:10.000': { image: 'second' }, + '00:00:15.000': { image: 'third' }, + '00:00:05.000': { image: 'first' }, + }; + + describe('response: undefined', () => { + it('empty {timestamps}', () => { + const range = Timestamp.range({}, 0); + expect(range).to.eql(undefined); + }); + + it('out of range', () => { + const range = Timestamp.range(timestamps, 99, { unit: 'secs' }); + expect(range).to.eql(undefined); + }); + }); + + describe('create', () => { + it('from inferred 0-point', () => { + const range = Timestamp.range(timestamps, 3_000); + expect(range?.start).to.eql('00:00:00.000'); + expect(range?.end).to.eql('00:00:05.000'); + }); + + it('between to timestamps', () => { + const range = Timestamp.range(timestamps, 6_000); + expect(range?.start).to.eql('00:00:05.000'); + expect(range?.end).to.eql('00:00:10.000'); + }); + + it('from "timestamp" string', () => { + const range = Timestamp.range(timestamps, '00:00:06.001'); + expect(range?.start).to.eql('00:00:05.000'); + expect(range?.end).to.eql('00:00:10.000'); + }); + }); + + describe('progress (percentage)', () => { + it('should calculate progress correctly at the bounds', () => { + const res = Timestamp.range(timestamps, 7000); + expect(res).to.not.be.undefined; + if (res) { + // At the start of the range (5000 msec), progress should be 0. + expect(res.progress(5000)).to.equal(0); + // At the end of the range (10000 msec), progress should be 1. + expect(res.progress(10000)).to.equal(1); + } + }); + + it('should calculate progress correctly for an intermediate value', () => { + const res = Timestamp.range(timestamps, 7000); + expect(res).to.not.be.undefined; + + // For a progress time of 7500 msec, progress = (7500 - 5000) / (10000 - 5000) = 0.5 + expect(res?.progress(7500)).to.equal(0.5); + }); + + it('should respect unit conversion for progress calculations', () => { + // Use seconds instead of milliseconds. + // The location is 7 seconds (7000 msec). In the sorted timestamps, + // the range remains from 5000 msec to 10000 msec. + const res = Timestamp.range(timestamps, 7, { unit: 'secs' }); + expect(res).to.not.be.undefined; + + // When passing times in seconds, progress(5) should be calculated as 5000 msec, so progress = 0. + expect(res?.progress(5, { unit: 'secs' })).to.equal(0); + + // progress(10) (i.e. 10000 msec) should equal 1. + expect(res?.progress(10, { unit: 'secs' })).to.equal(1); + + // progress(7.5) seconds (7500 msec) should equal 0.5. + expect(res?.progress(7.5, { unit: 'secs' })).to.equal(0.5); + }); + + it('should round progress correctly when the round option is provided', () => { + // Use a round option (e.g., 2 decimal places) and choose a time that results in a non-integer fraction. + const res = Timestamp.range(timestamps, 7000, { round: 2 }); + expect(res).to.not.be.undefined; + + // For a progress time of 6666 msec, the unrounded progress is: + // (6666 - 5000) / (10000 - 5000) = 1666 / 5000 ā 0.3332 + // With rounding to 2 decimals, we expect ā 0.33. + const prog = res?.progress(6666); + expect(prog).to.be.closeTo(0.33, 0.01); + }); + }); + }); +}); diff --git a/code/sys/std/src/m.DateTime/-Until.-unit.test.ts b/code/sys/std/src/m.DateTime/-Until.test.ts similarity index 71% rename from code/sys/std/src/m.DateTime/-Until.-unit.test.ts rename to code/sys/std/src/m.DateTime/-Until.test.ts index 9e93d8892f..a7816878ef 100644 --- a/code/sys/std/src/m.DateTime/-Until.-unit.test.ts +++ b/code/sys/std/src/m.DateTime/-Until.test.ts @@ -1,4 +1,4 @@ -import { describe, expect, it } from '../-test.ts'; +import { type t, describe, expect, it } from '../-test.ts'; import { rx } from '../m.Rx/mod.ts'; import { Time } from './mod.ts'; @@ -33,7 +33,6 @@ describe('Time.until', () => { time.dispose$.subscribe(() => disposeFired++); expect(time.disposed).to.eql(false); - const res = time.delay(10, () => (count += 1)); dispose(); @@ -43,5 +42,22 @@ describe('Time.until', () => { expect(time.disposed).to.eql(true); expect(disposeFired).to.eql(1); }); + + it('constructs from ({ lifecycle/disposable }) object', async () => { + const test = async (input: t.Disposable) => { + let count = 0; + + const time = Time.until(input.dispose$); + const res = time.delay(10, () => (count += 1)); + input.dispose(); + + await res; + await Time.wait(20); + expect(count).to.eql(0); + }; + + await test(rx.lifecycle()); + await test(rx.disposable()); + }); }); }); diff --git a/code/sys/std/src/m.DateTime/m.Time.delay.ts b/code/sys/std/src/m.DateTime/m.Time.delay.ts index 4d7cf69e10..80c5c521eb 100644 --- a/code/sys/std/src/m.DateTime/m.Time.delay.ts +++ b/code/sys/std/src/m.DateTime/m.Time.delay.ts @@ -12,29 +12,47 @@ export function delay(...args: any[]): t.TimeDelayPromise { const { signal } = controller; const cancel = () => controller.abort('delay cancelled'); - const is: t.DeepMutable<T['is']> = { completed: false, cancelled: false, done: false }; + const is: t.DeepMutable<T['is']> = { done: false, completed: false, cancelled: false }; + let promise: any; - const promise: any = new Promise<void>((resolve) => { - const done = () => { - is.done = true; - is.cancelled = signal.aborted; - resolve(); - }; + if (msecs === undefined) { + /** + * Micro-task queue ("tick"). + */ + promise = Promise.resolve(); + promise + .then(() => { + fn?.(); + is.completed = true; + is.done = true; + }) + .catch(() => (is.done = true)); + } else { + /** + * Macro-task queue. + */ + promise = new Promise<void>((resolve) => { + const done = () => { + is.done = true; + is.cancelled = signal.aborted; + resolve(); + }; - signal.onabort = done; - const complete = () => { - fn?.(); - is.completed = true; - done(); - }; + signal.onabort = done; + const complete = () => { + fn?.(); + is.completed = true; + done(); + }; - denoDelay(msecs, { signal }).then(complete).catch(done); - }); + denoDelay(msecs, { signal }).then(complete).catch(done); + }); + } // Decorate the promise with extra time/delay controller fields. promise.cancel = cancel; promise.is = is; - promise.timeout = msecs; + promise.timeout = msecs || 0; return promise as T; } @@ -43,7 +61,7 @@ export function delay(...args: any[]): t.TimeDelayPromise { */ export const Wrangle = { delayArgs(input: any[]) { - let msecs = 0; + let msecs = undefined; let fn: t.TimeDelayCallback | undefined; if (typeof input[0] === 'number') msecs = input[0]; if (typeof input[0] === 'function') fn = input[0]; diff --git a/code/sys/std/src/m.DateTime/m.Time.until.ts b/code/sys/std/src/m.DateTime/m.Time.until.ts index 72b13f0b8c..9ac96078f7 100644 --- a/code/sys/std/src/m.DateTime/m.Time.until.ts +++ b/code/sys/std/src/m.DateTime/m.Time.until.ts @@ -2,36 +2,39 @@ import type { t } from './common.ts'; import { Subject, take, takeUntil } from 'rxjs'; import { Dispose } from '../m.Dispose/mod.ts'; -import { delay as baseDelay, Wrangle as DelayWrangle } from './m.Time.delay.ts'; +import { delay, Wrangle } from './m.Time.delay.ts'; /** * Exposes timer functions that cease after a dispose signal is received. */ export function until(until$: t.UntilObservable) { const life = Dispose.lifecycle(until$); - const { dispose$ } = life; const api: t.TimeUntil = { delay(...args: any[]): t.TimeDelayPromise { - const { msecs, fn } = DelayWrangle.delayArgs(args); + const { msecs, fn } = Wrangle.delayArgs(args); const done$ = new Subject<void>(); - const res = baseDelay(msecs, () => { + const res = delay(msecs, () => { done$.next(); return fn?.(); }); + life.dispose$.pipe(takeUntil(done$), take(1)).subscribe(() => res.cancel()); return res; }, wait(msecs) { - return api.delay(msecs ?? 0); + return typeof msecs === 'number' ? api.delay(msecs) : api.delay(); }, /** * Lifecycle */ - dispose$, + dispose: life.dispose, + get dispose$() { + return life.dispose$; + }, get disposed() { return life.disposed; }, diff --git a/code/sys/std/src/m.DateTime/m.Time.wait.ts b/code/sys/std/src/m.DateTime/m.Time.wait.ts index 0c1d27d316..fb9ec7ea66 100644 --- a/code/sys/std/src/m.DateTime/m.Time.wait.ts +++ b/code/sys/std/src/m.DateTime/m.Time.wait.ts @@ -6,6 +6,6 @@ import { delay } from './m.Time.delay.ts'; * (NB: use with `await`.) * @param msecs: delay in milliseconds. */ -export const wait: t.TimeLib['wait'] = (msecs = 0) => { +export const wait: t.TimeLib['wait'] = (msecs) => { return delay(msecs); }; diff --git a/code/sys/std/src/m.DateTime/m.Timestamp.parse.ts b/code/sys/std/src/m.DateTime/m.Timestamp.parse.ts new file mode 100644 index 0000000000..a6c8f53526 --- /dev/null +++ b/code/sys/std/src/m.DateTime/m.Timestamp.parse.ts @@ -0,0 +1,71 @@ +import { type t, isRecord } from './common.ts'; +import { Duration } from './m.Time.Duration.ts'; + +/** + * Parse a "HH:MM:DD:mmm" string into a structured object. + */ +export function parseTime( + timestamp: t.StringTimestamp, + options: { round?: number } = {}, +): t.TimeDuration { + // Split the timestamp into "hour", "minute", and "second.millisecond" parts. + const parts = timestamp.split(':'); + if (parts.length !== 3) throw new Error(`Invalid time format: ${timestamp}`); + + // Parse hours and minutes. + const hours = parseInt(parts[0], 10); + const minutes = parseInt(parts[1], 10); + + // Split the seconds and milliseconds (expected format "SS.mmm"). + const timeParts = parts[2].split('.'); + if (timeParts.length !== 2) { + throw new Error(`Invalid seconds/milliseconds format: ${parts[2]}`); + } + const seconds = parseInt(timeParts[0], 10); + const milliseconds = parseInt(timeParts[1], 10); + + // Check that all parts are valid numbers. + if (isNaN(hours) || isNaN(minutes) || isNaN(seconds) || isNaN(milliseconds)) { + throw new Error(`Invalid number in timestamp: ${timestamp}`); + } + + // Calculate the total milliseconds. + const msecs = + hours * 3_600_000 + // 1 hour = 3,600,000 msecs + minutes * 60_000 + // 1 minute = 60,000 msecs + seconds * 1_000 + // 1 second = 1,000 msecs + milliseconds; + + const { round } = options; + return Duration.create(msecs, { round }); +} + +/** + * Convert the set of { "HH:MM:SS:mmm":<value> } timestamp + * strings into list of sorted stuctures. + */ +export function parseMap<T>( + timestamps: t.Timestamps<T>, + options: { round?: number; ensureZero?: boolean } = {}, +): t.Timestamp<T>[] { + if (!isRecord(timestamps)) return []; + + const ZERO = '00:00:00.000'; + if (options.ensureZero && !timestamps[ZERO]) { + timestamps = { ...timestamps, [ZERO]: {} as T }; + } + + const parse = (timestamp: string, data: T) => { + const total = parseTime(timestamp, options); + return { + timestamp, + data, + get total() { + return total; + }, + }; + }; + return Object.entries(timestamps) + .map(([timestamp, data]) => parse(timestamp, data)) + .sort((a, b) => a.total.sec - b.total.sec); +} diff --git a/code/sys/std/src/m.DateTime/m.Timestamp.range.ts b/code/sys/std/src/m.DateTime/m.Timestamp.range.ts new file mode 100644 index 0000000000..106ae95fbc --- /dev/null +++ b/code/sys/std/src/m.DateTime/m.Timestamp.range.ts @@ -0,0 +1,53 @@ +import { type t } from './common.ts'; +import { parseMap, parseTime } from './m.Timestamp.parse.ts'; + +/** + * Generate a sub-range for a timestamp within a map of timestamps. + */ +export const range: t.TimestampLib['range'] = (timestamps, location, options = {}) => { + const { unit = 'msecs', round } = options; + const map = parseMap(timestamps, { round, ensureZero: true }); + const msecs = wrangle.msecs(location, unit); + + // Ensure there are enough timestamps. + if (map.length < 2) return undefined; + + // Iterate over the sorted timestamps to find where currentTime fits. + for (let i = 0; i < map.length - 1; i++) { + const start = map[i].total.msec; + const end = map[i + 1].total.msec; + + if (msecs >= start && msecs <= end) { + const api: t.TimestampRange = { + start: map[i].timestamp, + end: map[i + 1].timestamp, + progress(time, opt = {}) { + const progressTime = wrangle.msecs(time, opt.unit ?? unit, round); + let prog = (progressTime - start) / (end - start); + if (round !== undefined) { + const factor = Math.pow(10, round); + prog = Math.round(prog * factor) / factor; + } + return prog as t.Percent; + }, + }; + return api; + } + } + + // Out-of-range. + return undefined; +}; + +/** + * Helpers + */ +const wrangle = { + msecs(input: t.NumberTime | t.StringTimestamp, unit: t.TimestampUnit, round?: number): t.Msecs { + if (typeof input === 'string') { + return parseTime(input, { round }).msec; + } else { + return unit === 'secs' ? input * 1000 : input; + } + }, +} as const; diff --git a/code/sys/std/src/m.DateTime/m.Timestamp.ts b/code/sys/std/src/m.DateTime/m.Timestamp.ts new file mode 100644 index 0000000000..9c0916e951 --- /dev/null +++ b/code/sys/std/src/m.DateTime/m.Timestamp.ts @@ -0,0 +1,89 @@ +import { type t, isRecord } from './common.ts'; +import { parseMap, parseTime } from './m.Timestamp.parse.ts'; +import { range } from './m.Timestamp.range.ts'; + +/** + * Tools for working with timestamps ("HH:MM:SS.mmm"). + */ +export const Timestamp: t.TimestampLib = { + range, + + parse(input: any, options: any) { + if (input === undefined || input === null) return []; + if (isRecord(input)) return parseMap(input, options) as any; // NB: type-hack. + if (typeof input === 'string') return parseTime(input); + throw new Error(`Input type not supported: ${typeof input}`); + }, + + find<T>( + timestamps: t.Timestamps<T>, + time: t.Msecs, + options: { unit?: t.TimestampUnit; round?: number } = {}, + ): t.Timestamp<T> | undefined { + const { unit, round } = options; + const msecs = wrangle.msecs(time, { unit }); + const parsedTimes = parseMap(timestamps, { round }); + let candidate: t.Timestamp<T> | undefined = undefined; + for (const entry of parsedTimes) { + // Match the last timestamp with time <= elapsed. + if (entry.total.msec <= msecs) candidate = entry; + else break; + } + return candidate; + }, + + isCurrent<T>( + timestamps: t.Timestamps<T>, + timestamp: t.StringTimestamp, + current: t.Secs, + options: { unit?: t.TimestampUnit; round?: number } = {}, + ) { + const { unit, round } = options; + const msecs = wrangle.msecs(current, { unit }); + const parsedTimes = parseMap(timestamps, { round }); + let candidate: t.Timestamp<T> | undefined = undefined; + + // Find the last timestamp with total time <= currentTime. + for (const entry of parsedTimes) { + if (entry.total.msec <= msecs) { + candidate = entry; + } else { + break; + } + } + + return candidate ? candidate.timestamp === timestamp : false; + }, + + toString(input) { + const duration = isRecord(input) ? input : Timestamp.parse(input); + const msec = duration.msec; + + // Calculate hours, minutes, seconds and milliseconds from the total milliseconds. + const hours = Math.floor(msec / 3600000); + const remainderAfterHours = msec % 3600000; + const minutes = Math.floor(remainderAfterHours / 60000); + const remainderAfterMinutes = remainderAfterHours % 60000; + const seconds = Math.floor(remainderAfterMinutes / 1000); + const milliseconds = remainderAfterMinutes % 1000; + + // Format with leading zeros. + const hoursStr = hours.toString().padStart(2, '0'); + const minutesStr = minutes.toString().padStart(2, '0'); + const secondsStr = seconds.toString().padStart(2, '0'); + const millisStr = milliseconds.toString().padStart(3, '0'); + + return `${hoursStr}:${minutesStr}:${secondsStr}.${millisStr}`; + }, +}; + +/** + * Helpers + */ +const wrangle = { + msecs(value: number, options: { unit?: t.TimestampUnit } = {}) { + const { unit = 'msecs' } = options; + if (unit === 'secs') value = value * 1000; + return value; + }, +} as const; diff --git a/code/sys/std/src/m.DateTime/mod.ts b/code/sys/std/src/m.DateTime/mod.ts index 75fd14b657..e5a4f2ec84 100644 --- a/code/sys/std/src/m.DateTime/mod.ts +++ b/code/sys/std/src/m.DateTime/mod.ts @@ -26,6 +26,7 @@ * elapsed.toString('m'); // ā "210m" * ``` */ -export { Date, Date as D, Day, Format } from './m.Date.ts'; +export { Date as D, Date, Day, Format } from './m.Date.ts'; export { Duration } from './m.Time.Duration.ts'; export { Time } from './m.Time.ts'; +export { Timestamp } from './m.Timestamp.ts'; diff --git a/code/sys/std/src/m.DateTime/t.Time.Timestamp.ts b/code/sys/std/src/m.DateTime/t.Time.Timestamp.ts new file mode 100644 index 0000000000..ef6028c8d3 --- /dev/null +++ b/code/sys/std/src/m.DateTime/t.Time.Timestamp.ts @@ -0,0 +1,64 @@ +import type { t } from './common.ts'; + +export type TimestampUnit = 'msecs' | 'secs'; + +/** + * Tools for working with timestamps ("HH:MM:SS.mmm"). + */ +export type TimestampLib = { + /** + * Convert a parsed [TimeDuration] back into a timestamp ("HH:MM:SS.mmm") + */ + toString(input: t.TimeDuration | t.StringTimestamp): string; + + /** + * Parse a "HH:MM:DD:mmm" string into a structured object. + */ + parse(timestamp: t.StringTimestamp, options?: { round?: number }): t.TimeDuration; + + /** + * Convert the map of { "HH:MM:SS:mmm": <T> } timestamps + * into a sorted list stuctured objects. + */ + parse<T>( + timestamps?: t.Timestamps<T>, + options?: { round?: number; ensureZero?: boolean }, + ): t.Timestamp<T>[]; + + /** + * Lookup a timestamp from an elapsed time within a {timestamps} map. + */ + find<T>( + timestamps: t.Timestamps<T>, + time: number, + options?: { unit?: t.TimestampUnit; round?: number }, + ): t.Timestamp<T> | undefined; + + /** + * Check if a given timestamp is the current one within a set based on the given time. + */ + isCurrent<T>( + timestamps: t.Timestamps<T>, + timestamp: t.StringTimestamp, + time: t.NumberTime, + options?: { unit?: t.TimestampUnit; round?: number }, + ): boolean; + + /** + * Generate a sub-range for a timestamp within a map of timestamps. + */ + range<T>( + timestamps: t.Timestamps<T>, + location: t.NumberTime | t.StringTimestamp, + options?: { unit?: t.TimestampUnit; round?: number }, + ): TimestampRange | undefined; +}; + +/** + * A timestamp range (start/end). + */ +export type TimestampRange = { + readonly start: t.StringTimestamp; + readonly end: t.StringTimestamp; + progress(time: t.NumberTime, options?: { unit?: t.TimestampUnit }): t.Percent; +}; diff --git a/code/sys/std/src/m.DateTime/t.Time.ts b/code/sys/std/src/m.DateTime/t.Time.ts index 33ebbf507a..5e3daf809d 100644 --- a/code/sys/std/src/m.DateTime/t.Time.ts +++ b/code/sys/std/src/m.DateTime/t.Time.ts @@ -36,7 +36,7 @@ export type TimeLib = { utc(input?: t.DateTimeInput): t.DateTime; /** A Time helper that runs only until it has been disposed. */ - until(until$?: t.UntilObservable): t.TimeUntil; + until(until$?: t.DisposeInput): t.TimeUntil; }; /** @@ -71,17 +71,11 @@ export type TimeDelay = { /** * Exposes timer functions that cease after a dispose signal is received. */ -export type TimeUntil = { - /** Fires when the transient time helper is disposed. */ - readonly dispose$: t.Observable<void>; - - /** Flag indicating if the transient time heper is disposed. */ - readonly disposed: boolean; - +export type TimeUntil = t.Lifecycle & { /** Delay for the specified milliseconds. */ delay: t.TimeLib['delay']; - /** Wait for the specified milliseconds pass. */ + /** Wait for the specified milliseconds to pass. */ wait: t.TimeLib['wait']; }; diff --git a/code/sys/std/src/m.DateTime/t.ts b/code/sys/std/src/m.DateTime/t.ts index d897477a39..f9ef061209 100644 --- a/code/sys/std/src/m.DateTime/t.ts +++ b/code/sys/std/src/m.DateTime/t.ts @@ -1,3 +1,4 @@ export type * from './t.Date.ts'; -export type * from './t.Time.ts'; export type * from './t.Time.Duration.ts'; +export type * from './t.Time.Timestamp.ts'; +export type * from './t.Time.ts'; diff --git a/code/sys/std/src/m.Dispose/-.test.ts b/code/sys/std/src/m.Dispose/-.test.ts index 50fd0a6a14..65cfd396b4 100644 --- a/code/sys/std/src/m.Dispose/-.test.ts +++ b/code/sys/std/src/m.Dispose/-.test.ts @@ -1,12 +1,13 @@ import { Subject } from 'rxjs'; import { describe, expect, it, type t } from '../-test.ts'; import { Time } from '../m.DateTime/mod.ts'; +import { Is, rx } from '../mod.ts'; import { Dispose } from './mod.ts'; describe('Disposable', () => { describe('sync', () => { it('disposable: create ā dispose', () => { - const test = (until?: t.UntilObservable) => { + const test = (until?: t.DisposeInput) => { const obj = Dispose.disposable(until); let count = 0; @@ -14,7 +15,12 @@ describe('Disposable', () => { obj.dispose(); obj.dispose(); - until?.forEach((subject) => subject.next()); + + if (Is.disposable(until)) { + until?.dispose(); + } else { + until?.forEach((subject) => subject.next()); + } expect(count).to.eql(1); }; @@ -22,10 +28,13 @@ describe('Disposable', () => { test(); test(new Subject<void>()); test([new Subject<void>(), new Subject<void>()]); + + test(rx.disposable()); + test(rx.lifecycle()); }); it('lifecycle: create ā dispose', () => { - const test = (until?: t.UntilObservable) => { + const test = (until?: t.DisposeInput) => { const obj = Dispose.lifecycle(until); expect(obj.disposed).to.eql(false); @@ -34,13 +43,22 @@ describe('Disposable', () => { obj.dispose(); obj.dispose(); - until?.forEach((subject) => subject.next()); + + if (Is.disposable(until)) { + until?.dispose(); + } else { + until?.forEach((subject) => subject.next()); + } + expect(count).to.eql(1); // NB: multiple calls to dispose to not refire the cleanup handler. }; test(); test(new Subject<void>()); test([new Subject<void>(), new Subject<void>()]); + + test(rx.disposable()); + test(rx.lifecycle()); }); }); @@ -215,14 +233,24 @@ describe('Disposable', () => { }); describe('Dispose.until', () => { - it('single', () => { + it('Input: single (observable)', () => { const $ = new Subject<void>(); const res = Dispose.until($); expect(res.length).to.eql(1); expect(res[0]).to.equal($); }); - it('list', () => { + it('Input: <t.Disposable>', () => { + const test = (input: t.Disposable) => { + const res = Dispose.until(input); + expect(res.length).to.eql(1); + expect(res[0]).to.equal(input.dispose$); + }; + test(rx.disposable()); + test(rx.lifecycle()); + }); + + it('Input: list', () => { const $1 = new Subject<void>(); const $2 = new Subject<void>(); const res = Dispose.until([$1, undefined, $2]); @@ -232,7 +260,7 @@ describe('Disposable', () => { expect(res[1]).to.eql($2); }); - it('deep list ā flattens', () => { + it('Input: deep list ā flattens', () => { const $1 = new Subject<void>(); const $2 = new Subject<void>(); const res = Dispose.until([$1, undefined, [undefined, [undefined, $2]]]); @@ -257,4 +285,59 @@ describe('Disposable', () => { expect(count).to.eql(1); }); }); + + describe('Dispose.toLifecycle', () => { + it('lifecycle', () => { + type T = t.Lifecycle & { count: number }; + const life = rx.lifecycle(); + const api = Dispose.toLifecycle<T>(life, { count: 123 }); + + let fired = 0; + api.dispose$.subscribe(() => fired++); + + expect(api.count).to.eql(123); + expect(api.disposed).to.eql(false); + + life.dispose(); + api.dispose(); + expect(fired).to.eql(1); + expect(api.disposed).to.eql(true); + }); + + it('(no param) generate lifecycle', () => { + type T = t.Lifecycle & { count: number }; + const api = Dispose.toLifecycle<T>({ count: 123 }); + let fired = 0; + api.dispose$.subscribe(() => fired++); + + expect(api.count).to.eql(123); + expect(api.disposed).to.eql(false); + + api.dispose(); + expect(fired).to.eql(1); + expect(api.disposed).to.eql(true); + }); + }); + + describe('Dispose.omitDispose', () => { + type T = t.Lifecycle & { count: number }; + + it('from lifecycle', () => { + const life = Dispose.lifecycle(); + const a = Dispose.toLifecycle<T>(life, { count: 123 }); + expect(typeof a.dispose === 'function').to.be.true; + + const b = Dispose.omitDispose(a); + expect(a).to.not.equal(b); + expect((b as any).dispose === undefined).to.be.true; + + let fired = 0; + b.dispose$.subscribe(() => fired++); + + expect(b.disposed).to.eql(false); + life.dispose(); + expect(b.disposed).to.eql(true); + expect(fired).to.eql(1); + }); + }); }); diff --git a/code/sys/std/src/m.Dispose/common.ts b/code/sys/std/src/m.Dispose/common.ts index 246d5f3cce..5adc5e3a35 100644 --- a/code/sys/std/src/m.Dispose/common.ts +++ b/code/sys/std/src/m.Dispose/common.ts @@ -1,4 +1,7 @@ export { flatten } from 'rambda'; -export { Subject, filter, take } from 'rxjs'; +export { filter, Subject, take } from 'rxjs'; export * from '../common.ts'; +export { Delete } from '../m.Delete/mod.ts'; +export { Err } from '../m.Err/mod.ts'; +export { Is } from '../m.Is/mod.ts'; diff --git a/code/sys/std/src/m.Dispose/m.Dispose.ts b/code/sys/std/src/m.Dispose/m.Dispose.ts index aec2b6e214..74190bc74e 100644 --- a/code/sys/std/src/m.Dispose/m.Dispose.ts +++ b/code/sys/std/src/m.Dispose/m.Dispose.ts @@ -1,138 +1,24 @@ -import { Delete } from '../m.Delete/mod.ts'; -import { Err } from '../m.Err/mod.ts'; -import { Is } from '../m.Rx/m.Is.ts'; -import { Subject, filter, flatten, take, type t } from './common.ts'; +import { type t } from './common.ts'; + +import { disposable, disposableAsync } from './u.dispose.ts'; +import { done } from './u.done.ts'; +import { lifecycle, lifecycleAsync, toLifecycle } from './u.lifecycle.ts'; +import { until } from './u.until.ts'; +import { omitDispose } from './u.omitDispose.ts'; /** * Toolkit for working with disposable interfaces. */ export const Dispose: t.DisposeLib = { - /** - * Generates a generic disposable interface that is - * typically mixed into a wider interface of some kind. - */ - disposable(until$): t.Disposable { - const dispose$ = new Subject<void>(); - const disposable: t.Disposable = { - dispose$: dispose$.asObservable(), - dispose: () => Dispose.done(dispose$), - }; - Dispose.until(until$).forEach(($) => $.subscribe(disposable.dispose)); - return disposable; - }, - - /** - * Generates an asnchronous Disposable interface. - */ - disposableAsync(...args: any[]) { - const { until$, onDispose } = wrangle.disposableAsyncArgs(args); - const dispose$ = new Subject<t.DisposeAsyncEvent>(); - let _disposing = false; - - type P = t.DisposeAsyncEventArgs; - const asPayload = (stage: t.DisposeAsyncStage, error?: t.DisposeError): P => { - const ok = !error; - const done = stage === 'complete' || stage === 'error'; - return Delete.undefined({ stage, is: { ok, done }, error }); - }; - const fire = (stage: t.DisposeAsyncStage, error?: t.DisposeError) => { - const payload = asPayload(stage, error); - dispose$.next({ type: 'dispose', payload }); - }; - - const disposable: t.DisposableAsync = { - dispose$: dispose$.asObservable(), - async dispose() { - if (_disposing) return; - _disposing = true; - - fire('start'); - try { - await onDispose?.(); // Invoke handler ("clean up resources"). - fire('complete'); - } catch (err: any) { - fire('error', { - name: 'DisposeError', - message: 'Failed while disposing asynchronously', - cause: Err.std(err), - }); - } - }, - }; - - Dispose.until(until$).forEach(($) => $.subscribe(disposable.dispose)); - return disposable; - }, + done, + until, - /** - * Generates a disposable interface that maintains - * and exposes it's disposed state. - */ - lifecycle(until$) { - const { dispose, dispose$ } = Dispose.disposable(until$); - let _disposed = false; - dispose$.pipe(take(1)).subscribe(() => (_disposed = true)); - return { - dispose$, - dispose, - get disposed() { - return _disposed; - }, - }; - }, + disposable, + disposableAsync, - lifecycleAsync(...args) { - const { until$, onDispose } = wrangle.disposableAsyncArgs(args); - const { dispose, dispose$ } = Dispose.disposableAsync(until$, onDispose); - let _disposed = false; - dispose$ - .pipe( - filter((e) => e.payload.stage === 'complete' || e.payload.stage === 'error'), - take(1), - ) - .subscribe(() => (_disposed = true)); - return { - dispose$, - dispose, - get disposed() { - return _disposed; - }, - }; - }, + lifecycle, + lifecycleAsync, + toLifecycle, - /** - * Listens to an observable (or set of observbles) and - * disposes of the target when any of them fire. - */ - until($) { - const list = Array.isArray($) ? $ : [$]; - return flatten(list).filter(Boolean) as t.Observable<unknown>[]; - }, - - /** - * "Completes" a subject by running: - * 1. subject.next(); - * 2. subject.complete(); - */ - done(dispose$) { - dispose$?.next?.(); - dispose$?.complete?.(); - }, + omitDispose, }; - -/** - * Helpers - */ -export const wrangle = { - disposableAsyncArgs(args: any[]) { - type Fn = () => Promise<void>; - let onDispose: Fn | undefined; - let until$: t.UntilObservable | undefined; - - if (typeof args[0] === 'function') onDispose = args[0]; - if (typeof args[1] === 'function') onDispose = args[1]; - if (Is.observable(args[0]) || Array.isArray(args[0])) until$ = Dispose.until(args[0]); - - return { onDispose, until$ }; - }, -} as const; diff --git a/code/sys/std/src/m.Dispose/t.ts b/code/sys/std/src/m.Dispose/t.ts index 0c559d656b..20a5a47227 100644 --- a/code/sys/std/src/m.Dispose/t.ts +++ b/code/sys/std/src/m.Dispose/t.ts @@ -11,7 +11,7 @@ export type DisposeLib = { * Generates a generic disposable interface that is * typically mixed into a wider interface of some kind. */ - disposable(until$?: t.UntilObservable): t.Disposable; + disposable(until$?: t.DisposeInput): t.Disposable; /** An async variant of the dispose pattern. */ disposableAsync(onDispose?: t.LifecycleStageHandler): t.DisposableAsync; @@ -21,16 +21,20 @@ export type DisposeLib = { * Generates a disposable interface that maintains * and exposes it's disposed state. */ - lifecycle(until$?: t.UntilObservable): t.Lifecycle; + lifecycle(until$?: t.DisposeInput): t.Lifecycle; /** An async variant of the lifecycle pattern. */ lifecycleAsync(onDispose?: LifecycleStageHandler): t.LifecycleAsync; - lifecycleAsync(until$?: t.UntilObservable, onDispose?: LifecycleStageHandler): t.LifecycleAsync; + lifecycleAsync(until$?: t.DisposeInput, onDispose?: LifecycleStageHandler): t.LifecycleAsync; + + /** Extend the given object to be expose the lifecycle API. */ + toLifecycle<T extends t.Lifecycle>(life: t.Lifecycle, api: t.OmitLifecycle<T>): T; + toLifecycle<T extends t.Lifecycle>(api: t.OmitLifecycle<T>): T; /** * Listens to an observable and disposes of the object when fires. */ - until($?: t.UntilObservable): t.Observable<unknown>[]; + until(dispose$?: t.DisposeInput): t.Observable<unknown>[]; /** * "Completes" a subject by running: @@ -39,4 +43,11 @@ export type DisposeLib = { * 2. subject.complete(); */ done(dispose$?: t.Subject<void>): void; + + /** + * Safely remove the `dispose` method from a disposable. + * NB: useful for surfacing from an API where you don't want + * callers to be able to disose of the resource. + */ + omitDispose<T extends t.Disposable | t.DisposableAsync>(obj: T): Omit<T, 'dispose'>; }; diff --git a/code/sys/std/src/m.Dispose/u.dispose.ts b/code/sys/std/src/m.Dispose/u.dispose.ts new file mode 100644 index 0000000000..be59215d6e --- /dev/null +++ b/code/sys/std/src/m.Dispose/u.dispose.ts @@ -0,0 +1,78 @@ +import { type t, Delete, Err, Is, Subject } from './common.ts'; +import { done } from './u.done.ts'; +import { until } from './u.until.ts'; + +/** + * Generates a generic disposable interface that is + * typically mixed into a wider interface of some kind. + */ +export function disposable(until$?: t.UntilInput) { + const subject$ = new Subject<void>(); + const dispose$ = subject$.asObservable(); + const disposable: t.Disposable = { + dispose: () => done(subject$), + get dispose$() { + return dispose$; + }, + }; + until(until$).forEach(($) => $.subscribe(disposable.dispose)); + return disposable; +} + +/** + * Generates an asnchronous Disposable interface. + */ +export function disposableAsync(...args: any[]) { + const { until$, onDispose } = toDisposableAsyncArgs(args); + const dispose$ = new Subject<t.DisposeAsyncEvent>(); + let _disposing = false; + + type P = t.DisposeAsyncEventArgs; + const asPayload = (stage: t.DisposeAsyncStage, error?: t.DisposeError): P => { + const ok = !error; + const done = stage === 'complete' || stage === 'error'; + return Delete.undefined({ stage, is: { ok, done }, error }); + }; + const fire = (stage: t.DisposeAsyncStage, error?: t.DisposeError) => { + const payload = asPayload(stage, error); + dispose$.next({ type: 'dispose', payload }); + }; + + const disposable: t.DisposableAsync = { + dispose$: dispose$.asObservable(), + async dispose() { + if (_disposing) return; + _disposing = true; + + fire('start'); + try { + await onDispose?.(); // Invoke handler ("clean up resources"). + fire('complete'); + } catch (err: any) { + fire('error', { + name: 'DisposeError', + message: 'Failed while disposing asynchronously', + cause: Err.std(err), + }); + } + }, + }; + + until(until$).forEach(($) => $.subscribe(disposable.dispose)); + return disposable; +} + +/** + * Helpers + */ +export function toDisposableAsyncArgs(args: any[]) { + type Fn = () => Promise<void>; + let onDispose: Fn | undefined; + let until$: t.UntilObservable | undefined; + + if (typeof args[0] === 'function') onDispose = args[0]; + if (typeof args[1] === 'function') onDispose = args[1]; + if (Is.observable(args[0]) || Array.isArray(args[0])) until$ = until(args[0]); + + return { onDispose, until$ }; +} diff --git a/code/sys/std/src/m.Dispose/u.done.ts b/code/sys/std/src/m.Dispose/u.done.ts new file mode 100644 index 0000000000..f45413a6cc --- /dev/null +++ b/code/sys/std/src/m.Dispose/u.done.ts @@ -0,0 +1,11 @@ +import { type t } from './common.ts'; + +/** + * "Completes" a subject by running: + * 1. subject.next(); + * 2. subject.complete(); + */ +export function done(dispose$?: t.Subject<void>) { + dispose$?.next?.(); + dispose$?.complete?.(); +} diff --git a/code/sys/std/src/m.Dispose/u.lifecycle.ts b/code/sys/std/src/m.Dispose/u.lifecycle.ts new file mode 100644 index 0000000000..2a08f1ff3e --- /dev/null +++ b/code/sys/std/src/m.Dispose/u.lifecycle.ts @@ -0,0 +1,81 @@ +import { type t, filter, take } from './common.ts'; +import { disposable, disposableAsync, toDisposableAsyncArgs } from './u.dispose.ts'; + +type L = t.Lifecycle; + +/** + * Generates a disposable interface that maintains + * and exposes it's disposed state. + */ +export function lifecycle(until$?: t.UntilInput) { + const { dispose, dispose$ } = disposable(until$); + let _disposed = false; + dispose$.pipe(take(1)).subscribe(() => (_disposed = true)); + return { + dispose, + get dispose$() { + return dispose$; + }, + get disposed() { + return _disposed; + }, + }; +} + +/** + * An async variant of the lifecycle pattern. + */ +export function lifecycleAsync(...args: any[]) { + const { until$, onDispose } = toDisposableAsyncArgs(args); + const { dispose, dispose$ } = disposableAsync(until$, onDispose); + let _disposed = false; + dispose$ + .pipe( + filter((e) => e.payload.stage === 'complete' || e.payload.stage === 'error'), + take(1), + ) + .subscribe(() => (_disposed = true)); + return { + dispose$, + dispose, + get disposed() { + return _disposed; + }, + }; +} + +/** + * Extend the given object to be expose the lifecycle API. + */ +export const toLifecycle: t.DisposeLib['toLifecycle'] = <T extends L>(...input: any[]): T => { + const { api, life } = wrangle.toLifecycleParams(input); + const obj = api as T & L; + + Object.defineProperties(obj, { + dispose: { + value: life.dispose.bind(life), + enumerable: true, + }, + dispose$: { + get: () => life.dispose$, + enumerable: true, + }, + disposed: { + get: () => life.disposed, + enumerable: true, + }, + }); + + return obj; +}; + +/** + * Helpers + */ +const wrangle = { + toLifecycleParams<T extends t.Lifecycle>(input: any[]): { life: t.Lifecycle; api: T } { + if (input.length === 1) return { life: lifecycle(), api: input[0] as T }; + if (input.length >= 2) return { life: input[0], api: input[1] }; + throw new Error('Failed to parse overloads: toLifecycle'); + }, +} as const; diff --git a/code/sys/std/src/m.Dispose/u.omitDispose.ts b/code/sys/std/src/m.Dispose/u.omitDispose.ts new file mode 100644 index 0000000000..4ad046b895 --- /dev/null +++ b/code/sys/std/src/m.Dispose/u.omitDispose.ts @@ -0,0 +1,21 @@ +import { type t } from './common.ts'; + +type D = t.Disposable | t.DisposableAsync; + +/** + * Safely remove the `dispose` method from a disposable. + * NB: useful for surfacing from an API where you don't want + * callers to be able to disose of the resource. + */ +export function omitDispose<T extends D>(obj: T): Omit<T, 'dispose'> { + const proto = Object.getPrototypeOf(obj); + const allDescs = Object.getOwnPropertyDescriptors(obj); + const newDescs: PropertyDescriptorMap = {}; + + for (const [key, desc] of Object.entries(allDescs)) { + if (key === 'dispose') continue; // NB: skip it. + newDescs[key] = desc; + } + + return Object.create(proto, newDescs) as Omit<T, 'dispose'>; +} diff --git a/code/sys/std/src/m.Dispose/u.until.ts b/code/sys/std/src/m.Dispose/u.until.ts new file mode 100644 index 0000000000..e4fd10efd4 --- /dev/null +++ b/code/sys/std/src/m.Dispose/u.until.ts @@ -0,0 +1,15 @@ +import { type t, flatten, Is } from './common.ts'; + +/** + * Listens to an observable (or set of observbles) and + * disposes of the target when any of them fire. + */ +export function until(input?: t.DisposeInput) { + if (Is.disposable(input)) { + return [input.dispose$]; + } else { + const $ = input; + const list = Array.isArray($) ? $ : [$]; + return flatten(list).filter(Boolean) as t.Observable<unknown>[]; + } +} diff --git a/code/sys/std/src/m.Err/-Err.catch.test.ts b/code/sys/std/src/m.Err/-Err.tryCatch.test.ts similarity index 88% rename from code/sys/std/src/m.Err/-Err.catch.test.ts rename to code/sys/std/src/m.Err/-Err.tryCatch.test.ts index 5af3425b1a..61e1f0c0d1 100644 --- a/code/sys/std/src/m.Err/-Err.catch.test.ts +++ b/code/sys/std/src/m.Err/-Err.tryCatch.test.ts @@ -9,7 +9,7 @@ describe('Err.catch', () => { }; it('success', async () => { - const res = await Err.catch(getUser(1)); + const res = await Err.tryCatch(getUser(1)); expect(res.ok).to.eql(true); expect(res.data).to.eql({ id: 1 }); expect(res.data?.id).to.equal(1); @@ -17,7 +17,7 @@ describe('Err.catch', () => { }); it('fail (error)', async () => { - const res = await Err.catch(getUser(123, true)); + const res = await Err.tryCatch(getUser(123, true)); expect(res.ok).to.eql(false); expect(res.data).to.eql(undefined); diff --git a/code/sys/std/src/m.Err/m.Err.ts b/code/sys/std/src/m.Err/m.Err.ts index 3f1bb9911b..660fffa82d 100644 --- a/code/sys/std/src/m.Err/m.Err.ts +++ b/code/sys/std/src/m.Err/m.Err.ts @@ -3,7 +3,7 @@ import { Is } from './m.Is.ts'; import { Name } from './m.Name.ts'; import { errors } from './u.errors.ts'; import { std } from './u.std.ts'; -import { catchError } from './u.catchError.ts'; +import { tryCatch } from './u.tryCatchError.ts'; /** * Helpers for working with errors. @@ -13,5 +13,5 @@ export const Err: t.ErrLib = { Name, std, errors, - catch: catchError, + tryCatch, }; diff --git a/code/sys/std/src/m.Err/t.ts b/code/sys/std/src/m.Err/t.ts index 4375774264..fc487f181d 100644 --- a/code/sys/std/src/m.Err/t.ts +++ b/code/sys/std/src/m.Err/t.ts @@ -20,19 +20,19 @@ export type ErrLib = { errors(): t.ErrorCollection; /** - * Principled way to handle try/catch/(error) execution - * on async functions avoiding the proliferation of - * try/catch statements. + * Function strongly typed way to handle try/catch (error) + * execution on async functions avoiding the proliferation of + * problematic native try/catch statements around a codebase. */ - catch<T>(promise: Promise<T>): Promise<t.ErrCatch<T>>; + tryCatch<T>(promise: Promise<T>): Promise<t.ErrCatch<T>>; }; /** * The response (and/or error) from an [Err.catch] method call. */ -export type ErrCatch<T> = ErrCatchSuccess<T> | ErrCatchFail<T>; -export type ErrCatchSuccess<T> = { ok: true; data: T; error: undefined }; -export type ErrCatchFail<T> = { ok: false; data?: T; error: t.StdError }; +export type ErrCatch<T> = ErrSuccess<T> | ErrFail<T>; +export type ErrSuccess<T> = { ok: true; data: T; error: undefined }; +export type ErrFail<T> = { ok: false; data?: T; error: t.StdError }; /** * Options passed to the `ErrLib.stdErr` method. diff --git a/code/sys/std/src/m.Err/u.catchError.ts b/code/sys/std/src/m.Err/u.tryCatchError.ts similarity index 71% rename from code/sys/std/src/m.Err/u.catchError.ts rename to code/sys/std/src/m.Err/u.tryCatchError.ts index a7a64abad0..34c68f62fc 100644 --- a/code/sys/std/src/m.Err/u.catchError.ts +++ b/code/sys/std/src/m.Err/u.tryCatchError.ts @@ -1,9 +1,7 @@ import type { t } from './common.ts'; import { std } from './u.std.ts'; -type F = t.ErrLib['catch']; - -export const catchError: F = async <T>(promise: Promise<T>) => { +export const tryCatch: t.ErrLib['tryCatch'] = async <T>(promise: Promise<T>) => { try { const data = await promise; return { ok: true, data, error: undefined }; diff --git a/code/sys/std/src/m.Immutable/-T.example.test.ts b/code/sys/std/src/m.Immutable/-T.example.test.ts new file mode 100644 index 0000000000..17bef78232 --- /dev/null +++ b/code/sys/std/src/m.Immutable/-T.example.test.ts @@ -0,0 +1,94 @@ +import { type t, describe, expect, it, slug } from '../-test.ts'; +import { single } from 'rxjs'; +import { Immutable } from './mod.ts'; + +describe('T:Immutable', () => { + /** + * Types: + */ + type P = t.PatchOperation; + + type MyState = { tmp: number }; + type MyStateEvent = t.InferImmutableEvent<MyStateEvents>; + type MyStateEvents = t.ImmutableEvents<MyState, P>; + type MyStateImmutable = t.ImmutableRef<MyState, P, MyStateEvents>; + + /** + * Sample implementation: + */ + const DEFAULT_ID = `default:${slug()}`; + let refs: Map<string, MyStateImmutable>; + const reset = () => (refs = new Map<string, MyStateImmutable>()); + reset(); + + const factory = (instanceId: string = DEFAULT_ID) => { + if (refs.has(instanceId)) return refs.get(instanceId)!; + + const model = Immutable.clonerRef<MyState>({ tmp: 0 }); + refs.set(instanceId, model); + return model; + }; + + const Store = { + state: factory, + }; + + /** + * State Factory (instantiation): + */ + it('example type declaration: /t.ts', () => { + const singleton = Store.state(); + const another = Store.state('something else'); + expect(singleton).to.equal(Store.state()); // NB: no-param ā singleton factory. + expect(singleton).to.not.equal(another); + + /** + * Immutable change: + */ + singleton.change((d) => d.tmp++); + expect(singleton.current.tmp).to.eql(1); + expect(another.current.tmp).to.eql(0); + }); + + describe('Store.state: (sample function)', () => { + it('default (singleton)', () => { + reset(); + const a = Store.state(); + const b = Store.state(); + expect(a).to.equal(b); // NB: same instance + }); + + it('custom: instance-id', () => { + reset(); + const id = 'foo'; + const a = Store.state(id); + const b = Store.state(id); + const c = Store.state(); + const d = Store.state('something else'); + expect(a).to.equal(b); // NB: same instance + expect(a).to.not.equal(c); + expect(a).to.not.equal(d); + }); + + it('change', () => { + reset(); + const a = Store.state(); + const b = Store.state(); + expect(a.current.tmp).to.eql(0); + a.change((d) => d.tmp++); + expect(b.current.tmp).to.eql(1); + }); + + it('shared events', () => { + reset(); + const a = Store.state(); + const b = Store.state(); + const bEvents = a.events(); + const bFired: MyStateEvent[] = []; + bEvents.changed$.pipe().subscribe((e) => bFired.push(e)); + + a.change((d) => d.tmp++); + expect(bFired[0].after).to.eql(a.current); + }); + }); +}); diff --git a/code/sys/std/src/m.Immutable/Immutable.events.ts b/code/sys/std/src/m.Immutable/Immutable.events.ts index 7d503a32bf..d0f8ffe33a 100644 --- a/code/sys/std/src/m.Immutable/Immutable.events.ts +++ b/code/sys/std/src/m.Immutable/Immutable.events.ts @@ -13,17 +13,11 @@ type DefaultPatch = t.PatchOperation; */ export function viaObservable<T, P = DefaultPatch>( $: t.Observable<t.ImmutableChange<T, P>>, - dispose$?: t.UntilObservable, + dispose$?: t.UntilInput, ): t.ImmutableEvents<T, P> { const life = rx.lifecycle(dispose$); - return { - changed$: $.pipe(rx.takeUntil(life.dispose$)), - dispose: life.dispose, - dispose$: life.dispose$, - get disposed() { - return life.disposed; - }, - }; + const changed$ = $.pipe(rx.takeUntil(life.dispose$)); + return rx.toLifecycle<t.ImmutableEvents<T, P>>(life, { changed$ }); } /** @@ -32,7 +26,7 @@ export function viaObservable<T, P = DefaultPatch>( */ export function viaOverride<T, P = DefaultPatch>( source: t.Immutable<T, P>, - dispose$?: t.UntilObservable, + dispose$?: t.UntilInput, ): t.ImmutableEvents<T, P> { const $ = rx.subject<t.ImmutableChange<T, P>>(); const api = viaObservable<T, P>($, dispose$); diff --git a/code/sys/std/src/m.Immutable/t.ts b/code/sys/std/src/m.Immutable/t.ts index 0a1134b478..a4278d7f71 100644 --- a/code/sys/std/src/m.Immutable/t.ts +++ b/code/sys/std/src/m.Immutable/t.ts @@ -17,12 +17,12 @@ type ClonerRef = <T>( type EventsViaOverride = <T, P = t.PatchOperation>( source: t.Immutable<T, P>, - dispose$?: t.UntilObservable, + dispose$?: t.UntilInput, ) => t.ImmutableEvents<T, P>; type EventsViaObservable = <T, P = t.PatchOperation>( $: t.Observable<t.ImmutableChange<T, P>>, - dispose$?: t.UntilObservable, + dispose$?: t.UntilInput, ) => t.ImmutableEvents<T, P>; /** diff --git a/code/sys/std/src/m.Is/-.test.ts b/code/sys/std/src/m.Is/-.test.ts index 1f5932b45c..806b263a2c 100644 --- a/code/sys/std/src/m.Is/-.test.ts +++ b/code/sys/std/src/m.Is/-.test.ts @@ -1,5 +1,5 @@ import { describe, expect, it } from '../-test.ts'; -import { Err, Is, Rx } from '../mod.ts'; +import { Err, Is, rx, Rx } from '../mod.ts'; describe('Is (common flags)', () => { it('rx: observable | subject', () => { @@ -139,6 +139,20 @@ describe('Is (common flags)', () => { }); }); + describe('Is.uint8Array', () => { + const binary = new Uint8Array([1, 2, 3]); + + it('is not Uint8Array', () => { + const NON = ['', 123, true, null, undefined, BigInt(0), Symbol('foo'), {}, []]; + NON.forEach((v) => expect(Is.uint8Array(v)).to.eql(false)); + expect(Is.uint8Array(binary.buffer)).to.eql(false); + }); + + it('is Uint8Array', () => { + expect(Is.uint8Array(binary)).to.equal(true); + }); + }); + describe('Is.blank', () => { describe('blank', () => { it('is blank (nothing)', () => { @@ -194,11 +208,38 @@ describe('Is (common flags)', () => { }); }); - it('Is.statusOK', () => { - const NON = ['foo', 123, false, null, undefined, {}, [], Symbol('foo'), BigInt(0)]; - NON.forEach((v: any) => expect(Is.statusOK(v)).to.eql(false)); - expect(Is.statusOK(200)).to.eql(true); - expect(Is.statusOK(201)).to.eql(true); - expect(Is.statusOK(404)).to.eql(false); + describe('Is.statusOK', () => { + it('Is.statusOK: true', () => { + expect(Is.statusOK(200)).to.eql(true); + expect(Is.statusOK(201)).to.eql(true); + }); + + it('Is.statusOK: false', () => { + const NON = ['foo', 123, false, null, undefined, {}, [], Symbol('foo'), BigInt(0)]; + NON.forEach((v: any) => expect(Is.statusOK(v)).to.eql(false)); + expect(Is.statusOK(404)).to.eql(false); + }); + }); + + describe('Is.browser', () => { + it('Is.browser: false', () => { + expect(Is.browser()).to.eql(false); + }); + }); + + describe('Is.disposable', () => { + it('Is.disposable: true', () => { + const disposable = rx.disposable(); + const life = rx.lifecycle(); + expect(Is.disposable(disposable)).to.be.true; + expect(Is.disposable(life)).to.be.true; + }); + + it('Is.disposable: false', () => { + const NON = ['', 123, true, null, undefined, BigInt(0), Symbol('foo'), {}, []]; + NON.forEach((value) => { + expect(Is.disposable(value)).to.eql(false); + }); + }); }); }); diff --git a/code/sys/std/src/m.Is/m.Is.ts b/code/sys/std/src/m.Is/m.Is.ts index ec347ec59a..3181610c2c 100644 --- a/code/sys/std/src/m.Is/m.Is.ts +++ b/code/sys/std/src/m.Is/m.Is.ts @@ -1,19 +1,35 @@ import { type t, isObject } from '../common.ts'; import { Err } from '../m.Err/mod.ts'; -import { Is as RxIs } from '../m.Rx/mod.ts'; -const { observable, subject } = RxIs; const { errorLike, stdError } = Err.Is; /** * Common flag evaluators. */ export const Is: t.StdIsLib = { - observable, - subject, errorLike, stdError, + disposable(input?: any): input is t.Disposable { + if (!isObject(input)) return false; + const obj = input as t.Disposable; + return typeof obj.dispose === 'function' && Is.observable(obj.dispose$); + }, + + /** + * Determine if the given input is an Observable. + */ + observable<T = unknown>(input?: any): input is t.Observable<T> { + return typeof input === 'object' && typeof input?.subscribe === 'function'; + }, + + /** + * Determine if the given input is an observable Subject. + */ + subject<T = unknown>(input?: any): input is t.Subject<T> { + return Is.observable(input) && typeof (input as any)?.next === 'function'; + }, + falsy(input?: any): input is t.Falsy | typeof NaN { return ( input === false || @@ -61,7 +77,12 @@ export const Is: t.StdIsLib = { }, arrayBufferLike(input?: any): input is ArrayBufferLike { - return input instanceof ArrayBuffer || input instanceof SharedArrayBuffer; + const tag = Object.prototype.toString.call(input); + return tag === '[object ArrayBuffer]' || tag === '[object SharedArrayBuffer]'; + }, + + uint8Array(input?: any): input is Uint8Array { + return Object.prototype.toString.call(input) === '[object Uint8Array]'; }, /** @@ -97,4 +118,12 @@ export const Is: t.StdIsLib = { if (typeof input !== 'number') return false; return String(input)[0] === '2'; }, + + /** + * Determines if currently running within a browser environment. + */ + browser() { + const g = globalThis; + return typeof g.window === 'object' && typeof g.document === 'object'; + }, }; diff --git a/code/sys/std/src/m.Is/t.ts b/code/sys/std/src/m.Is/t.ts index 58f4ab6b8a..133941f1e5 100644 --- a/code/sys/std/src/m.Is/t.ts +++ b/code/sys/std/src/m.Is/t.ts @@ -15,6 +15,11 @@ export type StdIsLib = { */ nil(input?: unknown): boolean; + /** + * Determine if the input is an object implementing the <t.Disposable> interface. + */ + disposable(input?: unknown): input is t.Disposable; + /** * Determine if the value is a Promise. */ @@ -23,12 +28,12 @@ export type StdIsLib = { /** * Determine if the value is an observable Subject. */ - subject: t.RxIs['subject']; + subject<T = unknown>(input?: any): input is t.Subject<T>; /** * Determine if the value is an Observable. */ - observable: t.RxIs['observable']; + observable<T = unknown>(input?: any): input is t.Observable<T>; /** * Determine if the value is like an Error object. @@ -51,10 +56,15 @@ export type StdIsLib = { json(input?: unknown): input is t.Json; /** - * Determine if the input is ArrayBufferLike. + * Determine if the input is [ArrayBufferLike]. */ arrayBufferLike(input?: unknown): input is ArrayBufferLike; + /** + * Determine if the inut is a [Uint8Array]. + */ + uint8Array(input?: unknown): input is Uint8Array; + /** * A safe way to test any value as to wheather is is 'blank' * meaning it can be either: @@ -70,4 +80,7 @@ export type StdIsLib = { /** Determine if the HTTP status code is within the 200 range. */ statusOK(status: number): boolean; + + /** Determines if currently running within a browser environment. */ + browser(): boolean; }; diff --git a/code/sys/std/src/m.Pkg/-Pkg.Dist.test.ts b/code/sys/std/src/m.Pkg/-Pkg.Dist.test.ts index 66d11be773..d5d53e6347 100644 --- a/code/sys/std/src/m.Pkg/-Pkg.Dist.test.ts +++ b/code/sys/std/src/m.Pkg/-Pkg.Dist.test.ts @@ -12,6 +12,7 @@ describe('Pkg.Dist', () => { const SAMPLE = { dist(): t.DistPkg { return { + '-type:': 'jsr:@sys/types:DistPkg', pkg: { name: `@ns/foo-${slug()}`, version: '1.2.3' }, size: { bytes: 1234 }, entry: './main.js', diff --git a/code/sys/std/src/m.Pkg/-Pkg.test.ts b/code/sys/std/src/m.Pkg/-Pkg.test.ts index f585d5b865..b84c885094 100644 --- a/code/sys/std/src/m.Pkg/-Pkg.test.ts +++ b/code/sys/std/src/m.Pkg/-Pkg.test.ts @@ -91,6 +91,7 @@ describe('Pkg', () => { it('true', () => { const dist: t.DistPkg = { + '-type:': 'jsr:@sys/types:DistPkg', pkg: { name: 'foo', version: '1.2.3' }, size: { bytes: 123_456 }, entry: 'pkg/entry.js', diff --git a/code/sys/std/src/m.Random/.test.ts b/code/sys/std/src/m.Random/.test.ts index f1107dd124..535f1597bf 100644 --- a/code/sys/std/src/m.Random/.test.ts +++ b/code/sys/std/src/m.Random/.test.ts @@ -13,7 +13,7 @@ describe('Random', () => { }); describe('Random.base36: string ā [0-9] and [A-Z]', () => { - const test = (length: number, total = 1000) => { + const test = (length: number, total = 100) => { const list = Array.from({ length: total }).map(() => Random.base36(length)); expect(R.uniq(list)).to.eql(list); // NB: no repeating random numbers. list.forEach((value) => expect(value.length).to.eql(length)); diff --git a/code/sys/std/src/m.Rx/-Rx.test.ts b/code/sys/std/src/m.Rx/-Rx.test.ts index 1d5204827e..802f3ef569 100644 --- a/code/sys/std/src/m.Rx/-Rx.test.ts +++ b/code/sys/std/src/m.Rx/-Rx.test.ts @@ -1,7 +1,15 @@ -import { describe, expect, it, Time, Testing } from '../-test.ts'; +import { describe, expect, it, Testing, Time } from '../-test.ts'; import { Dispose, rx, Rx } from '../mod.ts'; describe('Observable/rx', () => { + describe('API', () => { + expect(rx.toLifecycle).to.equal(Dispose.toLifecycle); + expect(rx.lifecycle).to.equal(Dispose.lifecycle); + expect(rx.lifecycleAsync).to.equal(Dispose.lifecycleAsync); + expect(rx.disposable).to.equal(Dispose.disposable); + expect(rx.disposableAsync).to.equal(Dispose.disposableAsync); + }); + it('dual cased names', () => { expect(Rx).to.equal(rx); }); diff --git a/code/sys/std/src/m.Rx/common.ts b/code/sys/std/src/m.Rx/common.ts index 903a07f5e9..3f144d4ef8 100644 --- a/code/sys/std/src/m.Rx/common.ts +++ b/code/sys/std/src/m.Rx/common.ts @@ -1,3 +1,5 @@ export * from '../common.ts'; -export * from './u.Rx.libs.ts'; export { Dispose } from '../m.Dispose/mod.ts'; +export { Is as StdIs } from '../m.Is/mod.ts'; + +export * from './u.Rx.libs.ts'; diff --git a/code/sys/std/src/m.Rx/m.Is.ts b/code/sys/std/src/m.Rx/m.Is.ts index f6a7ab2512..7ede83f475 100644 --- a/code/sys/std/src/m.Rx/m.Is.ts +++ b/code/sys/std/src/m.Rx/m.Is.ts @@ -1,4 +1,4 @@ -import type { t } from '../common.ts'; +import { t, StdIs } from './common.ts'; type Event = { type: string; payload: unknown }; @@ -6,19 +6,8 @@ type Event = { type: string; payload: unknown }; * Type guards (boolean evaluators). */ export const Is: t.RxIs = { - /** - * Determine if the given input is an Observable. - */ - observable<T = unknown>(input?: any): input is t.Observable<T> { - return typeof input === 'object' && typeof input?.subscribe === 'function'; - }, - - /** - * Determine if the given input is an observable Subject. - */ - subject<T = unknown>(input?: any): input is t.Subject<T> { - return Is.observable(input) && typeof (input as any)?.next === 'function'; - }, + observable: StdIs.observable, + subject: StdIs.subject, /** * Determine if the object structure matches that of the diff --git a/code/sys/std/src/m.Rx/m.Rx.ts b/code/sys/std/src/m.Rx/m.Rx.ts index aa30270d3a..f6831959f8 100644 --- a/code/sys/std/src/m.Rx/m.Rx.ts +++ b/code/sys/std/src/m.Rx/m.Rx.ts @@ -2,12 +2,12 @@ import { type t, Dispose } from './common.ts'; import * as lib from './u.Rx.libs.ts'; import { Is } from './m.Is.ts'; +import { bus } from './u.bus.ts'; import { event, payload } from './u.payload.ts'; import { asPromise } from './u.promise.ts'; import { withinTimeThreshold } from './u.time.ts'; -import { bus } from './u.bus.ts'; -const { disposable, disposableAsync, lifecycle, lifecycleAsync, done } = Dispose; +const { disposable, disposableAsync, lifecycle, lifecycleAsync, done, toLifecycle } = Dispose; /** Tools for working with Observables (via the RXJS library). */ export const Rx: t.RxLib = { @@ -20,9 +20,10 @@ export const Rx: t.RxLib = { bus, done, disposable, - lifecycle, disposableAsync, + lifecycle, lifecycleAsync, + toLifecycle, event, payload, diff --git a/code/sys/std/src/m.Rx/t.ts b/code/sys/std/src/m.Rx/t.ts index 01b49c1dfe..dc3f8d895e 100644 --- a/code/sys/std/src/m.Rx/t.ts +++ b/code/sys/std/src/m.Rx/t.ts @@ -24,6 +24,7 @@ export type RxLib = Rxjs & { disposableAsync: t.DisposeLib['disposableAsync']; lifecycle: t.DisposeLib['lifecycle']; lifecycleAsync: t.DisposeLib['lifecycleAsync']; + toLifecycle: t.DisposeLib['toLifecycle']; withinTimeThreshold<T>( $: t.Observable<T>, @@ -56,8 +57,8 @@ export type RxPromiseResponse<E extends Event> = { */ export type RxIs = { event(input: any, type?: string | { startsWith: string }): boolean; - observable<T = unknown>(input?: any): input is t.Observable<T>; - subject<T = unknown>(input?: any): input is t.Subject<T>; + observable: t.StdIsLib['observable']; + subject: t.StdIsLib['subject']; }; /** diff --git a/code/sys/std/src/m.Rx/u.time.ts b/code/sys/std/src/m.Rx/u.time.ts index b0a249e865..7d545d0c6c 100644 --- a/code/sys/std/src/m.Rx/u.time.ts +++ b/code/sys/std/src/m.Rx/u.time.ts @@ -9,8 +9,10 @@ import { Subject, filter, take, takeUntil } from './u.Rx.libs.ts'; export function withinTimeThreshold<T>( $: t.Observable<T>, timeout: t.Msecs, - options: { dispose$?: t.UntilObservable } = {}, + options: { dispose$?: t.UntilInput } = {}, ): t.TimeThreshold<T> { + const life = Dispose.lifecycle(options.dispose$); + const listen = (timeout: number) => { type R = { result: boolean; value?: T }; const startedAt = Date.now(); @@ -36,32 +38,23 @@ export function withinTimeThreshold<T>( }; /** - * Response listener. + * Response listener: */ const timeout$ = new Subject<void>(); const $$ = new Subject<T>(); $.subscribe((e) => { const listen$ = listen(timeout).pipe( - takeUntil(dispose$), + takeUntil(life.dispose$), filter((e) => !!e.result), ); listen$.subscribe((e) => $$.next(e.value!)); }); - let _disposed = false; - const { dispose, dispose$ } = Dispose.disposable(options.dispose$); - dispose$.subscribe(() => { - $$.complete(); - _disposed = true; + /** + * API: + */ + return Dispose.toLifecycle<t.TimeThreshold<T>>(life, { + $: $$.pipe(takeUntil(life.dispose$)), + timeout$: timeout$.pipe(takeUntil(life.dispose$)), }); - - return { - $: $$.pipe(takeUntil(dispose$)), - timeout$: timeout$.pipe(takeUntil(dispose$)), - dispose, - dispose$, - get disposed() { - return _disposed; - }, - }; } diff --git a/code/sys/std/src/m.Signal/-.test.ts b/code/sys/std/src/m.Signal/-.test.ts new file mode 100644 index 0000000000..f4f1bac4d7 --- /dev/null +++ b/code/sys/std/src/m.Signal/-.test.ts @@ -0,0 +1,368 @@ +import { Time, describe, expect, it } from '../-test.ts'; +import { Signal } from './mod.ts'; +import { rx } from '../m.Rx/mod.ts'; + +import * as Preact from '@preact/signals-core'; + +describe('Signal', () => { + it('API', () => { + expect(Signal.create).to.equal(Preact.signal); + expect(Signal.effect).to.equal(Preact.effect); + }); + + describe('Core "Signal" API', () => { + describe('signal: update', () => { + it('should create a signal with an initial value and update correctly', () => { + const s = Signal.create(0); + expect(s.value).to.eql(0); + s.value = 42; + expect(s.value).to.eql(42); + }); + + it('updates object', async () => { + type T = { count: number }; + const initial = { count: 0 }; + const s = Signal.create<T>(initial); + expect(s.value).to.eql(initial); + expect(s.value).to.equal(initial); // NB: actual instance. + + let fired: T[] = []; + const stop = Signal.effect(() => { + fired.push(s.value); + }); + expect(fired).to.eql([initial]); + + // Replace object. + s.value = { count: 123 }; + + await Time.wait(); + expect(fired.length).to.eql(2); + expect(fired).to.eql([initial, { count: 123 }]); + + // Mutate object. + s.value.count = 456; + await Time.wait(); + + expect(s.value).to.eql({ count: 456 }); + expect(fired).to.eql([initial, { count: 456 }]); // ā ļø NB: the object IS NOT immutable. + expect(fired.length).to.eql(2); // ...and no change event was fired. + + stop(); + }); + }); + + describe('signal: effect (reactivity)', () => { + it('should run the effect whenever the signal value changes', async () => { + let dummy = 0; + const s = Signal.create(0); + + const stop = Signal.effect(() => { + dummy = s.value; + }); + + expect(dummy).to.eql(0); + s.value = 123; + + await Time.wait(); // NB: Wait for the effect to propagate (micro-task queue, aka. "tick"). + expect(dummy).to.eql(123); + + stop(); // NB: Stop the effect to release listener (dispose). + }); + }); + + describe('signal: computed', () => { + it('should create a derived signal that updates based on dependencies', () => { + const a = Signal.create(2); + const b = Signal.create(3); + const sum = Signal.computed(() => a.value + b.value); + expect(sum.value).to.eql(5); + + b.value = 10; + expect(sum.value).to.eql(12); + }); + }); + + describe('signal: batch (change)', () => { + it('should group updates so that effects run only once', () => { + let count = 0; + const x = Signal.create(1); + const y = Signal.create(2); + + Signal.effect(() => { + count++; + // NB: Access signals so this effect depends on them. + x.value; + y.value; + }); + + expect(count).to.eql(1); + + Signal.batch(() => { + x.value = 10; + y.value = 20; + }); + + // NB: Both updates happened in one batch, so the effect runs only once more. + expect(count).to.equal(2); + }); + }); + }); + + describe('value helpers', () => { + describe('Signal.toggle', () => { + it('should toggle boolean', () => { + const s = Signal.create(false); + expect(s.value).to.eql(false); + const res = Signal.toggle(s); + expect(s.value).to.eql(true); + expect(res).to.eql(true); + }); + + it('should toggle boolean from <undefined>', () => { + const s = Signal.create<boolean | undefined>(); + expect(s.value).to.eql(undefined); + const res = Signal.toggle(s); + expect(s.value).to.eql(true); + expect(res).to.eql(true); + Signal.toggle(s); + expect(s.value).to.eql(false); + Signal.toggle(s); + expect(s.value).to.eql(true); + }); + + it('force: true', () => { + const s = Signal.create(false); + const res1 = Signal.toggle(s, true); + const res2 = Signal.toggle(s, true); + expect(res1).to.eql(true); + expect(res2).to.eql(true); + expect(s.value).to.eql(true); + }); + + it('force: false', () => { + const s = Signal.create(true); + const res1 = Signal.toggle(s, false); + const res2 = Signal.toggle(s, false); + expect(res1).to.eql(false); + expect(res2).to.eql(false); + expect(s.value).to.eql(false); + }); + }); + + describe('Signal.cycle', () => { + type T = 'a' | 'b' | 'c'; + + it('should cycle union [string] signal', () => { + const s = Signal.create<T>('a'); + expect(s.value).to.eql('a'); + + const values: T[] = ['a', 'b', 'c']; + + // Cycle from "a" to "b". + const res1 = Signal.cycle(s, values); + expect(res1).to.eql('b'); + expect(s.value).to.eql('b'); + + // Cycle from "b" to "c". + const res2 = Signal.cycle(s, values); + expect(res2).to.eql('c'); + expect(s.value).to.eql('c'); + + // Cycle from "c" back to "a". + const res3 = Signal.cycle(s, values); + expect(res3).to.eql('a'); + expect(s.value).to.eql('a'); + }); + + it('cycles [number] array', () => { + const s = Signal.create<number>(1); + expect(s.value).to.eql(1); + + const values: number[] = [1, 2, 3]; + + // Cycle from 1 to 2. + const res1 = Signal.cycle(s, values); + expect(res1).to.eql(2); + expect(s.value).to.eql(2); + + // Cycle from 2 to 3. + const res2 = Signal.cycle(s, values); + expect(res2).to.eql(3); + expect(s.value).to.eql(3); + + // Cycle from 3 back to 1. + const res3 = Signal.cycle(s, values); + expect(res3).to.eql(1); + expect(s.value).to.eql(1); + }); + + it('cycles array of arrays (1)', () => { + type V = string | number; + const s = Signal.create<V[] | undefined>(); + expect(s.value).to.eql(undefined); + + const values: V[][] = [ + [1, 2], + [3, 4], + ['1fr', 100, 'auto'], + ]; + + const res = Signal.cycle(s, values); + expect(res).to.eql([1, 2]); + expect(s.value).to.eql([1, 2]); + + Signal.cycle(s, values); + expect(s.value).to.eql([3, 4]); + + Signal.cycle(s, values); + expect(s.value).to.eql(['1fr', 100, 'auto']); + }); + + it('cycle array of arrays (2)', () => { + type V = string | number; + type T = V | V[] | undefined; + const s = Signal.create<T>(); + const values = [undefined, 0, 10, [50, 15], ['1fr', 100, 'auto']]; + + const test = (expected: T) => { + Signal.cycle(s, values); + expect(s.value).to.eql(expected); + }; + + test(0); + test(10); + test([50, 15]); + test(['1fr', 100, 'auto']); + test(undefined); + }); + + it('cycles from <undefined>', () => { + const s = Signal.create<T | undefined>(); + expect(s.value).to.eql(undefined); + + const values: (T | undefined)[] = [undefined, 'a', 'b']; + const res = Signal.cycle(s, values); + expect(res).to.eql('a'); + expect(s.value).to.eql('a'); + + Signal.cycle(s, values); + expect(s.value).to.eql('b'); + + Signal.cycle(s, values); + expect(s.value).to.eql(undefined); + + Signal.cycle(s, values); + expect(s.value).to.eql('a'); + }); + + it('cycles from different initial value', () => { + const s = Signal.create<T>('b'); + expect(s.value).to.eql('b'); + + const values: T[] = ['a', 'b', 'c']; + const res = Signal.cycle(s, values); + expect(res).to.eql('c'); + expect(s.value).to.eql('c'); + }); + + it('cycles from <undefined> initial value', () => { + const s = Signal.create<T>(); + expect(s.value).to.eql(undefined); + + const values: T[] = ['a', 'b', 'c']; + const res1 = Signal.cycle<T>(s, values); + expect(res1).to.eql('a'); + expect(s.value).to.eql('a'); + + Signal.cycle<T>(s, values); + Signal.cycle<T>(s, values); + expect(s.value).to.eql('c'); + Signal.cycle<T>(s, values); + expect(s.value).to.eql('a'); + + // Edge-case: no values specified ā returns <undefined>. + s.value = undefined; + const res2 = Signal.cycle<T>(s, []); + expect(res2).to.eql(undefined); + expect(s.value).to.eql(undefined); + }); + + it('force value works', () => { + const s = Signal.create<T>('a'); + expect(s.value).to.eql('a'); + + const values: T[] = ['a', 'b', 'c']; + + // Force to "c" + const res1 = Signal.cycle(s, values, 'c'); + expect(res1).to.eql('c'); + expect(s.value).to.eql('c'); + + // Force to "b" + const res2 = Signal.cycle(s, values, 'b'); + expect(res2).to.eql('b'); + expect(s.value).to.eql('b'); + }); + + it('should default to first element if current value is not in values array', () => { + const s = Signal.create('z'); + expect(s.value).to.eql('z'); + + const values: T[] = ['a', 'b', 'c']; + + const res = Signal.cycle(s, values); + expect(res).to.eql('a'); + expect(s.value).to.eql('a'); + }); + }); + }); + + describe('Listeners', () => { + it('create ā <change> ā dispose', () => { + const life = rx.disposable(); + const a = Signal.listeners(); + const b = Signal.listeners(life); + + expect(a.disposed).to.eql(false); + expect(b.disposed).to.eql(false); + + expect(a.count).to.eql(0); + expect(b.count).to.eql(0); + + const signal = Signal.create(0); + const fired = { a: 0, b: 0 }; + const resA = a.effect(() => { + signal.value; + fired.a++; + }); + const resB = b.effect(() => { + signal.value; + fired.b++; + }); + + expect(resA.count).to.eql(1); + expect(resB.count).to.eql(1); + + expect(fired).to.eql({ a: 1, b: 1 }); // NB: initial run. + signal.value++; + expect(fired).to.eql({ a: 2, b: 2 }); + + life.dispose(); + expect(a.disposed).to.eql(false); + expect(b.disposed).to.eql(true); + expect(b.count).to.eql(0); + + signal.value++; + expect(fired).to.eql({ a: 3, b: 2 }); + + a.dispose(); + signal.value++; + expect(fired).to.eql({ a: 3, b: 2 }); // NB: no change. + + expect(a.disposed).to.eql(true); + expect(b.disposed).to.eql(true); + expect(a.count).to.eql(0); + expect(b.count).to.eql(0); + }); + }); +}); diff --git a/code/sys/std/src/m.Signal/common.ts b/code/sys/std/src/m.Signal/common.ts new file mode 100644 index 0000000000..579e1f4a09 --- /dev/null +++ b/code/sys/std/src/m.Signal/common.ts @@ -0,0 +1,2 @@ +export * from '../common.ts'; +export { Dispose } from '../m.Dispose/mod.ts'; diff --git a/code/sys/std/src/m.Signal/m.Signal.ts b/code/sys/std/src/m.Signal/m.Signal.ts new file mode 100644 index 0000000000..83437fd241 --- /dev/null +++ b/code/sys/std/src/m.Signal/m.Signal.ts @@ -0,0 +1,32 @@ +import { batch, computed, effect, signal } from '@preact/signals-core'; + +import type { t } from './common.ts'; +import { cycle } from './u.cycle.ts'; +import { toggle } from './u.toggle.ts'; +import { listeners } from './u.listeners.ts'; + +export { signal }; + +/** + * Reactive Signals. + * See: + * https://github.com/tc39/proposal-signals + * https://preactjs.com/blog/introducing-signals/ + * https://preactjs.com/guide/v10/signals + */ +export const Signal: t.SignalLib = { + /** + * Primary API: + */ + create: signal, + effect, + computed, + batch, + + /** + * Helpers: + */ + listeners, + toggle, + cycle, +} as const; diff --git a/code/sys/std/src/m.Signal/mod.ts b/code/sys/std/src/m.Signal/mod.ts new file mode 100644 index 0000000000..c75d94b0e9 --- /dev/null +++ b/code/sys/std/src/m.Signal/mod.ts @@ -0,0 +1,9 @@ +/** + * @module + * Reactive Signals. + * See: + * https://github.com/tc39/proposal-signals + * https://preactjs.com/blog/introducing-signals/ + * https://preactjs.com/guide/v10/signals + */ +export { Signal } from './m.Signal.ts'; diff --git a/code/sys/std/src/m.Signal/t.ts b/code/sys/std/src/m.Signal/t.ts new file mode 100644 index 0000000000..2d01dc195b --- /dev/null +++ b/code/sys/std/src/m.Signal/t.ts @@ -0,0 +1,64 @@ +import type { t } from './common.ts'; +import type Preact from '@preact/signals-core'; +import type { ReadonlySignal, Signal } from '@preact/signals-core'; + +export { ReadonlySignal, Signal }; + +/** + * Utility type to extract the type of a signal value. + * @example + * ```ts + * const mySignal = Signal.create<'Foo' | 'Bar'>('Foo'); + * type T = ExtractSignalValue<typeof mySignal>; + * ``` + */ +export type ExtractSignalValue<T> = T extends Signal<infer U> ? U : never; + +/** Callback passed into a signal effect. */ +export type SignalEffectFn = () => void | (() => void); + +/** + * Reactive Signals. + * See: + * https://github.com/tc39/proposal-signals + * https://preactjs.com/blog/introducing-signals/ + * https://preactjs.com/guide/v10/signals + */ +export type SignalLib = { + /** Create a new plain signal. */ + create: typeof Preact.signal; + + /** Create an effect to run arbitrary code in response to signal changes. */ + effect: typeof Preact.effect; + + /** Combine multiple value updates into one "commit" at the end of the provided callback. */ + batch: typeof Preact.batch; + + /** Create a new signal that is computed based on the values of other signals. */ + computed: typeof Preact.computed; + + /** Create a new listeners collection. */ + listeners(dispose$?: t.UntilInput): t.SignalListeners; + + // +} & t.SignalValueHelpersLib; + +/** + * Utility helpers for operating on Signal values. + */ +export type SignalValueHelpersLib = { + /** Toggle a boolean signal. */ + toggle(signal: Signal<boolean | undefined>, forceValue?: boolean): boolean; + + /** Cycle a union string signal through a list of possible values. */ + cycle<T>(signal: Signal<T | undefined>, values: T[], forceValue?: T): T; +}; + +/** + * Helper for managing the disposal of a collection + * of signal effect listeners. + */ +export type SignalListeners = t.Lifecycle & { + readonly count: number; + effect(fn: t.SignalEffectFn): SignalListeners; +}; diff --git a/code/sys/std/src/m.Signal/u.cycle.ts b/code/sys/std/src/m.Signal/u.cycle.ts new file mode 100644 index 0000000000..3ae6132ba9 --- /dev/null +++ b/code/sys/std/src/m.Signal/u.cycle.ts @@ -0,0 +1,24 @@ +import { type t, R } from './common.ts'; + +/** + * Cycle a union string signal through a list of possible values. + */ +export const cycle: t.SignalLib['cycle'] = <T>( + signal: t.Signal<T | undefined>, + values: T[], + forceValue?: T, +): T => { + const next = forceValue !== undefined ? forceValue : wrangle.next(signal, values); + signal.value = next; + return next; +}; + +/** + * Helpers + */ +const wrangle = { + next<T>(signal: t.Signal<T | undefined>, values: T[]): T { + const index = values.findIndex((item) => R.equals(item, signal.value)); + return values[(index + 1) % values.length]; + }, +} as const; diff --git a/code/sys/std/src/m.Signal/u.listeners.ts b/code/sys/std/src/m.Signal/u.listeners.ts new file mode 100644 index 0000000000..2e2ca62282 --- /dev/null +++ b/code/sys/std/src/m.Signal/u.listeners.ts @@ -0,0 +1,38 @@ +import { effect as preactEffect } from '@preact/signals-core'; +import { type t, Dispose } from './common.ts'; + +export const listeners: t.SignalLib['listeners'] = (until$) => { + const life = Dispose.lifecycle(until$); + const disposers = new Set<() => void>(); + + const effect = (fn: t.SignalEffectFn) => { + const dispose = preactEffect(fn); + disposers.add(dispose); + + const sub = life.dispose$.subscribe(() => { + dispose(); + disposers.delete(dispose); + sub.unsubscribe(); + }); + + return api; + }; + + life.dispose$.subscribe(() => { + disposers.forEach((dispose) => dispose()); + disposers.clear(); + }); + + /** + * API: + */ + + const api = Dispose.toLifecycle<t.SignalListeners>(life, { + effect, + get count() { + return disposers.size; + }, + }); + + return api; +}; diff --git a/code/sys/std/src/m.Signal/u.toggle.ts b/code/sys/std/src/m.Signal/u.toggle.ts new file mode 100644 index 0000000000..4c92be764d --- /dev/null +++ b/code/sys/std/src/m.Signal/u.toggle.ts @@ -0,0 +1,10 @@ +import { type t } from './common.ts'; + +/** + * Toggle a boolean signal. + */ +export const toggle: t.SignalLib['toggle'] = (signal, forceValue) => { + const next = typeof forceValue === 'boolean' ? forceValue : !signal.value; + signal.value = next; + return next; +}; diff --git a/code/sys/std/src/m.Testing.Server/m.Server.ts b/code/sys/std/src/m.Testing.Server/m.Server.ts index 7f0646ed44..aa19a98983 100644 --- a/code/sys/std/src/m.Testing.Server/m.Server.ts +++ b/code/sys/std/src/m.Testing.Server/m.Server.ts @@ -1,5 +1,6 @@ -import type { t } from '../common/mod.ts'; +import type { t } from '../common.ts'; import { Url } from '../m.Url/mod.ts'; +import { Dispose } from '../m.Dispose/mod.ts'; type M = 'GET' | 'PUT' | 'POST' | 'DELETE'; type H = { method: M; handler: Deno.ServeHandler }; diff --git a/code/sys/std/src/m.Testing.Server/mod.ts b/code/sys/std/src/m.Testing.Server/mod.ts index 8634571532..4ccf4516fa 100644 --- a/code/sys/std/src/m.Testing.Server/mod.ts +++ b/code/sys/std/src/m.Testing.Server/mod.ts @@ -7,7 +7,17 @@ import type { t } from '../common.ts'; import { Testing as Base } from '../m.Testing/mod.ts'; import { TestHttpServer as Http } from './m.HttpServer.ts'; -export { describe, expect, expectError, it } from '../m.Testing/mod.ts'; +export { + afterAll, + afterEach, + Bdd, + beforeAll, + beforeEach, + describe, + expect, + expectError, + it, +} from '../m.Testing/mod.ts'; /** * Testing helpers including light-weight HTTP server helpers (Deno). diff --git a/code/sys/std/src/m.Testing/-Testing.test.ts b/code/sys/std/src/m.Testing/-.test.ts similarity index 56% rename from code/sys/std/src/m.Testing/-Testing.test.ts rename to code/sys/std/src/m.Testing/-.test.ts index cc31e7ed38..7946b036a7 100644 --- a/code/sys/std/src/m.Testing/-Testing.test.ts +++ b/code/sys/std/src/m.Testing/-.test.ts @@ -1,5 +1,5 @@ import { Random } from '../m.Random/mod.ts'; -import { Testing, describe, expect, it, expectError } from './mod.ts'; +import { Time, Testing, describe, expect, it, expectError } from './mod.ts'; Deno.test('Deno.test: sample (down at the test runner metal)', async (test) => { await test.step('eql', () => { @@ -8,10 +8,18 @@ Deno.test('Deno.test: sample (down at the test runner metal)', async (test) => { }); describe('Testing', () => { - it('exports BDD semantics', () => { + it('exports BDD semantics', async () => { + const { describe, it } = await import('@std/testing/bdd'); + const { afterAll, afterEach, beforeAll, beforeEach } = await import('@std/testing/bdd'); + + expect(Testing.Bdd.expect).to.equal(expect); expect(Testing.Bdd.describe).to.equal(describe); expect(Testing.Bdd.it).to.equal(it); - expect(Testing.Bdd.expect).to.equal(expect); + + expect(Testing.Bdd.beforeAll).to.equal(beforeAll); + expect(Testing.Bdd.afterAll).to.equal(afterAll); + expect(Testing.Bdd.beforeEach).to.equal(beforeEach); + expect(Testing.Bdd.afterEach).to.equal(afterEach); }); it('randomPort', () => { @@ -77,4 +85,32 @@ describe('Testing', () => { expect(count).to.eql(3); }); }); + + describe('wait', () => { + it('milliseconds (macro-task queue)', async () => { + const timer = Time.timer(); + await Testing.wait(10); + expect(timer.elapsed.msec).to.be.above(9); + }); + + it('no param (micro-task queue) ā "tick"', async () => { + await Testing.retry(3, async () => { + const timer = Time.timer(); + let microtaskResolved = false; + let macrotaskResolved = false; + + const stop = setTimeout(() => (macrotaskResolved = true), 0); // ā Schedule a macro-task for comparison. + await Testing.wait(); // ā the micro-task delay. + + const elapsed = timer.elapsed.msec; + microtaskResolved = true; + + expect(microtaskResolved).to.be.true; + expect(macrotaskResolved).to.be.false; // Microtasks should run before macrotasks + expect(elapsed).to.eql(0); // Should be ~0ms or very close + + clearTimeout(stop); + }); + }); + }); }); diff --git a/code/sys/std/src/m.Testing/libs.ts b/code/sys/std/src/m.Testing/libs.ts index 80c3b9f62e..f3f7ff0ee6 100644 --- a/code/sys/std/src/m.Testing/libs.ts +++ b/code/sys/std/src/m.Testing/libs.ts @@ -1,2 +1,2 @@ -export { describe, it } from '@std/testing/bdd'; +export { afterAll, afterEach, beforeAll, beforeEach, describe, it } from '@std/testing/bdd'; export { expect } from 'chai'; diff --git a/code/sys/std/src/m.Testing/m.Bdd.ts b/code/sys/std/src/m.Testing/m.Bdd.ts index 7483f4c734..8756c81f08 100644 --- a/code/sys/std/src/m.Testing/m.Bdd.ts +++ b/code/sys/std/src/m.Testing/m.Bdd.ts @@ -1,7 +1,16 @@ -import { describe, expect, it, type t } from './common.ts'; +import { + afterAll, + afterEach, + beforeAll, + beforeEach, + describe, + expect, + it, + type t, +} from './common.ts'; import { expectError } from './u.ts'; -export { describe, expect, it, expectError }; +export { afterAll, afterEach, beforeAll, beforeEach, describe, expect, expectError, it }; /** * BDD semantics ("Behavior Driven Development") helpers. @@ -9,6 +18,12 @@ export { describe, expect, it, expectError }; export const Bdd: t.BddLib = { describe, it, + + beforeAll, + beforeEach, + afterAll, + afterEach, + expect, expectError, }; diff --git a/code/sys/std/src/m.Testing/m.Testing.ts b/code/sys/std/src/m.Testing/m.Testing.ts index 0d0f47f291..d3450cf6d9 100644 --- a/code/sys/std/src/m.Testing/m.Testing.ts +++ b/code/sys/std/src/m.Testing/m.Testing.ts @@ -11,9 +11,10 @@ export const Testing: t.TestingLib = { slug, /** - * Wait for n-milliseconds + * Wait for n-milliseconds. */ wait(msecs): Promise<void> { + if (msecs === undefined) return Promise.resolve(); return new Promise((resolve) => setTimeout(resolve, msecs)); }, diff --git a/code/sys/std/src/m.Testing/mod.ts b/code/sys/std/src/m.Testing/mod.ts index eef702a60d..9f6fd1658c 100644 --- a/code/sys/std/src/m.Testing/mod.ts +++ b/code/sys/std/src/m.Testing/mod.ts @@ -20,12 +20,23 @@ * }); * }); */ -export { Bdd, describe, expect, expectError, it } from './m.Bdd.ts'; +export { + afterAll, + afterEach, + Bdd, + beforeAll, + beforeEach, + describe, + expect, + expectError, + it, +} from './m.Bdd.ts'; + export { Testing } from './m.Testing.ts'; /** * Common utility helpers. */ +export { Time } from '../m.DateTime/mod.ts'; export { Path } from '../m.Path/mod.ts'; export { slug } from '../m.Random/mod.ts'; -export { Time } from '../m.DateTime/mod.ts'; diff --git a/code/sys/std/src/m.Testing/t.ts b/code/sys/std/src/m.Testing/t.ts index 363a0ef60d..1c8bb7e218 100644 --- a/code/sys/std/src/m.Testing/t.ts +++ b/code/sys/std/src/m.Testing/t.ts @@ -1,4 +1,4 @@ -import type { describe, it } from '@std/testing/bdd'; +import type { describe, it, beforeAll, beforeEach, afterAll, afterEach } from '@std/testing/bdd'; import type { expect } from 'chai'; import type { t } from './common.ts'; @@ -9,7 +9,11 @@ export type TestingLib = { readonly FALSY: t.Falsy[]; readonly Bdd: BddLib; slug: t.RandomLib['slug']; - wait(delay: t.Msecs): Promise<void>; + + /** Wait for n-milliseconds, or a "tick" (micrso-task queue) if no delay specified. */ + wait(delay?: t.Msecs): Promise<void>; + + /** Generate a random (unused) port number. */ randomPort(): number; /** Attempt to run the test function <n>-times before throwing. */ @@ -36,12 +40,29 @@ export type Expect = typeof expect; /** Expect an error asyncronously */ export type ExpectError = (fn: () => Promise<any> | any, message?: string) => Promise<any>; +/** Run some shared setup before all of the tests in the group. */ +export type BeforeAll = typeof beforeAll; +/** Run some shared setup before each test in the suite. */ +export type BeforeEach = typeof beforeEach; + +/** Run some shared teardown after all of the tests in the suite. */ +export type AfterAll = typeof afterAll; +/** Run some shared teardown after each test in the suite. */ +export type AfterEach = typeof afterEach; + /** * BDD semantics ("Behavior Driven Development") helpers. */ export type BddLib = { - readonly describe: typeof describe; - readonly it: typeof it; - readonly expect: typeof expect; + readonly describe: Describe; + readonly it: It; + + readonly beforeAll: BeforeAll; + readonly afterAll: AfterAll; + + readonly beforeEach: BeforeEach; + readonly afterEach: AfterEach; + + readonly expect: Expect; readonly expectError: t.ExpectError; }; diff --git a/code/sys/std/src/m.Value.Obj/-.test.ts b/code/sys/std/src/m.Value.Obj/-.test.ts index 08448c83bb..799bd3c7f1 100644 --- a/code/sys/std/src/m.Value.Obj/-.test.ts +++ b/code/sys/std/src/m.Value.Obj/-.test.ts @@ -7,7 +7,7 @@ describe('Value.Obj', () => { expect(Value.Obj).to.equal(Obj); }); - describe('Value.Obj.walk', () => { + describe('Obj.walk', () => { type T = { key: string | number; value: any; path: (string | number)[] }; it('processes object', () => { @@ -141,7 +141,7 @@ describe('Value.Obj', () => { }); }); - describe('Value.Obj.build', () => { + describe('Obj.build', () => { it('return default root object (no keyPath)', () => { expect(Value.Obj.build('', {})).to.eql({}); expect(Value.Obj.build(' ', {})).to.eql({}); @@ -213,7 +213,7 @@ describe('Value.Obj', () => { }); }); - describe('Value.Obj.pluck', () => { + describe('Obj.pluck', () => { it('returns [undefined] when no match', () => { expect(Value.Obj.pluck('foo', {})).to.eql(undefined); expect(Value.Obj.pluck('foo.bar', {})).to.eql(undefined); @@ -244,7 +244,7 @@ describe('Value.Obj', () => { }); }); - describe('Value.Obj.remove', () => { + describe('Obj.remove', () => { const test = (keyPath: string, root: any, expected: any) => { const result = Value.Obj.remove(keyPath, root); const msg = `keyPath: "${keyPath}"`; @@ -281,7 +281,7 @@ describe('Value.Obj', () => { }); }); - describe('Value.Obj.prune', () => { + describe('Obj.prune', () => { const test = (keyPath: string, root: any, expected: any) => { const result = Value.Obj.prune(keyPath, root); const msg = `keyPath: "${keyPath}"`; @@ -329,7 +329,7 @@ describe('Value.Obj', () => { }); }); - describe('Value.Obj.toArray', () => { + describe('Obj.toArray', () => { type IFoo = { count: number }; type IFoos = { one: IFoo; @@ -364,7 +364,7 @@ describe('Value.Obj', () => { }); }); - describe('Value.Obj.trimStringsDeep', () => { + describe('Obj.trimStringsDeep', () => { it('shallow', () => { const name = 'foo'.repeat(100); const obj = { @@ -437,7 +437,7 @@ describe('Value.Obj', () => { }); }); - describe('Value.Obj.pick', () => { + describe('Obj.pick', () => { type T = { a: number; b: number; c: number }; const Sample = { create(): T { @@ -475,7 +475,7 @@ describe('Value.Obj', () => { }); }); - describe('Value.Obj.sortKeys', () => { + describe('Obj.sortKeys', () => { it('empty', () => { const obj = {}; const res = Value.Obj.sortKeys(obj); @@ -489,4 +489,167 @@ describe('Value.Obj', () => { expect(Object.keys(res).sort()).to.eql(Object.keys(obj).sort()); }); }); + + describe('Obj.clone', () => { + it('return different instance', () => { + const obj = { foo: 123, bar: { msg: 'hello' } }; + const res = Obj.clone(obj); + expect(res).to.eql(obj); + expect(res).to.not.equal(obj); // NB: different instance. + expect(res.bar).to.not.equal(obj.bar); + }); + + it('circular-reference safe', () => { + type Cycle = { self?: Cycle; msg: string; list: (number | Cycle)[] }; + + const obj: Cycle = { msg: 'š', list: [1] }; + obj.self = obj; + obj.list.push(obj); + obj.list.push(2); + obj.list.push(obj.list[1]); + + const res = Obj.clone(obj); + expect(res).to.eql(obj); + expect(res).to.not.equal(obj); + expect(res.list).to.not.equal(obj.list); + expect(res.list[1]).to.not.equal(obj.list[1]); + }); + + it('should return the same value for primitives', () => { + expect(Obj.clone(null)).to.equal(null); + expect(Obj.clone(undefined)).to.equal(undefined); + expect(Obj.clone(42)).to.equal(42); + expect(Obj.clone('hello')).to.equal('hello'); + expect(Obj.clone(true)).to.equal(true); + }); + + it('should clone arrays properly', () => { + const arr = [1, 2, { a: 3 }]; + const clonedArr = Obj.clone(arr); + expect(clonedArr).to.eql(arr); + expect(clonedArr).to.not.equal(arr); + expect(clonedArr[2]).to.not.equal(arr[2]); + }); + + it('should clone plain objects deeply', () => { + const obj = { a: 1, b: { c: 2 } }; + const clonedObj = Obj.clone(obj); + expect(clonedObj).to.eql(obj); + expect(clonedObj).to.not.equal(obj); + expect(clonedObj.b).to.not.equal(obj.b); + }); + + it('should clone objects with symbol keys', () => { + const sym = Symbol('key'); + const obj = { foo: 'bar', [sym]: 'baz' }; + const clonedObj = Obj.clone(obj); + expect(clonedObj).to.eql(obj); + expect(clonedObj).to.not.equal(obj); + }); + + it('should clone objects with non-enumerable properties', () => { + const obj: any = { visible: 'yes' }; + Object.defineProperty(obj, 'hidden', { + value: 'secret', + enumerable: false, + configurable: true, + writable: true, + }); + const clonedObj = Obj.clone(obj); + expect(clonedObj.visible).to.equal('yes'); + const desc = Object.getOwnPropertyDescriptor(clonedObj, 'hidden'); + expect(desc).to.exist; + expect(desc!.value).to.equal('secret'); + }); + + it('should preserve custom prototypes', () => { + class Custom { + prop: number; + constructor(prop: number) { + this.prop = prop; + } + } + const obj = new Custom(10); + (obj as any).extra = 'test'; + const clonedObj = Obj.clone(obj); + expect(clonedObj).to.eql(obj); + expect(clonedObj).to.not.equal(obj); + expect(Object.getPrototypeOf(clonedObj)).to.equal(Custom.prototype); + }); + + it('should not clone functions, but preserve the same function reference', () => { + const fn = function () { + return 'test'; + }; + const obj = { fn }; + const clonedObj = Obj.clone(obj); + expect(clonedObj.fn).to.equal(fn); + }); + + it('should handle circular references in objects', () => { + type Cycle = { self?: Cycle; msg: string; list: (number | Cycle)[] }; + const obj: Cycle = { msg: 'š', list: [1] }; + obj.self = obj; + obj.list.push(obj); + obj.list.push(2); + obj.list.push(obj.list[1]); // Add an additional cycle: list[1] is the same as obj. + + const cloned = Obj.clone(obj); + expect(cloned).to.eql(obj); + expect(cloned).to.not.equal(obj); + expect(cloned.list).to.not.equal(obj.list); + + // NB: The cloned object's self reference should point to the cloned object. + expect(cloned.self).to.equal(cloned); + + // NB: Verify that cyclic references within the array are maintained. + expect(cloned.list[1]).to.equal(cloned); + }); + + it('should handle circular references in arrays', () => { + const arr: any[] = [1, 2]; + arr.push(arr); + const clonedArr = Obj.clone(arr); + expect(clonedArr).to.eql(arr); + expect(clonedArr).to.not.equal(arr); + + // The cloned array's third element should reference the cloned array itself. + expect(clonedArr[2]).to.equal(clonedArr); + }); + + it('should clone nested objects and arrays', () => { + const obj = { + a: { b: [1, { c: 'hello' }] }, + d: 'world', + }; + const clonedObj = Obj.clone(obj); + expect(clonedObj).to.eql(obj); + expect(clonedObj.a).to.not.equal(obj.a); + expect(clonedObj.a.b).to.not.equal(obj.a.b); + expect(clonedObj.a.b[1]).to.not.equal(obj.a.b[1]); + }); + + it('should clone Date objects (note: date value may not be preserved)', () => { + const date = new Date(); + (date as any).extra = 'data'; + const clonedDate = Obj.clone(date); + expect(clonedDate).to.be.an.instanceof(Date); + + // NB: Extra properties are cloned. + expect((clonedDate as any).extra).to.equal('data'); + expect(clonedDate.getTime()).to.equal(date.getTime()); + }); + + it('should clone RegExp objects (note: pattern and flags may not be preserved)', () => { + const regex = /abc/gi; + (regex as any).extra = 'data'; + const clonedRegex = Obj.clone(regex); + expect(clonedRegex).to.be.an.instanceof(RegExp); + + // NB: The source and flags should match if cloned correctly. + expect(clonedRegex.source).to.equal(regex.source); + expect(clonedRegex.flags).to.equal(regex.flags); + expect((clonedRegex as any).extra).to.equal('data'); + }); + }); }); diff --git a/code/sys/std/src/m.Value.Obj/m.Obj.clone.ts b/code/sys/std/src/m.Value.Obj/m.Obj.clone.ts new file mode 100644 index 0000000000..4b5b4db594 --- /dev/null +++ b/code/sys/std/src/m.Value.Obj/m.Obj.clone.ts @@ -0,0 +1,69 @@ +import { type t } from '../common.ts'; + +/** + * Deeply clone the given object (circular-reference safe) + * with support for Date and RegExp. + */ +export const clone: t.ObjLib['clone'] = (obj) => deepClone(obj); + +/** + * Helpers + */ + +function deepClone<T>(obj: T, visited = new WeakMap()): T { + // Primitives and functions not cloned. + if (obj === null || typeof obj !== 'object') return obj; + + // Return cached clone if available (handle cycles safely). + if (visited.has(obj)) return visited.get(obj); + + // Handle Date objects: + if (obj instanceof Date) { + const clonedDate = new Date(obj.getTime()); + visited.set(obj, clonedDate); + // Clone any extra properties on the Date: + for (const key of Reflect.ownKeys(obj)) { + // NB: Optionally skip built-in keys, though most Date objects won't have extra ones. + if (typeof key === 'string' && ['getTime', 'toString', 'valueOf', 'toJSON'].includes(key)) { + continue; + } + (clonedDate as any)[key] = deepClone((obj as any)[key], visited); // ā š³ RECURSION + } + return clonedDate as unknown as T; + } + + // Handle RegExp objects: + if (obj instanceof RegExp) { + const clonedRegExp = new RegExp(obj.source, obj.flags); + visited.set(obj, clonedRegExp); + + // Clone any extra properties on the RegExp: + for (const key of Reflect.ownKeys(obj)) { + (clonedRegExp as any)[key] = deepClone((obj as any)[key], visited); // ā š³ RECURSION + } + return clonedRegExp as unknown as T; + } + + // Handle arrays: + if (Array.isArray(obj)) { + const arrClone: any[] = []; + visited.set(obj, arrClone); + for (const item of obj) { + arrClone.push(deepClone(item, visited)); // ā š³ RECURSION + } + return arrClone as unknown as T; + } + + // Handle plain objects: + const cloneObj = Object.create(Object.getPrototypeOf(obj)); + visited.set(obj, cloneObj); + + for (const key of Reflect.ownKeys(obj)) { + // Ensure proper copying of both string and symbol keys. + // Also, copying non-enumerable properties might be necessary in some cases: + (cloneObj as any)[key] = deepClone((obj as any)[key], visited); // ā š³ RECURSION + } + + // Finish up. + return cloneObj; +} diff --git a/code/sys/std/src/m.Value.Obj/mod.ts b/code/sys/std/src/m.Value.Obj/mod.ts index 80720b582c..90507a7ae8 100644 --- a/code/sys/std/src/m.Value.Obj/mod.ts +++ b/code/sys/std/src/m.Value.Obj/mod.ts @@ -1,6 +1,10 @@ import type { t } from '../common.ts'; + +import { clone } from './m.Obj.clone.ts'; import { build, pluck, prune, remove } from './m.Obj.path.ts'; -import { pick, toArray, trimStringsDeep, walk, sortKeys } from './m.Obj.ts'; +import { pick, sortKeys, toArray, trimStringsDeep, walk } from './m.Obj.ts'; + +export { sortKeys }; export const Obj: t.ObjLib = { walk, @@ -12,4 +16,5 @@ export const Obj: t.ObjLib = { remove, prune, sortKeys, + clone, }; diff --git a/code/sys/std/src/m.Value.Obj/t.ts b/code/sys/std/src/m.Value.Obj/t.ts index 40773f88fb..f5265ce1ba 100644 --- a/code/sys/std/src/m.Value.Obj/t.ts +++ b/code/sys/std/src/m.Value.Obj/t.ts @@ -65,6 +65,12 @@ export type ObjLib = { * Sort the keys of an object. */ sortKeys<T extends O>(obj: T): T; + + /** + * Deeply clone the given object (circular-reference safe) + * with support for Date and RegExp. + */ + clone<T>(obj: T): T; }; /** A callback passed to the object walker function. */ diff --git a/code/sys/std/src/m.Value.Str/-.test.ts b/code/sys/std/src/m.Value.Str/-.test.ts index 57a9cc9ecb..6619c9dd70 100644 --- a/code/sys/std/src/m.Value.Str/-.test.ts +++ b/code/sys/std/src/m.Value.Str/-.test.ts @@ -246,4 +246,111 @@ describe('Str (Text)', () => { test('--camel-Case', '--camel-case'); }); }); + + describe('String.truncate', () => { + it('returns the original text when length is less than max', () => { + expect(Str.truncate('abc', 5)).to.eql('abc'); + }); + + it('returns the original text when length equals max', () => { + expect(Str.truncate('hello', 5)).to.eql('hello'); + }); + + it('truncates the text when length is greater than max', () => { + // For max = 5, it takes the first 4 characters and appends an ellipsis. + expect(Str.truncate('abcdef', 5)).to.eql('abcdā¦'); + }); + + it('handles max = 1 correctly', () => { + // For a string longer than 1, it returns just the ellipsis. + expect(Str.truncate('abc', 1)).to.eql('ā¦'); + // When the text length equals max, no truncation happens. + expect(Str.truncate('a', 1)).to.eql('a'); + }); + + it('returns an empty string when given an empty string', () => { + expect(Str.truncate('', 3)).to.eql(''); + }); + + it('handles edge case when max is 0', () => { + // With max = 0, "abc" becomes "abc".slice(0, -1) which is "ab", then an ellipsis is appended. + expect(Str.truncate('abc', 0)).to.eql('abā¦'); + }); + + it('handles undefined', () => { + expect(Str.truncate(undefined, 5)).to.eql(''); + expect(Str.truncate(undefined, 0)).to.eql(''); + }); + }); + + describe('Str.Lorem', () => { + const LOREM = `Lorem ipsum dolor sit amet, consectetur adipiscing elit. Quisque nec quam lorem. Praesent fermentum, augue ut porta varius, eros nisl euismod ante, ac suscipit elit libero nec dolor. Morbi magna enim, molestie non arcu id, varius sollicitudin neque. In sed quam mauris. Aenean mi nisl, elementum non arcu quis, ultrices tincidunt augue. Vivamus fermentum iaculis tellus finibus porttitor. Nulla eu purus id dolor auctor suscipit. Integer lacinia sapien at ante tempus volutpat.`; + const Lorem = Str.Lorem; + + it('Str.lorem (string)', () => { + expect(Str.lorem).to.eql(Str.Lorem.text); + }); + + it('Lorem.text | toString', () => { + expect(Str.Lorem.text).to.eql(LOREM); + expect(Str.Lorem.toString()).to.eql(LOREM); + expect(String(Str.Lorem)).to.eql(LOREM); + }); + + describe('Str.Lorem.words', () => { + it('should return an empty string when count is negative', () => { + const result = Str.Lorem.words(-5); + expect(result).to.equal(''); + }); + + it('should return an empty string when count is zero', () => { + const result = Str.Lorem.words(0); + expect(result).to.equal(''); + }); + + it('should return the first N words with a trailing period when count > 0', () => { + // For example, if count = 5, we expect the result to have 5 words and end with a period. + const count = 5; + const result = Str.Lorem.words(count); + + // Remove the final character (period) to check the word count. + const words = result.split(' '); + expect(words.length).to.equal(count); + expect(result.endsWith('.')).to.be.true; + }); + + it('should repeat the text if count exceeds the total number of words', () => { + const count = 1000; + const result = Str.Lorem.words(count); + + // Remove the trailing period (if present) and split the result into words. + const resultWords = result.endsWith('.') + ? result.slice(0, -1).split(' ') + : result.split(' '); + expect(resultWords.length).to.equal(count); + + // Get the original words from Lorem.text (removing the trailing period). + const originalWords = Str.Lorem.text.endsWith('.') + ? Str.Lorem.text.slice(0, -1).split(' ') + : Str.Lorem.text.split(' '); + + const trimPeriod = (input: string) => input.replace(/\.$/, ''); + + // Verify that the output words repeat the original words in sequence. + for (let i = 0; i < resultWords.length; i++) { + const a = resultWords[i]; + const b = originalWords[i % originalWords.length]; + expect(trimPeriod(a)).to.equal(trimPeriod(b)); + } + + // Ensure the result ends with a period. + expect(result.endsWith('.')).to.be.true; + }); + + it('words - no param', () => { + const result = Str.Lorem.words(); + expect(result.split(' ').length).to.eql(Lorem.text.split(' ').length); + }); + }); + }); }); diff --git a/code/sys/std/src/m.Value.Str/m.Lorem.ts b/code/sys/std/src/m.Value.Str/m.Lorem.ts new file mode 100644 index 0000000000..26afc3fc59 --- /dev/null +++ b/code/sys/std/src/m.Value.Str/m.Lorem.ts @@ -0,0 +1,23 @@ +import type { t } from './common.ts'; + +const LOREM = `Lorem ipsum dolor sit amet, consectetur adipiscing elit. Quisque nec quam lorem. Praesent fermentum, augue ut porta varius, eros nisl euismod ante, ac suscipit elit libero nec dolor. Morbi magna enim, molestie non arcu id, varius sollicitudin neque. In sed quam mauris. Aenean mi nisl, elementum non arcu quis, ultrices tincidunt augue. Vivamus fermentum iaculis tellus finibus porttitor. Nulla eu purus id dolor auctor suscipit. Integer lacinia sapien at ante tempus volutpat.`; +const total = LOREM.split(' ').length; + +export const Lorem: t.StrLoremLib = { + get text() { + return LOREM; + }, + + toString() { + return LOREM; + }, + + words(count = total) { + if (count < 0) return ''; + const list = LOREM.split(/\s+/).filter((word) => word.length > 0); + const repeats = Math.ceil(count / list.length); + const repeatedWords = Array(repeats).fill(list).flat(); + const result = repeatedWords.slice(0, count).join(' ').trim(); + return result && !result.endsWith('.') ? result + '.' : result; + }, +}; diff --git a/code/sys/std/src/m.Value.Str/m.Str.ts b/code/sys/std/src/m.Value.Str/m.Str.ts index c240b90c6b..6ce5b5ab73 100644 --- a/code/sys/std/src/m.Value.Str/m.Str.ts +++ b/code/sys/std/src/m.Value.Str/m.Str.ts @@ -7,10 +7,14 @@ import { diff } from './u.diff.ts'; import { plural } from './u.plural.ts'; import { shorten } from './u.shorten.ts'; import { replace, splice } from './u.splice.ts'; +import { truncate } from './u.truncate.ts'; +import { Lorem } from './m.Lorem.ts'; export { bytes, capitalize, diff, plural, replace, shorten, splice }; export const Str: t.StrLib = { + Lorem, + lorem: Lorem.text, diff, splice, replace, @@ -19,4 +23,5 @@ export const Str: t.StrLib = { camelToKebab, plural, bytes, + truncate, } as const; diff --git a/code/sys/std/src/m.Value.Str/mod.ts b/code/sys/std/src/m.Value.Str/mod.ts index 314897d00a..e17c5f0d0f 100644 --- a/code/sys/std/src/m.Value.Str/mod.ts +++ b/code/sys/std/src/m.Value.Str/mod.ts @@ -13,4 +13,5 @@ * expect(Value.Str).to.equal(Str); * ``` */ -export { Str, bytes, capitalize, diff, plural, replace, shorten, splice } from './m.Str.ts'; +export { Lorem } from './m.Lorem.ts'; +export { bytes, capitalize, diff, plural, replace, shorten, splice, Str } from './m.Str.ts'; diff --git a/code/sys/std/src/m.Value.Str/t.ts b/code/sys/std/src/m.Value.Str/t.ts index 904f0800ec..379e348bd6 100644 --- a/code/sys/std/src/m.Value.Str/t.ts +++ b/code/sys/std/src/m.Value.Str/t.ts @@ -13,6 +13,11 @@ export type FormatBytesOptions = FormatOptions & {}; * Tools for working on strings of text. */ export type StrLib = { + /** The "lorem ipsum" helper library. */ + readonly Lorem: StrLoremLib; + /** The "lorem ipsum" string. */ + readonly lorem: string; + /** * Calculate a difference between two strings. */ @@ -52,4 +57,18 @@ export type StrLib = { * Convert bytes to a human-readable string, eg: 1337 ā "1.34 kB". */ bytes: t.FormatBytes; + + /** + * Truncates a string with ellipsis if over a maximum length. + */ + truncate(text: string | undefined, max: number): string; +}; + +/** + * Tools for working with sample "lorem ipsum..." text. + */ +export type StrLoremLib = { + readonly text: string; + toString(): string; + words(count?: number): string; }; diff --git a/code/sys/std/src/m.Value.Str/u.truncate.ts b/code/sys/std/src/m.Value.Str/u.truncate.ts new file mode 100644 index 0000000000..274133fb91 --- /dev/null +++ b/code/sys/std/src/m.Value.Str/u.truncate.ts @@ -0,0 +1,5 @@ +import { type t } from './common.ts'; + +export const truncate: t.StrLib['truncate'] = (text = '', max) => { + return text.length > max ? `${text.slice(0, max - 1)}ā¦` : text; +}; diff --git a/code/sys/std/src/m.Value/m.Value.ts b/code/sys/std/src/m.Value/m.Value.ts index 3b97767ac9..2b5cf8fd71 100644 --- a/code/sys/std/src/m.Value/m.Value.ts +++ b/code/sys/std/src/m.Value/m.Value.ts @@ -3,10 +3,10 @@ import { type t, isEmptyRecord, isObject, isRecord } from '../common.ts'; import { Array } from '../m.Value.Array/mod.ts'; import { Num } from '../m.Value.Num/mod.ts'; import { Obj } from '../m.Value.Obj/mod.ts'; -import { Str } from '../m.Value.Str/mod.ts'; +import { Lorem, Str } from '../m.Value.Str/mod.ts'; import { toggle } from './u.toggle.ts'; -export { Array, isEmptyRecord, Num, Obj, Str }; +export { Array, isEmptyRecord, Lorem, Num, Obj, Str }; /** * Tools for evaluating and manipulating types of values. diff --git a/code/sys/std/src/m.Value/mod.ts b/code/sys/std/src/m.Value/mod.ts index a988cb3ef0..981bf8926f 100644 --- a/code/sys/std/src/m.Value/mod.ts +++ b/code/sys/std/src/m.Value/mod.ts @@ -5,5 +5,5 @@ export { isEmptyRecord, isObject, isRecord } from '../common.ts'; export { V } from '../m.Validation/mod.ts'; export { asArray } from '../m.Value.Array/mod.ts'; -export { sortKeys } from '../m.Value.Obj/m.Obj.ts'; -export { Array, Num, Str, Value } from './m.Value.ts'; +export { Obj, sortKeys } from '../m.Value.Obj/mod.ts'; +export { Array, Num, Str, Value, Lorem } from './m.Value.ts'; diff --git a/code/sys/std/src/mod.ts b/code/sys/std/src/mod.ts index 773e19874f..3acfb78738 100644 --- a/code/sys/std/src/mod.ts +++ b/code/sys/std/src/mod.ts @@ -8,7 +8,7 @@ export type * as t from './types.ts'; export { Args } from './m.Args/mod.ts'; export { Async } from './m.Async/mod.ts'; -export { D, Date, Duration, Time } from './m.DateTime/mod.ts'; +export { D, Date, Duration, Time, Timestamp } from './m.DateTime/mod.ts'; export { Delete } from './m.Delete/mod.ts'; export { Dispose } from './m.Dispose/mod.ts'; export { Err } from './m.Err/mod.ts'; @@ -22,7 +22,8 @@ export { maybeWait, Promise } from './m.Promise/mod.ts'; export { slug } from './m.Random/mod.ts'; export { Regex } from './m.Regex/mod.ts'; export { Rx, rx } from './m.Rx/mod.ts'; +export { Signal } from './m.Signal/mod.ts'; export { Url } from './m.Url/mod.ts'; -export { Array, asArray, isObject, isRecord, Num, Str, V, Value } from './m.Value/mod.ts'; +export { Array, asArray, isObject, isRecord, Num, Obj, Str, V, Value } from './m.Value/mod.ts'; export { R } from './common.ts'; diff --git a/code/sys/std/src/pkg.ts b/code/sys/std/src/pkg.ts index 9e70de2172..076d4a2016 100644 --- a/code/sys/std/src/pkg.ts +++ b/code/sys/std/src/pkg.ts @@ -1,8 +1,16 @@ -import type { Pkg as TPkg } from '@sys/types'; -import { default as deno } from '../deno.json' with { type: 'json' }; -import { Pkg } from './m.Pkg/mod.ts'; +import type { Pkg } from '@sys/types'; /** * Package meta-data. -*/ -export const pkg: TPkg = Pkg.fromJson(deno) + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). + */ +export const pkg: Pkg = { name: '@sys/std', version: '0.0.141' }; diff --git a/code/sys/std/src/types.ts b/code/sys/std/src/types.ts index 16c798b457..310a0a9ace 100644 --- a/code/sys/std/src/types.ts +++ b/code/sys/std/src/types.ts @@ -24,6 +24,7 @@ export type * from './m.Regex/t.ts'; export type * from './m.Rx/t.ts'; export type * from './m.Semver.Server/t.ts'; export type * from './m.Semver/t.ts'; +export type * from './m.Signal/t.ts'; export type * from './m.Testing.Server/t.ts'; export type * from './m.Testing/t.ts'; export type * from './m.Url/t.ts'; diff --git a/code/sys/sys/deno.json b/code/sys/sys/deno.json index b215cb9fee..34ecc7df62 100644 --- a/code/sys/sys/deno.json +++ b/code/sys/sys/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/sys", - "version": "0.0.55", + "version": "0.0.65", "license": "MIT", "tasks": { "lint": "deno lint", diff --git a/code/sys/sys/src/pkg.ts b/code/sys/sys/src/pkg.ts index 79cb3f9c80..2aa6d36ded 100644 --- a/code/sys/sys/src/pkg.ts +++ b/code/sys/sys/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/sys', version: '0.0.65' }; diff --git a/code/sys/testing/deno.json b/code/sys/testing/deno.json index d4935bf1e7..e85545d466 100644 --- a/code/sys/testing/deno.json +++ b/code/sys/testing/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/testing", - "version": "0.0.75", + "version": "0.0.85", "license": "MIT", "tasks": { "test": "deno test -RW", diff --git a/code/sys/testing/src/ns.server/common.ts b/code/sys/testing/src/ns.server/common.ts index cc7ffd504c..8c0e74c00d 100644 --- a/code/sys/testing/src/ns.server/common.ts +++ b/code/sys/testing/src/ns.server/common.ts @@ -2,4 +2,16 @@ export * from '../common.ts'; export { c } from '@sys/color/ansi'; export { Fs, Path } from '@sys/fs'; -export { describe, expect, expectError, it, slug, Time } from '@sys/std/testing'; +export { + afterAll, + afterEach, + Bdd, + beforeAll, + beforeEach, + describe, + expect, + expectError, + it, + slug, + Time, +} from '@sys/std/testing'; diff --git a/code/sys/testing/src/ns.server/m.DomMock/u.polyfill.ts b/code/sys/testing/src/ns.server/m.DomMock/u.polyfill.ts index 6883cb5857..00d1ed38e1 100644 --- a/code/sys/testing/src/ns.server/m.DomMock/u.polyfill.ts +++ b/code/sys/testing/src/ns.server/m.DomMock/u.polyfill.ts @@ -1,7 +1,5 @@ import { Window } from 'happy-dom'; - import type { t } from '../common.ts'; -import { Keyboard } from './m.Keyboard.ts'; const g = globalThis as any; let _window: Window | undefined; diff --git a/code/sys/testing/src/pkg.ts b/code/sys/testing/src/pkg.ts index 79cb3f9c80..20e73e2f50 100644 --- a/code/sys/testing/src/pkg.ts +++ b/code/sys/testing/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/testing', version: '0.0.85' }; diff --git a/code/sys/text/deno.json b/code/sys/text/deno.json index ad594c8633..e753eeb062 100644 --- a/code/sys/text/deno.json +++ b/code/sys/text/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/text", - "version": "0.0.73", + "version": "0.0.83", "license": "MIT", "tasks": { "test": "deno test -RW", diff --git a/code/sys/text/src/pkg.ts b/code/sys/text/src/pkg.ts index 79cb3f9c80..ebf828bb64 100644 --- a/code/sys/text/src/pkg.ts +++ b/code/sys/text/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/text', version: '0.0.83' }; diff --git a/code/sys/tmpl/deno.json b/code/sys/tmpl/deno.json index 4f4f84f26c..bacb9e914b 100644 --- a/code/sys/tmpl/deno.json +++ b/code/sys/tmpl/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/tmpl", - "version": "0.0.79", + "version": "0.0.90", "license": "MIT", "tasks": { "lint": "deno lint", diff --git a/code/sys/tmpl/src/-test/mod.ts b/code/sys/tmpl/src/-test/mod.ts index bf35f0f3b4..d498f63fbe 100644 --- a/code/sys/tmpl/src/-test/mod.ts +++ b/code/sys/tmpl/src/-test/mod.ts @@ -1,3 +1,3 @@ -export { describe, expect, it, Testing } from '@sys/testing/server'; +export { describe, expect, expectError, it, Testing } from '@sys/testing/server'; export * from '../common.ts'; export * from './u.SAMPLE.ts'; diff --git a/code/sys/tmpl/src/-test/-sample/.gitignore b/code/sys/tmpl/src/-test/sample-1/.gitignore similarity index 100% rename from code/sys/tmpl/src/-test/-sample/.gitignore rename to code/sys/tmpl/src/-test/sample-1/.gitignore diff --git a/code/sys/tmpl/src/-test/-sample/deno.json b/code/sys/tmpl/src/-test/sample-1/deno.json similarity index 100% rename from code/sys/tmpl/src/-test/-sample/deno.json rename to code/sys/tmpl/src/-test/sample-1/deno.json diff --git a/code/sys/tmpl/src/-test/-sample/docs/index.md b/code/sys/tmpl/src/-test/sample-1/docs/index.md similarity index 100% rename from code/sys/tmpl/src/-test/-sample/docs/index.md rename to code/sys/tmpl/src/-test/sample-1/docs/index.md diff --git a/code/sys/tmpl/src/-test/-sample/mod.ts b/code/sys/tmpl/src/-test/sample-1/mod.ts similarity index 100% rename from code/sys/tmpl/src/-test/-sample/mod.ts rename to code/sys/tmpl/src/-test/sample-1/mod.ts diff --git a/code/sys/tmpl/src/-test/sample-2/README.md b/code/sys/tmpl/src/-test/sample-2/README.md new file mode 100644 index 0000000000..adb251ade3 --- /dev/null +++ b/code/sys/tmpl/src/-test/sample-2/README.md @@ -0,0 +1 @@ +# Sample Readme diff --git a/code/sys/tmpl/src/-test/sample-2/images/volcano.jpg b/code/sys/tmpl/src/-test/sample-2/images/volcano.jpg new file mode 100644 index 0000000000..ab0cef116f Binary files /dev/null and b/code/sys/tmpl/src/-test/sample-2/images/volcano.jpg differ diff --git a/code/sys/tmpl/src/-test/u.SAMPLE.ts b/code/sys/tmpl/src/-test/u.SAMPLE.ts index 636c3b2f7c..b829e76315 100644 --- a/code/sys/tmpl/src/-test/u.SAMPLE.ts +++ b/code/sys/tmpl/src/-test/u.SAMPLE.ts @@ -1,21 +1,39 @@ import { type t, Fs, slug } from '../common.ts'; +type Options = { slug?: boolean }; + +/** + * Test folder/file samples. + */ export const SAMPLE = { - init(options: { source?: t.StringDir; slug?: boolean } = {}) { - const source = Fs.resolve(options.source ?? './src/-test/-sample'); - const target = Fs.resolve(`./.tmp/test/m.Tmpl`, options.slug ?? true ? slug() : ''); - const exists = (dir: t.StringDir, path: string[]) => Fs.exists(Fs.join(dir, ...path)); + fs(namespace: string) { + const target = Fs.join('./.tmp', namespace); return { - source, - target, - ls: { - source: () => Fs.ls(source), - target: () => Fs.ls(target), - }, - exists: { - source: (...path: string[]) => exists(source, path), - target: (...path: string[]) => exists(target, path), - }, + sample1: (options?: Options) => init('./src/-test/sample-1', target, options), + sample2: (options?: Options) => init('./src/-test/sample-2', target, options), } as const; }, } as const; + +/** + * Initialize a test folder. + */ +function init(source: t.StringDir, target: t.StringDir, options: Options = {}) { + const randomDir = options.slug ?? true ? slug() : ''; + source = Fs.resolve(source); + target = Fs.resolve(target, randomDir); + + const exists = (dir: t.StringDir, path: t.StringPath[]) => Fs.exists(Fs.join(dir, ...path)); + return { + source, + target, + ls: { + source: () => Fs.ls(source), + target: () => Fs.ls(target), + }, + exists: { + source: (...path: t.StringPath[]) => exists(source, path), + target: (...path: t.StringPath[]) => exists(target, path), + }, + } as const; +} diff --git a/code/sys/tmpl/src/common/libs.ts b/code/sys/tmpl/src/common/libs.ts index 5adf8e2704..c831ae83ae 100644 --- a/code/sys/tmpl/src/common/libs.ts +++ b/code/sys/tmpl/src/common/libs.ts @@ -1,3 +1,3 @@ -export { Err, Is, rx, slug, Time } from '@sys/std'; export { c, Cli, stripAnsi } from '@sys/cli'; export { Fs, Path } from '@sys/fs'; +export { Err, Is, isRecord, R, rx, slug, Time } from '@sys/std'; diff --git a/code/sys/tmpl/src/m.Log/-.test.ts b/code/sys/tmpl/src/m.Log/-.test.ts index 2961dab25e..cc7cbb4ad0 100644 --- a/code/sys/tmpl/src/m.Log/-.test.ts +++ b/code/sys/tmpl/src/m.Log/-.test.ts @@ -3,22 +3,24 @@ import { Tmpl } from '../m.Tmpl/mod.ts'; import { Log } from './mod.ts'; describe('Tmpl.Log', () => { + const Test = SAMPLE.fs('m.Log'); + it('API', () => { expect(Tmpl.Log).to.equal(Log); }); describe('Log.table', () => { it('log table', async () => { - const test = SAMPLE.init(); + const test = Test.sample1(); let change = false; const tmpl = Tmpl.create(test.source, (e) => { if (change) e.modify('// foo'); }); - const res1 = await tmpl.copy(test.target); - const res2 = await tmpl.copy(test.target); + const res1 = await tmpl.write(test.target); + const res2 = await tmpl.write(test.target); change = true; - const res3 = await tmpl.copy(test.target); + const res3 = await tmpl.write(test.target); const table1 = Log.table(res1.ops); const table2 = Log.table(res2.ops); @@ -38,17 +40,17 @@ describe('Tmpl.Log', () => { }); it('empty (no operations)', async () => { - const test = SAMPLE.init(); + const test = Test.sample1(); const tmpl = Tmpl.create(test.source).filter(() => false); - const res = await tmpl.copy(test.target); + const res = await tmpl.write(test.target); const table = Log.table(res.ops); expect(table).to.include('No items to display'); }); it('option: { trimBase:<path> }', async () => { - const test = SAMPLE.init(); + const test = Test.sample1(); const tmpl = Tmpl.create(test.source, (e) => {}); - const res = await tmpl.copy(test.target); + const res = await tmpl.write(test.target); const trimPathLeft = Path.trimCwd(test.target) + '/'; const table = Log.table(res.ops, { trimPathLeft }); @@ -56,9 +58,9 @@ describe('Tmpl.Log', () => { }); it('option: { note: Ęn }', async () => { - const test = SAMPLE.init(); + const test = Test.sample1(); const tmpl = Tmpl.create(test.source, (e) => {}); - const res = await tmpl.copy(test.target); + const res = await tmpl.write(test.target); const table = Log.table(res.ops, { note(op) { diff --git a/code/sys/tmpl/src/m.Tmpl/-.test.ts b/code/sys/tmpl/src/m.Tmpl/-.test.ts index 75c7486190..79fe01ae3c 100644 --- a/code/sys/tmpl/src/m.Tmpl/-.test.ts +++ b/code/sys/tmpl/src/m.Tmpl/-.test.ts @@ -1,11 +1,12 @@ -import { describe, expect, Fs, it, SAMPLE, Time, type t } from '../-test.ts'; +import { type t, describe, expect, expectError, Fs, it, R, SAMPLE, Time } from '../-test.ts'; import { Tmpl } from './mod.ts'; describe('Tmpl', () => { const readFile = async (path: string) => (await Fs.readText(path)).data; + const Test = SAMPLE.fs('m.Tmpl'); const logOps = ( - res: t.TmplCopyResponse, + res: t.TmplWriteResponse, title: string, options: { indent?: number; hideExcluded?: boolean } = {}, ) => { @@ -15,21 +16,21 @@ describe('Tmpl', () => { }; it('init: paths', async () => { - const test = SAMPLE.init(); + const test = Test.sample1(); const tmpl = Tmpl.create(test.source); expect(tmpl.source.absolute).to.eql(test.source); expect(await tmpl.source.ls()).to.eql(await test.ls.source()); tmpl; }); - describe('tmpl.copy:', () => { + describe('tmpl.write:', () => { it('copies all source files', async () => { - const test = SAMPLE.init(); + const test = Test.sample1(); const tmpl = Tmpl.create(test.source); expect(await test.ls.target()).to.eql([]); - const a = await tmpl.copy(test.target); - const b = await tmpl.copy(test.target); + const a = await tmpl.write(test.target); + const b = await tmpl.write(test.target); expect(a.source.absolute).to.eql(test.source); expect(await a.target.ls()).to.eql(await test.ls.target()); @@ -46,21 +47,64 @@ describe('Tmpl', () => { logOps(a, 'Copy:', { indent: 2 }); }); - it('tmpl.copy(): ā create ā update', async () => { - const test = SAMPLE.init(); + it('copies binary files (eg ".jpg")', async () => { + const test = Test.sample2(); + const tmpl = Tmpl.create(test.source); + + const res = await tmpl.write(test.target); + const sourcePath = Fs.join(res.source.absolute, 'images/volcano.jpg'); + const targetPath = Fs.join(res.target.absolute, 'images/volcano.jpg'); + + const source = await Fs.read(sourcePath); + const target = await Fs.read(targetPath); + + expect(String(source.data).startsWith('255,216,255,224,0,16')).to.be.true; // NB: source file contains data. + expect(target.exists).to.eql(true); + expect(target.data).to.eql(source.data); + }); + + it('writes when file-processor function is NOT specified', async () => { + const sample1 = Test.sample1(); + const sample2 = Test.sample2(); + const tmpl1 = Tmpl.create(sample1.source); + const tmpl2 = Tmpl.create(sample2.source); + + await tmpl1.write(sample1.target); + await tmpl2.write(sample2.target); + + const assertIncludes = (paths: string[], path: string) => { + const match = paths.some((p) => p.endsWith(path)); + expect(match).to.be.true; + }; + + const targets1 = await sample1.ls.target(); + const targets2 = await sample2.ls.target(); + + assertIncludes(targets1, '.gitignore'); + assertIncludes(targets1, 'deno.json'); + assertIncludes(targets1, 'mod.ts'); + assertIncludes(targets1, 'docs/index.md'); + + assertIncludes(targets2, 'README.md'); + assertIncludes(targets2, 'images/volcano.jpg'); + }); + + it('tmpl.write(): ā create ā update', async () => { + const test = Test.sample1(); let foo = 0; let count = 0; const tmpl = Tmpl.create(test.source, (e) => { if (e.target.file.name !== 'mod.ts') return e.exclude(); e.modify(`const foo = ${foo}`); expect(e.target.exists).to.eql(count === 0 ? false : true); + expect(e.ctx).to.eql(undefined); count++; }); - const resA = await tmpl.copy(test.target); - foo = 123; // NB: cuase change in file. - const resB = await tmpl.copy(test.target); // NB: "updated" via change flag. - const resC = await tmpl.copy(test.target); // NB: no changes. + const resA = await tmpl.write(test.target); + foo = 123; // NB: cause change in file. + const resB = await tmpl.write(test.target); // NB: "updated" via change flag. + const resC = await tmpl.write(test.target); // NB: no changes. const a = resA.ops.filter((e) => e.written); const b = resB.ops.filter((e) => e.written); @@ -85,14 +129,14 @@ describe('Tmpl', () => { describe('fn: processFile (callback)', () => { it('fn: exclude', async () => { - const { source, target } = SAMPLE.init(); + const { source, target } = Test.sample1(); const tmpl = Tmpl.create(source, async (e) => { await Time.wait(0); // NB: ensure the async variant of the function waits for completion. if (e.target.file.name.endsWith('.md')) e.exclude('user-space'); if (e.target.file.name === '.gitignore') e.exclude(); }); - const res = await tmpl.copy(target); + const res = await tmpl.write(target); for (const op of res.ops) { if (op.file.target.file.name.endsWith('.md')) { @@ -110,23 +154,23 @@ describe('Tmpl', () => { }); it('fn: file source/target', async () => { - const { source, target } = SAMPLE.init(); + const { source, target } = Test.sample1(); let count = 0; const tmpl = Tmpl.create(source, (e) => { count++; expect(e.tmpl.base).to.eql(source); expect(e.target.base).to.eql(target); }); - await tmpl.copy(target); + await tmpl.write(target); expect(count).to.greaterThan(0); // NB: ensure the callback ran. }); it('fn: rename file', async () => { - const test = SAMPLE.init(); + const test = Test.sample1(); const tmpl = Tmpl.create(test.source, (e) => { if (e.target.file.name === 'mod.ts') e.rename('main.ts'); }); - const res = await tmpl.copy(test.target); + const res = await tmpl.write(test.target); const match = res.ops.find((m) => m.file.target.file.name === 'main.ts'); expect(match?.file.tmpl.file.name).to.eql('mod.ts'); expect(match?.file.target.file.name).to.eql('main.ts'); @@ -135,79 +179,198 @@ describe('Tmpl', () => { }); it('fn: modify (file text)', async () => { - const { source, target } = SAMPLE.init(); + const { source, target } = Test.sample1(); const tmpl = Tmpl.create(source, (e) => { + if (!e.text) return; if (e.target.file.name === 'mod.ts') { const next = e.text.tmpl.replace(/\{FOO_BAR\}/g, 'š Hello'); e.modify(next); } }); - const a = await tmpl.copy(target); - const b = await tmpl.copy(target); + const a = await tmpl.write(target); + const b = await tmpl.write(target); const matchA = a.ops.find((m) => m.file.target.file.name === 'mod.ts'); const matchB = b.ops.find((m) => m.file.target.file.name === 'mod.ts'); - expect(matchA?.text.tmpl).to.include(`name: '{FOO_BAR}'`); - expect(matchA?.text.target.before).to.include(''); // NB: Nothing has been written yet. - expect(matchA?.text.target.after).to.include(`name: 'š Hello'`); - expect(matchB?.text.target.before).to.include(`name: 'š Hello'`); // NB: prior written modification (already exists). - expect(await readFile(matchA?.file.target.absolute ?? '')).to.include(`name: 'š Hello'`); + expect(!!(matchA?.text && matchB?.text)).to.be.true; + + if (matchA?.contentType === 'text' && matchB?.contentType === 'text') { + expect(matchA.text.tmpl).to.include(`name: '{FOO_BAR}'`); + expect(matchA.text.target.before).to.include(''); // NB: Nothing has been written yet. + expect(matchA.text.target.after).to.include(`name: 'š Hello'`); + expect(matchA.text.target.isDiff).to.eql(true); + expect(matchB.text.target.before).to.include(`name: 'š Hello'`); // NB: prior written modification (already exists). + expect(await readFile(matchA.file.target.absolute ?? '')).to.include(`name: 'š Hello'`); + } const writtenA = await readFile(a.target.join('mod.ts')); - const writtenB = await readFile(a.target.join('mod.ts')); + const writtenB = await readFile(b.target.join('mod.ts')); expect(writtenA).to.include(`name: 'š Hello'`); expect(writtenB).to.include(`name: 'š Hello'`); }); + + it('fn: modify (binary file)', async () => { + const { source, target } = Test.sample2(); + const jpg = (await Fs.read('./src/-test/sample-2/images/volcano.jpg')).data; + let replaceWith: Uint8Array | undefined; + + const tmpl = Tmpl.create(source, (e) => { + if (e.contentType !== 'binary') return; + if (replaceWith) e.modify(replaceWith); + }); + + const a = await tmpl.write(target); + replaceWith = new Uint8Array([1, 2, 3]); + const b = await tmpl.write(target); + + const matchA = a.ops.find((m) => m.file.target.file.name === 'volcano.jpg'); + const matchB = b.ops.find((m) => m.file.target.file.name === 'volcano.jpg'); + + if (matchA?.contentType === 'binary' && matchB?.contentType === 'binary') { + expect(matchA.binary.tmpl).to.eql(jpg); + expect(matchA.binary.target.before).to.eql(new Uint8Array(0)); // NB: Nothing has been written yet. + expect(matchA.binary.target.after).to.eql(jpg); + expect(matchA.binary.target.isDiff).to.eql(true); + expect(matchB.binary.target.after).to.eql(replaceWith); + } + + const writtenFile = (await Fs.read(b.target.join('images/volcano.jpg'))).data; + expect(writtenFile).to.eql(replaceWith); + }); + + it('fn: modify with wrong type (throws)', async () => { + const sample1 = Test.sample1(); + const sample2 = Test.sample2(); + + const a = Tmpl.create(sample1.source, (e) => { + if (e.contentType === 'text') e.modify(new Uint8Array([1, 2, 3])); // NB: error. + }); + const b = Tmpl.create(sample2.source, (e) => { + if (e.contentType === 'binary') e.modify('fail'); // NB: error. + }); + + await expectError( + () => a.write(sample1.target), + 'Expected string content to update text-file', + ); + + await expectError( + () => b.write(sample2.target), + 'Expected Uint8Array content to update binary-file', + ); + }); + + it('fn: {ctx} passed constructor param', async () => { + const { source, target } = Test.sample1(); + const ctx = { foo: 'root' }; + const fired = [] as any[]; + const tmpl = Tmpl.create(source, { ctx, processFile: (e) => fired.push(e.ctx) }); + + const a = await tmpl.write(target); + const b = await tmpl.write(target, { ctx: { bar: 456 } }); + const c = await tmpl.write(target, { ctx: { foo: 123, bar: 456 } }); + + expect(fired.every((m) => !!m)).to.be.true; + expect(a.ctx).to.eql(ctx); + expect(b.ctx).to.eql({ foo: 'root', bar: 456 }); + expect(c.ctx).to.eql({ foo: 123, bar: 456 }); // NB: root {ctx} overwritten. + }); + + it('fn: {ctx} passed via .write() param', async () => { + const { source, target } = Test.sample1(); + const ctx = { foo: 123 }; + const fired = [] as any[]; + const tmpl = Tmpl.create(source, (e) => fired.push(e.ctx)); + + const a = await tmpl.write(target, { ctx }); + expect(fired.filter(Boolean).length).to.be.greaterThan(1); + expect(fired.every((m) => R.equals(m, ctx))).to.be.true; + expect(a.ctx).to.eql(ctx); + + // No context provided. + const beforeLength = fired.filter(Boolean).length; + const b = await tmpl.write(target); + expect(fired.filter(Boolean).length).to.eql(beforeLength); + expect(b.ctx).to.eql(undefined); + }); }); - describe('fn: beforeCopy | afterCopy (callbacks)', () => { - it('beforeCopy: sync/async', async () => { - const { source, target } = SAMPLE.init(); - const fired: t.TmplCopyHandlerArgs[] = []; + describe('fn: beforeWrite | afterWrite (callbacks)', () => { + it('beforeWrite: sync/async', async () => { + const { source, target } = Test.sample1(); + const fired: t.TmplWriteHandlerArgs[] = []; - const a: t.TmplCopyHandler = (e) => fired.push(e); - const b: t.TmplCopyHandler = async (e) => { + const a: t.TmplWriteHandler = (e) => fired.push(e); + const b: t.TmplWriteHandler = async (e) => { expect((await fired[0].dir.target.ls()).length).to.greaterThan(0); // NB: files already exist. fired.push(e); }; - const tmpl = Tmpl.create(source, { beforeCopy: a }); + const tmpl = Tmpl.create(source, { beforeWrite: a }).filter(() => true); // NB: .filter tests that the before/after handlers are passed through. - await tmpl.copy(target); + await tmpl.write(target); expect(fired.length).to.eql(1); - await tmpl.copy(target, { beforeCopy: [b] }); + await tmpl.write(target, { onBefore: [b] }); expect(fired.length).to.eql(3); // NB: 2-more (the constructor callback PLUS callback passed to the copy paramemter). + expect(fired.every((m) => m.ctx === undefined)).to.eql(true); // NB: {ctx} not passed to the [Tmpl.write] method. }); - it('afterCopy: sync/async', async () => { - const { source, target } = SAMPLE.init(); - const fired: t.TmplCopyHandlerArgs[] = []; + it('afterWrite: sync/async', async () => { + const { source, target } = Test.sample1(); + const fired: t.TmplWriteHandlerArgs[] = []; - const a: t.TmplCopyHandler = (e) => fired.push(e); - const b: t.TmplCopyHandler = async (e) => { + const a: t.TmplWriteHandler = (e) => fired.push(e); + const b: t.TmplWriteHandler = async (e) => { expect((await fired[0].dir.target.ls()).length).to.greaterThan(0); // NB: files already exist. fired.push(e); }; - const tmpl = Tmpl.create(source, { afterCopy: a }); + const tmpl = Tmpl.create(source, { afterWrite: a }); - await tmpl.copy(target); + await tmpl.write(target); expect(fired.length).to.eql(1); - await tmpl.copy(target, { afterCopy: [b] }); + await tmpl.write(target, { onAfter: [b] }); expect(fired.length).to.eql(3); // NB: 2-more (the constructor callback PLUS callback passed to the copy paramemter). + expect(fired.every((m) => m.ctx === undefined)).to.eql(true); // NB: {ctx} not passed to the [Tmpl.write] method. + }); + + it('{ctx} passed via .write() param', async () => { + const { source, target } = Test.sample1(); + const ctx = { foo: 123 }; + const fired = { + before: [] as any[], + after: [] as any[], + }; + const onBefore: t.TmplWriteHandler = (e) => fired.before.push(e.ctx); + const onAfter: t.TmplWriteHandler = (e) => fired.after.push(e.ctx); + const tmpl = Tmpl.create(source); + + // No context provided, so {ctx} not passed through before/after callbacks. + await tmpl.write(target, { onBefore, onAfter }); + expect(fired.before.length).to.greaterThan(0); + expect(fired.after.length).to.greaterThan(0); + expect(fired.before.filter(Boolean).length).to.eql(0); + expect(fired.after.filter(Boolean).length).to.eql(0); + + // Before/after callbacks provided with {ctx}. + await tmpl.write(target, { ctx, onBefore, onAfter }); + expect(fired.before.filter(Boolean).length).to.eql(1); + expect(fired.after.filter(Boolean).length).to.eql(1); + expect(fired.before.filter(Boolean).every((m) => R.equals(m, ctx))).to.be.true; + expect(fired.after.filter(Boolean).every((m) => R.equals(m, ctx))).to.be.true; }); }); describe('flag: force', () => { it('force', async () => { - const test = SAMPLE.init(); + const test = Test.sample1(); const tmpl = Tmpl.create(test.source); - const resA = await tmpl.copy(test.target); - const resB = await tmpl.copy(test.target); // NB: no changes (already written). - const resC = await tmpl.copy(test.target, { force: true }); + const resA = await tmpl.write(test.target); + const resB = await tmpl.write(test.target); // NB: no changes (already written). + const resC = await tmpl.write(test.target, { force: true }); expect(resA.ops.every((m) => m.forced === false)).to.be.true; expect(resB.ops.every((m) => m.forced === false)).to.be.true; @@ -224,14 +387,14 @@ describe('Tmpl', () => { }); it('force ā (with exclusions)', async () => { - const test = SAMPLE.init(); + const test = Test.sample1(); const tmpl = Tmpl.create(test.source, (e) => { if (!e.target.exists) return; // NB: create the user-space file if it does not yet exist if (e.target.file.name === 'doc.md') e.exclude('user-space'); }); - const resA = await tmpl.copy(test.target); - const resB = await tmpl.copy(test.target, { force: true }); + const resA = await tmpl.write(test.target); + const resB = await tmpl.write(test.target, { force: true }); expect(await test.exists.target('docs/index.md')).to.eql(true); const indent = 2; @@ -242,11 +405,11 @@ describe('Tmpl', () => { }); }); - describe('flag: write (default: true)', () => { - it('write: false', async () => { - const test = SAMPLE.init(); + describe('flag: dryRun (default: false)', () => { + it('dryRun: true (does not write)', async () => { + const test = Test.sample1(); const tmpl = Tmpl.create(test.source); - const res = await tmpl.copy(test.target, { write: false }); + const res = await tmpl.write(test.target, { dryRun: true }); expect(res.ops.every((m) => m.written === false)).to.be.true; for (const op of res.ops) { expect(await Fs.exists(op.file.target.absolute)).to.eql(false); @@ -254,9 +417,9 @@ describe('Tmpl', () => { }); it('logs as "dry run"', async () => { - const test = SAMPLE.init(); + const test = Test.sample1(); const tmpl = Tmpl.create(test.source); - const res = await tmpl.copy(test.target, { write: false }); + const res = await tmpl.write(test.target, { dryRun: true }); const table = Tmpl.Log.table(res.ops); expect(table).to.include('dry-run'); console.info(table); @@ -270,7 +433,7 @@ describe('Tmpl', () => { }; it('single-level filter', async () => { - const test = SAMPLE.init(); + const test = Test.sample1(); const tmpl1 = Tmpl.create(test.source); const tmpl2 = Tmpl.create(test.source).filter((e) => e.file.name !== '.gitignore'); expect(includes(await tmpl1.source.ls(), '/.gitignore')).to.be.true; @@ -278,7 +441,7 @@ describe('Tmpl', () => { }); it('multi-level filter', async () => { - const test = SAMPLE.init(); + const test = Test.sample1(); const tmpl1 = Tmpl.create(test.source).filter((e) => e.file.name !== '.gitignore'); const tmpl2 = tmpl1.filter((e) => !e.file.name.endsWith('.md')); expect(includes(await tmpl1.source.ls(), '/.gitignore')).to.be.false; @@ -287,9 +450,9 @@ describe('Tmpl', () => { }); it('does not copy filtered files', async () => { - const test = SAMPLE.init(); + const test = Test.sample1(); const tmpl = Tmpl.create(test.source).filter((e) => !e.file.name.endsWith('.md')); - await tmpl.copy(test.target); + await tmpl.write(test.target); const paths = await test.ls.target(); expect(paths.length).to.greaterThan(2); diff --git a/code/sys/tmpl/src/m.Tmpl/t.ts b/code/sys/tmpl/src/m.Tmpl/t.ts index fae8511684..1fc14debe5 100644 --- a/code/sys/tmpl/src/m.Tmpl/t.ts +++ b/code/sys/tmpl/src/m.Tmpl/t.ts @@ -1,5 +1,7 @@ import type { t } from './common.ts'; +type O = Record<string, unknown>; + /** * Library for copying template files. */ @@ -21,9 +23,17 @@ export type TmplFactory = ( /** Options passed to the template engine factory. */ export type TmplFactoryOptions = { + /** Handler to run after the write operation completes. */ + beforeWrite?: t.TmplWriteHandler; + + /** Handler to process each file in the template. */ processFile?: t.TmplProcessFile; - beforeCopy?: t.TmplCopyHandler; - afterCopy?: t.TmplCopyHandler; + + /** Handler to run after the write operation completes. */ + afterWrite?: t.TmplWriteHandler; + + /** Context data passed to the process handler. */ + ctx?: O; }; /** @@ -34,7 +44,7 @@ export type Tmpl = { readonly source: t.FsDir; /** Perform a copy of the templates to a target directory. */ - copy(target: t.StringDir, options?: t.TmplCopyOptions): Promise<t.TmplCopyResponse>; + write(target: t.StringDir, options?: t.TmplWriteOptions): Promise<t.TmplWriteResponse>; /** Clones the template filtering down to a subset of source files. */ filter(fn: t.TmplFilter): t.Tmpl; @@ -58,24 +68,44 @@ export type TmplFilter = t.FsFileFilter; */ export type TmplProcessFile = (args: TmplProcessFileArgs) => TmplProcessFileResponse; export type TmplProcessFileResponse = t.IgnoredResult | Promise<t.IgnoredResult>; -export type TmplProcessFileArgs = { +export type TmplProcessFileArgs = t.TmplProcessTextFileArgs | TmplProcessBinaryFileArgs; + +type FileArgs = { + /** Optional context passed to the `Tmpl.write` operation. */ + readonly ctx?: O; + /** The source template file. */ readonly tmpl: t.FsFile; /** The target location being copied to. */ readonly target: t.FsFile & { exists: boolean }; - /** The text body of the file. */ - readonly text: { tmpl: string; current: string }; - /** Filter out the file from being copied. */ exclude(reason?: string): TmplProcessFileArgs; /** Adjust the name of the file. */ rename(filename: string): TmplProcessFileArgs; - /** Adjust the text within the file. */ - modify(text: string): TmplProcessFileArgs; + /** Adjust the content of the file. */ + modify(next: string | Uint8Array): TmplProcessTextFileArgs; +}; + +/** Arguments passed to a text-file for processing. */ +export type TmplProcessTextFileArgs = FileArgs & { + /** The content-type of the template file. */ + readonly contentType: t.TmplTextFileOperation['contentType']; + /** The text body of the file. */ + readonly text: { tmpl: string; current: string }; + readonly binary: undefined; +}; + +/** Arguments passed to a binary-file for processing. */ +export type TmplProcessBinaryFileArgs = FileArgs & { + /** The content-type of the template file. */ + readonly contentType: t.TmplBinaryFileOperation['contentType']; + /** The text body of the file. */ + readonly binary: { tmpl: Uint8Array; current: Uint8Array }; + readonly text: undefined; }; /** @@ -83,40 +113,46 @@ export type TmplProcessFileArgs = { * Use this to do either clean up, or additional setup actions not handled * directly by the template-copy engine. */ -export type TmplCopyHandler = (e: TmplCopyHandlerArgs) => t.IgnoredResult; -/** Arguments passed to the `afterCopy` callback. */ -export type TmplCopyHandlerArgs = { +export type TmplWriteHandler = (e: TmplWriteHandlerArgs) => t.IgnoredResult; +/** Arguments passed to the write handler. */ +export type TmplWriteHandlerArgs = { readonly dir: { readonly source: t.FsDir; readonly target: t.FsDir }; + readonly ctx?: O; }; /** Options passed to the `tmpl.copy` method. */ -export type TmplCopyOptions = { +export type TmplWriteOptions = { /** Flag indicating if the copy operation should be forced. (NB: "excluded" paths will never be written). */ force?: boolean; - /** Flag indicating if the files should be written. Default: true (pass false for a "dry-run"). */ - write?: boolean; + /** Flag indicating if the files should be written. Default: false. */ + dryRun?: boolean; /** Handler(s) to run before the copy operation starts. */ - beforeCopy?: t.TmplCopyHandler | t.TmplCopyHandler[]; + onBefore?: t.TmplWriteHandler | t.TmplWriteHandler[]; /** Handler(s) to run after the copy operation completes. */ - afterCopy?: t.TmplCopyHandler | t.TmplCopyHandler[]; + onAfter?: t.TmplWriteHandler | t.TmplWriteHandler[]; + + /** Context data passed to the process handler. */ + ctx?: O; }; /** * The reponse returned from the `tmpl.copy` method. */ -export type TmplCopyResponse = { +export type TmplWriteResponse = { readonly source: t.FsDir; readonly target: t.FsDir; readonly ops: t.TmplFileOperation[]; + readonly ctx?: O; }; /** * Details about a file update. */ -export type TmplFileOperation = { +export type TmplFileOperation = TmplTextFileOperation | TmplBinaryFileOperation; +type Operation = { /** If excluded, contains the reason for the exclusion, otherwise `boolean` flag. */ excluded: boolean | { reason: string }; @@ -134,10 +170,42 @@ export type TmplFileOperation = { /** File path details. */ file: { tmpl: t.FsFile; target: t.FsFile }; +}; + +/** The content-type contained within the template file. */ +export type TmplFileContentType = TmplFileOperation['contentType']; + +export type TmplTextFileOperation = Operation & { + /** The content-type of the template file. */ + readonly contentType: 'text'; + readonly binary: undefined; /** The text content of the file. */ - text: { - tmpl: string; - target: { before: string; after: string; isDiff: boolean }; - }; + readonly text: t.TmplFileOperationText; +}; + +export type TmplBinaryFileOperation = Operation & { + /** The content-type of the template file. */ + readonly contentType: 'binary'; + readonly text: undefined; + + /** The binary content of the file. */ + readonly binary: t.TmplFileOperationBinary; +}; + +/** The text content of the file. */ +export type TmplFileOperationText = { + /** The source template before transform. */ + readonly tmpl: string; + /** Details about the template file at the target location. */ + readonly target: { before: string; after: string; isDiff: boolean }; +}; + +/** The binary content of the file. */ +export type TmplFileOperationBinary = { + /** The source template before transform. */ + readonly tmpl: Uint8Array; + + /** Details about the template file at the target location. */ + readonly target: { before: Uint8Array; after: Uint8Array; isDiff: boolean }; }; diff --git a/code/sys/tmpl/src/m.Tmpl/u.copy.ts b/code/sys/tmpl/src/m.Tmpl/u.copy.ts deleted file mode 100644 index cdbce42ccd..0000000000 --- a/code/sys/tmpl/src/m.Tmpl/u.copy.ts +++ /dev/null @@ -1,172 +0,0 @@ -import { type t, Fs } from './common.ts'; - -type Changes = { - excluded: t.TmplFileOperation['excluded']; - filename: string; - text: string; -}; - -export async function copy( - source: t.FsDir, - target: t.FsDir, - fn: t.TmplProcessFile | undefined, - options: t.TmplCopyOptions = {}, -) { - const forced = options.force ?? false; - const ops: t.TmplFileOperation[] = []; - const res: t.TmplCopyResponse = { - get source() { - return source; - }, - get target() { - return target; - }, - get ops() { - return ops; - }, - }; - - const copyArgs: t.TmplCopyHandlerArgs = { - get dir() { - return { source, target }; - }, - }; - - /** - * Run BEFORE handlers. - */ - for (const fn of wrangle.copyHandlers(options.beforeCopy)) { - await fn(copyArgs); - } - - /** - * Perform copy on files. - */ - for (const from of await source.ls()) { - if (await Fs.Is.dir(from)) continue; - - const to = Fs.join(target.absolute, from.slice(source.absolute.length + 1)); - const sourceText = (await Fs.readText(from)).data ?? ''; - const targetText = (await Fs.readText(to)).data ?? ''; - - type T = t.TmplFileOperation; - let _file: T['file']; - let _text: T['text']; - - const op: T = { - get file() { - if (_file) return _file; - return (_file = { - tmpl: Fs.toFile(from, source.absolute), - target: Fs.toFile(to, target.absolute), - }); - }, - get text() { - if (_text) return _text; - return (_text = { - tmpl: sourceText, - target: { - before: targetText, - after: targetText || sourceText, - get isDiff() { - return op.text.target.before !== op.text.target.after; - }, - }, - }); - }, - excluded: false, - written: false, - created: false, - updated: false, - forced, - }; - ops.push(op); - - if (typeof fn === 'function') { - const { args, changes } = await wrangle.args(op); - await fn(args); - if (changes.excluded) op.excluded = changes.excluded; - if (changes.filename) op.file.target = wrangle.rename(op.file.target, changes.filename); - if (changes.text) { - op.text.target.after = changes.text; // Update to modified output. - } else { - op.text.target.after = args.text.tmpl; // Update to current template. - } - } - - if (!op.excluded) { - const target = op.file.target; - const path = target.absolute; - const exists = await Fs.exists(path); - const isDiff = op.text.target.isDiff; - - if (!exists) { - op.created = true; - } else if (isDiff || op.forced) { - op.updated = true; - } - - if ((op.created || op.updated) && (options.write ?? true)) { - op.written = true; - - await Fs.ensureDir(Fs.dirname(path)); - await Deno.writeTextFile(path, op.text.target.after); - } - } - } - - /** - * Run AFTER handlers. - */ - for (const fn of wrangle.copyHandlers(options.afterCopy)) { - await fn(copyArgs); - } - - // Finish up. - return res; -} - -/** - * Helpers - */ -const wrangle = { - async args(op: t.TmplFileOperation) { - const changes: Changes = { excluded: false, filename: '', text: '' }; - const { tmpl, target } = op.file; - const exists = await Fs.exists(target.absolute); - const args: t.TmplProcessFileArgs = { - get tmpl() { - return tmpl; - }, - get target() { - return { ...target, exists }; - }, - get text() { - return { tmpl: op.text.tmpl, current: op.text.target.before }; - }, - exclude(reason) { - changes.excluded = typeof reason === 'string' ? { reason } : true; - return args; - }, - rename(filename) { - changes.filename = filename; - return args; - }, - modify(text) { - changes.text = text; - return args; - }, - }; - return { args, changes } as const; - }, - - rename(input: t.FsFile, newFilename: string): t.FsFile { - return Fs.toFile(Fs.join(input.dir, newFilename), input.base); - }, - - copyHandlers(input?: t.TmplCopyHandler | t.TmplCopyHandler[]): t.TmplCopyHandler[] { - if (!input) return []; - const res = Array.isArray(input) ? input : [input]; - return res.flat(Infinity).filter(Boolean); - }, -} as const; diff --git a/code/sys/tmpl/src/m.Tmpl/u.create.ts b/code/sys/tmpl/src/m.Tmpl/u.create.ts index a6f2106aa8..cd975eb273 100644 --- a/code/sys/tmpl/src/m.Tmpl/u.create.ts +++ b/code/sys/tmpl/src/m.Tmpl/u.create.ts @@ -1,12 +1,14 @@ -import { type t, Fs } from './common.ts'; -import { copy } from './u.copy.ts'; +import { type t, Fs, isRecord } from './common.ts'; +import { write } from './u.write.ts'; + +type O = Record<string, unknown>; /** * Create a new directory template. */ export const create: t.TmplFactory = (sourceDir, opt) => { - const { processFile, beforeCopy, afterCopy } = wrangle.options(opt); - return factory({ sourceDir, beforeCopy, processFile, afterCopy }); + const { processFile, beforeWrite, afterWrite, ctx } = wrangle.options(opt); + return factory({ sourceDir, beforeWrite, processFile, afterWrite, ctx }); }; /** @@ -14,10 +16,11 @@ export const create: t.TmplFactory = (sourceDir, opt) => { */ function factory(args: { sourceDir: t.StringDir; - beforeCopy?: t.TmplCopyHandler; + beforeWrite?: t.TmplWriteHandler; processFile?: t.TmplProcessFile; - afterCopy?: t.TmplCopyHandler; + afterWrite?: t.TmplWriteHandler; filter?: t.FsFileFilter[]; + ctx?: O; }): t.Tmpl { const { sourceDir, processFile } = args; const source = Fs.toDir(sourceDir, args.filter); @@ -25,15 +28,16 @@ function factory(args: { get source() { return source; }, - copy(target, options = {}) { - const beforeCopy = wrangle.copyHandlers(args.beforeCopy, options.beforeCopy); - const afterCopy = wrangle.copyHandlers(args.afterCopy, options.afterCopy); - return copy(source, Fs.toDir(target), processFile, { ...options, beforeCopy, afterCopy }); + write(target, options = {}) { + const onBefore = wrangle.writeHandlers(args.beforeWrite, options.onBefore); + const onAfter = wrangle.writeHandlers(args.afterWrite, options.onAfter); + const ctx = wrangle.ctx(args.ctx, options.ctx); + return write(source, Fs.toDir(target), processFile, { ...options, onBefore, onAfter, ctx }); }, filter(next) { - const { sourceDir, processFile } = args; + const { sourceDir, processFile, beforeWrite, afterWrite } = args; const filter = [...(args.filter ?? []), next]; - return factory({ sourceDir, processFile, filter }); + return factory({ sourceDir, processFile, beforeWrite, afterWrite, filter }); }, }; return tmpl; @@ -49,8 +53,12 @@ const wrangle = { return input; }, - copyHandlers(base?: t.TmplCopyHandler, param?: t.TmplCopyHandler | t.TmplCopyHandler[]) { - type T = t.TmplCopyHandler; + writeHandlers(base?: t.TmplWriteHandler, param?: t.TmplWriteHandler | t.TmplWriteHandler[]) { + type T = t.TmplWriteHandler; return [param, base].flat(Infinity).filter(Boolean) as T[]; }, + + ctx(root?: O, write?: O): O | undefined { + return isRecord(root) || isRecord(write) ? { ...root, ...write } : undefined; + }, } as const; diff --git a/code/sys/tmpl/src/m.Tmpl/u.write.args.ts b/code/sys/tmpl/src/m.Tmpl/u.write.args.ts new file mode 100644 index 0000000000..f1ccc437fb --- /dev/null +++ b/code/sys/tmpl/src/m.Tmpl/u.write.args.ts @@ -0,0 +1,62 @@ +import { type t, Fs, Is } from './common.ts'; + +type O = Record<string, unknown>; + +export type Changes = { + excluded: t.TmplFileOperation['excluded']; + filename?: string; + text?: string; + binary?: Uint8Array; +}; + +export async function createArgs(op: t.TmplFileOperation, ctx?: O) { + const changes: Changes = { excluded: false }; + + const { tmpl, target } = op.file; + const exists = await Fs.exists(target.absolute); + const isText = op.contentType === 'text'; + const isBinary = op.contentType === 'binary'; + + const args: t.TmplProcessFileArgs = { + contentType: op.contentType as any, + get ctx() { + return ctx; + }, + get tmpl() { + return tmpl; + }, + get target() { + return { ...target, exists }; + }, + get text() { + if (!isText) return undefined; + return { tmpl: op.text!.tmpl, current: op.text!.target.before }; + }, + get binary() { + if (!isBinary) return undefined; + const tmpl = op.binary!.tmpl; + const current = op.binary!.target.before; + return { tmpl, current } as any; // NB: type hack. + }, + exclude(reason) { + changes.excluded = typeof reason === 'string' ? { reason } : true; + return args; + }, + rename(filename) { + changes.filename = filename; + return args; + }, + modify(input: any) { + if (isText && typeof input !== 'string') { + throw new Error(`Expected string content to update text-file: ${target.relative}`); + } + if (isBinary && !Is.uint8Array(input)) { + throw new Error(`Expected Uint8Array content to update binary-file: ${target.relative}`); + } + if (isText) changes.text = input; + if (isBinary) changes.binary = input; + return args as any; + }, + }; + return { args, changes } as const; +} diff --git a/code/sys/tmpl/src/m.Tmpl/u.write.ts b/code/sys/tmpl/src/m.Tmpl/u.write.ts new file mode 100644 index 0000000000..fd66cc5e46 --- /dev/null +++ b/code/sys/tmpl/src/m.Tmpl/u.write.ts @@ -0,0 +1,212 @@ +import { type t, Fs } from './common.ts'; +import { createArgs } from './u.write.args.ts'; + +type O = Record<string, unknown>; + +export async function write( + source: t.FsDir, + target: t.FsDir, + fn: t.TmplProcessFile | undefined, + options: t.TmplWriteOptions = {}, +) { + const { ctx, dryRun } = options; + const forced = options.force ?? false; + + const ops: t.TmplFileOperation[] = []; + const res: t.TmplWriteResponse = { + get source() { + return source; + }, + get target() { + return target; + }, + get ops() { + return ops; + }, + get ctx() { + return ctx; + }, + }; + + const copyArgs: t.TmplWriteHandlerArgs = { + ctx, + get dir() { + return { source, target }; + }, + }; + + /** + * Run BEFORE handlers. + */ + for (const fn of wrangle.copyHandlers(options.onBefore)) { + await fn(copyArgs); + } + + /** + * Perform copy on files. + */ + for (const from of await source.ls()) { + if (await Fs.Is.dir(from)) continue; + const isBinary = await Fs.Is.binary(from); + const isText = !isBinary; + const to = Fs.join(target.absolute, from.slice(source.absolute.length + 1)); + + const Lazy = { + file() { + let _prop: t.TmplFileOperation['file']; + return () => { + if (_prop) return _prop; + return (_prop = { + tmpl: Fs.toFile(from, source.absolute), + target: Fs.toFile(to, target.absolute), + }); + }; + }, + async text() { + type R = t.TmplTextFileOperation['text']; + const source = isText ? (await Fs.readText(from)).data ?? '' : ''; + const target = isText ? (await Fs.readText(to)).data ?? '' : ''; + let _prop: R; + return (): R => { + if (_prop) return _prop; + return (_prop = { + tmpl: source, + target: { + before: target, + after: target || source, + get isDiff() { + return _prop?.target.before !== _prop?.target.after; + }, + }, + }); + }; + }, + async binary() { + type R = t.TmplBinaryFileOperation['binary']; + const empty = new Uint8Array(0); + const source = isBinary ? (await Fs.read(from)).data ?? empty : empty; + const target = isBinary ? (await Fs.read(to)).data ?? empty : empty; + let _prop: R; + return (): R => { + if (_prop) return _prop; + return (_prop = { + tmpl: source, + target: { + before: target, + after: target || source, + get isDiff() { + return !arraysEqual(_prop?.target.before, _prop?.target.after); + }, + }, + }); + }; + }, + } as const; + + const fileProp = Lazy.file(); + const textProp = await Lazy.text(); + const binaryProp = await Lazy.binary(); + + const op: t.TmplFileOperation = { + contentType: (isText ? 'text' : 'binary') as any, + get file() { + return fileProp(); + }, + get text() { + return (isText ? textProp() : undefined) as any; // NB: type-hack. + }, + get binary() { + return (isBinary ? binaryProp() : undefined) as any; // NB: type-hack. + }, + excluded: false, + written: false, + created: false, + updated: false, + forced, + }; + ops.push(op); + + /** + * Run "process file" handler. + */ + const { args, changes } = await createArgs(op, ctx); + if (typeof fn === 'function') { + await fn(args); + + if (changes.excluded) op.excluded = changes.excluded; + if (changes.filename) op.file.target = wrangle.rename(op.file.target, changes.filename); + if (isText) { + if (changes.text) { + op.text!.target.after = changes.text; // Update to: modified output. + } else { + op.text!.target.after = args.text!.tmpl; // Update to: current template. + } + } + if (isBinary) { + if (changes.binary) { + op.binary!.target.after = changes.binary; // Update to: modified output. + } else { + op.binary!.target.after = args.binary!.tmpl; // Update to: current template. + } + } + } else { + if (isText) op.text!.target.after = args.text!.tmpl; // Update to: current template. + if (isBinary) op.binary!.target.after = args.binary!.tmpl; // Update to: current template. + } + + if (!op.excluded) { + const target = op.file.target; + const path = target.absolute; + const exists = await Fs.exists(path); + + let isDiff = false; + if (isText) isDiff = op.text!.target.isDiff; + if (isBinary) isDiff = op.binary!.target.isDiff; + + if (!exists) { + op.created = true; + } else if (isDiff || op.forced) { + op.updated = true; + } + + if (!dryRun && (op.created || op.updated)) { + let data: string | Uint8Array | undefined; + if (isText) data = op.text!.target.after; + if (isBinary) data = op.binary!.target.after; + if (data) { + await Fs.write(path, data, { throw: true }); + op.written = true; + } + } + } + } + + /** + * Run AFTER handlers. + */ + for (const fn of wrangle.copyHandlers(options.onAfter)) { + await fn(copyArgs); + } + + // Finish up. + return res; +} + +/** + * Helpers + */ +const wrangle = { + rename(input: t.FsFile, newFilename: string): t.FsFile { + return Fs.toFile(Fs.join(input.dir, newFilename), input.base); + }, + + copyHandlers(input?: t.TmplWriteHandler | t.TmplWriteHandler[]): t.TmplWriteHandler[] { + if (!input) return []; + const res = Array.isArray(input) ? input : [input]; + return res.flat(Infinity).filter(Boolean); + }, +} as const; + +function arraysEqual(a: Uint8Array, b: Uint8Array): boolean { + return a.length === b.length && a.every((value, index) => value === b[index]); +} diff --git a/code/sys/tmpl/src/pkg.ts b/code/sys/tmpl/src/pkg.ts index 79cb3f9c80..d0d1339d3d 100644 --- a/code/sys/tmpl/src/pkg.ts +++ b/code/sys/tmpl/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@sys/tmpl', version: '0.0.90' }; diff --git a/code/sys/types/deno.json b/code/sys/types/deno.json index ac4c591cf1..4c95229dd0 100644 --- a/code/sys/types/deno.json +++ b/code/sys/types/deno.json @@ -1,6 +1,6 @@ { "name": "@sys/types", - "version": "0.0.82", + "version": "0.0.92", "license": "MIT", "tasks": { "test": "deno test -RW", diff --git a/code/sys/types/src/pkg.ts b/code/sys/types/src/pkg.ts index dfc4462424..40a7a990a7 100644 --- a/code/sys/types/src/pkg.ts +++ b/code/sys/types/src/pkg.ts @@ -1,8 +1,16 @@ -import type { t } from '@sys/types'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = {name: deno.name, version: deno.version }; +export const pkg: Pkg = { name: '@sys/types', version: '0.0.92' }; diff --git a/code/sys/types/src/types/t.Dispose.ts b/code/sys/types/src/types/t.Dispose.ts index 397939d9f0..ac8bd027d1 100644 --- a/code/sys/types/src/types/t.Dispose.ts +++ b/code/sys/types/src/types/t.Dispose.ts @@ -13,6 +13,10 @@ export type Disposable = { dispose(): void; }; +/** The "until this fires" input for a disposable resource factory. */ +export type DisposeInput = t.UntilObservable | t.Disposable; +export type UntilInput = DisposeInput; + /** * An object that provides a standard asynchronous destructor pattern. */ @@ -54,7 +58,7 @@ export type LifecycleAsync = DisposableAsync & { readonly disposed: boolean }; /** * Utility Type: remove fields from composite Dispose object. */ -export type OmitDisposable<T extends Disposable | DisposableAsync> = Omit< +export type OmitDisposable<T extends Disposable | DisposableAsync | object> = Omit< T, 'dispose' | 'dispose$' >; @@ -62,7 +66,7 @@ export type OmitDisposable<T extends Disposable | DisposableAsync> = Omit< /** * Utility Type: remove fields from composite Lifecycle object. */ -export type OmitLifecycle<T extends Lifecycle | LifecycleAsync> = Omit< +export type OmitLifecycle<T extends Lifecycle | LifecycleAsync | object> = Omit< T, 'dispose' | 'dispose$' | 'disposed' >; diff --git a/code/sys/types/src/types/t.Dom.ts b/code/sys/types/src/types/t.Dom.ts index d82e6b19ec..231cca7511 100644 --- a/code/sys/types/src/types/t.Dom.ts +++ b/code/sys/types/src/types/t.Dom.ts @@ -12,3 +12,6 @@ export type DomRect = { bottom: number; left: number; }; + +/** Readonly version of the DomRect type. */ +export type DomRectReadonly = Readonly<DomRect>; diff --git a/code/sys/types/src/types/t.Immutable.ts b/code/sys/types/src/types/t.Immutable.ts index ad7ded5cab..25c9806b26 100644 --- a/code/sys/types/src/types/t.Immutable.ts +++ b/code/sys/types/src/types/t.Immutable.ts @@ -55,6 +55,15 @@ export type ImmutableEvents< C extends ImmutableChange<D, P> = ImmutableChange<D, P>, > = t.Lifecycle & { readonly changed$: t.Observable<C> }; +/** + * Utility type to infer the event-type contained within the ImmutableEvents type. + */ +export type InferImmutableEvent<T extends { changed$: t.Observable<any> }> = T extends { + changed$: t.Observable<infer E>; +} + ? E + : never; + /** * Represents a before/after patched change to the immutable state. */ diff --git a/code/sys/types/src/types/t.Number.ts b/code/sys/types/src/types/t.Number.ts index 0fc24f1180..96debd733c 100644 --- a/code/sys/types/src/types/t.Number.ts +++ b/code/sys/types/src/types/t.Number.ts @@ -1,29 +1,26 @@ -/** - * A number representing an 0-based index. - */ +/** A number representing an 0-based index. */ export type Index = number; -/** - * A number representing a network port. - */ +/** A number representing a network port. */ export type PortNumber = number; -/** - * A number representing screen pixels. - */ +/** A number representing screen pixels. */ export type Pixels = number; -/** - * Number representing a percentage: 0..1 ā (0=0%, 1=100%). - */ +/** Number representing a percentage: 0..1 ā (0=0%, 1=100%). */ export type Percent = number; -/** - * A pixel OR a percentage number: 0..1 = percent, >1 = pixels. - */ +/** A pixel OR a percentage number: 0..1 = percent, >1 = pixels. */ export type PixelOrPercent = Pixels | Percent; // -/** - * A number that represents a "total" (typically 1-based). - */ +/** A number that represents an "offset" of another value. */ +export type NumberOffset = number; + +/** A number that represents a "total" (typically 1-based). */ export type NumberTotal = number; + +/** A number that represents a width. */ +export type NumberWidth = number; + +/** A number that represents a height. */ +export type NumberHeight = number; diff --git a/code/sys/types/src/types/t.Pkg.dist.ts b/code/sys/types/src/types/t.Pkg.dist.ts index 1f180ed427..17f17ea1f0 100644 --- a/code/sys/types/src/types/t.Pkg.dist.ts +++ b/code/sys/types/src/types/t.Pkg.dist.ts @@ -7,6 +7,8 @@ import type { t } from './common.ts'; * produced during a build/bundle operation for a module. */ export type DistPkg = { + '-type:': 'jsr:@sys/types:DistPkg'; + /** The package meta-data info. */ pkg: t.Pkg; diff --git a/code/sys/types/src/types/t.String.ts b/code/sys/types/src/types/t.String.ts index 4592276cb5..f96d1996a0 100644 --- a/code/sys/types/src/types/t.String.ts +++ b/code/sys/types/src/types/t.String.ts @@ -64,6 +64,9 @@ export type StringJson = string; /** A raw string of unparsed YAML. */ export type StringYaml = string; +/** String representing a timestamp in the form "HH:MM:SS:mmm". */ +export type StringTimestamp = string; + /** * The name (module-specifier) of an ESM import. * eg: diff --git a/code/sys/types/src/types/t.Time.ts b/code/sys/types/src/types/t.Time.ts index 594ded265a..4d463be44a 100644 --- a/code/sys/types/src/types/t.Time.ts +++ b/code/sys/types/src/types/t.Time.ts @@ -1,27 +1,34 @@ +import type { t } from './common.ts'; + +/** + * A number representing a time (eg. msecs, secs etc). + */ +export type NumberTime = number; + /** * A number representing milliseconds. */ -export type Msecs = number; +export type Msecs = NumberTime; /** * Number represening seconds. */ -export type Secs = number; +export type Secs = NumberTime; /** * Number represening minutes. */ -export type Mins = number; +export type Mins = NumberTime; /** * Number represening hours. */ -export type Hours = number; +export type Hours = NumberTime; /** * Number represening days. */ -export type Days = number; +export type Days = NumberTime; /** * Represents a Unix Epoch timestamp in seconds. The Unix Epoch time is @@ -52,3 +59,19 @@ export type TimeDuration = { readonly day: Days; toString(unit?: TimeUnit | { unit?: TimeUnit; round?: number }): string; }; + +/** + * A index/map of timestamp related data-objects. + */ +export type Timestamps<T> = { + [HH_MM_SS_mmm: t.StringTimestamp]: T; +}; + +/** + * A single timestamp with data and duration props. + */ +export type Timestamp<T> = { + timestamp: t.StringTimestamp; + total: t.TimeDuration; + data: T; +}; diff --git a/deno.json b/deno.json index 700859c013..55324a8f9b 100644 --- a/deno.json +++ b/deno.json @@ -12,8 +12,9 @@ "reload:clear": "rm -rf node_modules", "reload:install": "deno install --reload", - "prep": "deno run -RWE --allow-ffi --allow-run ./scripts/main.ts --prep", - "bump": "deno run -RWE --allow-sys --allow-ffi --allow-run ./scripts/main.ts --bump", + "prep": "deno run -RWE --allow-ffi --allow-run ./scripts/main.ts --prep", + "bump": "deno run -RWE --allow-sys --allow-ffi --allow-run ./scripts/main.ts --bump", + "tmpl": "deno run -RWE ./scripts/main.ts --tmpl", "tmp": "deno run -A ./scripts/-tmp.ts" }, @@ -25,7 +26,7 @@ "jsxImportSource": "react" }, "nodeModulesDir": "auto", - "importMap": "./deno.imports.json", + "importMap": "./imports.json", "workspace": [ "./code/sys/types", "./code/sys/std", @@ -63,8 +64,8 @@ "./code/sys/main", "./code/-samples/deno.vite-react", + "./deploy/@tdb.slc", "./deploy/api.db.team", - "./deploy/slc.db.team", "./deploy/tmp.db.team", "./code/-tmpl/deno", diff --git a/deno.lock b/deno.lock index 519c656b1b..9dfa4b3c69 100644 --- a/deno.lock +++ b/deno.lock @@ -7,48 +7,30 @@ "jsr:@cliffy/keypress@1.0.0-rc.7": "1.0.0-rc.7", "jsr:@cliffy/prompt@1.0.0-rc.7": "1.0.0-rc.7", "jsr:@cliffy/table@1.0.0-rc.7": "1.0.0-rc.7", - "jsr:@deno/cache-dir@0.13.2": "0.13.2", - "jsr:@deno/emit@*": "0.46.0", - "jsr:@deno/graph@~0.73.1": "0.73.1", - "jsr:@std/assert@0.223": "0.223.0", - "jsr:@std/assert@^1.0.10": "1.0.11", "jsr:@std/assert@~1.0.6": "1.0.11", "jsr:@std/async@1.0.10": "1.0.10", - "jsr:@std/bytes@0.223": "0.223.0", - "jsr:@std/bytes@^1.0.2": "1.0.4", - "jsr:@std/crypto@^1.0.3": "1.0.3", - "jsr:@std/data-structures@^1.0.6": "1.0.6", "jsr:@std/datetime@0.225.3": "0.225.3", "jsr:@std/dotenv@0.225.3": "0.225.3", "jsr:@std/encoding@1.0.7": "1.0.7", "jsr:@std/encoding@~1.0.5": "1.0.7", - "jsr:@std/fmt@0.223": "0.223.0", - "jsr:@std/fmt@~1.0.2": "1.0.5", - "jsr:@std/fs@0.223": "0.223.0", - "jsr:@std/fs@1.0.11": "1.0.11", + "jsr:@std/fmt@~1.0.2": "1.0.6", "jsr:@std/fs@1.0.13": "1.0.13", - "jsr:@std/fs@^1.0.9": "1.0.11", - "jsr:@std/internal@^1.0.5": "1.0.5", - "jsr:@std/io@0.223": "0.223.0", "jsr:@std/io@~0.224.9": "0.224.9", - "jsr:@std/path@0.223": "0.223.0", "jsr:@std/path@1.0.8": "1.0.8", "jsr:@std/path@^1.0.8": "1.0.8", "jsr:@std/path@~1.0.6": "1.0.8", "jsr:@std/semver@1.0.4": "1.0.4", - "jsr:@std/testing@1.0.9": "1.0.9", - "jsr:@std/text@~1.0.7": "1.0.10", - "jsr:@std/uuid@1.0.4": "1.0.4", + "jsr:@std/text@~1.0.7": "1.0.11", "npm:@automerge/automerge-repo-network-broadcastchannel@1.2.1": "1.2.1", "npm:@automerge/automerge-repo-storage-indexeddb@1.2.1": "1.2.1", "npm:@automerge/automerge-repo-storage-nodefs@1.2.1": "1.2.1", "npm:@automerge/automerge-repo@1.2.1": "1.2.1", "npm:@automerge/automerge@2.2.8": "2.2.8", + "npm:@deno/vite-plugin@1.0.4": "1.0.4_vite@6.1.1", "npm:@jsr/std__async@1.0.10": "1.0.10", "npm:@jsr/std__datetime@0.225.3": "0.225.3", "npm:@jsr/std__dotenv@0.225.3": "0.225.3", "npm:@jsr/std__encoding@1.0.7": "1.0.7", - "npm:@jsr/std__fs@1.0.11": "1.0.11", "npm:@jsr/std__fs@1.0.13": "1.0.13", "npm:@jsr/std__path@1.0.8": "1.0.8", "npm:@jsr/std__semver@1.0.4": "1.0.4", @@ -56,12 +38,15 @@ "npm:@jsr/std__uuid@1.0.4": "1.0.4", "npm:@noble/hashes@1.7.1": "1.7.1", "npm:@onsetsoftware/automerge-patcher@0.14.0": "0.14.0_@automerge+automerge@2.2.8", + "npm:@preact/signals-core@1.8.0": "1.8.0", + "npm:@preact/signals-react@3.0.1": "3.0.1_react@18.3.1", + "npm:@svgdotjs/svg.js@3.2.4": "3.2.4", "npm:@types/diff@7.0.1": "7.0.1", "npm:@types/node@*": "22.12.0", "npm:@types/react-dom@18.3.5": "18.3.5_@types+react@18.3.18", "npm:@types/react@18.3.18": "18.3.18", "npm:@vidstack/react@1.12.12": "1.12.12_@types+react@18.3.18_react@18.3.1", - "npm:@vitejs/plugin-react-swc@3.8.0": "3.8.0_vite@6.1.1__@types+node@22.12.0__yaml@2.7.0_@types+node@22.12.0_yaml@2.7.0", + "npm:@vitejs/plugin-react-swc@3.8.0": "3.8.0_vite@6.1.1", "npm:approx-string-match@2": "2.0.0", "npm:chai@5": "5.2.0", "npm:csstype@3": "3.1.3", @@ -75,6 +60,7 @@ "npm:hono@4.7.2": "4.7.2", "npm:ignore@7": "7.0.3", "npm:immer@10": "10.1.1", + "npm:motion@12.5.0": "12.5.0_react@18.3.1_react-dom@18.3.1__react@18.3.1", "npm:ollama@0.5.13": "0.5.13", "npm:ora@8.2.0": "8.2.0", "npm:pretty-bytes@6.1.1": "6.1.1", @@ -83,10 +69,9 @@ "npm:react-dom@18.3.1": "18.3.1_react@18.3.1", "npm:react-error-boundary@5": "5.0.0_react@18.3.1", "npm:react-icons@5.5.0": "5.5.0_react@18.3.1", - "npm:react-inspector@6": "6.0.2_react@18.3.1", + "npm:react-inspector@6.0.2": "6.0.2_react@18.3.1", "npm:react-spinners@0.15.0": "0.15.0_react@18.3.1_react-dom@18.3.1__react@18.3.1", "npm:react@18.3.1": "18.3.1", - "npm:rollup@4.34.6": "4.34.6", "npm:rollup@4.34.8": "4.34.8", "npm:rxjs@7.8.2": "7.8.2", "npm:strip-ansi@7": "7.1.0", @@ -95,11 +80,11 @@ "npm:ts-essentials@10.0.4": "10.0.4", "npm:ua-parser-js@2.0.2": "2.0.2", "npm:valibot@1.0.0-rc.1": "1.0.0-rc.1", - "npm:vite-plugin-wasm@3.4.1": "3.4.1_vite@6.1.1__@types+node@22.12.0__yaml@2.7.0_@types+node@22.12.0_yaml@2.7.0", - "npm:vite@*": "6.1.1_@types+node@22.12.0_yaml@2.7.0", - "npm:vite@6.1.1": "6.1.1_@types+node@22.12.0_yaml@2.7.0", - "npm:vitepress@*": "1.6.3_vite@5.4.14_vue@3.5.13_focus-trap@7.6.4_@types+react@18.3.18_react@18.3.1_react-dom@18.3.1__react@18.3.1", - "npm:vitepress@1.6.3": "1.6.3_vite@5.4.14_vue@3.5.13_focus-trap@7.6.4_@types+react@18.3.18_react@18.3.1_react-dom@18.3.1__react@18.3.1", + "npm:vite-plugin-wasm@3.4.1": "3.4.1_vite@6.1.1", + "npm:vite@*": "6.1.1_yaml@2.7.0", + "npm:vite@6.1.1": "6.1.1_yaml@2.7.0", + "npm:vitepress@*": "1.6.3_vite@5.4.14__@types+node@22.12.0_vue@3.5.13_focus-trap@7.6.4_@types+node@22.12.0_@types+react@18.3.18_react@18.3.1_react-dom@18.3.1__react@18.3.1", + "npm:vitepress@1.6.3": "1.6.3_vite@5.4.14__@types+node@22.12.0_vue@3.5.13_focus-trap@7.6.4_@types+node@22.12.0_@types+react@18.3.18_react@18.3.1_react-dom@18.3.1__react@18.3.1", "npm:vue@3.5.13": "3.5.13", "npm:yaml@2.7.0": "2.7.0" }, @@ -109,7 +94,7 @@ "dependencies": [ "jsr:@cliffy/internal", "jsr:@std/encoding@~1.0.5", - "jsr:@std/io@~0.224.9" + "jsr:@std/io" ] }, "@cliffy/internal@1.0.0-rc.7": { @@ -131,9 +116,9 @@ "jsr:@cliffy/ansi", "jsr:@cliffy/internal", "jsr:@cliffy/keycode", - "jsr:@std/assert@~1.0.6", - "jsr:@std/fmt@~1.0.2", - "jsr:@std/io@~0.224.9", + "jsr:@std/assert", + "jsr:@std/fmt", + "jsr:@std/io", "jsr:@std/path@~1.0.6", "jsr:@std/text" ] @@ -141,53 +126,15 @@ "@cliffy/table@1.0.0-rc.7": { "integrity": "9fdd9776eda28a0b397981c400eeb1aa36da2371b43eefe12e6ff555290e3180", "dependencies": [ - "jsr:@std/fmt@~1.0.2" + "jsr:@std/fmt" ] }, - "@deno/cache-dir@0.13.2": { - "integrity": "c22419dfe27ab85f345bee487aaaadba498b005cce3644e9d2528db035c5454d", - "dependencies": [ - "jsr:@deno/graph", - "jsr:@std/fmt@0.223", - "jsr:@std/fs@0.223", - "jsr:@std/io@0.223", - "jsr:@std/path@0.223" - ] - }, - "@deno/emit@0.46.0": { - "integrity": "e276be2c77bac1b93caf775762e2a49a54cb00da2d48ca2b01ed8d7cba9d082c", - "dependencies": [ - "jsr:@deno/cache-dir", - "jsr:@std/path@0.223" - ] - }, - "@deno/graph@0.73.1": { - "integrity": "cd69639d2709d479037d5ce191a422eabe8d71bb68b0098344f6b07411c84d41" - }, - "@std/assert@0.223.0": { - "integrity": "eb8d6d879d76e1cc431205bd346ed4d88dc051c6366365b1af47034b0670be24" - }, "@std/assert@1.0.11": { - "integrity": "2461ef3c368fe88bc60e186e7744a93112f16fd110022e113a0849e94d1c83c1", - "dependencies": [ - "jsr:@std/internal" - ] + "integrity": "2461ef3c368fe88bc60e186e7744a93112f16fd110022e113a0849e94d1c83c1" }, "@std/async@1.0.10": { "integrity": "2ff1b1c7d33d1416159989b0f69e59ec7ee8cb58510df01e454def2108b3dbec" }, - "@std/bytes@0.223.0": { - "integrity": "84b75052cd8680942c397c2631318772b295019098f40aac5c36cead4cba51a8" - }, - "@std/bytes@1.0.4": { - "integrity": "11a0debe522707c95c7b7ef89b478c13fb1583a7cfb9a85674cd2cc2e3a28abc" - }, - "@std/crypto@1.0.3": { - "integrity": "a2a32f51ddef632d299e3879cd027c630dcd4d1d9a5285d6e6788072f4e51e7f" - }, - "@std/data-structures@1.0.6": { - "integrity": "76a7fd8080c66604c0496220a791860492ab21a04a63a969c0b9a0609bbbb760" - }, "@std/datetime@0.225.3": { "integrity": "fda0b7791b43d9a46764986fec39922a8ceb70ffca642651fa7d0683501a171a" }, @@ -197,20 +144,8 @@ "@std/encoding@1.0.7": { "integrity": "f631247c1698fef289f2de9e2a33d571e46133b38d042905e3eac3715030a82d" }, - "@std/fmt@0.223.0": { - "integrity": "6deb37794127dfc7d7bded2586b9fc6f5d50e62a8134846608baf71ffc1a5208" - }, - "@std/fmt@1.0.5": { - "integrity": "0cfab43364bc36650d83c425cd6d99910fc20c4576631149f0f987eddede1a4d" - }, - "@std/fs@0.223.0": { - "integrity": "3b4b0550b2c524cbaaa5a9170c90e96cbb7354e837ad1bdaf15fc9df1ae9c31c" - }, - "@std/fs@1.0.11": { - "integrity": "ba674672693340c5ebdd018b4fe1af46cb08741f42b4c538154e97d217b55bdd", - "dependencies": [ - "jsr:@std/path@^1.0.8" - ] + "@std/fmt@1.0.6": { + "integrity": "a2c56a69a2369876ddb3ad6a500bb6501b5bad47bb3ea16bfb0c18974d2661fc" }, "@std/fs@1.0.13": { "integrity": "756d3ff0ade91c9e72b228e8012b6ff00c3d4a4ac9c642c4dac083536bf6c605", @@ -218,68 +153,35 @@ "jsr:@std/path@^1.0.8" ] }, - "@std/internal@1.0.5": { - "integrity": "54a546004f769c1ac9e025abd15a76b6671ddc9687e2313b67376125650dc7ba" - }, - "@std/io@0.223.0": { - "integrity": "2d8c3c2ab3a515619b90da2c6ff5ea7b75a94383259ef4d02116b228393f84f1", - "dependencies": [ - "jsr:@std/assert@0.223", - "jsr:@std/bytes@0.223" - ] - }, "@std/io@0.224.9": { "integrity": "4414664b6926f665102e73c969cfda06d2c4c59bd5d0c603fd4f1b1c840d6ee3" }, - "@std/path@0.223.0": { - "integrity": "593963402d7e6597f5a6e620931661053572c982fc014000459edc1f93cc3989", - "dependencies": [ - "jsr:@std/assert@0.223" - ] - }, "@std/path@1.0.8": { "integrity": "548fa456bb6a04d3c1a1e7477986b6cffbce95102d0bb447c67c4ee70e0364be" }, "@std/semver@1.0.4": { "integrity": "a62af791917d8fd6c48d6ebbb872f83fad3fc6671ffadbbd39ea229c2d34d175" }, - "@std/testing@1.0.9": { - "integrity": "9bdd4ac07cb13e7594ac30e90f6ceef7254ac83a9aeaa089be0008f33aab5cd4", - "dependencies": [ - "jsr:@std/assert@^1.0.10", - "jsr:@std/data-structures", - "jsr:@std/fs@^1.0.9", - "jsr:@std/internal", - "jsr:@std/path@^1.0.8" - ] - }, - "@std/text@1.0.10": { - "integrity": "9dcab377450253c0efa9a9a0c731040bfd4e1c03f8303b5934381467b7954338" - }, - "@std/uuid@1.0.4": { - "integrity": "f4233149cc8b4753cc3763fd83a7c4101699491f55c7be78dc7b30281946d7a0", - "dependencies": [ - "jsr:@std/bytes@^1.0.2", - "jsr:@std/crypto" - ] + "@std/text@1.0.11": { + "integrity": "f191fa22590cac8b1cdba6cc4ab97940e720f7cc67b3084e54405b428bf5843d" } }, "npm": { - "@algolia/autocomplete-core@1.17.7_algoliasearch@5.20.3": { + "@algolia/autocomplete-core@1.17.7_algoliasearch@5.21.0": { "integrity": "sha512-BjiPOW6ks90UKl7TwMv7oNQMnzU+t/wk9mgIDi6b1tXpUek7MW0lbNOUHpvam9pe3lVCf4xPFT+lK7s+e+fs7Q==", "dependencies": [ "@algolia/autocomplete-plugin-algolia-insights", "@algolia/autocomplete-shared" ] }, - "@algolia/autocomplete-plugin-algolia-insights@1.17.7_search-insights@2.17.3_algoliasearch@5.20.3": { + "@algolia/autocomplete-plugin-algolia-insights@1.17.7_search-insights@2.17.3_algoliasearch@5.21.0": { "integrity": "sha512-Jca5Ude6yUOuyzjnz57og7Et3aXjbwCSDf/8onLHSQgw1qW3ALl9mrMWaXb5FmPVkV3EtkD2F/+NkT6VHyPu9A==", "dependencies": [ "@algolia/autocomplete-shared", "search-insights" ] }, - "@algolia/autocomplete-preset-algolia@1.17.7_@algolia+client-search@5.20.3_algoliasearch@5.20.3": { + "@algolia/autocomplete-preset-algolia@1.17.7_@algolia+client-search@5.21.0_algoliasearch@5.21.0": { "integrity": "sha512-ggOQ950+nwbWROq2MOCIL71RE0DdQZsceqrg32UqnhDz8FlO9rL8ONHNsI2R1MH0tkgVIDKI/D0sMiUchsFdWA==", "dependencies": [ "@algolia/autocomplete-shared", @@ -287,15 +189,15 @@ "algoliasearch" ] }, - "@algolia/autocomplete-shared@1.17.7_@algolia+client-search@5.20.3_algoliasearch@5.20.3": { + "@algolia/autocomplete-shared@1.17.7_@algolia+client-search@5.21.0_algoliasearch@5.21.0": { "integrity": "sha512-o/1Vurr42U/qskRSuhBH+VKxMvkkUVTLU6WZQr+L5lGZZLYWyhdzWjW0iGXY7EkwRTjBqvN2EsR81yCTGV/kmg==", "dependencies": [ "@algolia/client-search", "algoliasearch" ] }, - "@algolia/client-abtesting@5.20.3": { - "integrity": "sha512-wPOzHYSsW+H97JkBLmnlOdJSpbb9mIiuNPycUCV5DgzSkJFaI/OFxXfZXAh1gqxK+hf0miKue1C9bltjWljrNA==", + "@algolia/client-abtesting@5.21.0": { + "integrity": "sha512-I239aSmXa3pXDhp3AWGaIfesqJBNFA7drUM8SIfNxMIzvQXUnHRf4rW1o77QXLI/nIClNsb8KOLaB62gO9LnlQ==", "dependencies": [ "@algolia/client-common", "@algolia/requester-browser-xhr", @@ -303,8 +205,8 @@ "@algolia/requester-node-http" ] }, - "@algolia/client-analytics@5.20.3": { - "integrity": "sha512-XE3iduH9lA7iTQacDGofBQyIyIgaX8qbTRRdj1bOCmfzc9b98CoiMwhNwdTifmmMewmN0EhVF3hP8KjKWwX7Yw==", + "@algolia/client-analytics@5.21.0": { + "integrity": "sha512-OxoUfeG9G4VE4gS7B4q65KkHzdGsQsDwxQfR5J9uKB8poSGuNlHJWsF3ABqCkc5VliAR0m8KMjsQ9o/kOpEGnQ==", "dependencies": [ "@algolia/client-common", "@algolia/requester-browser-xhr", @@ -312,11 +214,11 @@ "@algolia/requester-node-http" ] }, - "@algolia/client-common@5.20.3": { - "integrity": "sha512-IYRd/A/R3BXeaQVT2805lZEdWo54v39Lqa7ABOxIYnUvX2vvOMW1AyzCuT0U7Q+uPdD4UW48zksUKRixShcWxA==" + "@algolia/client-common@5.21.0": { + "integrity": "sha512-iHLgDQFyZNe9M16vipbx6FGOA8NoMswHrfom/QlCGoyh7ntjGvfMb+J2Ss8rRsAlOWluv8h923Ku3QVaB0oWDQ==" }, - "@algolia/client-insights@5.20.3": { - "integrity": "sha512-QGc/bmDUBgzB71rDL6kihI2e1Mx6G6PxYO5Ks84iL3tDcIel1aFuxtRF14P8saGgdIe1B6I6QkpkeIddZ6vWQw==", + "@algolia/client-insights@5.21.0": { + "integrity": "sha512-y7XBO9Iwb75FLDl95AYcWSLIViJTpR5SUUCyKsYhpP9DgyUqWbISqDLXc96TS9shj+H+7VsTKA9cJK8NUfVN6g==", "dependencies": [ "@algolia/client-common", "@algolia/requester-browser-xhr", @@ -324,8 +226,8 @@ "@algolia/requester-node-http" ] }, - "@algolia/client-personalization@5.20.3": { - "integrity": "sha512-zuM31VNPDJ1LBIwKbYGz/7+CSm+M8EhlljDamTg8AnDilnCpKjBebWZR5Tftv/FdWSro4tnYGOIz1AURQgZ+tQ==", + "@algolia/client-personalization@5.21.0": { + "integrity": "sha512-6KU658lD9Tss4oCX6c/O15tNZxw7vR+WAUG95YtZzYG/KGJHTpy2uckqbMmC2cEK4a86FAq4pH5azSJ7cGMjuw==", "dependencies": [ "@algolia/client-common", "@algolia/requester-browser-xhr", @@ -333,8 +235,8 @@ "@algolia/requester-node-http" ] }, - "@algolia/client-query-suggestions@5.20.3": { - "integrity": "sha512-Nn872PuOI8qzi1bxMMhJ0t2AzVBqN01jbymBQOkypvZHrrjZPso3iTpuuLLo9gi3yc/08vaaWTAwJfPhxPwJUw==", + "@algolia/client-query-suggestions@5.21.0": { + "integrity": "sha512-pG6MyVh1v0X+uwrKHn3U+suHdgJ2C+gug+UGkNHfMELHMsEoWIAQhxMBOFg7hCnWBFjQnuq6qhM3X9X5QO3d9Q==", "dependencies": [ "@algolia/client-common", "@algolia/requester-browser-xhr", @@ -342,8 +244,8 @@ "@algolia/requester-node-http" ] }, - "@algolia/client-search@5.20.3": { - "integrity": "sha512-9+Fm1ahV8/2goSIPIqZnVitV5yHW5E5xTdKy33xnqGd45A9yVv5tTkudWzEXsbfBB47j9Xb3uYPZjAvV5RHbKA==", + "@algolia/client-search@5.21.0": { + "integrity": "sha512-nZfgJH4njBK98tFCmCW1VX/ExH4bNOl9DSboxeXGgvhoL0fG1+4DDr/mrLe21OggVCQqHwXBMh6fFInvBeyhiQ==", "dependencies": [ "@algolia/client-common", "@algolia/requester-browser-xhr", @@ -351,8 +253,8 @@ "@algolia/requester-node-http" ] }, - "@algolia/ingestion@1.20.3": { - "integrity": "sha512-5GHNTiZ3saLjTNyr6WkP5hzDg2eFFAYWomvPcm9eHWskjzXt8R0IOiW9kkTS6I6hXBwN5H9Zna5mZDSqqJdg+g==", + "@algolia/ingestion@1.21.0": { + "integrity": "sha512-k6MZxLbZphGN5uRri9J/krQQBjUrqNcScPh985XXEFXbSCRvOPKVtjjLdVjGVHXXPOQgKrIZHxIdRNbHS+wVuA==", "dependencies": [ "@algolia/client-common", "@algolia/requester-browser-xhr", @@ -360,8 +262,8 @@ "@algolia/requester-node-http" ] }, - "@algolia/monitoring@1.20.3": { - "integrity": "sha512-KUWQbTPoRjP37ivXSQ1+lWMfaifCCMzTnEcEnXwAmherS5Tp7us6BAqQDMGOD4E7xyaS2I8pto6WlOzxH+CxmA==", + "@algolia/monitoring@1.21.0": { + "integrity": "sha512-FiW5nnmyHvaGdorqLClw3PM6keXexAMiwbwJ9xzQr4LcNefLG3ln82NafRPgJO/z0dETAOKjds5aSmEFMiITHQ==", "dependencies": [ "@algolia/client-common", "@algolia/requester-browser-xhr", @@ -369,8 +271,8 @@ "@algolia/requester-node-http" ] }, - "@algolia/recommend@5.20.3": { - "integrity": "sha512-oo/gG77xTTTclkrdFem0Kmx5+iSRFiwuRRdxZETDjwzCI7svutdbwBgV/Vy4D4QpYaX4nhY/P43k84uEowCE4Q==", + "@algolia/recommend@5.21.0": { + "integrity": "sha512-+JXavbbliaLmah5QNgc/TDW/+r0ALa+rGhg5Y7+pF6GpNnzO0L+nlUaDNE8QbiJfz54F9BkwFUnJJeRJAuzTFw==", "dependencies": [ "@algolia/client-common", "@algolia/requester-browser-xhr", @@ -378,20 +280,20 @@ "@algolia/requester-node-http" ] }, - "@algolia/requester-browser-xhr@5.20.3": { - "integrity": "sha512-BkkW7otbiI/Er1AiEPZs1h7lxbtSO9p09jFhv3/iT8/0Yz0CY79VJ9iq+Wv1+dq/l0OxnMpBy8mozrieGA3mXQ==", + "@algolia/requester-browser-xhr@5.21.0": { + "integrity": "sha512-Iw+Yj5hOmo/iixHS94vEAQ3zi5GPpJywhfxn1el/zWo4AvPIte/+1h9Ywgw/+3M7YBj4jgAkScxjxQCxzLBsjA==", "dependencies": [ "@algolia/client-common" ] }, - "@algolia/requester-fetch@5.20.3": { - "integrity": "sha512-eAVlXz7UNzTsA1EDr+p0nlIH7WFxo7k3NMxYe8p38DH8YVWLgm2MgOVFUMNg9HCi6ZNOi/A2w/id2ZZ4sKgUOw==", + "@algolia/requester-fetch@5.21.0": { + "integrity": "sha512-Z00SRLlIFj3SjYVfsd9Yd3kB3dUwQFAkQG18NunWP7cix2ezXpJqA+xAoEf9vc4QZHdxU3Gm8gHAtRiM2iVaTQ==", "dependencies": [ "@algolia/client-common" ] }, - "@algolia/requester-node-http@5.20.3": { - "integrity": "sha512-FqR3pQPfHfQyX1wgcdK6iyqu86yP76MZd4Pzj1y/YLMj9rRmRCY0E0AffKr//nrOFEwv6uY8BQY4fd9/6b0ZCg==", + "@algolia/requester-node-http@5.21.0": { + "integrity": "sha512-WqU0VumUILrIeVYCTGZlyyZoC/tbvhiyPxfGRRO1cSjxN558bnJLlR2BvS0SJ5b75dRNK7HDvtXo2QoP9eLfiA==", "dependencies": [ "@algolia/client-common" ] @@ -442,20 +344,20 @@ "@babel/helper-validator-identifier@7.25.9": { "integrity": "sha512-Ed61U6XJc3CVRfkERJWDz4dJwKe7iLmmJsbOGu9wSloNSFttHV0I8g6UAgb7qnK5ly5bGLPd4oXZlxCdANBOWQ==" }, - "@babel/parser@7.26.9": { - "integrity": "sha512-81NWa1njQblgZbQHxWHpxxCzNsa3ZwvFqpUg7P+NNUU6f3UU2jBEg4OlF/J6rl8+PQGh1q6/zWScd001YwcA5A==", + "@babel/parser@7.26.10": { + "integrity": "sha512-6aQR2zGE/QFi8JpDLjUZEPYOs7+mhKXm86VaKFiLP35JQwQb6bwUE+XbvkH0EptsYhbNBSUGaUBLKqxH1xSgsA==", "dependencies": [ "@babel/types" ] }, - "@babel/runtime@7.26.9": { - "integrity": "sha512-aA63XwOkcl4xxQa3HjPMqOP6LiK0ZDv3mUPYEFXkpHbaFjtGggE1A61FjFzJnB+p7/oy2gA8E+rcBNl/zC1tMg==", + "@babel/runtime@7.26.10": { + "integrity": "sha512-2WJMeRQPHKSPemqk/awGrAiuFfzBmOIPXKizAsVhWH9YJqLZ0H+HS4c8loHGgW6utJ3E/ejXQUsiGaQy2NZ9Fw==", "dependencies": [ "regenerator-runtime" ] }, - "@babel/types@7.26.9": { - "integrity": "sha512-Y3IR1cRnOxOCDvMmNiym7XpXQ93iGDDPHx+Zj+NM+rg0fBaShfQLkg+hKPaZCEvg5N/LeCo4+Rj/i3FuJsIQaw==", + "@babel/types@7.26.10": { + "integrity": "sha512-emqcG3vHrpxUKTrxcblR36dcrcoRDvKmnL/dCL6ZsHaShW80qxCAcNhzQZrpeM765VzEos+xOi4s+r4IXzTwdQ==", "dependencies": [ "@babel/helper-string-parser", "@babel/helper-validator-identifier" @@ -485,6 +387,12 @@ "@jridgewell/trace-mapping" ] }, + "@deno/vite-plugin@1.0.4_vite@6.1.1": { + "integrity": "sha512-xg8YT8Wn2sGXSnJgiGTpBGX1Dov0c6fd1rAp8VsfrCUtyBRRWzwVMAnd3fQ4yq8h7LSVvJUxEFN4U421k/DQLA==", + "dependencies": [ + "vite@6.1.1" + ] + }, "@docsearch/css@3.8.2": { "integrity": "sha512-y05ayQFyUmCXze79+56v/4HpycYF3uFqB78pLPrSV5ZKAlDuIAAJNhaRi8tTdRNXh05yxX/TyNnzD6LwSM89vQ==" }, @@ -495,7 +403,7 @@ "preact" ] }, - "@docsearch/react@3.8.2_@types+react@18.3.18_react@18.3.1_react-dom@18.3.1__react@18.3.1_algoliasearch@5.20.3": { + "@docsearch/react@3.8.2_@types+react@18.3.18_react@18.3.1_react-dom@18.3.1__react@18.3.1_algoliasearch@5.21.0": { "integrity": "sha512-xCRrJQlTt8N9GU0DG4ptwHRkfnSnD/YpdeaXe02iKfqs97TkZJv60yE+1eq/tjPcVnTW8dP5qLP7itifFVV5eg==", "dependencies": [ "@algolia/autocomplete-core", @@ -667,8 +575,8 @@ "@floating-ui/utils@0.2.9": { "integrity": "sha512-MDWhGtE+eHw5JW7lq4qhc5yRLS11ERl1c7Z6Xd0a58DozHES6EnNNwUWbMiG4J9Cgj053Bhk8zvlhFYKVhULwg==" }, - "@iconify-json/simple-icons@1.2.25": { - "integrity": "sha512-2E1/gOCO97rF6usfhhiXxwzCb+UhdEsxW3lW1Sew+xZY0COY6dp82Z/r1rUt2fWKneWjuoGcNeJHHXQyG8mIuw==", + "@iconify-json/simple-icons@1.2.29": { + "integrity": "sha512-KYrxmxtRz6iOAulRiUsIBMUuXek+H+Evwf8UvYPIkbQ+KDoOqTegHx3q/w3GDDVC0qJYB+D3hXPMZcpm78qIuA==", "dependencies": [ "@iconify/types" ] @@ -727,12 +635,6 @@ "@jsr/std__encoding@1.0.7": { "integrity": "sha512-eySTZkCAHjiKGgOMvhUyTL4aT22svaCO0eO6CAbz48kRECMtRGEfe3KWXkVy4fhlZ/4OTw/RQLrTYyEqpGQp/Q==" }, - "@jsr/std__fs@1.0.11": { - "integrity": "sha512-Ynnw/D9hhsgeooOY4ak5oPgm9P7kXK/L2wNyWkDMfN7Pe5c9jtg2REkANhQU7axpp4Sct39/2/fayXP5FIfTcg==", - "dependencies": [ - "@jsr/std__path" - ] - }, "@jsr/std__fs@1.0.13": { "integrity": "sha512-3hvGsQwDd0ljtwaqG2psox4W0/t8vf0U1MQi6qczWoqajhZ3kgfRv7YexIZHCgWn7IP7nlRbARwhKbStJP7PmQ==", "dependencies": [ @@ -754,7 +656,7 @@ "@jsr/std__assert", "@jsr/std__async", "@jsr/std__data-structures", - "@jsr/std__fs@1.0.11", + "@jsr/std__fs", "@jsr/std__internal", "@jsr/std__path" ] @@ -779,117 +681,71 @@ "@pkgjs/parseargs@0.11.0": { "integrity": "sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==" }, - "@rollup/rollup-android-arm-eabi@4.34.6": { - "integrity": "sha512-+GcCXtOQoWuC7hhX1P00LqjjIiS/iOouHXhMdiDSnq/1DGTox4SpUvO52Xm+div6+106r+TcvOeo/cxvyEyTgg==" + "@preact/signals-core@1.8.0": { + "integrity": "sha512-OBvUsRZqNmjzCZXWLxkZfhcgT+Fk8DDcT/8vD6a1xhDemodyy87UJRJfASMuSD8FaAIeGgGm85ydXhm7lr4fyA==" + }, + "@preact/signals-react@3.0.1_react@18.3.1": { + "integrity": "sha512-HkM5Q2IsETO1M0bUzy4JB0EPQCf99SMbWP9K6GYlYVHfOX1HLKJ6Dl9L1/1rQnmrQpUsTw0R+IJQDh6tYWar2g==", + "dependencies": [ + "@preact/signals-core", + "react", + "use-sync-external-store" + ] }, "@rollup/rollup-android-arm-eabi@4.34.8": { "integrity": "sha512-q217OSE8DTp8AFHuNHXo0Y86e1wtlfVrXiAlwkIvGRQv9zbc6mE3sjIVfwI8sYUyNxwOg0j/Vm1RKM04JcWLJw==" }, - "@rollup/rollup-android-arm64@4.34.6": { - "integrity": "sha512-E8+2qCIjciYUnCa1AiVF1BkRgqIGW9KzJeesQqVfyRITGQN+dFuoivO0hnro1DjT74wXLRZ7QF8MIbz+luGaJA==" - }, "@rollup/rollup-android-arm64@4.34.8": { "integrity": "sha512-Gigjz7mNWaOL9wCggvoK3jEIUUbGul656opstjaUSGC3eT0BM7PofdAJaBfPFWWkXNVAXbaQtC99OCg4sJv70Q==" }, - "@rollup/rollup-darwin-arm64@4.34.6": { - "integrity": "sha512-z9Ib+OzqN3DZEjX7PDQMHEhtF+t6Mi2z/ueChQPLS/qUMKY7Ybn5A2ggFoKRNRh1q1T03YTQfBTQCJZiepESAg==" - }, "@rollup/rollup-darwin-arm64@4.34.8": { "integrity": "sha512-02rVdZ5tgdUNRxIUrFdcMBZQoaPMrxtwSb+/hOfBdqkatYHR3lZ2A2EGyHq2sGOd0Owk80oV3snlDASC24He3Q==" }, - "@rollup/rollup-darwin-x64@4.34.6": { - "integrity": "sha512-PShKVY4u0FDAR7jskyFIYVyHEPCPnIQY8s5OcXkdU8mz3Y7eXDJPdyM/ZWjkYdR2m0izD9HHWA8sGcXn+Qrsyg==" - }, "@rollup/rollup-darwin-x64@4.34.8": { "integrity": "sha512-qIP/elwR/tq/dYRx3lgwK31jkZvMiD6qUtOycLhTzCvrjbZ3LjQnEM9rNhSGpbLXVJYQ3rq39A6Re0h9tU2ynw==" }, - "@rollup/rollup-freebsd-arm64@4.34.6": { - "integrity": "sha512-YSwyOqlDAdKqs0iKuqvRHLN4SrD2TiswfoLfvYXseKbL47ht1grQpq46MSiQAx6rQEN8o8URtpXARCpqabqxGQ==" - }, "@rollup/rollup-freebsd-arm64@4.34.8": { "integrity": "sha512-IQNVXL9iY6NniYbTaOKdrlVP3XIqazBgJOVkddzJlqnCpRi/yAeSOa8PLcECFSQochzqApIOE1GHNu3pCz+BDA==" }, - "@rollup/rollup-freebsd-x64@4.34.6": { - "integrity": "sha512-HEP4CgPAY1RxXwwL5sPFv6BBM3tVeLnshF03HMhJYCNc6kvSqBgTMmsEjb72RkZBAWIqiPUyF1JpEBv5XT9wKQ==" - }, "@rollup/rollup-freebsd-x64@4.34.8": { "integrity": "sha512-TYXcHghgnCqYFiE3FT5QwXtOZqDj5GmaFNTNt3jNC+vh22dc/ukG2cG+pi75QO4kACohZzidsq7yKTKwq/Jq7Q==" }, - "@rollup/rollup-linux-arm-gnueabihf@4.34.6": { - "integrity": "sha512-88fSzjC5xeH9S2Vg3rPgXJULkHcLYMkh8faix8DX4h4TIAL65ekwuQMA/g2CXq8W+NJC43V6fUpYZNjaX3+IIg==" - }, "@rollup/rollup-linux-arm-gnueabihf@4.34.8": { "integrity": "sha512-A4iphFGNkWRd+5m3VIGuqHnG3MVnqKe7Al57u9mwgbyZ2/xF9Jio72MaY7xxh+Y87VAHmGQr73qoKL9HPbXj1g==" }, - "@rollup/rollup-linux-arm-musleabihf@4.34.6": { - "integrity": "sha512-wM4ztnutBqYFyvNeR7Av+reWI/enK9tDOTKNF+6Kk2Q96k9bwhDDOlnCUNRPvromlVXo04riSliMBs/Z7RteEg==" - }, "@rollup/rollup-linux-arm-musleabihf@4.34.8": { "integrity": "sha512-S0lqKLfTm5u+QTxlFiAnb2J/2dgQqRy/XvziPtDd1rKZFXHTyYLoVL58M/XFwDI01AQCDIevGLbQrMAtdyanpA==" }, - "@rollup/rollup-linux-arm64-gnu@4.34.6": { - "integrity": "sha512-9RyprECbRa9zEjXLtvvshhw4CMrRa3K+0wcp3KME0zmBe1ILmvcVHnypZ/aIDXpRyfhSYSuN4EPdCCj5Du8FIA==" - }, "@rollup/rollup-linux-arm64-gnu@4.34.8": { "integrity": "sha512-jpz9YOuPiSkL4G4pqKrus0pn9aYwpImGkosRKwNi+sJSkz+WU3anZe6hi73StLOQdfXYXC7hUfsQlTnjMd3s1A==" }, - "@rollup/rollup-linux-arm64-musl@4.34.6": { - "integrity": "sha512-qTmklhCTyaJSB05S+iSovfo++EwnIEZxHkzv5dep4qoszUMX5Ca4WM4zAVUMbfdviLgCSQOu5oU8YoGk1s6M9Q==" - }, "@rollup/rollup-linux-arm64-musl@4.34.8": { "integrity": "sha512-KdSfaROOUJXgTVxJNAZ3KwkRc5nggDk+06P6lgi1HLv1hskgvxHUKZ4xtwHkVYJ1Rep4GNo+uEfycCRRxht7+Q==" }, - "@rollup/rollup-linux-loongarch64-gnu@4.34.6": { - "integrity": "sha512-4Qmkaps9yqmpjY5pvpkfOerYgKNUGzQpFxV6rnS7c/JfYbDSU0y6WpbbredB5cCpLFGJEqYX40WUmxMkwhWCjw==" - }, "@rollup/rollup-linux-loongarch64-gnu@4.34.8": { "integrity": "sha512-NyF4gcxwkMFRjgXBM6g2lkT58OWztZvw5KkV2K0qqSnUEqCVcqdh2jN4gQrTn/YUpAcNKyFHfoOZEer9nwo6uQ==" }, - "@rollup/rollup-linux-powerpc64le-gnu@4.34.6": { - "integrity": "sha512-Zsrtux3PuaxuBTX/zHdLaFmcofWGzaWW1scwLU3ZbW/X+hSsFbz9wDIp6XvnT7pzYRl9MezWqEqKy7ssmDEnuQ==" - }, "@rollup/rollup-linux-powerpc64le-gnu@4.34.8": { "integrity": "sha512-LMJc999GkhGvktHU85zNTDImZVUCJ1z/MbAJTnviiWmmjyckP5aQsHtcujMjpNdMZPT2rQEDBlJfubhs3jsMfw==" }, - "@rollup/rollup-linux-riscv64-gnu@4.34.6": { - "integrity": "sha512-aK+Zp+CRM55iPrlyKiU3/zyhgzWBxLVrw2mwiQSYJRobCURb781+XstzvA8Gkjg/hbdQFuDw44aUOxVQFycrAg==" - }, "@rollup/rollup-linux-riscv64-gnu@4.34.8": { "integrity": "sha512-xAQCAHPj8nJq1PI3z8CIZzXuXCstquz7cIOL73HHdXiRcKk8Ywwqtx2wrIy23EcTn4aZ2fLJNBB8d0tQENPCmw==" }, - "@rollup/rollup-linux-s390x-gnu@4.34.6": { - "integrity": "sha512-WoKLVrY9ogmaYPXwTH326+ErlCIgMmsoRSx6bO+l68YgJnlOXhygDYSZe/qbUJCSiCiZAQ+tKm88NcWuUXqOzw==" - }, "@rollup/rollup-linux-s390x-gnu@4.34.8": { "integrity": "sha512-DdePVk1NDEuc3fOe3dPPTb+rjMtuFw89gw6gVWxQFAuEqqSdDKnrwzZHrUYdac7A7dXl9Q2Vflxpme15gUWQFA==" }, - "@rollup/rollup-linux-x64-gnu@4.34.6": { - "integrity": "sha512-Sht4aFvmA4ToHd2vFzwMFaQCiYm2lDFho5rPcvPBT5pCdC+GwHG6CMch4GQfmWTQ1SwRKS0dhDYb54khSrjDWw==" - }, "@rollup/rollup-linux-x64-gnu@4.34.8": { "integrity": "sha512-8y7ED8gjxITUltTUEJLQdgpbPh1sUQ0kMTmufRF/Ns5tI9TNMNlhWtmPKKHCU0SilX+3MJkZ0zERYYGIVBYHIA==" }, - "@rollup/rollup-linux-x64-musl@4.34.6": { - "integrity": "sha512-zmmpOQh8vXc2QITsnCiODCDGXFC8LMi64+/oPpPx5qz3pqv0s6x46ps4xoycfUiVZps5PFn1gksZzo4RGTKT+A==" - }, "@rollup/rollup-linux-x64-musl@4.34.8": { "integrity": "sha512-SCXcP0ZpGFIe7Ge+McxY5zKxiEI5ra+GT3QRxL0pMMtxPfpyLAKleZODi1zdRHkz5/BhueUrYtYVgubqe9JBNQ==" }, - "@rollup/rollup-win32-arm64-msvc@4.34.6": { - "integrity": "sha512-3/q1qUsO/tLqGBaD4uXsB6coVGB3usxw3qyeVb59aArCgedSF66MPdgRStUd7vbZOsko/CgVaY5fo2vkvPLWiA==" - }, "@rollup/rollup-win32-arm64-msvc@4.34.8": { "integrity": "sha512-YHYsgzZgFJzTRbth4h7Or0m5O74Yda+hLin0irAIobkLQFRQd1qWmnoVfwmKm9TXIZVAD0nZ+GEb2ICicLyCnQ==" }, - "@rollup/rollup-win32-ia32-msvc@4.34.6": { - "integrity": "sha512-oLHxuyywc6efdKVTxvc0135zPrRdtYVjtVD5GUm55I3ODxhU/PwkQFD97z16Xzxa1Fz0AEe4W/2hzRtd+IfpOA==" - }, "@rollup/rollup-win32-ia32-msvc@4.34.8": { "integrity": "sha512-r3NRQrXkHr4uWy5TOjTpTYojR9XmF0j/RYgKCef+Ag46FWUTltm5ziticv8LdNsDMehjJ543x/+TJAek/xBA2w==" }, - "@rollup/rollup-win32-x64-msvc@4.34.6": { - "integrity": "sha512-0PVwmgzZ8+TZ9oGBmdZoQVXflbvuwzN/HRclujpl4N/q3i+y0lqLw8n1bXA8ru3sApDjlmONaNAuYr38y1Kr9w==" - }, "@rollup/rollup-win32-x64-msvc@4.34.8": { "integrity": "sha512-U0FaE5O1BCpZSeE6gBl3c5ObhePQSfk9vDRToMmTkbhCOgW4jqvtS5LGyQ76L1fH8sM0keRp4uDTsbjiUyjk0g==" }, @@ -948,38 +804,41 @@ "@shikijs/vscode-textmate@10.0.2": { "integrity": "sha512-83yeghZ2xxin3Nj8z1NMd/NCuca+gsYXswywDy5bHvwlWL8tpTQmzGeUuHd9FC3E/SBEMvzJRwWEOz5gGes9Qg==" }, - "@swc/core-darwin-arm64@1.10.18": { - "integrity": "sha512-FdGqzAIKVQJu8ROlnHElP59XAUsUzCFSNsou+tY/9ba+lhu8R9v0OI5wXiPErrKGZpQFMmx/BPqqhx3X4SuGNg==" + "@svgdotjs/svg.js@3.2.4": { + "integrity": "sha512-BjJ/7vWNowlX3Z8O4ywT58DqbNRyYlkk6Yz/D13aB7hGmfQTvGX4Tkgtm/ApYlu9M7lCQi15xUEidqMUmdMYwg==" }, - "@swc/core-darwin-x64@1.10.18": { - "integrity": "sha512-RZ73gZRituL/ZVLgrW6BYnQ5g8tuStG4cLUiPGJsUZpUm0ullSH6lHFvZTCBNFTfpQChG6eEhi2IdG6DwFp1lw==" + "@swc/core-darwin-arm64@1.11.11": { + "integrity": "sha512-vJcjGVDB8cZH7zyOkC0AfpFYI/7GHKG0NSsH3tpuKrmoAXJyCYspKPGid7FT53EAlWreN7+Pew+bukYf5j+Fmg==" }, - "@swc/core-linux-arm-gnueabihf@1.10.18": { - "integrity": "sha512-8iJqI3EkxJuuq21UHoen1VS+QlS23RvynRuk95K+Q2HBjygetztCGGEc+Xelx9a0uPkDaaAtFvds4JMDqb9SAA==" + "@swc/core-darwin-x64@1.11.11": { + "integrity": "sha512-/N4dGdqEYvD48mCF3QBSycAbbQd3yoZ2YHSzYesQf8usNc2YpIhYqEH3sql02UsxTjEFOJSf1bxZABDdhbSl6A==" }, - "@swc/core-linux-arm64-gnu@1.10.18": { - "integrity": "sha512-8f1kSktWzMB6PG+r8lOlCfXz5E8Qhsmfwonn77T/OfjvGwQaWrcoASh2cdjpk3dydbf8jsKGPQE1lSc7GyjXRQ==" + "@swc/core-linux-arm-gnueabihf@1.11.11": { + "integrity": "sha512-hsBhKK+wVXdN3x9MrL5GW0yT8o9GxteE5zHAI2HJjRQel3HtW7m5Nvwaq+q8rwMf4YQRd8ydbvwl4iUOZx7i2Q==" }, - "@swc/core-linux-arm64-musl@1.10.18": { - "integrity": "sha512-4rv+E4VLdgQw6zjbTAauCAEExxChvxMpBUMCiZweTNPKbJJ2dY6BX2WGJ1ea8+RcgqR/Xysj3AFbOz1LBz6dGA==" + "@swc/core-linux-arm64-gnu@1.11.11": { + "integrity": "sha512-YOCdxsqbnn/HMPCNM6nrXUpSndLXMUssGTtzT7ffXqr7WuzRg2e170FVDVQFIkb08E7Ku5uOnnUVAChAJQbMOQ==" }, - "@swc/core-linux-x64-gnu@1.10.18": { - "integrity": "sha512-vTNmyRBVP+sZca+vtwygYPGTNudTU6Gl6XhaZZ7cEUTBr8xvSTgEmYXoK/2uzyXpaTUI4Bmtp1x81cGN0mMoLQ==" + "@swc/core-linux-arm64-musl@1.11.11": { + "integrity": "sha512-nR2tfdQRRzwqR2XYw9NnBk9Fdvff/b8IiJzDL28gRR2QiJWLaE8LsRovtWrzCOYq6o5Uu9cJ3WbabWthLo4jLw==" }, - "@swc/core-linux-x64-musl@1.10.18": { - "integrity": "sha512-1TZPReKhFCeX776XaT6wegknfg+g3zODve+r4oslFHI+g7cInfWlxoGNDS3niPKyuafgCdOjme2g3OF+zzxfsQ==" + "@swc/core-linux-x64-gnu@1.11.11": { + "integrity": "sha512-b4gBp5HA9xNWNC5gsYbdzGBJWx4vKSGybGMGOVWWuF+ynx10+0sA/o4XJGuNHm8TEDuNh9YLKf6QkIO8+GPJ1g==" }, - "@swc/core-win32-arm64-msvc@1.10.18": { - "integrity": "sha512-o/2CsaWSN3bkzVQ6DA+BiFKSVEYvhWGA1h+wnL2zWmIDs2Knag54sOEXZkCaf8YQyZesGeXJtPEy9hh/vjJgkA==" + "@swc/core-linux-x64-musl@1.11.11": { + "integrity": "sha512-dEvqmQVswjNvMBwXNb8q5uSvhWrJLdttBSef3s6UC5oDSwOr00t3RQPzyS3n5qmGJ8UMTdPRmsopxmqaODISdg==" }, - "@swc/core-win32-ia32-msvc@1.10.18": { - "integrity": "sha512-eTPASeJtk4mJDfWiYEiOC6OYUi/N7meHbNHcU8e+aKABonhXrIo/FmnTE8vsUtC6+jakT1TQBdiQ8fzJ1kJVwA==" + "@swc/core-win32-arm64-msvc@1.11.11": { + "integrity": "sha512-aZNZznem9WRnw2FbTqVpnclvl8Q2apOBW2B316gZK+qxbe+ktjOUnYaMhdCG3+BYggyIBDOnaJeQrXbKIMmNdw==" }, - "@swc/core-win32-x64-msvc@1.10.18": { - "integrity": "sha512-1Dud8CDBnc34wkBOboFBQud9YlV1bcIQtKSg7zC8LtwR3h+XAaCayZPkpGmmAlCv1DLQPvkF+s0JcaVC9mfffQ==" + "@swc/core-win32-ia32-msvc@1.11.11": { + "integrity": "sha512-DjeJn/IfjgOddmJ8IBbWuDK53Fqw7UvOz7kyI/728CSdDYC3LXigzj3ZYs4VvyeOt+ZcQZUB2HA27edOifomGw==" }, - "@swc/core@1.10.18": { - "integrity": "sha512-IUWKD6uQYGRy8w2X9EZrtYg1O3SCijlHbCXzMaHQYc1X7yjijQh4H3IVL9ssZZyVp2ZDfQZu4bD5DWxxvpyjvg==", + "@swc/core-win32-x64-msvc@1.11.11": { + "integrity": "sha512-Gp/SLoeMtsU4n0uRoKDOlGrRC6wCfifq7bqLwSlAG8u8MyJYJCcwjg7ggm0rhLdC2vbiZ+lLVl3kkETp+JUvKg==" + }, + "@swc/core@1.11.11": { + "integrity": "sha512-pCVY2Wn6dV/labNvssk9b3Owi4WOYsapcbWm90XkIj4xH/56Z6gzja9fsU+4MdPuEfC2Smw835nZHcdCFGyX6A==", "dependencies": [ "@swc/core-darwin-arm64", "@swc/core-darwin-x64", @@ -998,8 +857,8 @@ "@swc/counter@0.1.3": { "integrity": "sha512-e2BR4lsJkkRlKZ/qCHPw9ZaSxc0MVUd7gtbtaB7aMvHeJVYe8sOB8DBZkP2DtISHGSku9sCK6T6cnY0CtXrOCQ==" }, - "@swc/types@0.1.17": { - "integrity": "sha512-V5gRru+aD8YVyCOMAjMpWR1Ui577DD5KSJsHP8RAxopAH22jFz6GZd/qxqjO6MJHQhcsjvjOFXyDhyLQUnMveQ==", + "@swc/types@0.1.19": { + "integrity": "sha512-WkAZaAfj44kh/UFdAQcrMP1I0nwRqpt27u+08LMBYMqmQfwwMofYoMh/48NGkMMRfC4ynpfwRbJuu8ErfNloeA==", "dependencies": [ "@swc/counter" ] @@ -1054,8 +913,8 @@ "form-data" ] }, - "@types/node@18.19.76": { - "integrity": "sha512-yvR7Q9LdPz2vGpmpJX5LolrgRdWvB67MJKDPSgIIzpFbaf9a1j/f5DnLp5VDyHGMR0QZHlTr1afsD87QCXFHKw==", + "@types/node@18.19.80": { + "integrity": "sha512-kEWeMwMeIvxYkeg1gTc01awpwLbfMRZXdIhwRcakd/KlK53jmRC26LqcbIt7fnAQTu5GzlnWmzA3H6+l1u6xxQ==", "dependencies": [ "undici-types@5.26.5" ] @@ -1085,8 +944,8 @@ "@types/unist@3.0.3": { "integrity": "sha512-ko/gIFJRv177XgZsZcBwnqJN5x/Gien8qNOn0D5bQU/zAzVf9Zt3BlcUiLqhV9y4ARk0GbT3tnUiPNgnTXzc/Q==" }, - "@types/web-bluetooth@0.0.20": { - "integrity": "sha512-g9gZnnXVq7gM7v3tJCWV/qw7w+KeOlSHAhgF9RytFyifW6AF61hdT2ucrYhPq9hLs5JIryeupHV3qGk95dH9ow==" + "@types/web-bluetooth@0.0.21": { + "integrity": "sha512-oIQLCGWtcFZy2JW77j9k8nHzAOpqMHLQejDA48XXMWH6tjCQHz5RCFz1bzsmROyL6PUm+LLnUiI4BCn221inxA==" }, "@ungap/structured-clone@1.3.0": { "integrity": "sha512-WmoN8qaIAo7WTYWbAZuG8PYEhn5fkz7dZrqTBZ7dtt//lL2Gwms1IcnQ5yHqjDfX8Ft5j4YzDM23f87zBfDe9g==" @@ -1100,15 +959,22 @@ "react" ] }, - "@vitejs/plugin-react-swc@3.8.0_vite@6.1.1__@types+node@22.12.0__yaml@2.7.0_@types+node@22.12.0_yaml@2.7.0": { + "@vitejs/plugin-react-swc@3.8.0_vite@6.1.1": { "integrity": "sha512-T4sHPvS+DIqDP51ifPqa9XIRAz/kIvIi8oXcnOZZgHmMotgmmdxe/DD5tMFlt5nuIRzT0/QuiwmKlH0503Aapw==", "dependencies": [ "@swc/core", - "vite@6.1.1_@types+node@22.12.0_yaml@2.7.0" + "vite@6.1.1" + ] + }, + "@vitejs/plugin-vue@5.2.3_vite@5.4.14__@types+node@22.12.0_vue@3.5.13_@types+node@22.12.0": { + "integrity": "sha512-IYSLEQj4LgZZuoVpdSUCw3dIynTWQgPlaRP6iAvMle4My0HdYwr5g5wQAfwOeHQBmYwEkqF70nRpSilr6PoUDg==", + "dependencies": [ + "vite@5.4.14_@types+node@22.12.0", + "vue" ] }, - "@vitejs/plugin-vue@5.2.1_vite@5.4.14_vue@3.5.13": { - "integrity": "sha512-cxh314tzaWwOLqVes2gnnCtvBDcM1UMdn+iFR+UjAn411dPT3tOmqrJjbMd7koZpMAmBM/GqeV4n9ge7JSiJJQ==", + "@vitejs/plugin-vue@5.2.3_vite@5.4.14_vue@3.5.13": { + "integrity": "sha512-IYSLEQj4LgZZuoVpdSUCw3dIynTWQgPlaRP6iAvMle4My0HdYwr5g5wQAfwOeHQBmYwEkqF70nRpSilr6PoUDg==", "dependencies": [ "vite@5.4.14", "vue" @@ -1209,8 +1075,8 @@ "@vue/shared@3.5.13": { "integrity": "sha512-/hnE/qP5ZoGpol0a5mDi45bOd7t3tjYJBjsgCsivow7D48cJeV5l05RD82lPqi7gRiphZM37rnhW1l6ZoCNNnQ==" }, - "@vueuse/core@12.7.0": { - "integrity": "sha512-jtK5B7YjZXmkGNHjviyGO4s3ZtEhbzSgrbX+s5o+Lr8i2nYqNyHuPVOeTdM1/hZ5Tkxg/KktAuAVDDiHMraMVA==", + "@vueuse/core@12.8.2": { + "integrity": "sha512-HbvCmZdzAu3VGi/pWYm5Ut+Kd9mn1ZHnn4L5G8kOQTPs/IwIAmJoBrmYk2ckLArgMXZj0AW3n5CAejLUO+PhdQ==", "dependencies": [ "@types/web-bluetooth", "@vueuse/metadata", @@ -1218,8 +1084,8 @@ "vue" ] }, - "@vueuse/integrations@12.7.0_focus-trap@7.6.4": { - "integrity": "sha512-IEq7K4bCl7mn3uKJaWtNXnd1CAPaHLUMuyj5K1/k/pVcItt0VONZW8xiGxdIovJcQjkzOHjImhX5t6gija+0/g==", + "@vueuse/integrations@12.8.2_focus-trap@7.6.4": { + "integrity": "sha512-fbGYivgK5uBTRt7p5F3zy6VrETlV9RtZjBqd1/HxGdjdckBgBM4ugP8LHpjolqTj14TXTxSK1ZfgPbHYyGuH7g==", "dependencies": [ "@vueuse/core", "@vueuse/shared", @@ -1227,11 +1093,11 @@ "vue" ] }, - "@vueuse/metadata@12.7.0": { - "integrity": "sha512-4VvTH9mrjXqFN5LYa5YfqHVRI6j7R00Vy4995Rw7PQxyCL3z0Lli86iN4UemWqixxEvYfRjG+hF9wL8oLOn+3g==" + "@vueuse/metadata@12.8.2": { + "integrity": "sha512-rAyLGEuoBJ/Il5AmFHiziCPdQzRt88VxR+Y/A/QhJ1EWtWqPBBAxTAFaSkviwEuOEZNtW8pvkPgoCZQ+HxqW1A==" }, - "@vueuse/shared@12.7.0": { - "integrity": "sha512-coLlUw2HHKsm7rPN6WqHJQr18WymN4wkA/3ThFaJ4v4gWGWAQQGK+MJxLuJTBs4mojQiazlVWAKNJNpUWGRkNw==", + "@vueuse/shared@12.8.2": { + "integrity": "sha512-dznP38YzxZoNloI0qpEfpkms8knDtaoQ6Y/sfS0L7Yki4zh40LFHEhur0odJC6xTHG5dxWVPiUWBXn+wCG2s5w==", "dependencies": [ "vue" ] @@ -1248,8 +1114,8 @@ "acorn" ] }, - "acorn@8.14.0": { - "integrity": "sha512-cl669nCJTZBsL97OF4kUQm5g5hC2uihk0NxY3WENAC0TYdILVkAyHymAntgxGkl7K+t0cXIrH5siy5S4XkFycA==" + "acorn@8.14.1": { + "integrity": "sha512-OvQ/2pUDKmgfCg++xsTX1wGxfTaszcHVcTctW4UJB4hibJx2HXxxO5UmVgyjMa+ZDsiaf5wWLXYpRWMmBI0QHg==" }, "agentkeepalive@4.6.0": { "integrity": "sha512-kja8j7PjmncONqaTsB8fQ+wE2mSU2DJ9D4XKoJ5PFWIdRMa6SLSN1ff4mOr4jCbfRSsxR4keIiySJU0N9T5hIQ==", @@ -1257,8 +1123,8 @@ "humanize-ms" ] }, - "algoliasearch@5.20.3": { - "integrity": "sha512-iNC6BGvipaalFfDfDnXUje8GUlW5asj0cTMsZJwO/0rhsyLx1L7GZFAY8wW+eQ6AM4Yge2p5GSE5hrBlfSD90Q==", + "algoliasearch@5.21.0": { + "integrity": "sha512-hexLq2lSO1K5SW9j21Ubc+q9Ptx7dyRTY7se19U8lhIlVMLCNXWCyQ6C22p9ez8ccX0v7QVmwkl2l1CnuGoO2Q==", "dependencies": [ "@algolia/client-abtesting", "@algolia/client-analytics", @@ -1308,8 +1174,8 @@ "base-64@0.1.0": { "integrity": "sha512-Y5gU45svrR5tI2Vt/X9GPd3L0HNIKzGu202EjxrXMpuc2V2CiKgemAbUUsqYmZJvPtCXoUKjNZwBJzsNScUbXA==" }, - "base-x@4.0.0": { - "integrity": "sha512-FuwxlW4H5kh37X/oW59pwTzzTKRzfrrQwhmyspRM7swOEZcHtDZSCt45U6oKgtuFE+WYPblePMVIPR4RZrh/hw==" + "base-x@4.0.1": { + "integrity": "sha512-uAZ8x6r6S3aUM9rbHGVOIsR15U/ZSc82b3ymnCPsT45Gk1DDvhDPdIgB5MrhirZWt+5K0EEPQH985kNqZgNPFw==" }, "birpc@0.2.19": { "integrity": "sha512-5WeXXAvTmitV1RqJFppT5QtUiz2p1mRSYU000Jkft5ZUCLJIk4uQriYNO50HknxKwM6jd8utNc66K1qGIwwWBQ==" @@ -1608,8 +1474,8 @@ "tabbable" ] }, - "foreground-child@3.3.0": { - "integrity": "sha512-Ld2g8rrAyMYFXBhEqMz8ZAHBi4J4uS1i/CxGMDnjyFWddMXLVcDp051DZfu+t7+ab7Wv6SMqpWmyFIj5UbfFvg==", + "foreground-child@3.3.1": { + "integrity": "sha512-gIXjKqtFuWEgzFRJA9WCQeSJLZDjgJUOMCMzxtvFq/37KojM1BFGufqsCy0r4qSQmYLsZYMeyRqzIWOMup03sw==", "dependencies": [ "cross-spawn", "signal-exit" @@ -1634,6 +1500,16 @@ "web-streams-polyfill@4.0.0-beta.3" ] }, + "framer-motion@12.5.0_react@18.3.1_react-dom@18.3.1__react@18.3.1": { + "integrity": "sha512-buPlioFbH9/W7rDzYh1C09AuZHAk2D1xTA1BlounJ2Rb9aRg84OXexP0GLd+R83v0khURdMX7b5MKnGTaSg5iA==", + "dependencies": [ + "motion-dom", + "motion-utils", + "react", + "react-dom", + "tslib" + ] + }, "fsevents@2.3.3": { "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==" }, @@ -1643,8 +1519,8 @@ "get-east-asian-width@1.3.0": { "integrity": "sha512-vpeMIQKxczTD/0s2CdEWHcb0eeJe6TFjxb+J5xgX7hScxqrGuyjmv4c1D4A/gelKfyox0gJJwIHF+fLjeaM8kQ==" }, - "get-intrinsic@1.2.7": { - "integrity": "sha512-VW6Pxhsrk0KAOqs3WEd0klDiF/+V7gQOpAvY1jVU/LHmaD/kQO4523aiJuikX/QAKYiW6x8Jh+RJej1almdtCA==", + "get-intrinsic@1.3.0": { + "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==", "dependencies": [ "call-bind-apply-helpers", "es-define-property", @@ -1864,8 +1740,8 @@ "micromark-util-symbol@2.0.1": { "integrity": "sha512-vs5t8Apaud9N28kgCrRUdEed4UJ+wWNvicHLPxCa9ENlYuAY31M0ETy5y1vA33YoNPDFTghEbnh6efaE8h4x0Q==" }, - "micromark-util-types@2.0.1": { - "integrity": "sha512-534m2WhVTddrcKVepwmVEVnUAmtrx9bfIjNoQHRqfnvdaHQiFytEhJoTgpWJvDEXCO5gLTQh3wYC1PgOJA4NSQ==" + "micromark-util-types@2.0.2": { + "integrity": "sha512-Yw0ECSpJoViF1qTU4DC6NwtC4aWGt1EkzaQB8KPPyCRR8z9TWeV0HbEFGTO+ZY1wB22zmxnJqhPyTpOVCpeHTA==" }, "mime-db@1.52.0": { "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==" @@ -1894,11 +1770,29 @@ "mitt@3.0.1": { "integrity": "sha512-vKivATfr97l2/QBCYAkXYDbrIWPM2IIKEl7YPhjCvKlG3kE2gm+uBo6nEXK3M5/Ffh/FLpKExzOQ3JJoJGFKBw==" }, + "motion-dom@12.5.0": { + "integrity": "sha512-uH2PETDh7m+Hjd1UQQ56yHqwn83SAwNjimNPE/kC+Kds0t4Yh7+29rfo5wezVFpPOv57U4IuWved5d1x0kNhbQ==", + "dependencies": [ + "motion-utils" + ] + }, + "motion-utils@12.5.0": { + "integrity": "sha512-+hFFzvimn0sBMP9iPxBa9OtRX35ZQ3py0UHnb8U29VD+d8lQ8zH3dTygJWqK7av2v6yhg7scj9iZuvTS0f4+SA==" + }, + "motion@12.5.0_react@18.3.1_react-dom@18.3.1__react@18.3.1": { + "integrity": "sha512-BTAYKszMmTvXSsIyeHNMPSicjWgUA4j7OmZv1xPpthm4sPub3ch66fy9U7BhJ1uXNL3YeprsIegzuvps3FkEMw==", + "dependencies": [ + "framer-motion", + "react", + "react-dom", + "tslib" + ] + }, "ms@2.1.3": { "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" }, - "nanoid@3.3.8": { - "integrity": "sha512-WNLf5Sd8oZxOm+TzppcYk8gVOgP+l58xNy58D0nbUnOxOWRWvlcCV4kUF7ltmI6PsrLl/BgKEyS4mqsGChFN0w==" + "nanoid@3.3.11": { + "integrity": "sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w==" }, "node-domexception@1.0.0": { "integrity": "sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ==" @@ -1979,8 +1873,8 @@ "source-map-js" ] }, - "preact@10.26.2": { - "integrity": "sha512-0gNmv4qpS9HaN3+40CLBAnKe0ZfyE4ZWo5xKlC1rVrr0ckkEvJvAQqKaHANdFKsGstoxrY4AItZ7kZSGVoVjgg==" + "preact@10.26.4": { + "integrity": "sha512-KJhO7LBFTjP71d83trW+Ilnjbo+ySsaAgCfXOXUlmGzJ4ygYPWmysm77yg4emwfmoz3b22yvH5IsVFHbhUaH5w==" }, "pretty-bytes@6.1.1": { "integrity": "sha512-mQUvGU6aUFQ+rNvTIAcZuWGRT9a6f6Yrg9bHs4ImKF+HZCEK+plBvnAZYSIQztknZF2qnzNtr6F8s0+IuptdlQ==" @@ -2068,54 +1962,28 @@ "glob" ] }, - "rollup@4.34.6": { - "integrity": "sha512-wc2cBWqJgkU3Iz5oztRkQbfVkbxoz5EhnCGOrnJvnLnQ7O0WhQUYyv18qQI79O8L7DdHrrlJNeCHd4VGpnaXKQ==", - "dependencies": [ - "@rollup/rollup-android-arm-eabi@4.34.6", - "@rollup/rollup-android-arm64@4.34.6", - "@rollup/rollup-darwin-arm64@4.34.6", - "@rollup/rollup-darwin-x64@4.34.6", - "@rollup/rollup-freebsd-arm64@4.34.6", - "@rollup/rollup-freebsd-x64@4.34.6", - "@rollup/rollup-linux-arm-gnueabihf@4.34.6", - "@rollup/rollup-linux-arm-musleabihf@4.34.6", - "@rollup/rollup-linux-arm64-gnu@4.34.6", - "@rollup/rollup-linux-arm64-musl@4.34.6", - "@rollup/rollup-linux-loongarch64-gnu@4.34.6", - "@rollup/rollup-linux-powerpc64le-gnu@4.34.6", - "@rollup/rollup-linux-riscv64-gnu@4.34.6", - "@rollup/rollup-linux-s390x-gnu@4.34.6", - "@rollup/rollup-linux-x64-gnu@4.34.6", - "@rollup/rollup-linux-x64-musl@4.34.6", - "@rollup/rollup-win32-arm64-msvc@4.34.6", - "@rollup/rollup-win32-ia32-msvc@4.34.6", - "@rollup/rollup-win32-x64-msvc@4.34.6", - "@types/estree", - "fsevents" - ] - }, "rollup@4.34.8": { "integrity": "sha512-489gTVMzAYdiZHFVA/ig/iYFllCcWFHMvUHI1rpFmkoUtRlQxqh6/yiNqnYibjMZ2b/+FUQwldG+aLsEt6bglQ==", "dependencies": [ - "@rollup/rollup-android-arm-eabi@4.34.8", - "@rollup/rollup-android-arm64@4.34.8", - "@rollup/rollup-darwin-arm64@4.34.8", - "@rollup/rollup-darwin-x64@4.34.8", - "@rollup/rollup-freebsd-arm64@4.34.8", - "@rollup/rollup-freebsd-x64@4.34.8", - "@rollup/rollup-linux-arm-gnueabihf@4.34.8", - "@rollup/rollup-linux-arm-musleabihf@4.34.8", - "@rollup/rollup-linux-arm64-gnu@4.34.8", - "@rollup/rollup-linux-arm64-musl@4.34.8", - "@rollup/rollup-linux-loongarch64-gnu@4.34.8", - "@rollup/rollup-linux-powerpc64le-gnu@4.34.8", - "@rollup/rollup-linux-riscv64-gnu@4.34.8", - "@rollup/rollup-linux-s390x-gnu@4.34.8", - "@rollup/rollup-linux-x64-gnu@4.34.8", - "@rollup/rollup-linux-x64-musl@4.34.8", - "@rollup/rollup-win32-arm64-msvc@4.34.8", - "@rollup/rollup-win32-ia32-msvc@4.34.8", - "@rollup/rollup-win32-x64-msvc@4.34.8", + "@rollup/rollup-android-arm-eabi", + "@rollup/rollup-android-arm64", + "@rollup/rollup-darwin-arm64", + "@rollup/rollup-darwin-x64", + "@rollup/rollup-freebsd-arm64", + "@rollup/rollup-freebsd-x64", + "@rollup/rollup-linux-arm-gnueabihf", + "@rollup/rollup-linux-arm-musleabihf", + "@rollup/rollup-linux-arm64-gnu", + "@rollup/rollup-linux-arm64-musl", + "@rollup/rollup-linux-loongarch64-gnu", + "@rollup/rollup-linux-powerpc64le-gnu", + "@rollup/rollup-linux-riscv64-gnu", + "@rollup/rollup-linux-s390x-gnu", + "@rollup/rollup-linux-x64-gnu", + "@rollup/rollup-linux-x64-musl", + "@rollup/rollup-win32-arm64-msvc", + "@rollup/rollup-win32-ia32-msvc", + "@rollup/rollup-win32-x64-msvc", "@types/estree", "fsevents" ] @@ -2218,7 +2086,7 @@ "subhosting@0.1.0-alpha.1": { "integrity": "sha512-uJGozPd5gcUcJVqBjGatFjbPlBfs3osfSFhhcpvl4FhqAZ1mqUoL6fbF1l+HAN74Z5+wOWmkowxguJznp6DKRA==", "dependencies": [ - "@types/node@18.19.76", + "@types/node@18.19.80", "@types/node-fetch", "abort-controller", "agentkeepalive", @@ -2253,7 +2121,7 @@ "ts-essentials@10.0.4": { "integrity": "sha512-lwYdz28+S4nicm+jFi6V58LaAIpxzhg9rLdgNC1VsdP/xiFBseGhF1M/shwCk6zMmwahBZdXcl34LVHrEang3A==" }, - "ts-node@10.9.2_@types+node@22.12.0_typescript@5.7.3": { + "ts-node@10.9.2_@types+node@22.12.0_typescript@5.8.2": { "integrity": "sha512-f0FFpIdcHgn8zcPSbf1dRevwt047YMnaiJM3u2w2RewrB+fob/zePZcrOyQoLMMO7aBIddLcQIEK5dYjkLnGrQ==", "dependencies": [ "@cspotcode/source-map-support", @@ -2276,8 +2144,8 @@ "tslib@2.8.1": { "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==" }, - "typescript@5.7.3": { - "integrity": "sha512-84MVSjMEHP+FQRPy3pX9sTVV/INIex71s9TL2Gm5FG/WG1SqXeKyZ0k7/blY/4FdOzI12CBy1vGc4og/eus0fw==" + "typescript@5.8.2": { + "integrity": "sha512-aJn6wq13/afZp/jT9QZmwEjDqqvSGp1VT5GVg+f/t6/oVyrgXM6BY1h9BRh/O5p3PlUPAe+WuiEZOmb/49RqoQ==" }, "ua-is-frozen@0.1.2": { "integrity": "sha512-RwKDW2p3iyWn4UbaxpP2+VxwqXh0jpvdxsYpZ5j/MLLiQOfbsV5shpgQiw93+KMYQPcteeMQ289MaAFzs3G9pw==" @@ -2331,6 +2199,12 @@ "unist-util-visit-parents" ] }, + "use-sync-external-store@1.4.0_react@18.3.1": { + "integrity": "sha512-9WXSPC5fMv61vaupRkCKCxsPxBocVnwakBEkMIHHpkTTg6icbJtg6jzgtLDm4bl3cSHAca52rYWih0k4K3PfHw==", + "dependencies": [ + "react" + ] + }, "uuid@9.0.1": { "integrity": "sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA==" }, @@ -2354,10 +2228,10 @@ "vfile-message" ] }, - "vite-plugin-wasm@3.4.1_vite@6.1.1__@types+node@22.12.0__yaml@2.7.0_@types+node@22.12.0_yaml@2.7.0": { + "vite-plugin-wasm@3.4.1_vite@6.1.1": { "integrity": "sha512-ja3nSo2UCkVeitltJGkS3pfQHAanHv/DqGatdI39ja6McgABlpsZ5hVgl6wuR8Qx5etY3T5qgDQhOWzc5RReZA==", "dependencies": [ - "vite@6.1.1_@types+node@22.12.0_yaml@2.7.0" + "vite@6.1.1" ] }, "vite@5.4.14": { @@ -2366,20 +2240,61 @@ "esbuild@0.21.5", "fsevents", "postcss", - "rollup@4.34.6" + "rollup" ] }, - "vite@6.1.1_@types+node@22.12.0_yaml@2.7.0": { - "integrity": "sha512-4GgM54XrwRfrOp297aIYspIti66k56v16ZnqHvrIM7mG+HjDlAwS7p+Srr7J6fGvEdOJ5JcQ/D9T7HhtdXDTzA==", + "vite@5.4.14_@types+node@22.12.0": { + "integrity": "sha512-EK5cY7Q1D8JNhSaPKVK4pwBFvaTmZxEnoKXLG/U9gmdDcihQGNzFlgIvaxezFR4glP1LsuiedwMBqCXH3wZccA==", "dependencies": [ "@types/node@22.12.0", + "esbuild@0.21.5", + "fsevents", + "postcss", + "rollup" + ] + }, + "vite@6.1.1": { + "integrity": "sha512-4GgM54XrwRfrOp297aIYspIti66k56v16ZnqHvrIM7mG+HjDlAwS7p+Srr7J6fGvEdOJ5JcQ/D9T7HhtdXDTzA==", + "dependencies": [ + "esbuild@0.24.2", + "fsevents", + "postcss", + "rollup" + ] + }, + "vite@6.1.1_yaml@2.7.0": { + "integrity": "sha512-4GgM54XrwRfrOp297aIYspIti66k56v16ZnqHvrIM7mG+HjDlAwS7p+Srr7J6fGvEdOJ5JcQ/D9T7HhtdXDTzA==", + "dependencies": [ "esbuild@0.24.2", "fsevents", "postcss", - "rollup@4.34.8", + "rollup", "yaml" ] }, + "vitepress@1.6.3_vite@5.4.14__@types+node@22.12.0_vue@3.5.13_focus-trap@7.6.4_@types+node@22.12.0_@types+react@18.3.18_react@18.3.1_react-dom@18.3.1__react@18.3.1": { + "integrity": "sha512-fCkfdOk8yRZT8GD9BFqusW3+GggWYZ/rYncOfmgcDtP3ualNHCAg+Robxp2/6xfH1WwPHtGpPwv7mbA3qomtBw==", + "dependencies": [ + "@docsearch/css", + "@docsearch/js", + "@iconify-json/simple-icons", + "@shikijs/core", + "@shikijs/transformers", + "@shikijs/types", + "@types/markdown-it", + "@vitejs/plugin-vue@5.2.3_vite@5.4.14__@types+node@22.12.0_vue@3.5.13_@types+node@22.12.0", + "@vue/devtools-api", + "@vue/shared", + "@vueuse/core", + "@vueuse/integrations", + "focus-trap", + "mark.js", + "minisearch", + "shiki", + "vite@5.4.14_@types+node@22.12.0", + "vue" + ] + }, "vitepress@1.6.3_vite@5.4.14_vue@3.5.13_focus-trap@7.6.4_@types+react@18.3.18_react@18.3.1_react-dom@18.3.1__react@18.3.1": { "integrity": "sha512-fCkfdOk8yRZT8GD9BFqusW3+GggWYZ/rYncOfmgcDtP3ualNHCAg+Robxp2/6xfH1WwPHtGpPwv7mbA3qomtBw==", "dependencies": [ @@ -2390,7 +2305,7 @@ "@shikijs/transformers", "@shikijs/types", "@types/markdown-it", - "@vitejs/plugin-vue", + "@vitejs/plugin-vue@5.2.3_vite@5.4.14_vue@3.5.13", "@vue/devtools-api", "@vue/shared", "@vueuse/core", @@ -2478,6 +2393,7 @@ "jsr:@cliffy/keypress@1.0.0-rc.7", "jsr:@cliffy/prompt@1.0.0-rc.7", "jsr:@cliffy/table@1.0.0-rc.7", + "jsr:@deno/dnt@0.41.3", "jsr:@std/async@1.0.10", "jsr:@std/datetime@0.225.3", "jsr:@std/dotenv@0.225.3", @@ -2494,6 +2410,9 @@ "npm:@automerge/automerge@2.2.8", "npm:@noble/hashes@1.7.1", "npm:@onsetsoftware/automerge-patcher@0.14.0", + "npm:@preact/signals-core@1.8.0", + "npm:@preact/signals-react@3.0.1", + "npm:@svgdotjs/svg.js@3.2.4", "npm:@types/diff@7.0.1", "npm:approx-string-match@2", "npm:chai@5", @@ -2512,7 +2431,7 @@ "npm:rambda@9.4.2", "npm:ramda@0.30.1", "npm:react-error-boundary@5", - "npm:react-inspector@6", + "npm:react-inspector@6.0.2", "npm:react-spinners@0.15.0", "npm:rxjs@7.8.2", "npm:strip-ansi@7", @@ -2525,6 +2444,7 @@ ], "packageJson": { "dependencies": [ + "npm:@deno/vite-plugin@1.0.4", "npm:@jsr/std__async@1.0.10", "npm:@jsr/std__datetime@0.225.3", "npm:@jsr/std__dotenv@0.225.3", @@ -2540,6 +2460,7 @@ "npm:@vidstack/react@1.12.12", "npm:@vitejs/plugin-react-swc@3.8.0", "npm:hono@4.7.2", + "npm:motion@12.5.0", "npm:react-dom@18.3.1", "npm:react-icons@5.5.0", "npm:react@18.3.1", diff --git a/deploy/@tdb.slc/-scripts/-main.ts b/deploy/@tdb.slc/-scripts/-main.ts new file mode 100644 index 0000000000..7c806709fa --- /dev/null +++ b/deploy/@tdb.slc/-scripts/-main.ts @@ -0,0 +1 @@ +import '../../../code/sys.driver/driver-vite/src/-entry/-main.ts'; diff --git a/deploy/@tdb.slc/-scripts/-tmp.ts b/deploy/@tdb.slc/-scripts/-tmp.ts new file mode 100644 index 0000000000..9e0f27eee7 --- /dev/null +++ b/deploy/@tdb.slc/-scripts/-tmp.ts @@ -0,0 +1 @@ +console.info('š', import.meta.url); diff --git a/deploy/@tdb.slc/README.md b/deploy/@tdb.slc/README.md new file mode 100644 index 0000000000..e892c53866 --- /dev/null +++ b/deploy/@tdb.slc/README.md @@ -0,0 +1,24 @@ +# SLC +- "Social Lean Canvas" UI component library. + +### Description +A shareable common language for modelling āSocial Enterpriseā and āImpact Driven Business.ā The dimensions of āpurposeā and āimpactā are woven into standard business modelling notation (a "canvas"). When done right this provides a clear pathway to simplifying the complexity inherent in building āimpact driven business.ā + + +### Deployments +- staging: https://slc.db.team +- production: https://socialleancanvas.com + +#### Legacy Domains + +``` + socialleancanvas.com +eusic.socialleancanvas.com + socialleancanvas.com/ember-slc +``` + +Prior deployment: +- https://slc-phil-tdb.vercel.app/ +- https://slc-phil-tdb.vercel.app/ember-slc + + diff --git a/deploy/@tdb.slc/deno.json b/deploy/@tdb.slc/deno.json new file mode 100644 index 0000000000..34f6bb2285 --- /dev/null +++ b/deploy/@tdb.slc/deno.json @@ -0,0 +1,24 @@ +{ + "name": "@tdb/slc", + "version": "0.0.1", + "license": "MIT", + "tasks": { + "dev": "deno run -RWNE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=dev --in=./src/-test/index.html", + "build": "deno run -RWE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=build --in=./src/-test/index.html", + "serve": "deno run -RNE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=serve", + "test": "deno test -RWNE --allow-run --allow-ffi --allow-sys", + "dry": "deno publish --allow-dirty --dry-run", + "clean": "deno run -RWE --allow-ffi ./-scripts/-main.ts --cmd=clean", + "upgrade": "deno run -RWNE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=upgrade", + "backup": "deno run -RWE --allow-run --allow-ffi ./-scripts/-main.ts --cmd=backup", + "help": "deno run -RE --allow-ffi ./-scripts/-main.ts --cmd=help", + "tmp": "deno run -A ./-scripts/-tmp.ts" + }, + "exports": { + ".": "./src/mod.ts", + "./t": "./src/types.ts", + "./types": "./src/types.ts", + "./ui": "./src/ui/mod.ts", + "./specs": "./src/-test/entry.Specs.ts" + } +} diff --git a/deploy/@tdb.slc/public/pdf/slc.pdf b/deploy/@tdb.slc/public/pdf/slc.pdf new file mode 100644 index 0000000000..17c0e687e2 Binary files /dev/null and b/deploy/@tdb.slc/public/pdf/slc.pdf differ diff --git a/deploy/@tdb.slc/src/-sample/-test.ui.ts b/deploy/@tdb.slc/src/-sample/-test.ui.ts new file mode 100644 index 0000000000..80e99d3a9c --- /dev/null +++ b/deploy/@tdb.slc/src/-sample/-test.ui.ts @@ -0,0 +1 @@ +export * from '../ui/-test.ui.ts'; diff --git a/deploy/@tdb.slc/src/-sample/common.ts b/deploy/@tdb.slc/src/-sample/common.ts new file mode 100644 index 0000000000..f19dabd006 --- /dev/null +++ b/deploy/@tdb.slc/src/-sample/common.ts @@ -0,0 +1 @@ +export * from '../ui/common.ts'; diff --git a/code/sys.ui/ui-react-devharness/src/m.Dev/-test.ts b/deploy/@tdb.slc/src/-test.ts similarity index 100% rename from code/sys.ui/ui-react-devharness/src/m.Dev/-test.ts rename to deploy/@tdb.slc/src/-test.ts diff --git a/deploy/@tdb.slc/src/-test/base.css b/deploy/@tdb.slc/src/-test/base.css new file mode 100644 index 0000000000..9f57ec5271 --- /dev/null +++ b/deploy/@tdb.slc/src/-test/base.css @@ -0,0 +1,3 @@ +body { + background-color: #293042; /* Inky Dark */ +} diff --git a/deploy/@tdb.slc/src/-test/common.ts b/deploy/@tdb.slc/src/-test/common.ts new file mode 100644 index 0000000000..8cae67176a --- /dev/null +++ b/deploy/@tdb.slc/src/-test/common.ts @@ -0,0 +1 @@ +export * from '../common.ts'; diff --git a/deploy/@tdb.slc/src/-test/entry.Specs.ts b/deploy/@tdb.slc/src/-test/entry.Specs.ts new file mode 100644 index 0000000000..19919936ae --- /dev/null +++ b/deploy/@tdb.slc/src/-test/entry.Specs.ts @@ -0,0 +1,23 @@ +/** + * @module + * DevHarness visual specs. + */ +import type { t } from './common.ts'; + +export const Specs = { + 'tdb.slc.entry.Landing-1': () => import('../ui/ui.Landing-1/-SPEC.tsx'), + 'tdb.slc.entry.Landing-2': () => import('../ui/ui.Landing-2/-SPEC.tsx'), + 'tdb.slc.entry.Landing-3': () => import('../ui/ui.Landing-3/-SPEC.tsx'), + + 'tdb.slc.ui.Layout': () => import('../ui/ui.Layout/-SPEC.tsx'), + + 'tdb.slc.ui.Logo.Wordmark': () => import('../ui/ui.Logo.Wordmark/-SPEC.tsx'), + 'tdb.slc.ui.Logo.Canvas': () => import('../ui/ui.Logo.Canvas/-SPEC.tsx'), + 'tdb.slc.ui.Video.Background': () => import('../ui/ui.Video.Background/-SPEC.tsx'), + 'tdb.slc.ui.TooSmall': () => import('../ui/ui.TooSmall/-SPEC.tsx'), + + 'tdb.slc.content.videos: (index)': () => import('../ui.Content/-sample/ui.Videos/-SPEC.tsx'), + 'tdb.slc.content.CanvasSlug': () => import('../ui.Content/ui/ui.CanvasSlug/-SPEC.tsx'), + 'tdb.slc.content.FadeText': () => import('../ui.Content/ui/ui.FadeText/-SPEC.tsx'), + 'tdb.slc.content.Image': () => import('../ui.Content/ui/ui.Image/-SPEC.tsx'), +} as t.SpecImports; diff --git a/deploy/@tdb.slc/src/-test/entry.tsx b/deploy/@tdb.slc/src/-test/entry.tsx new file mode 100644 index 0000000000..6c89e06973 --- /dev/null +++ b/deploy/@tdb.slc/src/-test/entry.tsx @@ -0,0 +1,69 @@ +import React, { StrictMode } from 'react'; +import { createRoot } from 'react-dom/client'; + +import type { t } from '../common.ts'; +import { pkg } from '../pkg.ts'; +import { useKeyboard } from '../ui/use/use.Keyboard.ts'; + +/** + * Render UI. + */ +const document = globalThis.document; +if (document) { + document.title = pkg.name; + document.body.style.overflow = 'hidden'; // NB: suppress rubber-band effect. +} + +/** + * Setup mounter: + */ +const Root = (props: { state: t.AppSignals; children?: t.ReactNode }) => { + useKeyboard(props.state); + return props.children; +}; + +/** + * MAIN entry. + */ +export async function main() { + const params = new URL(location.href).searchParams; + const isDev = params.has('dev') || params.has('d'); + const root = createRoot(document.getElementById('root')!); + + if (isDev) { + /** + * DevHarness: + */ + const { App } = await import('../ui/App/mod.ts'); + const { render } = await import('@sys/ui-react-devharness'); + const { Specs } = await import('./entry.Specs.ts'); + + const app = App.signals(); + const el = await render(pkg, Specs, { hrDepth: 3, style: { Absolute: 0 } }); + + root.render( + <StrictMode> + <Root state={app}>{el}</Root> + </StrictMode>, + ); + } else { + /** + * Landing (entry): + */ + const { Landing, App, Content } = await import('../ui/ui.Landing-3/mod.ts'); + const app = App.signals(); + + app.stack.push(await Content.Factory.entry()); + await App.Render.preload(app, Content.factory, 'Entry', 'Trailer', 'Overview'); + + root.render( + <StrictMode> + <Root state={app}> + <Landing state={app} style={{ Absolute: 0 }} /> + </Root> + </StrictMode>, + ); + } +} + +main().catch((err) => console.error(`Failed to render DevHarness`, err)); diff --git a/deploy/slc.db.team/src/-test/index.html b/deploy/@tdb.slc/src/-test/index.html similarity index 60% rename from deploy/slc.db.team/src/-test/index.html rename to deploy/@tdb.slc/src/-test/index.html index 1bca1020b7..b9285c2c69 100644 --- a/deploy/slc.db.team/src/-test/index.html +++ b/deploy/@tdb.slc/src/-test/index.html @@ -3,11 +3,6 @@ <head> <meta charset="UTF-8" /> <meta name="viewport" content="width=device-width, initial-scale=1.0" /> - - <meta name="apple-mobile-web-app-capable" content="yes" /> - <meta name="apple-mobile-web-app-status-bar-style" content="black" /> - <link rel="manifest" href="./manifest.json" /> - <title>loading...</title> </head> <body> diff --git a/deploy/@tdb.slc/src/-test/mod.ts b/deploy/@tdb.slc/src/-test/mod.ts new file mode 100644 index 0000000000..b58a897701 --- /dev/null +++ b/deploy/@tdb.slc/src/-test/mod.ts @@ -0,0 +1,2 @@ +export { c, describe, expect, expectError, it, Testing } from '@sys/testing/server'; +export * from '../common.ts'; diff --git a/deploy/@tdb.slc/src/.test.ts b/deploy/@tdb.slc/src/.test.ts new file mode 100644 index 0000000000..7348c03d0a --- /dev/null +++ b/deploy/@tdb.slc/src/.test.ts @@ -0,0 +1,8 @@ +import { type t, describe, it, expect, Pkg, pkg } from './-test.ts'; + +describe(`module: ${Pkg.toString(pkg)}`, () => { + it('exists', () => { + console.info(`š¦ Module`, pkg); + expect(typeof pkg.name === 'string').to.be.true; + }); +}); diff --git a/deploy/slc.db.team/src/common.ts b/deploy/@tdb.slc/src/common.ts similarity index 100% rename from deploy/slc.db.team/src/common.ts rename to deploy/@tdb.slc/src/common.ts diff --git a/deploy/@tdb.slc/src/common/libs.ts b/deploy/@tdb.slc/src/common/libs.ts new file mode 100644 index 0000000000..140530b65c --- /dev/null +++ b/deploy/@tdb.slc/src/common/libs.ts @@ -0,0 +1 @@ +export { Err, Is, isRecord, Path, Pkg, rx, Signal, slug, Str, Time, Timestamp } from '@sys/std'; diff --git a/deploy/@tdb.slc/src/common/m.CanvasPanel.ts b/deploy/@tdb.slc/src/common/m.CanvasPanel.ts new file mode 100644 index 0000000000..6aecb6ce23 --- /dev/null +++ b/deploy/@tdb.slc/src/common/m.CanvasPanel.ts @@ -0,0 +1,19 @@ +import type * as t from './t.ts'; + +export const CanvasPanel = { + get list(): t.CanvasPanel[] { + return [ + 'purpose', + 'impact', + 'problem', + 'solution', + 'metrics', + 'uvp', + 'advantage', + 'channels', + 'customers', + 'costs', + 'revenue', + ]; + }, +} as const; diff --git a/deploy/@tdb.slc/src/common/mod.ts b/deploy/@tdb.slc/src/common/mod.ts new file mode 100644 index 0000000000..2ea69b26b5 --- /dev/null +++ b/deploy/@tdb.slc/src/common/mod.ts @@ -0,0 +1,7 @@ +import type * as t from './t.ts'; + +export { pkg } from '../pkg.ts'; +export { type t }; + +export * from './libs.ts'; +export * from './m.CanvasPanel.ts'; diff --git a/deploy/@tdb.slc/src/common/t.ts b/deploy/@tdb.slc/src/common/t.ts new file mode 100644 index 0000000000..bb53c40cdd --- /dev/null +++ b/deploy/@tdb.slc/src/common/t.ts @@ -0,0 +1,29 @@ +/** + * @external + */ +export type { ReactElement, ReactNode } from 'react'; + +/** + * @system + */ +export type * from '@sys/types'; + +export type { ExtractSignalValue, Signal } from '@sys/std/t'; +export type { SpecImports } from '@sys/testing/t'; +export type { CssEdgesInput, CssInput, CssMarginArray, CssProps, CssValue } from '@sys/ui-css/t'; +export type { + SheetMarginInput, + SheetOrientation, + SheetOrientationY, + SheetSignalStack, + SvgElement, + SvgInstance, + VideoPlayerSignals, + VimeoIFrame, +} from '@sys/ui-react-components/t'; +export type { DevCtx } from '@sys/ui-react-devharness/t'; + +/** + * @local + */ +export type * from '../types.ts'; diff --git a/deploy/@tdb.slc/src/mod.ts b/deploy/@tdb.slc/src/mod.ts new file mode 100644 index 0000000000..5246807c9a --- /dev/null +++ b/deploy/@tdb.slc/src/mod.ts @@ -0,0 +1,13 @@ +/** + * @module + * The "Social Lean Canvas" product system. + */ +export { pkg } from './pkg.ts'; + +/** Module types. */ +export type * as t from './types.ts'; + +/** + * Library + */ +export { App } from './ui/App/mod.ts'; diff --git a/deploy/@tdb.slc/src/pkg.ts b/deploy/@tdb.slc/src/pkg.ts new file mode 100644 index 0000000000..495d9f86eb --- /dev/null +++ b/deploy/@tdb.slc/src/pkg.ts @@ -0,0 +1,16 @@ +import type { Pkg } from '@sys/types'; + +/** + * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). + */ +export const pkg: Pkg = { name: '@tdb/slc', version: '0.0.1' }; diff --git a/deploy/@tdb.slc/src/types.ts b/deploy/@tdb.slc/src/types.ts new file mode 100644 index 0000000000..dcf4e0b048 --- /dev/null +++ b/deploy/@tdb.slc/src/types.ts @@ -0,0 +1,59 @@ +/** + * @module + * Module types. + */ + +/** + * UI: logical "App" state. + */ +export type * from './ui/App.Layout/t.ts'; +export type * from './ui/App.Render/t.ts'; +export type * from './ui/App.Signals.Controller/t.ts'; +export type * from './ui/App.Signals/t.ts'; +export type * from './ui/App/t.ts'; + +/** + * UI Structure: + */ +export type * from './ui/ui.Layout/t.ts'; +export type * from './ui/ui.Logo.Canvas/t.ts'; +export type * from './ui/ui.Logo.Wordmark/t.ts'; +export type * from './ui/ui.Sheet/t.ts'; +export type * from './ui/ui.TooSmall/t.ts'; +export type * from './ui/ui.Video.Background/t.ts'; + +export type * from './ui/ui.Landing-1/t.ts'; +export type * from './ui/ui.Landing-2/t.ts'; +export type * from './ui/ui.Landing-3/t.ts'; + +export type * from './ui/use/t.ts'; + +/** + * UI Content: + */ +export type * from './ui.Content/m.Content/t.ts'; +export type * from './ui.Content/m.Factory/t.ts'; +export type * from './ui.Content/t.ts'; +export type * from './ui.Content/ui/t.ts'; +export type * from './ui.Content/ui/ui.CanvasSlug/t.ts'; +export type * from './ui.Content/ui/ui.FadeText/t.ts'; +export type * from './ui.Content/ui/ui.Image/t.ts'; +export type * from './ui.Content/ui/ui.Pulldown/t.ts'; + +export type * from './ui.Content/-sample/ui.Videos/t.ts'; + +/** + * SLC Panels. + */ +export type CanvasPanel = + | 'purpose' + | 'impact' + | 'problem' + | 'solution' + | 'metrics' + | 'uvp' + | 'advantage' + | 'channels' + | 'customers' + | 'costs' + | 'revenue'; diff --git a/deploy/@tdb.slc/src/ui.Content/-sample/ui.Videos/-SPEC.Debug.tsx b/deploy/@tdb.slc/src/ui.Content/-sample/ui.Videos/-SPEC.Debug.tsx new file mode 100644 index 0000000000..ef8a0c6856 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/-sample/ui.Videos/-SPEC.Debug.tsx @@ -0,0 +1,91 @@ +import React from 'react'; +import { type t, Button, Color, css, Player, Signal, VIDEO } from './common.ts'; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; +type P = DebugProps; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + const s = Signal.create; + const video = Player.Video.signals({ + src: VIDEO.Trailer.src, // Rowan: "group scale", + cornerRadius: 0, + }); + const props = { + theme: s<t.CommonTheme>('Light'), + video, + }; + + const api = { + props, + listen() { + const p = props; + p.theme.value; + }, + }; + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<P> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => { + p.theme.value; + p.video.props.src.value; + }); + + /** + * Render: + */ + const theme = Color.theme(p.theme.value); + const styles = { + base: css({ color: theme.fg }), + title: css({ fontWeight: 'bold', marginBottom: 10 }), + }; + + const selectVideo = (label: string, src: string) => { + const s = p.video.props; + const isCurrent = s.src.value === src; + const styles = { + base: css({ display: 'grid', gridTemplateColumns: `1fr auto`, marginLeft: 15 }), + label: css({ color: isCurrent ? Color.BLUE : undefined }), + }; + return ( + <Button block onClick={() => (s.src.value = src)}> + <div className={styles.base.class}> + <div className={styles.label.class}>{label}</div> + <div>{src}</div> + </div> + </Button> + ); + }; + + return ( + <div className={css(styles.base, props.style).class}> + <Button + block + label={`theme: "${p.theme}"`} + onClick={() => Signal.cycle<t.CommonTheme>(p.theme, ['Light', 'Dark'])} + /> + + <hr /> + <div className={styles.title.class}>Videos</div> + {selectVideo('Trailer', VIDEO.Trailer.src)} + {selectVideo('Overview', VIDEO.Overview.src)} + <hr /> + {selectVideo('ref: "Group Scale"', VIDEO.GroupScale.src)} + {selectVideo('sample: Hindi Translation', 'vimeo/1074559094')} + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui.Content/-sample/ui.Videos/-SPEC.tsx b/deploy/@tdb.slc/src/ui.Content/-sample/ui.Videos/-SPEC.tsx new file mode 100644 index 0000000000..de1aa8269e --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/-sample/ui.Videos/-SPEC.tsx @@ -0,0 +1,28 @@ +import { Dev, Signal, Spec } from '../../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { VideosIndex } from './mod.ts'; + +export default Spec.describe('VideosIndex', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + Dev.Theme.signalEffect(ctx, p.theme, 1); + Signal.effect(() => { + debug.listen(); + ctx.redraw(); + }); + + ctx.subject + .size([null, null]) + .display('grid') + .render((e) => <VideosIndex theme={p.theme.value} signals={p.video} />); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/deploy/@tdb.slc/src/ui.Content/-sample/ui.Videos/common.ts b/deploy/@tdb.slc/src/ui.Content/-sample/ui.Videos/common.ts new file mode 100644 index 0000000000..aee2707a58 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/-sample/ui.Videos/common.ts @@ -0,0 +1,7 @@ +export { VIDEO } from '../../VIDEO.ts'; +export * from '../../common.ts'; + +/** + * Constants: + */ +export const DEFAULTS = {} as const; diff --git a/deploy/@tdb.slc/src/ui.Content/-sample/ui.Videos/mod.ts b/deploy/@tdb.slc/src/ui.Content/-sample/ui.Videos/mod.ts new file mode 100644 index 0000000000..d2b68a3bf4 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/-sample/ui.Videos/mod.ts @@ -0,0 +1,4 @@ +/** + * @module + */ +export { VideosIndex } from './ui.tsx'; diff --git a/deploy/@tdb.slc/src/ui.Content/-sample/ui.Videos/t.ts b/deploy/@tdb.slc/src/ui.Content/-sample/ui.Videos/t.ts new file mode 100644 index 0000000000..42c7a97c09 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/-sample/ui.Videos/t.ts @@ -0,0 +1,10 @@ +import type { t } from './common.ts'; + +/** + * <Component>: + */ +export type VideosIndexProps = { + signals?: t.VideoPlayerSignals; + theme?: t.CommonTheme; + style?: t.CssInput; +}; diff --git a/deploy/@tdb.slc/src/ui.Content/-sample/ui.Videos/ui.tsx b/deploy/@tdb.slc/src/ui.Content/-sample/ui.Videos/ui.tsx new file mode 100644 index 0000000000..38aae9ad1e --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/-sample/ui.Videos/ui.tsx @@ -0,0 +1,24 @@ +import React from 'react'; +import { type t, Color, css, Player } from './common.ts'; + +export const VideosIndex: React.FC<t.VideosIndexProps> = (props) => { + const {} = props; + const signalsRef = React.useRef(props.signals); + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ + color: theme.fg, + width: 688, + }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <Player.Video.View signals={signalsRef.current} /> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui.Content/-test.ui.ts b/deploy/@tdb.slc/src/ui.Content/-test.ui.ts new file mode 100644 index 0000000000..80e99d3a9c --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/-test.ui.ts @@ -0,0 +1 @@ +export * from '../ui/-test.ui.ts'; diff --git a/deploy/@tdb.slc/src/ui.Content/VIDEO.ts b/deploy/@tdb.slc/src/ui.Content/VIDEO.ts new file mode 100644 index 0000000000..d50a5a1989 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/VIDEO.ts @@ -0,0 +1,11 @@ +import { vimeo, TUBES } from './common.ts'; + +/** + * Index of Video IDs. + */ +export const VIDEO = { + Tubes: TUBES, + GroupScale: vimeo(727951677), + Trailer: vimeo(1068502644), + Overview: vimeo(1068653222), +} as const; diff --git a/deploy/@tdb.slc/src/ui.Content/common.ts b/deploy/@tdb.slc/src/ui.Content/common.ts new file mode 100644 index 0000000000..45b6d3a168 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/common.ts @@ -0,0 +1 @@ +export * from './common/mod.ts'; diff --git a/deploy/@tdb.slc/src/ui.Content/common/libs.ts b/deploy/@tdb.slc/src/ui.Content/common/libs.ts new file mode 100644 index 0000000000..256ffc6f31 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/common/libs.ts @@ -0,0 +1,3 @@ +export { DEFAULTS } from '../../ui/App.Render/mod.ts'; +export { App, LogoCanvas, LogoWordmark, Sheet } from '../../ui/mod.ts'; +export { TooSmall } from '../../ui/ui.TooSmall/mod.ts'; diff --git a/deploy/@tdb.slc/src/ui.Content/common/mod.ts b/deploy/@tdb.slc/src/ui.Content/common/mod.ts new file mode 100644 index 0000000000..5a921e0ef1 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/common/mod.ts @@ -0,0 +1,14 @@ +export * from '../../ui/common.ts'; +export { Icons } from '../ui.Icons.ts'; + +export * from './libs.ts'; +export type * as t from './t.ts'; + +/** + * A curried image importer helpers. + */ +export function i(importer: () => Promise<any>) { + return async () => { + return (await importer()).default as string; + }; +} diff --git a/deploy/@tdb.slc/src/ui.Content/common/t.ts b/deploy/@tdb.slc/src/ui.Content/common/t.ts new file mode 100644 index 0000000000..1e20daf06b --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/common/t.ts @@ -0,0 +1 @@ +export type * from '../../common/t.ts'; diff --git a/deploy/@tdb.slc/src/ui.Content/m.Content/-.test.ts b/deploy/@tdb.slc/src/ui.Content/m.Content/-.test.ts new file mode 100644 index 0000000000..4b981d294e --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/m.Content/-.test.ts @@ -0,0 +1,29 @@ +import { describe, expect, it } from '../../-test.ts'; +import { Factory, factory } from '../m.Factory/mod.ts'; +import { Content } from './mod.ts'; + +describe('Content', () => { + it('API', () => { + expect(Content.Factory).to.equal(Factory); + expect(Content.factory).to.equal(factory); + }); + + describe('Content.Is', () => { + const Is = Content.Is; + const NON = ['', 123, true, null, undefined, BigInt(0), Symbol('foo'), {}, []]; + + it('Is.video', async () => { + const test = (input: any, expected: boolean) => expect(Is.video(input)).to.eql(expected); + test(await factory('Trailer'), true); + test(await factory('Overview'), true); + NON.forEach((value) => test(value, false)); + }); + + it('Is.static', async () => { + const test = (input: any, expected: boolean) => expect(Is.static(input)).to.eql(expected); + test(await factory('Entry'), true); + test(await factory('Programme'), true); + NON.forEach((value) => test(value, false)); + }); + }); +}); diff --git a/deploy/@tdb.slc/src/ui.Content/m.Content/common.ts b/deploy/@tdb.slc/src/ui.Content/m.Content/common.ts new file mode 100644 index 0000000000..8cae67176a --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/m.Content/common.ts @@ -0,0 +1 @@ +export * from '../common.ts'; diff --git a/deploy/@tdb.slc/src/ui.Content/m.Content/m.Content.ts b/deploy/@tdb.slc/src/ui.Content/m.Content/m.Content.ts new file mode 100644 index 0000000000..e3b54a439f --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/m.Content/m.Content.ts @@ -0,0 +1,14 @@ +/** + * @module + * Content library + */ +import type { t } from './common.ts'; + +import { Factory, factory } from '../m.Factory/mod.ts'; +import { Is } from './m.Is.ts'; + +export const Content: t.ContentLib = { + Is, + Factory, + factory, +} as const; diff --git a/deploy/@tdb.slc/src/ui.Content/m.Content/m.Is.ts b/deploy/@tdb.slc/src/ui.Content/m.Content/m.Is.ts new file mode 100644 index 0000000000..5defc28f4a --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/m.Content/m.Is.ts @@ -0,0 +1,15 @@ +/** + * @module + * Content library + */ +import { type t, isRecord } from './common.ts'; + +export const Is: t.ContentIs = { + video(input): input is t.VideoContent { + return isRecord(input) && (input as t.VideoContent)['-type'] === 'VideoContent'; + }, + + static(input): input is t.StaticContent { + return isRecord(input) && (input as t.StaticContent)['-type'] === 'StaticContent'; + }, +} as const; diff --git a/deploy/@tdb.slc/src/ui.Content/m.Content/mod.ts b/deploy/@tdb.slc/src/ui.Content/m.Content/mod.ts new file mode 100644 index 0000000000..25a26e0dcb --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/m.Content/mod.ts @@ -0,0 +1,5 @@ +/** + * @module + * Content library. + */ +export { Content } from './m.Content.ts'; diff --git a/deploy/@tdb.slc/src/ui.Content/m.Content/t.ts b/deploy/@tdb.slc/src/ui.Content/m.Content/t.ts new file mode 100644 index 0000000000..f944474a4d --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/m.Content/t.ts @@ -0,0 +1,18 @@ +import type { t } from './common.ts'; + +/** + * Content library API. + */ +export type ContentLib = { + readonly Is: t.ContentIs; + readonly Factory: t.ContentFactoryLib; + readonly factory: t.ContentFactory; +}; + +/** + * Content flags. + */ +export type ContentIs = { + video(input: any): input is t.VideoContent; + static(input: any): input is t.StaticContent; +}; diff --git a/deploy/@tdb.slc/src/ui.Content/m.Factory/-.test.ts b/deploy/@tdb.slc/src/ui.Content/m.Factory/-.test.ts new file mode 100644 index 0000000000..821b816853 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/m.Factory/-.test.ts @@ -0,0 +1,35 @@ +import { describe, expect, it } from '../../-test.ts'; +import { Content } from '../mod.ts'; +import { VIDEO } from '../VIDEO.ts'; + +describe('Content.Factory', () => { + it('Entry', async () => { + const a = await Content.Factory.entry(); + const b = await Content.factory('Entry'); + expect(a?.id).to.eql('Entry'); + expect(b?.id).to.equal(a?.id); + }); + + it('Trailer', async () => { + const a = await Content.Factory.trailer(); + const b = await Content.factory('Trailer'); + expect(a?.id).to.eql('Trailer'); + expect(b?.id).to.equal(a?.id); + expect(a?.video?.props.src.value).to.eql(VIDEO.Trailer.src); + }); + + it('Overview', async () => { + const a = await Content.Factory.overview(); + const b = await Content.factory('Overview'); + expect(a?.id).to.eql('Overview'); + expect(b?.id).to.equal(a?.id); + expect(a?.video?.props.src.value).to.eql(VIDEO.Overview.src); + }); + + it('Programme', async () => { + const a = await Content.Factory.programme(); + const b = await Content.factory('Programme'); + expect(a?.id).to.eql('Programme'); + expect(b?.id).to.equal(a?.id); + }); +}); diff --git a/deploy/@tdb.slc/src/ui.Content/m.Factory/m.Factory.tsx b/deploy/@tdb.slc/src/ui.Content/m.Factory/m.Factory.tsx new file mode 100644 index 0000000000..ef793acdcf --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/m.Factory/m.Factory.tsx @@ -0,0 +1,22 @@ +import type { t } from '../common.ts'; + +export const Factory: t.ContentFactoryLib = { + entry: async () => (await import('../ui.Entry/mod.ts')).factory(), + trailer: async () => (await import('../ui.Trailer/mod.ts')).factory(), + overview: async () => (await import('../ui.Overview/mod.ts')).factory(), + programme: async () => (await import('../ui.Programme/mod.ts')).factory(), +} as const; + +/** + * Look up and dynamically import the content for the given ID. + */ +export const factory: t.ContentFactory = async (id) => { + if (id === 'Entry') return Factory.entry(); + if (id === 'Trailer') return Factory.trailer(); + if (id === 'Overview') return Factory.overview(); + if (id === 'Programme') return Factory.programme(); + + // Not found. + console.warn(`Content with id "${id}" not found.`); + return undefined; +}; diff --git a/deploy/@tdb.slc/src/ui.Content/m.Factory/mod.ts b/deploy/@tdb.slc/src/ui.Content/m.Factory/mod.ts new file mode 100644 index 0000000000..2d3abcaf69 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/m.Factory/mod.ts @@ -0,0 +1,5 @@ +/** + * @module + * Factories for dynamic ESM content imports. + */ +export { factory, Factory } from './m.Factory.tsx'; diff --git a/deploy/@tdb.slc/src/ui.Content/m.Factory/t.ts b/deploy/@tdb.slc/src/ui.Content/m.Factory/t.ts new file mode 100644 index 0000000000..a77a3f4d08 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/m.Factory/t.ts @@ -0,0 +1,16 @@ +import type { t } from '../common.ts'; + +/** + * Content factory library. + */ +export type ContentFactoryLib = { + entry(): Promise<t.StaticContent>; + trailer(): Promise<t.VideoContent>; + overview(): Promise<t.VideoContent>; + programme(): Promise<t.StaticContent>; +}; + +/** + * Content factory. + */ +export type ContentFactory = (id: t.ContentStage) => Promise<t.Content | undefined>; diff --git a/deploy/@tdb.slc/src/ui.Content/mod.ts b/deploy/@tdb.slc/src/ui.Content/mod.ts new file mode 100644 index 0000000000..f75d47ea05 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/mod.ts @@ -0,0 +1,5 @@ +/** + * @module + * Content entry point. + */ +export { Content } from './m.Content/mod.ts'; diff --git a/deploy/@tdb.slc/src/ui.Content/t.ts b/deploy/@tdb.slc/src/ui.Content/t.ts new file mode 100644 index 0000000000..480b327115 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/t.ts @@ -0,0 +1,39 @@ +import type { t } from './common.ts'; + +/** + * The content stages of the view. + */ +export type ContentStage = 'Entry' | 'Trailer' | 'Overview' | 'Programme'; + +/** + * Time based content definition. + */ +export type ContentTimestamps = t.Timestamps<ContentTimestamp>; +export type ContentTimestamp = ContentTimestampProps | ContentTimestampProps['column']; +export type ContentTimestampProps = { + column?: t.VideoContentRenderer; + pulldown?: t.VideoContentRenderer; +}; + +/** + * Content variation: Video. + */ +export type VideoContent = t.Content<V & { '-type': 'VideoContent' }>; +export type VideoContentProps = t.ContentProps<V>; +type V = { + id: t.ContentStage; + timestamps: ContentTimestamps; + showElapsed?: boolean; + playOnLoad?: boolean; + video?: t.VideoPlayerSignals; +}; + +export type VideoContentRenderer = t.ContentRenderer<t.VideoContentProps>; + +/** + * Content variation: Static. + */ +export type StaticContent = t.Content<{ + '-type': 'StaticContent'; + id: t.ContentStage; +}>; diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Entry/common.ts b/deploy/@tdb.slc/src/ui.Content/ui.Entry/common.ts new file mode 100644 index 0000000000..8cae67176a --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Entry/common.ts @@ -0,0 +1 @@ +export * from '../common.ts'; diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Entry/m.Factory.tsx b/deploy/@tdb.slc/src/ui.Content/ui.Entry/m.Factory.tsx new file mode 100644 index 0000000000..eb82ddcbe5 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Entry/m.Factory.tsx @@ -0,0 +1,12 @@ +import React from 'react'; +import { type t, DEFAULTS } from './common.ts'; +import { Entry } from './ui.tsx'; + +export function factory() { + const content: t.StaticContent = { + '-type': 'StaticContent', + id: 'Entry', + render: (props) => <Entry {...props} theme={DEFAULTS.theme.base} />, + }; + return content; +} diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Entry/mod.ts b/deploy/@tdb.slc/src/ui.Content/ui.Entry/mod.ts new file mode 100644 index 0000000000..048c873c21 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Entry/mod.ts @@ -0,0 +1 @@ +export * from './m.Factory.tsx'; diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Entry/ui.Button.Rounded.tsx b/deploy/@tdb.slc/src/ui.Content/ui.Entry/ui.Button.Rounded.tsx new file mode 100644 index 0000000000..bfc4218d28 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Entry/ui.Button.Rounded.tsx @@ -0,0 +1,97 @@ +import React, { useEffect, useState } from 'react'; +import { type t, Button, Color, css, Time } from './common.ts'; + +export type RoundedButtonProps = { + label?: string; + pulse?: boolean | Pulse; + theme?: t.CommonTheme; + style?: t.CssInput; + onClick?: MouseHandler; +}; + +export type Pulse = { pulsing?: boolean; duration?: t.Msecs; opacity?: t.Percent }; + +type P = RoundedButtonProps; +type MouseHandler = React.MouseEventHandler; + +/** + * Component: + */ +export const RoundedButton: React.FC<P> = (props) => { + const { label = 'Unnamed' } = props; + const pulse = wrangle.pulse(props.pulse); + + const [isOver, setOver] = useState(false); + const [pulseOpacity, setPulseOpacity] = useState(0); // New state for opacity pulsing + + /** + * Effects: + * When pulse is active (pulsing is true), set up a cycle that toggles the pulseOpacity value. + */ + useEffect(() => { + if (!pulse.pulsing) { + setPulseOpacity(0); + return; + } + + const time = Time.until(); + const duration = pulse.duration; + const toggleOpacity = () => { + setPulseOpacity((prev) => (prev === 0 ? 1 : 0)); + time.delay(duration, toggleOpacity); + }; + + toggleOpacity(); + return time.dispose; + }, [pulse.duration, pulse.pulsing]); + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ position: 'relative', color: theme.fg }), + body: { + base: css({ + position: 'relative', + minWidth: 70, + backgroundColor: Color.alpha(theme.fg, isOver ? 1 : 0.15), + transition: `background-color 100ms ease-in-out`, + Padding: [12, 25], + borderRadius: 40, + border: `solid 1px ${Color.alpha(theme.fg, 0.8)}`, + display: 'grid', + }), + content: css({ display: 'grid', placeItems: 'center' }), + pulse: css({ + Absolute: 0, + Padding: [12, 25], + borderRadius: 40, + backgroundColor: Color.alpha(theme.fg, pulse.opacity), + opacity: pulse ? pulseOpacity : 1, + transition: `opacity ${pulse.duration}ms ease-in-out`, + }), + }, + }; + + return ( + <Button theme={theme.name} onClick={props.onClick} onMouse={(e) => setOver(e.isOver)}> + <div className={styles.body.base.class}> + <div className={styles.body.pulse.class} /> + <div className={styles.body.content.class}>{label}</div> + </div> + </Button> + ); +}; + +/** + * Helpers + */ +const wrangle = { + pulse(input?: P['pulse']): Required<Pulse> { + const DEFAULT: Required<Pulse> = { duration: 1500, pulsing: false, opacity: 0.3 }; + if (!input) return DEFAULT; + if (input === true) return { ...DEFAULT, pulsing: true }; + return { ...DEFAULT, ...input }; + }, +} as const; diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Entry/ui.Buttons.tsx b/deploy/@tdb.slc/src/ui.Content/ui.Entry/ui.Buttons.tsx new file mode 100644 index 0000000000..4ef0fec586 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Entry/ui.Buttons.tsx @@ -0,0 +1,48 @@ +import React from 'react'; +import { Factory } from '../m.Factory/mod.ts'; +import { type t, Color, css } from './common.ts'; +import { RoundedButton } from './ui.Button.Rounded.tsx'; + +export type ButtonsProps = { + state: t.AppSignals; + theme?: t.CommonTheme; + style?: t.CssInput; +}; + +/** + * Component: + */ +export const Buttons: React.FC<ButtonsProps> = (props) => { + const { state } = props; + + /** + * Handlers: + */ + const stack = state.stack; + const showTrailer = async () => stack.push(await Factory.trailer()); + const showOverview = async () => stack.push(await Factory.overview()); + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ + color: theme.fg, + display: 'grid', + gridTemplateColumns: `auto auto`, + columnGap: '15px', + }), + button: css({ + Padding: [15, 40], + borderRadius: 40, + }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <RoundedButton theme={theme.name} label={'Trailer'} onClick={showTrailer} pulse /> + <RoundedButton theme={theme.name} label={'Overview'} onClick={showOverview} /> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Entry/ui.Install.tsx b/deploy/@tdb.slc/src/ui.Content/ui.Entry/ui.Install.tsx new file mode 100644 index 0000000000..781af19776 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Entry/ui.Install.tsx @@ -0,0 +1,37 @@ +import React from 'react'; +import { type t, css, Icons } from './common.ts'; + +export type InstallProps = { + theme?: t.CommonTheme; + style?: t.CssInput; +}; + +/** + * Component: + */ +export const Install: React.FC<InstallProps> = (props) => { + const {} = props; + + /** + * Render: + */ + const styles = { + base: css({ + position: 'relative', + userSelect: 'none', + display: 'grid', + justifyItems: 'center', + }), + label: css({ opacity: 0.2 }), + icon: css({ Size: 44, display: 'grid', placeItems: 'center' }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={styles.label.class}>{`Install ( SLC System )`}</div> + <div className={styles.icon.class}> + <Icons.Arrow.Down /> + </div> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Entry/ui.tsx b/deploy/@tdb.slc/src/ui.Content/ui.Entry/ui.tsx new file mode 100644 index 0000000000..0a02835d6c --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Entry/ui.tsx @@ -0,0 +1,65 @@ +import React from 'react'; +import { type t, Color, css, Icons, LogoCanvas, LogoWordmark } from './common.ts'; +import { Buttons } from './ui.Buttons.tsx'; +import { Install } from './ui.Install.tsx'; + +export type EntryProps = t.ContentProps & {}; + +const heartRateBPM = 72; +const delay = 60_000 / heartRateBPM; // NB: 60_000 ms in a minute. + +/** + * Component: + */ +export const Entry: React.FC<EntryProps> = (props) => { + const { state } = props; + const breakpoint = state.breakpoint; + const isStandalone = window.matchMedia('(display-mode: standalone)').matches; + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ + color: theme.fg, + pointerEvents: 'auto', + display: 'grid', + gridTemplateRows: '44px 1fr auto', + }), + header: css({ + MarginX: 10, + borderBottom: `solid 1px ${Color.alpha(theme.fg, 0.1)}`, + display: 'grid', + placeItems: 'center', + }), + body: css({ display: 'grid', placeItems: 'center' }), + footer: css({ display: breakpoint.name === 'Mobile' ? 'grid' : 'none' }), + brand: { + base: css({ display: 'grid', placeItems: 'center', rowGap: '35px' }), + canvas: css({ MarginX: 70 }), + wordmark: css({ width: 120 }), + }, + }; + + return ( + <div className={css(styles.base, props.style).class} onClick={() => state.stack.clear(1)}> + <div className={styles.header.class}> + <Icons.Add.Plus opacity={0.2} /> + </div> + <div className={styles.body.class}> + <div className={styles.brand.base.class}> + <LogoCanvas + theme={theme.name} + style={styles.brand.canvas} + selected={'purpose'} + selectionAnimation={{ delay, loop: true }} + /> + <LogoWordmark theme={theme.name} style={styles.brand.wordmark} /> + <Buttons theme={theme.name} state={state} style={{ marginTop: 100 }} /> + </div> + </div> + <div className={styles.footer.class}>{!isStandalone && <Install theme={theme.name} />}</div> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Icons.ts b/deploy/@tdb.slc/src/ui.Content/ui.Icons.ts new file mode 100644 index 0000000000..75fbee79a4 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Icons.ts @@ -0,0 +1,10 @@ +import { MdFace } from 'react-icons/md'; +import { Icons as Base, icon } from '../ui/ui.Icons.ts'; + +/** + * Icon collection: + */ +export const Icons = { + ...Base, + Face: icon(MdFace), +} as const; diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Overview/common.ts b/deploy/@tdb.slc/src/ui.Content/ui.Overview/common.ts new file mode 100644 index 0000000000..41de695810 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Overview/common.ts @@ -0,0 +1,2 @@ +export * from '../common.ts'; +export { CanvasSlug, Image } from '../ui/mod.ts'; diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/build-measure-learn.png b/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/build-measure-learn.png new file mode 100644 index 0000000000..f618f185ea Binary files /dev/null and b/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/build-measure-learn.png differ diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/customer-model.png b/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/customer-model.png new file mode 100644 index 0000000000..e995d1c768 Binary files /dev/null and b/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/customer-model.png differ diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/defining-purpose.png b/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/defining-purpose.png new file mode 100644 index 0000000000..6cb4ec8f0b Binary files /dev/null and b/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/defining-purpose.png differ diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/economic-model.png b/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/economic-model.png new file mode 100644 index 0000000000..8db203df07 Binary files /dev/null and b/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/economic-model.png differ diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/failure.png b/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/failure.png new file mode 100644 index 0000000000..6977a3e11b Binary files /dev/null and b/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/failure.png differ diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/impact-model.png b/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/impact-model.png new file mode 100644 index 0000000000..fdfdcd6e1e Binary files /dev/null and b/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/impact-model.png differ diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/model-parts.png b/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/model-parts.png new file mode 100644 index 0000000000..8779bf87e3 Binary files /dev/null and b/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/model-parts.png differ diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/refine.png b/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/refine.png new file mode 100644 index 0000000000..034cb75a48 Binary files /dev/null and b/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/refine.png differ diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/strategy.png b/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/strategy.png new file mode 100644 index 0000000000..fae31a0406 Binary files /dev/null and b/deploy/@tdb.slc/src/ui.Content/ui.Overview/img/strategy.png differ diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Overview/m.Factory.tsx b/deploy/@tdb.slc/src/ui.Content/ui.Overview/m.Factory.tsx new file mode 100644 index 0000000000..9186166fc9 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Overview/m.Factory.tsx @@ -0,0 +1,28 @@ +import React from 'react'; +import { VIDEO } from '../VIDEO.ts'; + +import { type t, DEFAULTS, Player } from './common.ts'; +import { timestamps } from './u.timestamps.tsx'; +import { Overview } from './ui.tsx'; + +/** + * Content: "Overview" (2 minute summary). + */ +export function factory() { + const theme = DEFAULTS.theme.sheet; + const src = VIDEO.Overview.src; + + const content: t.VideoContent = { + '-type': 'VideoContent', + + id: 'Overview', + playOnLoad: true, + video: Player.Video.signals({ + src, + scale: (e) => e.enlargeBy(2), // NB: enlarge 2px to crop out noise/line at top of video. + }), + render: (props) => <Overview {...props} theme={theme} />, + timestamps, + }; + return content; +} diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Overview/m.Images.ts b/deploy/@tdb.slc/src/ui.Content/ui.Overview/m.Images.ts new file mode 100644 index 0000000000..6018194cf6 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Overview/m.Images.ts @@ -0,0 +1,13 @@ +import { i } from './common.ts'; + +export const Images = { + build: i(() => import('./img/build-measure-learn.png')), + impactModel: i(() => import('./img/impact-model.png')), + customerModel: i(() => import('./img/customer-model.png')), + modelParts: i(() => import('./img/model-parts.png')), + definingPurpose: i(() => import('./img/defining-purpose.png')), + refine: i(() => import('./img/refine.png')), + economicModel: i(() => import('./img/economic-model.png')), + strategy: i(() => import('./img/strategy.png')), + failure: i(() => import('./img/failure.png')), +} as const; diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Overview/mod.ts b/deploy/@tdb.slc/src/ui.Content/ui.Overview/mod.ts new file mode 100644 index 0000000000..6192e59e17 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Overview/mod.ts @@ -0,0 +1,2 @@ +export * from './m.Factory.tsx'; +export * from './m.Images.ts'; diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Overview/u.timestamps.tsx b/deploy/@tdb.slc/src/ui.Content/ui.Overview/u.timestamps.tsx new file mode 100644 index 0000000000..e3bebd5cbe --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Overview/u.timestamps.tsx @@ -0,0 +1,73 @@ +import React from 'react'; +import { type t, CanvasSlug, Image } from './common.ts'; +import { Images } from './m.Images.ts'; + +const Slug = CanvasSlug; + +/** + * Overview: + */ +export const timestamps: t.ContentTimestamps = { + '00:00:00.000': (p) => <Slug {...p} logo={'SLC'} />, + '00:00:01.000': (p) => <Slug {...p} />, + '00:00:02.300': (p) => <Slug {...p} text={'slow'} />, + '00:00:03.210': (p) => <Slug {...p} text={'risky'} />, + '00:00:04.300': (p) => <Slug {...p} text={'mostly\nunsuccessful'} />, + '00:00:06.350': (p) => <Slug {...p} />, + '00:00:08.000': { + column: (p) => <Slug {...p} logo={'SLC'} />, + pulldown: (p) => <Image.View src={Images.failure()} />, + }, + '00:00:16.000': (p) => <Slug {...p} />, + '00:00:16.850': (p) => <Slug {...p} text={'business model'} />, + '00:00:18.580': (p) => <Slug {...p} text={'economic\nfoundation'} />, + '00:00:23.550': (p) => <Slug {...p} text={'lasting\nmeasurable\nimpact'} selected={'impact'} />, + '00:00:36.000': (p) => <Slug {...p} />, + '00:00:36.250': (p) => <Slug {...p} text={'fast fail'} />, + '00:00:42.000': (p) => <Slug {...p} text={'living dead'} />, + '00:00:53.000': (p) => <Slug {...p} />, + '00:00:58.300': (p) => <Slug {...p} text={'shared\nsense-making'} />, + '00:01:04.000': (p) => <Slug {...p} logo={'SLC'} />, + '00:01:09.000': { + column: (p) => <Slug {...p} logo={'SLC'} />, + pulldown: (p) => <Image.View src={Images.build()} padding={'5%'} />, + }, + '00:01:23.000': (p) => <Slug {...p} logo={'SLC'} />, + '00:01:25.250': { + column: (p) => <Slug {...p} logo={'SLC'} />, + pulldown: (p) => <Image.View src={Images.definingPurpose()} />, + }, + '00:01:37.900': { + column: (p) => <Slug {...p} logo={'SLC'} />, + pulldown: (p) => <Image.View src={Images.modelParts()} padding={'10%'} />, + }, + '00:01:49.000': (p) => <Slug {...p} logo={'SLC'} />, + '00:01:50.810': { + column: (p) => <Slug {...p} logo={'SLC'} />, + pulldown: (p) => <Image.View src={Images.customerModel()} />, + }, + '00:02:11.000': (p) => <Slug {...p} text={'simple'} />, + '00:02:16.000': (p) => <Slug {...p} text={'complex'} />, + '00:02:23.360': (p) => <Slug {...p} text={'complex\nprotocol'} />, + '00:02:33.460': (p) => <Slug {...p} />, + '00:02:37.000': { + column: (p) => <Slug {...p} logo={'SLC'} />, + pulldown: (p) => <Image.View src={Images.impactModel()} />, + }, + '00:02:43.660': (p) => <Slug {...p} text={'proveable\nimpact'} />, + '00:02:48.000': { + column: (p) => <Slug {...p} logo={'SLC'} />, + pulldown: (p) => <Image.View src={Images.economicModel()} />, + }, + '00:03:04.660': (p) => <Slug {...p} logo={'SLC'} />, + '00:03:10.000': { + column: (p) => <Slug {...p} logo={'SLC'} />, + pulldown: (p) => <Image.View src={Images.refine()} />, + }, + '00:03:15.730': (p) => <Slug {...p} text={'genuinely\nhigh\npotential'} />, + '00:03:19.250': { + column: (p) => <Slug {...p} logo={'SLC'} />, + pulldown: (p) => <Image.View src={Images.strategy()} />, + }, + '00:03:33.000': (p) => <Slug {...p} logo={'CC'} />, +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Overview/ui.tsx b/deploy/@tdb.slc/src/ui.Content/ui.Overview/ui.tsx new file mode 100644 index 0000000000..e6d4f38dfa --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Overview/ui.tsx @@ -0,0 +1,59 @@ +import React from 'react'; +import { ElapsedTime, usePulldown, useTimestamps } from '../ui/mod.ts'; +import { type t, css, Player, Sheet, Time } from './common.ts'; + +export type OverviewProps = t.VideoContentProps; + +/** + * Component: + */ +export const Overview: React.FC<OverviewProps> = (props) => { + const { state, content } = props; + const { showElapsed = true } = content; + + const player = content.video; + const timestamp = useTimestamps(props, player); + usePulldown(props, timestamp); + + /** + * Effect: Play on load. + */ + React.useEffect(() => { + if (content.playOnLoad) player?.play(); + }, [player]); + + /** + * Render: + */ + const edge: t.SheetMarginInput = state.breakpoint.name === 'Desktop' ? ['1fr', 390, '1fr'] : 10; + const styles = { + base: css({ marginTop: 44 }), + body: css({ position: 'relative', display: 'grid', gridTemplateRows: '1fr auto' }), + content: css({ display: 'grid' }), + player: css({ marginBottom: -1 }), + }; + + const elBody = ( + <div className={styles.body.class}> + <div className={styles.content.class}>{timestamp.column}</div> + <Player.Video.View + signals={player} + style={styles.player} + onEnded={() => Time.delay(1000, () => state.stack.clear(1))} // NB: add time buffer before hiding. + /> + </div> + ); + + return ( + <Sheet + {...props} + style={styles.base} + theme={props.theme} + edgeMargin={edge} + orientation={'Bottom:Up'} + > + {elBody} + <ElapsedTime player={player} abs={true} show={showElapsed} /> + </Sheet> + ); +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Programme/m.Factory.tsx b/deploy/@tdb.slc/src/ui.Content/ui.Programme/m.Factory.tsx new file mode 100644 index 0000000000..f6633ee069 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Programme/m.Factory.tsx @@ -0,0 +1,27 @@ +import React from 'react'; +import { type t, Color, css, Sheet, App, Button, DEFAULTS } from '../common.ts'; + +/** + * Content: "Programme" + */ +export function factory() { + const sheetTheme = DEFAULTS.theme.sheet; + const content: t.StaticContent = { + '-type': 'StaticContent', + id: 'Programme', + + render(props) { + const styles = { + base: css({ padding: 10 }), + }; + + return ( + <Sheet {...props} theme={sheetTheme} orientation={'Top:Down'}> + <div className={styles.base.class}>Hello Programme</div> + {/* {props.children} */} + </Sheet> + ); + }, + }; + return content; +} diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Programme/mod.ts b/deploy/@tdb.slc/src/ui.Content/ui.Programme/mod.ts new file mode 100644 index 0000000000..048c873c21 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Programme/mod.ts @@ -0,0 +1 @@ +export * from './m.Factory.tsx'; diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Trailer/common.ts b/deploy/@tdb.slc/src/ui.Content/ui.Trailer/common.ts new file mode 100644 index 0000000000..370c668bd6 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Trailer/common.ts @@ -0,0 +1,2 @@ +export * from '../common.ts'; +export { CanvasSlug } from '../ui/mod.ts'; diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Trailer/m.Factory.tsx b/deploy/@tdb.slc/src/ui.Content/ui.Trailer/m.Factory.tsx new file mode 100644 index 0000000000..607b76746d --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Trailer/m.Factory.tsx @@ -0,0 +1,30 @@ +import React from 'react'; + +import { VIDEO } from '../VIDEO.ts'; +import { type t, DEFAULTS, Player } from './common.ts'; +import { timestamps } from './u.timestamps.tsx'; +import { Trailer } from './ui.tsx'; + +/** + * Content: "Trailer" (30 second intro). + */ +export function factory() { + const theme = DEFAULTS.theme.sheet; + const src = VIDEO.Trailer.src; + + const content: t.VideoContent = { + '-type': 'VideoContent', + id: 'Trailer', + + playOnLoad: true, + video: Player.Video.signals({ + src, + scale: (e) => e.enlargeBy(2), // NB: enlarge 2px to crop out noise/line at top of video. + }), + + render: (props) => <Trailer {...props} theme={theme} />, + timestamps, + }; + + return content; +} diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Trailer/mod.ts b/deploy/@tdb.slc/src/ui.Content/ui.Trailer/mod.ts new file mode 100644 index 0000000000..048c873c21 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Trailer/mod.ts @@ -0,0 +1 @@ +export * from './m.Factory.tsx'; diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Trailer/u.timestamps.tsx b/deploy/@tdb.slc/src/ui.Content/ui.Trailer/u.timestamps.tsx new file mode 100644 index 0000000000..d93224b8d5 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Trailer/u.timestamps.tsx @@ -0,0 +1,24 @@ +import React from 'react'; +import { type t, CanvasPanel, CanvasSlug } from './common.ts'; + +const Slug = CanvasSlug; +const panels = CanvasPanel.list; + +/** + * Trailer: + */ +export const timestamps: t.ContentTimestamps = { + '00:00:00.000': (p) => <Slug {...p} logo={'SLC'} />, + '00:00:00.001': (p) => <Slug {...p} text={'social ventures'} />, + '00:00:03.560': (p) => <Slug {...p} text={'good ideas'} />, + '00:00:07.000': (p) => <Slug {...p} text={'wrong priorities'} />, + + '00:00:11.870': (p) => <Slug {...p} selected={'purpose'} text={'purpose'} />, + '00:00:19.600': (p) => <Slug {...p} selected={panels} text={'decompose'} />, + '00:00:23.500': (p) => <Slug {...p} selected={panels.toReversed()} text={'recompose'} />, + '00:00:29.540': (p) => <Slug {...p} selected={'purpose'} logo={'SLC'} />, + '00:00:34.000': (p) => <Slug {...p} selected={'purpose'} text={'coherence'} />, + '00:00:37.590': (p) => <Slug {...p} selected={'purpose'} logo={'SLC'} />, + '00:00:47.350': (p) => <Slug {...p} selected={'purpose'} text={'shared clarity'} />, + '00:00:55.620': (p) => <Slug {...p} selected={'purpose'} logo={'CC'} />, +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui.Trailer/ui.tsx b/deploy/@tdb.slc/src/ui.Content/ui.Trailer/ui.tsx new file mode 100644 index 0000000000..3b997867dc --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui.Trailer/ui.tsx @@ -0,0 +1,53 @@ +import React from 'react'; +import { css, Player, Sheet, type t, Time } from '../common.ts'; +import { ElapsedTime, useTimestamps } from '../ui/mod.ts'; + +export type TrailerProps = t.VideoContentProps; + +/** + * Component: + */ +export const Trailer: React.FC<TrailerProps> = (props) => { + const { state, content } = props; + const { showElapsed = true } = content; + const player = content.video; + const timestamp = useTimestamps(props, player); + + /** + * Effect: Play on load. + */ + React.useEffect(() => { + if (content.playOnLoad) player?.play(); + }, [player]); + + /** + * Render: + */ + const edge: t.SheetMarginInput = state.breakpoint.name === 'Desktop' ? ['1fr', 390, '1fr'] : 10; + const styles = { + base: css({ marginTop: 44 }), + body: css({ position: 'relative', display: 'grid', gridTemplateRows: '1fr auto' }), + content: css({ display: 'grid' }), + player: css({ marginBottom: -1 }), + }; + + return ( + <Sheet + {...props} + theme={props.theme} + style={styles.base} + edgeMargin={edge} + orientation={'Bottom:Up'} + > + <div className={styles.body.class}> + <div className={styles.content.class}>{timestamp.column}</div> + <Player.Video.View + signals={player} + style={styles.player} + onEnded={() => Time.delay(1000, () => state.stack.clear(1))} // NB: add time buffer before hiding. + /> + </div> + <ElapsedTime player={player} abs={true} show={showElapsed} /> + </Sheet> + ); +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/common.ts b/deploy/@tdb.slc/src/ui.Content/ui/common.ts new file mode 100644 index 0000000000..8cae67176a --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/common.ts @@ -0,0 +1 @@ +export * from '../common.ts'; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/mod.ts b/deploy/@tdb.slc/src/ui.Content/ui/mod.ts new file mode 100644 index 0000000000..e6db1acb87 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/mod.ts @@ -0,0 +1,6 @@ +export * from './ui.CanvasSlug/mod.ts'; +export * from './ui.ElapsedTime.tsx'; +export * from './ui.Image/mod.ts'; +export * from './ui.Pulldown/mod.ts'; + +export * from './use.Timestamps.ts'; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/t.ts b/deploy/@tdb.slc/src/ui.Content/ui/t.ts new file mode 100644 index 0000000000..74bb1be51e --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/t.ts @@ -0,0 +1,14 @@ +import { type t } from './common.ts'; + +/** + * Hook for managing content. + */ +export type UseTimestamps = ( + props: t.VideoContentProps, + player?: t.VideoPlayerSignals, +) => TimestampsHook; + +export type TimestampsHook = { + readonly column?: t.ReactNode; + readonly pulldown?: t.ReactNode; +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.CanvasSlug/-SPEC.Debug.tsx b/deploy/@tdb.slc/src/ui.Content/ui/ui.CanvasSlug/-SPEC.Debug.tsx new file mode 100644 index 0000000000..003a03f433 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.CanvasSlug/-SPEC.Debug.tsx @@ -0,0 +1,87 @@ +import React from 'react'; +import { canvasSelectedButton } from '../../../ui/ui.Logo.Canvas/-SPEC.Debug.tsx'; +import { type t, Button, css, D, Signal } from './common.ts'; + +type P = t.CanvasSlugProps; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + const s = Signal.create; + const props = { + debug: s<P['debug']>(true), + theme: s<P['theme']>('Light'), + selected: s<P['selected']>(), + logo: s<P['logo']>(D.logo), + text: s<P['text']>(), + }; + const api = { + props, + listen() { + const p = props; + p.debug.value; + p.theme.value; + p.selected.value; + p.logo.value; + p.text.value; + }, + }; + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => debug.listen()); + + /** + * Render: + */ + const styles = { + base: css({}), + title: css({ fontWeight: 'bold', marginBottom: 10 }), + cols: css({ display: 'grid', gridTemplateColumns: 'auto 1fr auto' }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={css(styles.title, styles.cols).class}>{'CanvasSlug'}</div> + + <Button block label={() => `debug: ${p.debug}`} onClick={() => Signal.toggle(p.debug)} /> + <Button + block + label={() => `theme: ${p.theme}`} + onClick={() => Signal.cycle<P['theme']>(p.theme, ['Light', 'Dark'])} + /> + + <hr /> + <Button + block + label={() => `logo: "${p.logo}"`} + onClick={() => Signal.cycle<P['logo']>(p.logo, [undefined, 'SLC', 'CC'])} + /> + <Button + block + label={() => `text: ${p.text.value ?? '<undefined>'}`} + onClick={() => { + Signal.cycle<P['text']>(p.text, [undefined, 'hello', 'purpose\nimpact']); + }} + /> + {canvasSelectedButton(p.selected)} + + <hr /> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.CanvasSlug/-SPEC.tsx b/deploy/@tdb.slc/src/ui.Content/ui/ui.CanvasSlug/-SPEC.tsx new file mode 100644 index 0000000000..f54e8837f4 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.CanvasSlug/-SPEC.tsx @@ -0,0 +1,37 @@ +import { Dev, Signal, Spec } from '../../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { CanvasSlug } from './mod.ts'; + +export default Spec.describe('CanvasSlug', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + Dev.Theme.signalEffect(ctx, p.theme, 1); + Signal.effect(() => { + debug.listen(); + ctx.redraw(); + }); + + ctx.subject + .size('fill-y', 150) + .display('grid') + .render((e) => ( + <CanvasSlug + style={{ width: 390 }} + debug={p.debug.value} + theme={p.theme.value} + selected={p.selected.value} + logo={p.logo.value} + text={p.text.value} + /> + )); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.CanvasSlug/common.ts b/deploy/@tdb.slc/src/ui.Content/ui/ui.CanvasSlug/common.ts new file mode 100644 index 0000000000..24d7deccd0 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.CanvasSlug/common.ts @@ -0,0 +1,12 @@ +import type { t } from './common.ts'; +export * from '../common.ts'; + +/** + * Constants: + */ +const logo: t.LogoKind = 'SLC'; + +export const DEFAULTS = { + logo, +} as const; +export const D = DEFAULTS; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.CanvasSlug/mod.ts b/deploy/@tdb.slc/src/ui.Content/ui/ui.CanvasSlug/mod.ts new file mode 100644 index 0000000000..1a70bbc6fe --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.CanvasSlug/mod.ts @@ -0,0 +1,4 @@ +/** + * @module + */ +export { CanvasSlug } from './ui.tsx'; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.CanvasSlug/t.ts b/deploy/@tdb.slc/src/ui.Content/ui/ui.CanvasSlug/t.ts new file mode 100644 index 0000000000..c183a8a3ef --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.CanvasSlug/t.ts @@ -0,0 +1,13 @@ +import type { t } from './common.ts'; + +/** + * <Component>: + */ +export type CanvasSlugProps = { + selected?: t.CanvasPanel | t.CanvasPanel[]; + logo?: t.LogoKind; + text?: string; + debug?: boolean; + theme?: t.CommonTheme; + style?: t.CssInput; +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.CanvasSlug/ui.tsx b/deploy/@tdb.slc/src/ui.Content/ui/ui.CanvasSlug/ui.tsx new file mode 100644 index 0000000000..2a0a39f9d3 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.CanvasSlug/ui.tsx @@ -0,0 +1,81 @@ +import React, { useState } from 'react'; +import { + type t, + Color, + css, + LogoCanvas, + LogoWordmark, + Time, + TooSmall, + useSizeObserver, + ReactString, +} from './common.ts'; +import { FadeText } from '../ui.FadeText/mod.ts'; + +export const CanvasSlug: React.FC<t.CanvasSlugProps> = (props) => { + const { debug = false, logo } = props; + const size = useSizeObserver(); + + const [ready, setReady] = useState(false); + + /** + * Effect: + */ + React.useEffect(() => { + const time = Time.until(); + time.delay(500, () => setReady(true)); + return time.dispose; + }, [size.ready]); + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ + position: 'relative', + display: 'grid', + opacity: ready ? 1 : 0, + transition: '1200ms', + }), + body: css({ position: 'relative', display: 'grid', placeItems: 'center' }), + layout: css({ display: 'grid', placeItems: 'center', rowGap: '30px' }), + canvas: css({ position: 'relative', width: 280 }), + logo: css({ width: logo === 'SLC' ? 130 : 200 }), + footer: css({ + position: 'relative', + height: 115, + width: '100%', + display: 'grid', + placeItems: 'center', + }), + text: css({ Absolute: 0 }), + }; + + const elText = <FadeText text={logo ? '' : props.text} style={styles.text} />; + const elWordmark = logo && <LogoWordmark theme={theme.name} logo={logo} style={styles.logo} />; + const elFooter = ( + <div className={styles.footer.class}> + {elText} + {elWordmark} + </div> + ); + + const elBody = ( + <div className={styles.body.class}> + <div className={styles.layout.class}> + <LogoCanvas theme={theme.name} style={styles.canvas} selected={props.selected} /> + {elFooter} + </div> + </div> + ); + + const elTooSmall = size.ready && size.height < 320 && <TooSmall />; + + return ( + <div ref={size.ref} className={css(styles.base, props.style).class}> + {elTooSmall || elBody} + {debug && size.toElement({ Absolute: [6, 8, null, null], opacity: 0.3 })} + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.ElapsedTime.tsx b/deploy/@tdb.slc/src/ui.Content/ui/ui.ElapsedTime.tsx new file mode 100644 index 0000000000..68e058f49f --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.ElapsedTime.tsx @@ -0,0 +1,61 @@ +import React from 'react'; +import { type t, css, Signal } from './common.ts'; + +export type ElapsedTimeProps = { + player?: t.VideoPlayerSignals; + abs?: t.CssEdgesInput | boolean; + show?: boolean; + style?: t.CssInput; +}; + +type P = ElapsedTimeProps; + +/** + * Component: + */ +export const ElapsedTime: React.FC<P> = (props) => { + const { player, show = true } = props; + const currentTime = player?.props.currentTime.value ?? 0; + + Signal.useRedrawEffect(() => player?.props.currentTime.value); + if (!show) return null; + if (currentTime <= 0) return null; + + /** + * Render: + */ + const styles = { + base: css({ + Absolute: wrangle.abs(props), + userSelect: 'none', + fontSize: 11, + opacity: 0.5, + }), + }; + + return <div className={css(styles.base, props.style).class}>{formatTime(currentTime)}</div>; +}; + +/** + * Helpers + */ +const formatTime = (timeInSeconds: number): string => { + const mins = Math.floor(timeInSeconds / 60); + const secs = Math.floor(timeInSeconds % 60); + const centi = Math.floor((timeInSeconds % 1) * 100); + const fmt = (value: number) => String(value).padStart(2, '0'); + return `${fmt(mins)}:${fmt(secs)}:${fmt(centi)}`; +}; + +/** + * Helpers + */ +const wrangle = { + abs(props: P): t.CssEdgesInput { + const { abs } = props; + if (!abs) return; + if (abs === true) return [5, 6, null, null]; + if (Array.isArray(abs)) return abs; + return; + }, +} as const; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/-SPEC.Debug.tsx b/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/-SPEC.Debug.tsx new file mode 100644 index 0000000000..8b3225728b --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/-SPEC.Debug.tsx @@ -0,0 +1,97 @@ +import React from 'react'; +import { Lorem } from '../../-test.ui.ts'; +import { type t, Button, css, Signal } from './common.ts'; + +type P = t.FadeTextProps; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + const s = Signal.create; + const props = { + theme: s<P['theme']>('Light'), + text: s<P['text']>('Lorem'), + loremIndex: s(0), + }; + const api = { + props, + listen() { + const p = props; + p.theme.value; + p.text.value; + p.loremIndex.value; + }, + }; + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => debug.listen()); + + /** + * Render: + */ + const styles = { + base: css({}), + title: css({ fontWeight: 'bold', marginBottom: 10 }), + cols: css({ display: 'grid', gridTemplateColumns: 'auto 1fr auto' }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={css(styles.title, styles.cols).class}> + <div>{'FadeText'}</div> + </div> + + <Button + block + label={() => `theme: ${p.theme.value ?? '<undefined>'}`} + onClick={() => Signal.cycle<P['theme']>(p.theme, ['Light', 'Dark'])} + /> + + <hr /> + + <Button + block + label={() => { + const value = p.text.value; + return `text: ${value ? `"${value}"` : '<undefined>'}`; + }} + onClick={() => { + const index = (p.loremIndex.value += 1); + p.text.value = Lorem.text.split(' ')[index]; + }} + /> + + <Button + block + label={() => `text: (clear)`} + onClick={() => { + p.text.value = undefined; + }} + /> + + <Button + block + label={() => `text: (multi-line)`} + onClick={() => { + p.text.value = 'multi\nline'; + }} + /> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/-SPEC.tsx b/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/-SPEC.tsx new file mode 100644 index 0000000000..83835c0d34 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/-SPEC.tsx @@ -0,0 +1,28 @@ +import { Dev, Spec, Signal, Lorem } from '../../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { FadeText } from './mod.ts'; + +export default Spec.describe('FadeText', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + Dev.Theme.signalEffect(ctx, p.theme, 1); + Signal.effect(() => { + debug.listen(); + ctx.redraw(); + }); + + ctx.subject + .size([390, 200]) + .display('grid') + .render((e) => <FadeText theme={p.theme.value} text={p.text.value} />); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/common.ts b/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/common.ts new file mode 100644 index 0000000000..2bb91a0351 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/common.ts @@ -0,0 +1,13 @@ +export * from '../common.ts'; + +/** + * Constants: + */ +export const DEFAULTS = { + fontSize: 36, + fontWeight: 'bold', + letterSpacing: '-0.02em', + lineHeight: 0.95, + duration: 700, +} as const; +export const D = DEFAULTS; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/mod.ts b/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/mod.ts new file mode 100644 index 0000000000..e1484206a6 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/mod.ts @@ -0,0 +1,4 @@ +/** + * @module + */ +export { FadeText } from './ui.tsx'; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/t.ts b/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/t.ts new file mode 100644 index 0000000000..931e3c0220 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/t.ts @@ -0,0 +1,17 @@ +import type { t } from './common.ts'; + +/** + * <Component>: + */ +export type FadeTextProps = { + text?: string; + duration?: t.Msecs; + + // Style: + fontSize?: t.CssProps['fontSize']; + fontWeight?: t.CssProps['fontWeight']; + lineHeight?: t.CssProps['lineHeight']; + letterSpacing?: t.CssProps['letterSpacing']; + theme?: t.CommonTheme; + style?: t.CssInput; +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/ui.Item.tsx b/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/ui.Item.tsx new file mode 100644 index 0000000000..16406dd6c4 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/ui.Item.tsx @@ -0,0 +1,50 @@ +import React, { useEffect, useState } from 'react'; +import { type t, css, Time, ReactString } from './common.ts'; + +export type FadeItem = { id: number; text: string; fadingOut: boolean }; + +export type ItemProps = { + item: FadeItem; + duration: number; + style?: t.CssInput; +}; + +/** + * Component: + */ +export const Item: React.FC<ItemProps> = (props) => { + const { item, duration } = props; + const [visible, setVisible] = useState<boolean>(false); + + /** + * Effect: fade-out + */ + useEffect(() => { + if (!item.fadingOut) { + const time = Time.until(); + time.delay(0, () => setVisible(true)); + return time.dispose; + } + }, [item.fadingOut]); + + /** + * Effect: When the item is marked as fading out, update the visible flag. + */ + useEffect(() => { + if (item.fadingOut) setVisible(false); + }, [item.fadingOut]); + + /** + * Render: + */ + const styles = { + base: css({ display: 'grid', placeItems: 'center' }), + body: css({ transition: `opacity ${duration}ms`, opacity: visible ? 1 : 0 }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={styles.body.class}>{ReactString.break(item.text)}</div> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/ui.tsx b/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/ui.tsx new file mode 100644 index 0000000000..04c8145605 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.FadeText/ui.tsx @@ -0,0 +1,51 @@ +import React, { useEffect, useRef, useState } from 'react'; +import { Color, css, D, Time, type t } from './common.ts'; +import { Item, type FadeItem } from './ui.Item.tsx'; + +export const FadeText: React.FC<t.FadeTextProps> = (props) => { + const { + text = '', + fontSize = D.fontSize, + fontWeight = D.fontWeight, + letterSpacing = D.letterSpacing, + lineHeight = D.lineHeight, + duration = D.duration, + } = props; + + const [items, setItems] = useState<FadeItem[]>([{ id: 0, text, fadingOut: false }]); + const next = useRef(1); + + /** + * Effect: + */ + useEffect(() => { + setItems((prev) => { + if (prev.length && prev[prev.length - 1].text === text) return prev; // ā If the latest item already shows the same text, do nothing. + const updated = prev.map((item) => ({ ...item, fadingOut: true })); // ā Mark all existing items as fading out. + return [...updated, { id: next.current++, text, fadingOut: false }]; // ā Add the new text item with a unique id. + }); + + // Schedule removal of items that have faded out. + const time = Time.until(); + time.delay(duration, () => setItems((prev) => prev.filter((item) => !item.fadingOut))); + return time.dispose; + }, [text, duration]); + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const color = theme.fg; + const styles = { + base: css({ position: 'relative' }), + item: css({ Absolute: 0, color, fontSize, fontWeight, letterSpacing, lineHeight }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + {items.map((item) => { + return <Item key={item.id} item={item} duration={duration} style={styles.item} />; + })} + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/-SPEC.Debug.tsx b/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/-SPEC.Debug.tsx new file mode 100644 index 0000000000..3c9d8c3b26 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/-SPEC.Debug.tsx @@ -0,0 +1,89 @@ +import React from 'react'; +import { Images } from '../../ui.Overview/mod.ts'; +import { type t, Button, css, Signal } from './common.ts'; + +type P = t.ImageViewProps; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + const s = Signal.create; + const props = { + theme: s<P['theme']>('Light'), + padding: s<P['padding']>(), + src: s<string>(), + }; + const api = { + props, + listen() { + const p = props; + p.theme.value; + p.src.value; + p.padding.value; + }, + }; + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => debug.listen()); + + /** + * Render: + */ + const styles = { + base: css({}), + title: css({ fontWeight: 'bold', marginBottom: 10 }), + cols: css({ display: 'grid', gridTemplateColumns: 'auto 1fr auto' }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={css(styles.title, styles.cols).class}> + <div>{'Image'}</div> + </div> + + <Button + block + label={() => `theme: ${p.theme.value ?? '<undefined>'}`} + onClick={() => Signal.cycle<P['theme']>(p.theme, ['Light', 'Dark'])} + /> + + <Button + block + label={() => { + const value = p.src.value; + return `src: ${value ? `...${value.slice(-36)}` : '<undefined>'}`; + }} + onClick={async () => { + const images = await Promise.all(Object.values(Images).map((loader) => loader())); + Signal.cycle<P['src']>(p.src, images); + }} + /> + + <Button + block + label={() => `padding: ${p.padding.value ?? '<undefined>'}`} + onClick={() => { + Signal.cycle<P['padding']>(p.padding, [undefined, 30, '20%', [10, 20, 50, 80]]); + }} + /> + + <hr /> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/-SPEC.tsx b/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/-SPEC.tsx new file mode 100644 index 0000000000..cf650198e0 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/-SPEC.tsx @@ -0,0 +1,30 @@ +import { Dev, Signal, Spec } from '../../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { Image } from './mod.ts'; + +export default Spec.describe('Image', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + Dev.Theme.signalEffect(ctx, p.theme, 1); + Signal.effect(() => { + debug.listen(); + ctx.redraw(); + }); + + ctx.subject + .size('fill', [150, 100]) + .display('grid') + .render((e) => { + return <Image.View theme={p.theme.value} src={p.src.value} padding={p.padding.value} />; + }); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/common.ts b/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/common.ts new file mode 100644 index 0000000000..a45006948c --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/common.ts @@ -0,0 +1,7 @@ +export * from '../common.ts'; + +/** + * Constants: + */ +export const DEFAULTS = {} as const; +export const D = DEFAULTS; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/m.Image.ts b/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/m.Image.ts new file mode 100644 index 0000000000..a4b9978011 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/m.Image.ts @@ -0,0 +1,6 @@ +import { type t } from './common.ts'; +import { ImageView as View } from './ui.tsx'; + +export const Image: t.ImageLib = { + View, +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/mod.ts b/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/mod.ts new file mode 100644 index 0000000000..68587a4576 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/mod.ts @@ -0,0 +1,4 @@ +/** + * @module + */ +export { Image } from './m.Image.ts'; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/t.ts b/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/t.ts new file mode 100644 index 0000000000..ec0552c853 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/t.ts @@ -0,0 +1,18 @@ +import type { t } from './common.ts'; + +/** + * Image content render library. + */ +export type ImageLib = { + View: React.FC<t.ImageViewProps>; +}; + +/** + * <Component>: + */ +export type ImageViewProps = { + src?: string | Promise<string>; + padding?: t.CssEdgesInput; + theme?: t.CommonTheme; + style?: t.CssInput; +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/ui.tsx b/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/ui.tsx new file mode 100644 index 0000000000..692afe0e82 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.Image/ui.tsx @@ -0,0 +1,118 @@ +import React, { useState, useEffect } from 'react'; +import { type t, Is, Color, css, Icons, Spinners } from './common.ts'; + +type P = t.ImageViewProps; + +export const ImageView: React.FC<P> = (props) => { + const [loading, setLoading] = useState(true); + const [error, setError] = useState(false); + const [src, setSrc] = useState<string>(); + + /** + * Handlers: + */ + const handleLoad = () => { + setLoading(false); + }; + + const handleError = () => { + setLoading(false); + setError(true); + }; + + /** + * Effects: + */ + useEffect(() => { + wrangle.src(props.src, setSrc); + }, [props.src]); + + useEffect(() => { + setLoading(true); + setError(false); + }, [src]); + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ color: theme.fg, display: 'grid' }), + img: css({ + Absolute: [-99999, null, null, -99999], + opacity: 0, + pointerEvents: 'none', + }), + display: { + base: css({ Absolute: props.padding, display: 'grid' }), + img: css({ + backgroundImage: src ? `url(${src})` : undefined, + backgroundRepeat: 'no-repeat', + backgroundPosition: 'center', + backgroundSize: 'contain', + opacity: loading || error ? 0 : 1, + transition: `opacity 400ms`, + }), + }, + spinner: css({ + Absolute: 0, + display: 'grid', + placeItems: 'center', + pointerEvents: 'none', + }), + error: { + base: css({ + Absolute: 0, + pointerEvents: 'none', + opacity: error ? 1 : 0, + transition: `opacity 400ms`, + display: 'grid', + placeItems: 'center', + }), + body: css({ display: 'grid', placeItems: 'center', rowGap: '12px' }), + }, + }; + + const elImg = ( + <img className={styles.img.class} src={src} onLoad={handleLoad} onError={handleError} /> + ); + + const elDisplayImage = ( + <div className={styles.display.base.class}> + <div className={styles.display.img.class} /> + </div> + ); + + const elSpinner = loading && ( + <div className={styles.spinner.class}> + <Spinners.Bar theme={props.theme} /> + </div> + ); + + const elError = error && ( + <div className={styles.error.base.class}> + <div className={styles.error.body.class}> + <Icons.Error size={38} /> + <div>{'( failed to load image )'}</div> + </div> + </div> + ); + + return ( + <div className={css(styles.base, props.style).class}> + {elImg} + {elDisplayImage} + {elSpinner} + {elError} + </div> + ); +}; + +/** + * Helpers + */ +const wrangle = { + async src(value: P['src'], setState: (value?: string) => void) { + setState(Is.promise(value) ? await value : value); + }, +} as const; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.Pulldown/common.ts b/deploy/@tdb.slc/src/ui.Content/ui/ui.Pulldown/common.ts new file mode 100644 index 0000000000..2bb91a0351 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.Pulldown/common.ts @@ -0,0 +1,13 @@ +export * from '../common.ts'; + +/** + * Constants: + */ +export const DEFAULTS = { + fontSize: 36, + fontWeight: 'bold', + letterSpacing: '-0.02em', + lineHeight: 0.95, + duration: 700, +} as const; +export const D = DEFAULTS; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.Pulldown/mod.ts b/deploy/@tdb.slc/src/ui.Content/ui/ui.Pulldown/mod.ts new file mode 100644 index 0000000000..cf59c44b62 --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.Pulldown/mod.ts @@ -0,0 +1,2 @@ +export { Pulldown } from './ui.tsx'; +export { usePulldown } from './use.Pulldown.tsx'; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.Pulldown/t.ts b/deploy/@tdb.slc/src/ui.Content/ui/ui.Pulldown/t.ts new file mode 100644 index 0000000000..0c2cfcf45f --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.Pulldown/t.ts @@ -0,0 +1,9 @@ +import { type t } from './common.ts'; + +/** + * Hook for managing the pulldown. + */ +export type UsePulldown = (props: t.VideoContentProps, timestamp: t.TimestampsHook) => PulldownHook; +export type PulldownHook = { + readonly is: { readonly showing: boolean }; +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.Pulldown/ui.tsx b/deploy/@tdb.slc/src/ui.Content/ui/ui.Pulldown/ui.tsx new file mode 100644 index 0000000000..00cb3f904b --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.Pulldown/ui.tsx @@ -0,0 +1,55 @@ +import React from 'react'; +import { ElapsedTime } from '../ui.ElapsedTime.tsx'; +import { useTimestamps } from '../use.Timestamps.ts'; +import { type t, Cropmarks, css, Sheet } from './common.ts'; + +export type PullDownProps = t.VideoContentProps; + +/** + * Component: + */ +export const Pulldown: React.FC<PullDownProps> = (props) => { + const { state, content } = props; + const { showElapsed = true } = content; + + const player = content.video; + const breakpoint = state.breakpoint; + const timestamp = useTimestamps(props, player); + + /** + * Render: + */ + const gutter = breakpoint.name === 'Desktop' ? 40 : 10; + const edgeTrack = `minmax(${gutter}px, 1fr)`; + const centerTrack = `minmax(0px, 960px)`; + const edge: t.SheetMarginInput = [edgeTrack, centerTrack, edgeTrack]; + + const styles = { + base: css({ position: 'relative', marginBottom: 218 }), + body: css({ Absolute: 0, display: 'grid' }), + }; + + const elBody = ( + <div className={styles.body.class}> + <Cropmarks + size={{ mode: 'fill', x: true, y: true, margin: [30, 30, 30, 30] }} + borderOpacity={0.06} + > + {timestamp.pulldown} + </Cropmarks> + </div> + ); + + return ( + <Sheet + {...props} + style={styles.base} + theme={props.theme} + edgeMargin={edge} + orientation={'Top:Down'} + > + {elBody} + <ElapsedTime player={player} abs={[null, 15, 10, null]} show={showElapsed} /> + </Sheet> + ); +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/ui.Pulldown/use.Pulldown.tsx b/deploy/@tdb.slc/src/ui.Content/ui/ui.Pulldown/use.Pulldown.tsx new file mode 100644 index 0000000000..6840189d6a --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/ui.Pulldown/use.Pulldown.tsx @@ -0,0 +1,33 @@ +import React, { useCallback, useEffect } from 'react'; +import { type t } from './common.ts'; +import { Pulldown } from './ui.tsx'; + +export const usePulldown: t.UsePulldown = (props, timestamp) => { + const { state, content } = props; + const render = useCallback(() => <Pulldown {...props} />, [props, timestamp]); + + /** + * Effect: show/hide. + */ + useEffect(() => { + const id = `${content.id}:pulldown`; + const exists = state.stack.exists((m) => m.id === id); + + if (exists && !timestamp.pulldown) { + state.stack.pop(); + return; + } + + if (!exists && timestamp.pulldown) { + state.stack.push({ id, render }); + return; + } + }, [timestamp.pulldown, content.id, render]); + + /** + * API + */ + return { + is: { showing: !!timestamp.pulldown }, + }; +}; diff --git a/deploy/@tdb.slc/src/ui.Content/ui/use.Timestamps.ts b/deploy/@tdb.slc/src/ui.Content/ui/use.Timestamps.ts new file mode 100644 index 0000000000..cdcff11bca --- /dev/null +++ b/deploy/@tdb.slc/src/ui.Content/ui/use.Timestamps.ts @@ -0,0 +1,53 @@ +import { useState } from 'react'; +import { type t, Is, Signal, Timestamp } from './common.ts'; + +export const useTimestamps: t.UseTimestamps = (props, player) => { + const { state, content } = props; + + const [column, setColumn] = useState<t.ReactNode>(); + const [pulldown, setPulldown] = useState<t.ReactNode>(); + + Signal.useEffect(() => { + if (!player || !props.content?.timestamps) return; + + const exists = state.stack.exists((e) => e.id === content.id); + if (!exists) { + setColumn(undefined); + setPulldown(undefined); + return; + } + + const timestamps = props.content.timestamps; + const secs = player.props.currentTime.value; + const match = Timestamp.find(timestamps, secs, { unit: 'secs' }); + + const renderer = wrangle.renderer(match?.data); + render(props, setColumn, renderer.column); + render(props, setPulldown, renderer.pulldown); + }); + + return { + column, + pulldown, + }; +}; + +/** + * Helpers + */ +const wrangle = { + renderer(data?: t.ContentTimestamp): t.ContentTimestampProps { + if (!data) return {}; + if (typeof data === 'function') return { column: data }; + return data; + }, +} as const; + +async function render( + props: t.VideoContentProps, + setState: (value: t.ReactNode) => void, + renderer?: t.VideoContentRenderer, +) { + const res = renderer?.(props); + setState(Is.promise(res) ? await res : res); +} diff --git a/deploy/@tdb.slc/src/ui/-test.ui.ts b/deploy/@tdb.slc/src/ui/-test.ui.ts new file mode 100644 index 0000000000..e5bce83774 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/-test.ui.ts @@ -0,0 +1,7 @@ +/** + * @module + * Testing tools running in the browser/ui. + */ +export { expect } from '@sys/std/testing'; +export { Dev, Lorem, Spec } from '@sys/ui-react-devharness'; +export * from '../common.ts'; diff --git a/deploy/@tdb.slc/src/ui/App.Layout/common.ts b/deploy/@tdb.slc/src/ui/App.Layout/common.ts new file mode 100644 index 0000000000..579fb2c8be --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Layout/common.ts @@ -0,0 +1,3 @@ +export * from '../common.ts'; + +export const DEFAULTS = {}; diff --git a/deploy/@tdb.slc/src/ui/App.Layout/m.Breakpoint.ts b/deploy/@tdb.slc/src/ui/App.Layout/m.Breakpoint.ts new file mode 100644 index 0000000000..e80662dcd0 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Layout/m.Breakpoint.ts @@ -0,0 +1,38 @@ +import { type t } from './common.ts'; + +/** + * Layout breakpoints calculation/helpers. + */ +export const Breakpoint: t.BreakpointLib = { + from(input: t.BreakpointSizeInput): t.Breakpoint { + if (typeof input === 'number') return Breakpoint.fromWidth(input); + if (typeof input === 'string') return Breakpoint.fromName(input); + return input; + }, + + fromWidth(width?: number): t.Breakpoint { + const name = Breakpoint.name(width); + return Breakpoint.fromName(name); + }, + + fromName(name: t.BreakpointName): t.Breakpoint { + const is = Breakpoint.is(name); + return { name, is }; + }, + + name(width?: number): t.BreakpointName { + if (width == null || width < 0) return 'UNKNOWN'; // NB: pre useSizeObserver initial measurement. + if (width <= 430) return 'Mobile'; + if (width <= 767) return 'Intermediate'; + return 'Desktop'; + }, + + is(name: t.BreakpointName): t.Breakpoint['is'] { + return { + ready: name !== 'UNKNOWN', + mobile: name === 'Mobile', + intermediate: name === 'Intermediate', + desktop: name === 'Desktop', + }; + }, +} as const; diff --git a/deploy/@tdb.slc/src/ui/App.Layout/m.Layout.tsx b/deploy/@tdb.slc/src/ui/App.Layout/m.Layout.tsx new file mode 100644 index 0000000000..bca7ab9b88 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Layout/m.Layout.tsx @@ -0,0 +1,9 @@ +import { type t } from './common.ts'; +import { Breakpoint } from './m.Breakpoint.ts'; + +/** + * Main Layout API (logic). + */ +export const Layout: t.AppLayoutLib = { + Breakpoint, +} as const; diff --git a/deploy/@tdb.slc/src/ui/App.Layout/mod.ts b/deploy/@tdb.slc/src/ui/App.Layout/mod.ts new file mode 100644 index 0000000000..4f8139da39 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Layout/mod.ts @@ -0,0 +1,6 @@ +/** + * @module + * General (non-UI component) layout logic. + */ +export { Breakpoint } from './m.Breakpoint.ts'; +export { Layout } from './m.Layout.tsx'; diff --git a/deploy/@tdb.slc/src/ui/App.Layout/t.Breakpoint.ts b/deploy/@tdb.slc/src/ui/App.Layout/t.Breakpoint.ts new file mode 100644 index 0000000000..c4609fe3a1 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Layout/t.Breakpoint.ts @@ -0,0 +1,41 @@ +import { type t } from './common.ts'; + +export type BreakpointSizeInput = t.NumberWidth | t.BreakpointName | t.Breakpoint; + +/** + * Library for working with breakpoints. + */ +export type BreakpointLib = { + from(input: t.BreakpointSizeInput): t.Breakpoint; + fromWidth(width?: number): t.Breakpoint; + fromName(name: t.BreakpointName): t.Breakpoint; + name(width?: number): t.BreakpointName; + is(name: t.BreakpointName): t.Breakpoint['is']; +}; + +/** + * Layout breakpoints + */ +export type BreakpointName = 'Mobile' | 'Intermediate' | 'Desktop' | 'UNKNOWN'; +export type Breakpoint = { + name: BreakpointName; + is: { + /** Width is -1 prior to the ResizeObserver completing it's first measure pass. */ + ready: boolean; + /** + * Mobile breakpoint. + * True when the viewport width is 430px or less, typically for smartphones. + */ + mobile: boolean; + /** + * Intermediate breakpoint. + * True when the viewport width is between 431px and 767px, ideal for phablets or small tablets. + */ + intermediate: boolean; + /** + * Desktop breakpoint. + * True when the viewport width is 768px or greater, typically for desktops and larger tablets. + */ + desktop: boolean; + }; +}; diff --git a/deploy/@tdb.slc/src/ui/App.Layout/t.ts b/deploy/@tdb.slc/src/ui/App.Layout/t.ts new file mode 100644 index 0000000000..cf598c3269 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Layout/t.ts @@ -0,0 +1,10 @@ +import { type t } from './common.ts'; +export type * from './t.Breakpoint.ts'; + +/** + * Logical helpers for working with layouts. + */ +export type AppLayoutLib = { + /** The screen-size breakpoint library. */ + readonly Breakpoint: t.BreakpointLib; +}; diff --git a/deploy/@tdb.slc/src/ui/App.Render/-.test.ts b/deploy/@tdb.slc/src/ui/App.Render/-.test.ts new file mode 100644 index 0000000000..33a11f90ea --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Render/-.test.ts @@ -0,0 +1,4 @@ +import { type t, describe, expect, it } from '../../-test.ts'; +import { AppSignals } from '../App.Signals/mod.ts'; + +describe('AppRender', () => {}); diff --git a/deploy/@tdb.slc/src/ui/App.Render/common.ts b/deploy/@tdb.slc/src/ui/App.Render/common.ts new file mode 100644 index 0000000000..8007245bca --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Render/common.ts @@ -0,0 +1,16 @@ +import { type t } from './common.ts'; +export * from '../common.ts'; + +export { AppSignals } from '../App.Signals/mod.ts'; +export { Breakpoint } from '../App.Layout/mod.ts'; + +/** + * Constants + */ +export const DEFAULTS = { + get theme() { + const base: t.CommonTheme = 'Dark'; + const sheet: t.CommonTheme = 'Light'; + return { base, sheet }; + }, +} as const; diff --git a/deploy/@tdb.slc/src/ui/App.Render/m.Render.preload.tsx b/deploy/@tdb.slc/src/ui/App.Render/m.Render.preload.tsx new file mode 100644 index 0000000000..2644f8f5d2 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Render/m.Render.preload.tsx @@ -0,0 +1,25 @@ +import { type t, Preload } from './common.ts'; +import { render } from './m.Render.stack.tsx'; + +/** + * Ensure the specified ESM content modules have been dyanamically imported. + */ +export const preload: t.AppRenderLib['preload'] = async <T extends string>( + state: t.AppSignals, + factory: (flag: T) => Promise<t.Content | undefined>, + ...content: T[] +) => { + if (typeof document === 'undefined') return; + + /** + * Dynamic import of ESM: + */ + const loading = content.map((flag) => factory(flag)); + const imports = (await Promise.all(loading)) as t.Content[]; + const elements = imports.map((content, index) => render({ index, content, state })); + + /** + * Render Portal: + */ + Preload.render(elements, 5_000); +}; diff --git a/deploy/@tdb.slc/src/ui/App.Render/m.Render.stack.tsx b/deploy/@tdb.slc/src/ui/App.Render/m.Render.stack.tsx new file mode 100644 index 0000000000..3b0768ba36 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Render/m.Render.stack.tsx @@ -0,0 +1,52 @@ +import React from 'react'; +import { type t, AnimatePresence, css, DEFAULTS } from './common.ts'; + +/** + * Renders the body of the matching timestamp. + */ +export function stack(state: t.AppSignals | undefined): t.ReactNode { + if (!state) return []; + const stack = state.stack.items ?? []; + const nodes = stack.map((content, index) => render({ index, state, content })); + return <AnimatePresence>{nodes}</AnimatePresence>; +} + +/** + * Render a single level in the stack. + */ +export function render(args: { index: number; state: t.AppSignals; content: t.Content }) { + const { state, content, index } = args; + const theme = wrangle.theme(content, state); + const is = wrangle.is(state, index); + const el = content.render?.({ index, content, state, is, theme }); + + const style = css({ + Absolute: 0, + pointerEvents: 'none', + zIndex: 0, // NB: establish a new stacking context (prevents content jumping above higher stack levels). + display: 'grid', + }); + + return ( + <div key={`${content.id}.${index}`} className={style.class}> + {el} + </div> + ); +} + +/** + * Helpers + */ +const wrangle = { + theme(content: t.Content, state: t.AppSignals): t.CommonTheme { + // NB: hard-coded from default. + // Possibly expand this later to store theme state on the App signals. + return DEFAULTS.theme.base; + }, + + is(state: t.AppSignals, index: number): t.ContentFlags { + const top = index === state.stack.length - 1; + const bottom = index === 0; + return { top, bottom }; + }, +} as const; diff --git a/deploy/@tdb.slc/src/ui/App.Render/m.Render.ts b/deploy/@tdb.slc/src/ui/App.Render/m.Render.ts new file mode 100644 index 0000000000..7a90ac21ba --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Render/m.Render.ts @@ -0,0 +1,11 @@ +import { type t } from './common.ts'; +import { stack } from './m.Render.stack.tsx'; +import { preload } from './m.Render.preload.tsx'; + +/** + * Render functions for display content. + */ +export const AppRender: t.AppRenderLib = { + stack, + preload, +} as const; diff --git a/deploy/@tdb.slc/src/ui/App.Render/mod.ts b/deploy/@tdb.slc/src/ui/App.Render/mod.ts new file mode 100644 index 0000000000..9af91c36d9 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Render/mod.ts @@ -0,0 +1,5 @@ +/** + * @module + */ +export { DEFAULTS } from './common.ts'; +export { AppRender } from './m.Render.ts'; diff --git a/deploy/@tdb.slc/src/ui/App.Render/t.ts b/deploy/@tdb.slc/src/ui/App.Render/t.ts new file mode 100644 index 0000000000..514b7105cc --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Render/t.ts @@ -0,0 +1,17 @@ +import type { t } from './common.ts'; + +/** + * Tools for rendering the application structure. + */ +export type AppRenderLib = { + stack: (state: t.AppSignals | undefined) => t.ReactNode; + + /** + * Ensure the specified ESM content modules have been dyanamically imported. + */ + preload<T extends string>( + state: t.AppSignals, + factory: (flag: T) => Promise<t.Content | undefined>, + ...content: T[] + ): Promise<void>; +}; diff --git a/deploy/@tdb.slc/src/ui/App.Signals.Controller/-.test.ts b/deploy/@tdb.slc/src/ui/App.Signals.Controller/-.test.ts new file mode 100644 index 0000000000..f9ea3adcee --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Signals.Controller/-.test.ts @@ -0,0 +1,58 @@ +import { describe, expect, it, rx } from '../../-test.ts'; +import { AppSignals } from '../App.Signals/mod.ts'; + +describe('AppSignals.Controllers', () => { + describe('start', () => { + it('should add all children', () => { + const state = AppSignals.create(); + + const root = AppSignals.Controllers.start(state); + expect(root.disposed).to.eql(false); + expect(root.kind).to.eql('Controller:App'); + + expect(root.children.map((e) => e.kind)).to.eql(['Controller:App:Background']); + root.children.forEach((child) => expect(child.disposed).to.eql(false)); + + const listening = state.props.controllers.listening.value; + expect(listening.includes('Controller:App')).to.eql(true); + expect(listening.includes('Controller:App:Background')).to.eql(true); + + root.dispose(); + expect(root.disposed).to.eql(true); + root.children.forEach((child) => expect(child.disposed).to.eql(true)); + }); + }); + + describe('background', () => { + it('adjusts backbround blur when stack changes', () => { + const life = rx.disposable(); + const state = AppSignals.create(); + const p = state.props; + expect(p.background.video.blur.value).to.eql(0); + + const ctrl = AppSignals.Controllers.background(state, life); + expect(ctrl.disposed).to.eql(false); + expect(ctrl.kind).to.eql('Controller:App:Background'); + expect(ctrl.children).to.eql([]); + + const listening = state.props.controllers.listening.value; + expect(listening.includes('Controller:App:Background')).to.eql(true); + + // Add to the stack. + state.stack.push({ id: 'base' }); + expect(p.background.video.blur.value).to.eql(0); // NB: no change. + + state.stack.push({ id: 'one' }); + expect(p.background.video.blur.value).to.eql(20); // NB: no change. + + state.stack.clear(); + expect(p.background.video.blur.value).to.eql(0); + + life.dispose(); + expect(ctrl.disposed).to.eql(true); + + state.stack.push({ id: 'one' }, { id: 'two' }); + expect(p.background.video.blur.value).to.eql(0); // NB: no change - disposed. + }); + }); +}); diff --git a/deploy/@tdb.slc/src/ui/App.Signals.Controller/common.ts b/deploy/@tdb.slc/src/ui/App.Signals.Controller/common.ts new file mode 100644 index 0000000000..8cae67176a --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Signals.Controller/common.ts @@ -0,0 +1 @@ +export * from '../common.ts'; diff --git a/deploy/@tdb.slc/src/ui/App.Signals.Controller/m.Controllers.background.ts b/deploy/@tdb.slc/src/ui/App.Signals.Controller/m.Controllers.background.ts new file mode 100644 index 0000000000..12c798dedd --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Signals.Controller/m.Controllers.background.ts @@ -0,0 +1,25 @@ +import { type t, Signal, rx } from './common.ts'; + +export const background: t.AppControllersLib['background'] = (state, until$) => { + const kind: t.AppControllerKind = 'Controller:App:Background'; + const listeners = Signal.listeners(until$); + const controllers = state.props.controllers; + controllers.listening.value = [...controllers.listening.value, kind]; + + /** + * Blur background when higher layers are visible. + */ + listeners.effect(() => { + const totalLayers = state.stack.length; + const blur = state.props.background.video.blur; + blur.value = totalLayers > 1 ? 20 : 0; + }); + + /** + * API + */ + return rx.toLifecycle<t.AppController>(listeners, { + kind, + children: [], + }); +}; diff --git a/deploy/@tdb.slc/src/ui/App.Signals.Controller/m.Controllers.ts b/deploy/@tdb.slc/src/ui/App.Signals.Controller/m.Controllers.ts new file mode 100644 index 0000000000..707e2a3bcf --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Signals.Controller/m.Controllers.ts @@ -0,0 +1,27 @@ +import { type t, Signal, rx } from './common.ts'; +import { background } from './m.Controllers.background.ts'; + +export const Controllers: t.AppControllersLib = { + background, + + start(state, until$) { + const kind: t.AppControllerKind = 'Controller:App'; + const children = new Set<t.AppController>(); + const listeners = Signal.listeners(until$); + const controllers = state.props.controllers; + controllers.listening.value = [...controllers.listening.value, kind]; + + // Initialize child controllers. + children.add(background(state, listeners.dispose$)); + + /** + * API: + */ + return rx.toLifecycle<t.AppController>(listeners, { + kind, + get children() { + return [...children]; + }, + }); + }, +}; diff --git a/deploy/@tdb.slc/src/ui/App.Signals.Controller/mod.ts b/deploy/@tdb.slc/src/ui/App.Signals.Controller/mod.ts new file mode 100644 index 0000000000..8ef8f105e7 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Signals.Controller/mod.ts @@ -0,0 +1,5 @@ +/** + * @module + * Behavior controllers that manipulate a signals API. + */ +export { Controllers } from './m.Controllers.ts'; diff --git a/deploy/@tdb.slc/src/ui/App.Signals.Controller/t.ts b/deploy/@tdb.slc/src/ui/App.Signals.Controller/t.ts new file mode 100644 index 0000000000..9255532a82 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Signals.Controller/t.ts @@ -0,0 +1,26 @@ +import type { t } from './common.ts'; + +/** + * Signal controllers for the application state. + */ +export type AppControllersLib = { + /** Hooks up all baseline controllers */ + start(state: t.AppSignals, until$?: t.UntilInput): AppController; + + /** + * Hook into the given state and apply controller logic + * based on live changes to the application. + */ + background(state: t.AppSignals, until$?: t.UntilInput): AppController; +}; + +/** + * Common interface for application controllers. + */ +export type AppController = t.Lifecycle & { + readonly kind: AppControllerKind; + readonly children: AppController[]; +}; + +/** Controller identification codes. */ +export type AppControllerKind = 'Controller:App' | 'Controller:App:Background'; diff --git a/deploy/@tdb.slc/src/ui/App.Signals/-.test.ts b/deploy/@tdb.slc/src/ui/App.Signals/-.test.ts new file mode 100644 index 0000000000..b2eb478d5b --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Signals/-.test.ts @@ -0,0 +1,55 @@ +import { c, describe, expect, it, Signal } from '../../-test.ts'; +import { VIDEO } from '../../ui.Content/VIDEO.ts'; +import { AppSignals } from './mod.ts'; + +describe('AppSignals', () => { + describe('lifecycle', () => { + it('create', () => { + const app = AppSignals.create(); + const p = app.props; + + expect(p.dist.value).to.eql(undefined); + expect(p.screen.breakpoint.value).to.eql('UNKNOWN'); + expect(p.stack.value).to.eql([]); + expect(p.screen.breakpoint.value).to.eql(app.breakpoint.name); + expect(p.controllers.listening.value).to.eql([]); + + expect(p.background.video.src.value).to.eql(VIDEO.Tubes.src); + expect(p.background.video.playing.value).to.eql(true); + expect(p.background.video.opacity.value).to.eql(0.2); + expect(p.background.video.blur.value).to.eql(0); + + console.info(); + console.info(c.brightGreen('SLC:App.Signals:')); + console.info(app); + console.info(); + }); + + it('instance id', () => { + const a = AppSignals.create(); + const b = AppSignals.create(); + + expect(a.instance.startsWith('app-')).to.eql(true); + expect(a.instance.length).to.be.greaterThan('app-1ss9'.length); + expect(a.instance).to.not.eql(b.instance); + }); + }); + + describe('listen', () => { + it('listens to changes', () => { + const app = AppSignals.create(); + + let count = 0; + const dispose = Signal.effect(() => { + app.listen(); + count++; + }); + + expect(count).to.eql(1); + app.stack.push({ id: 'foo' }); + expect(count).to.eql(2); + + dispose(); + }); + }); +}); diff --git a/deploy/@tdb.slc/src/ui/App.Signals/-Controllers.test.ts b/deploy/@tdb.slc/src/ui/App.Signals/-Controllers.test.ts new file mode 100644 index 0000000000..4acf98d1c0 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Signals/-Controllers.test.ts @@ -0,0 +1,51 @@ +import { describe, expect, it, rx } from '../../-test.ts'; +import { AppSignals } from './mod.ts'; + +describe('AppSignals.Controllers', () => { + describe('start', () => { + it('should add all children', () => { + const state = AppSignals.create(); + + const root = AppSignals.Controllers.start(state); + expect(root.disposed).to.eql(false); + expect(root.kind).to.eql('Controller:App'); + + expect(root.children.map(({ kind: id }) => id)).to.eql(['Controller:App:Background']); + root.children.forEach((child) => expect(child.disposed).to.eql(false)); + + root.dispose(); + expect(root.disposed).to.eql(true); + root.children.forEach((child) => expect(child.disposed).to.eql(true)); + }); + }); + + describe('background', () => { + it('adjusts backbround blur when stack changes', () => { + const life = rx.disposable(); + const state = AppSignals.create(); + const p = state.props; + expect(p.background.video.blur.value).to.eql(0); + + const ctrl = AppSignals.Controllers.background(state, life); + expect(ctrl.disposed).to.eql(false); + expect(ctrl.kind).to.eql('Controller:App:Background'); + expect(ctrl.children).to.eql([]); + + // Add to the stack. + state.stack.push({ id: 'base' }); + expect(p.background.video.blur.value).to.eql(0); // NB: no change. + + state.stack.push({ id: 'one' }); + expect(p.background.video.blur.value).to.eql(20); // NB: no change. + + state.stack.clear(); + expect(p.background.video.blur.value).to.eql(0); + + life.dispose(); + expect(ctrl.disposed).to.eql(true); + + state.stack.push({ id: 'one' }, { id: 'two' }); + expect(p.background.video.blur.value).to.eql(0); // NB: no change - disposed. + }); + }); +}); diff --git a/deploy/@tdb.slc/src/ui/App.Signals/-Stack.test.ts b/deploy/@tdb.slc/src/ui/App.Signals/-Stack.test.ts new file mode 100644 index 0000000000..01f69781ce --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Signals/-Stack.test.ts @@ -0,0 +1,46 @@ +import { type t, describe, expect, it, Signal } from '../../-test.ts'; +import { AppSignals } from './mod.ts'; + +describe('AppSignals.stack', () => { + const a: t.Content = { id: 'a' }; + const b: t.Content = { id: 'b' }; + const c: t.Content = { id: 'b' }; + + /** + * NOTE: stack more fully tested in ui/components:Sheet + */ + it('push ā pop ā clear', () => { + const app = AppSignals.create(); + const fired: number[] = []; + const dispose = Signal.effect(() => { + fired.push(app.props.stack.value.length); + }); + + expect(app.stack.length).to.eql(0); + + // Push single. + app.stack.push(a); + expect(app.stack.length).to.eql(1); + expect(app.props.stack.value).to.eql([a]); + expect(fired).to.eql([0, 1]); + + // Push many. + app.stack.push(b, c); + expect(app.stack.length).to.eql(3); + expect(app.props.stack.value).to.eql([a, b, c]); + expect(fired).to.eql([0, 1, 3]); + + // Push <undefined>. + app.stack.push(); + app.stack.push(undefined, a, undefined, b); + expect(app.props.stack.value).to.eql([a, b, c, a, b]); + + app.stack.clear(1); + expect(app.props.stack.value).to.eql([a]); + + app.stack.clear(); + expect(app.props.stack.value).to.eql([]); + + dispose(); + }); +}); diff --git a/deploy/@tdb.slc/src/ui/App.Signals/common.ts b/deploy/@tdb.slc/src/ui/App.Signals/common.ts new file mode 100644 index 0000000000..259b40d01f --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Signals/common.ts @@ -0,0 +1,2 @@ +export { Breakpoint } from '../App.Layout/mod.ts'; +export * from '../common.ts'; diff --git a/deploy/@tdb.slc/src/ui/App.Signals/m.Signals.create.ts b/deploy/@tdb.slc/src/ui/App.Signals/m.Signals.create.ts new file mode 100644 index 0000000000..dd3090df1d --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Signals/m.Signals.create.ts @@ -0,0 +1,55 @@ +import { type t, Breakpoint, SheetBase, Signal, slug, TUBES } from './common.ts'; + +/** + * Create a new instance of the application-state signals API. + */ +export const create: t.AppSignalsLib['create'] = (until$) => { + const s = Signal.create; + + /** + * API: + */ + type T = t.AppSignals; + type P = T['props']; + const props: P = { + dist: s<t.DistPkg>(), + stack: s<t.Content[]>([]), + screen: { breakpoint: s<t.BreakpointName>('UNKNOWN') }, + background: { + video: { + src: s<string>(TUBES.src), + playing: s<boolean>(true), + opacity: s<t.Percent | undefined>(0.2), + blur: s<t.Percent | undefined>(0), + }, + }, + controllers: { listening: s<t.AppControllerKind[]>([]) }, + }; + + const stack = SheetBase.Signals.stack(props.stack); + const api: T = { + instance: `app-${slug()}`, + get props() { + return props; + }, + get stack() { + return stack; + }, + get breakpoint() { + return Breakpoint.from(props.screen.breakpoint.value); + }, + listen() { + const p = props; + p.stack.value; + p.dist.value; + p.screen.breakpoint.value; + p.background.video.src.value; + p.background.video.playing.value; + p.background.video.opacity.value; + p.background.video.blur.value; + }, + }; + + // Finish up. + return api; +}; diff --git a/deploy/@tdb.slc/src/ui/App.Signals/m.Signals.ts b/deploy/@tdb.slc/src/ui/App.Signals/m.Signals.ts new file mode 100644 index 0000000000..4795031007 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Signals/m.Signals.ts @@ -0,0 +1,8 @@ +import type { t } from './common.ts'; +import { Controllers } from '../App.Signals.Controller/mod.ts'; +import { create } from './m.Signals.create.ts'; + +export const AppSignals: t.AppSignalsLib = { + Controllers, + create, +}; diff --git a/deploy/@tdb.slc/src/ui/App.Signals/mod.ts b/deploy/@tdb.slc/src/ui/App.Signals/mod.ts new file mode 100644 index 0000000000..90f4eb9f77 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Signals/mod.ts @@ -0,0 +1,5 @@ +/** + * @module + * Application state (via Signals). + */ +export { AppSignals } from './m.Signals.ts'; diff --git a/deploy/@tdb.slc/src/ui/App.Signals/t.Content.ts b/deploy/@tdb.slc/src/ui/App.Signals/t.Content.ts new file mode 100644 index 0000000000..43336418fb --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Signals/t.Content.ts @@ -0,0 +1,46 @@ +import type { t } from './common.ts'; + +/** + * Definition of content structure. + */ +export type Content<P = {}> = P & { + /** Content identifier. */ + id: t.StringId; + + /** + * Render the base content. + * Additional items (such as the current timestamp) are + * rendered into the {children} property. + */ + render?(props: ContentProps<P>): t.ReactNode; +}; + +/** + * Component Props: + */ +export type ContentProps<P = {}> = { + /** The index within the content-stack. */ + index: t.Index; + is: t.ContentFlags; + content: t.Content<P>; + state: t.AppSignals; + theme: t.CommonTheme; + style?: t.CssInput; +}; + +/** + * Flags pertaining to content. + */ +export type ContentFlags = { + /** Flag indicating if this is the current top-level view in the stack. */ + top: boolean; + /** Flag indicating if this is the bottom most view in the stack. */ + bottom: boolean; +}; + +/** + * Syncronous of asynchronous content renderer. + */ +export type ContentRenderer<P extends t.ContentProps = t.ContentProps> = ( + props: P, +) => t.ReactNode | Promise<t.ReactNode>; diff --git a/deploy/@tdb.slc/src/ui/App.Signals/t.Signals.ts b/deploy/@tdb.slc/src/ui/App.Signals/t.Signals.ts new file mode 100644 index 0000000000..50f4b4f237 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Signals/t.Signals.ts @@ -0,0 +1,39 @@ +import type { t } from './common.ts'; + +/** + * Global application-state signals API. + */ +export type AppSignals = { + /** Unique instance of the signals. */ + readonly instance: t.StringId; + + /** Signal properties: */ + readonly props: { + readonly dist: t.Signal<t.DistPkg | undefined>; + readonly stack: t.Signal<t.Content[]>; + readonly screen: { readonly breakpoint: t.Signal<t.BreakpointName> }; + readonly background: { + readonly video: { + readonly src: t.Signal<string>; + readonly playing: t.Signal<boolean>; + readonly opacity: t.Signal<t.Percent | undefined>; + readonly blur: t.Signal<t.Pixels | undefined>; + }; + }; + readonly controllers: { + listening: t.Signal<t.AppControllerKind[]>; + }; + }; + + /** API for interacting with the stack. */ + readonly stack: t.AppSignalsStack; + readonly breakpoint: t.Breakpoint; + + /** Hook into all relevant value listeners. */ + listen(): void; +}; + +/** + * API for managing the screen stack. + */ +export type AppSignalsStack = t.SheetSignalStack<t.Content>; diff --git a/deploy/@tdb.slc/src/ui/App.Signals/t.ts b/deploy/@tdb.slc/src/ui/App.Signals/t.ts new file mode 100644 index 0000000000..d93bdc2a15 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App.Signals/t.ts @@ -0,0 +1,14 @@ +import type { t } from './common.ts'; + +export type * from './t.Content.ts'; +export type * from './t.Signals.ts'; + +/** + * Library for working with the [AppSignals] model. + */ +export type AppSignalsLib = { + readonly Controllers: t.AppControllersLib; + + /** Create a new instance of the application-state signals API. */ + create(dispose$?: t.UntilObservable): t.AppSignals; +}; diff --git a/deploy/@tdb.slc/src/ui/App/-.test.ts b/deploy/@tdb.slc/src/ui/App/-.test.ts new file mode 100644 index 0000000000..7e74c930b7 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App/-.test.ts @@ -0,0 +1,11 @@ +import { describe, expect, it } from '../../-test.ts'; +import { AppSignals, Layout } from './common.ts'; +import { App } from './mod.ts'; + +describe('App', () => { + it('API', () => { + expect(App.Signals).to.equal(AppSignals); + expect(App.signals).to.equal(AppSignals.create); + expect(App.Layout).to.equal(Layout); + }); +}); diff --git a/deploy/@tdb.slc/src/ui/App/common.ts b/deploy/@tdb.slc/src/ui/App/common.ts new file mode 100644 index 0000000000..be106c23ec --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App/common.ts @@ -0,0 +1,5 @@ +export * from '../common.ts'; + +export { AppSignals } from '../App.Signals/mod.ts'; +export { Layout } from '../App.Layout/mod.ts'; +export { AppRender } from '../App.Render/mod.ts'; diff --git a/deploy/@tdb.slc/src/ui/App/m.App.ts b/deploy/@tdb.slc/src/ui/App/m.App.ts new file mode 100644 index 0000000000..054c6d032a --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App/m.App.ts @@ -0,0 +1,11 @@ +/** + * @module + */ +import { type t, Layout, AppRender as Render, AppSignals as Signals } from './common.ts'; + +export const App: t.AppLib = { + Layout, + Render, + Signals, + signals: Signals.create, +}; diff --git a/deploy/@tdb.slc/src/ui/App/mod.ts b/deploy/@tdb.slc/src/ui/App/mod.ts new file mode 100644 index 0000000000..292d718a02 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App/mod.ts @@ -0,0 +1,4 @@ +/** + * @module + */ +export * from './m.App.ts'; diff --git a/deploy/@tdb.slc/src/ui/App/t.ts b/deploy/@tdb.slc/src/ui/App/t.ts new file mode 100644 index 0000000000..d3ba2de088 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/App/t.ts @@ -0,0 +1,12 @@ +import type { t } from './common.ts'; + +/** + * Root Application (logic) API. + */ +export type AppLib = { + readonly Signals: t.AppSignalsLib; + readonly signals: t.AppSignalsLib['create']; + + readonly Layout: t.AppLayoutLib; + readonly Render: t.AppRenderLib; +}; diff --git a/deploy/@tdb.slc/src/ui/common.ts b/deploy/@tdb.slc/src/ui/common.ts new file mode 100644 index 0000000000..45b6d3a168 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/common.ts @@ -0,0 +1 @@ +export * from './common/mod.ts'; diff --git a/deploy/@tdb.slc/src/ui/common/libs.ts b/deploy/@tdb.slc/src/ui/common/libs.ts new file mode 100644 index 0000000000..511f00e514 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/common/libs.ts @@ -0,0 +1,27 @@ +import { motion as Motion } from 'motion/react'; +export { AnimatePresence } from 'motion/react'; +export { Motion as M, Motion }; + +/** + * System + */ +export { Color, css, Style } from '@sys/ui-css'; +export { Keyboard } from '@sys/ui-dom'; +export { + ReactString, + useClickOutside, + useDist, + useIsTouchSupported, + useSizeObserver, +} from '@sys/ui-react'; +export { + Button, + Cropmarks, + ObjectView, + Player, + Preload, + Sheet as SheetBase, + Spinners, + Svg, + VimeoBackground, +} from '@sys/ui-react-components'; diff --git a/deploy/@tdb.slc/src/ui/common/mod.ts b/deploy/@tdb.slc/src/ui/common/mod.ts new file mode 100644 index 0000000000..cd48a34925 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/common/mod.ts @@ -0,0 +1,15 @@ +export * from '../../common.ts'; +export * from './libs.ts'; + +export { Signal } from '@sys/ui-react'; + +/** + * Local + */ +export * from '../ui.Icons.ts'; + +/** + * Common Video Refs + */ +export const vimeo = (id: number) => ({ id, src: `vimeo/${id}` } as const); +export const TUBES = vimeo(499921561); diff --git a/deploy/@tdb.slc/src/ui/mod.ts b/deploy/@tdb.slc/src/ui/mod.ts new file mode 100644 index 0000000000..f7d4ac5d16 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/mod.ts @@ -0,0 +1,13 @@ +/** + * @module + * UI Components. + */ +export { LogoCanvas } from './ui.Logo.Canvas/mod.ts'; +export { LogoWordmark } from './ui.Logo.Wordmark/mod.ts'; +export { VideoBackground } from './ui.Video.Background/mod.ts'; + +export { Landing as Landing1 } from './ui.Landing-1/mod.ts'; +export { Landing as Landing2 } from './ui.Landing-2/mod.ts'; +export { Sheet } from './ui.Sheet/mod.ts'; + +export { App } from './App/mod.ts'; diff --git a/deploy/@tdb.slc/src/ui/ui.Icons.ts b/deploy/@tdb.slc/src/ui/ui.Icons.ts new file mode 100644 index 0000000000..691ac9dbea --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Icons.ts @@ -0,0 +1,16 @@ +import { Icon } from '@sys/ui-react-components'; +import { MdAdd, MdArrowDownward, MdErrorOutline, MdOutlineAddBox } from 'react-icons/md'; +import { PiProjectorScreenLight } from 'react-icons/pi'; + +const icon = Icon.renderer; +export { icon }; + +/** + * Icon collection: + */ +export const Icons = { + Error: icon(MdErrorOutline), + Add: { Plus: icon(MdAdd), Square: icon(MdOutlineAddBox) }, + Arrow: { Down: icon(MdArrowDownward) }, + ProjectorScreen: icon(PiProjectorScreenLight), +} as const; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-1/-SPEC.Debug.tsx b/deploy/@tdb.slc/src/ui/ui.Landing-1/-SPEC.Debug.tsx new file mode 100644 index 0000000000..5051f18397 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-1/-SPEC.Debug.tsx @@ -0,0 +1,76 @@ +import React from 'react'; +import { type t, Button, Color, css, Signal } from './common.ts'; + +/** + * Types + */ +export type DebugProps = { + ctx: { debug: DebugSignals; landing: t.LandingSignals }; + style?: t.CssValue; +}; +export type DebugSignals = ReturnType<typeof createDebugSignals>; +type P = DebugProps; + +/** + * Signals + */ +export function createDebugSignals() { + const s = Signal.create; + const props = { theme: s<t.CommonTheme>('Dark') }; + const api = { props }; + return api; +} + +/** + * Component + */ +export const Debug: React.FC<P> = (props) => { + const { ctx } = props; + const d = ctx.debug.props; + const p = ctx.landing.props; + + Signal.useRedrawEffect(() => { + d.theme.value; + p.canvasPosition.value; + p.sidebarVisible.value; + }); + + /** + * Render: + */ + const theme = Color.theme(d.theme.value); + const styles = { + base: css({}), + title: css({ fontWeight: 'bold', marginBottom: 10 }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={styles.title.class}>{'Landing-1'}</div> + <hr /> + + <Button + block + label={`theme: ${d.theme}`} + onClick={() => Signal.cycle(d.theme, ['Light', 'Dark'])} + /> + <Button + block + label={`sidebarVisible: ${p.sidebarVisible}`} + onClick={() => Signal.toggle(p.sidebarVisible)} + /> + + <hr /> + + <Button + block + label={`canvasPosition: "${p.canvasPosition}"`} + onClick={() => { + Signal.cycle<t.LandingCanvasPosition>(p.canvasPosition, ['Center', 'Center:Bottom']); + }} + /> + + <hr /> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-1/-SPEC.tsx b/deploy/@tdb.slc/src/ui/ui.Landing-1/-SPEC.tsx new file mode 100644 index 0000000000..27cf18f6da --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-1/-SPEC.tsx @@ -0,0 +1,32 @@ +import { Dev, Spec, Signal } from '../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { Landing, signalsFactory } from './mod.ts'; + +export default Spec.describe('Landing', (e) => { + const debug = createDebugSignals(); + const landing = signalsFactory(); + const d = debug.props; + const p = landing.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + Dev.Theme.signalEffect(ctx, d.theme, 1); + Signal.effect(() => { + p.ready.value; + p.sidebarVisible.value; + ctx.redraw(); + }); + + ctx.subject + .size('fill') + .display('grid') + .render((e) => { + return <Landing theme={d.theme.value} signals={landing} />; + }); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug ctx={{ debug, landing }} />); + }); +}); diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-1/-Signals.test.ts b/deploy/@tdb.slc/src/ui/ui.Landing-1/-Signals.test.ts new file mode 100644 index 0000000000..0b22ff2d14 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-1/-Signals.test.ts @@ -0,0 +1,27 @@ +import { Time, describe, expect, it, Signal } from '../../-test.ts'; +import { DEFAULTS } from './common.ts'; +import { signalsFactory } from './mod.ts'; + +const D = DEFAULTS; + +describe('Landing (Screen): Signals API', () => { + describe('props', () => { + it('initial values (defaults)', () => { + const signals = signalsFactory(); + const p = signals.props; + expect(p.ready.value).to.eql(false); + expect(p.canvasPosition.value === 'Center').to.be.true; + expect(p.sidebarVisible.value === false).to.be.true; + }); + + it('param: custom { defaults }', () => { + const s = signalsFactory({ + canvasPosition: 'Center:Bottom', + sidebarVisible: true, + }); + const p = s.props; + expect(p.canvasPosition.value === 'Center:Bottom').to.be.true; + expect(p.sidebarVisible.value).to.eql(true); + }); + }); +}); diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-1/common.ts b/deploy/@tdb.slc/src/ui/ui.Landing-1/common.ts new file mode 100644 index 0000000000..d7460c06d1 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-1/common.ts @@ -0,0 +1,18 @@ +import { type t } from '../common.ts'; + +export * from '../common.ts'; +export { LogoWordmark } from '../ui.Logo.Wordmark/mod.ts'; + +/** + * Constants + */ +export const DEFAULTS = { + sidebarVisible: false, + get tubes() { + const id = 499921561; + return { id, src: `vimeo/${id}` }; + }, + get canvasPosition(): t.LandingCanvasPosition { + return 'Center'; + }, +} as const; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-1/m.Signals.ts b/deploy/@tdb.slc/src/ui/ui.Landing-1/m.Signals.ts new file mode 100644 index 0000000000..eae29f0728 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-1/m.Signals.ts @@ -0,0 +1,18 @@ +import { type t, DEFAULTS, Signal } from './common.ts'; + +type T = t.LandingSignals; +const D = DEFAULTS; + +/** + * Factory: create a new instance of signals + */ +export const signalsFactory: t.LandingSignalsFactory = (defaults = {}) => { + const s = Signal.create; + const props: T['props'] = { + ready: s<boolean>(false), + sidebarVisible: s<boolean>(defaults.sidebarVisible ?? D.sidebarVisible), + canvasPosition: s<t.LandingCanvasPosition>(defaults.canvasPosition ?? D.canvasPosition), + }; + const api: T = { props }; + return api; +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-1/mod.ts b/deploy/@tdb.slc/src/ui/ui.Landing-1/mod.ts new file mode 100644 index 0000000000..7a5c555eef --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-1/mod.ts @@ -0,0 +1,2 @@ +export { Landing } from './ui.tsx'; +export { signalsFactory } from './m.Signals.ts'; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-1/t.ts b/deploy/@tdb.slc/src/ui/ui.Landing-1/t.ts new file mode 100644 index 0000000000..f4d744e6a8 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-1/t.ts @@ -0,0 +1,33 @@ +import type { t } from './common.ts'; + +export type LandingCanvasPosition = 'Center' | 'Center:Bottom'; + +/** + * Signals: + */ +export type LandingSignalsFactory = (defaults?: LandingSignalsFactoryDefaults) => t.LandingSignals; +/** Defaults passed to the signals API factory. */ +export type LandingSignalsFactoryDefaults = { + canvasPosition?: t.LandingCanvasPosition; + sidebarVisible?: boolean; +}; + +/** + * Signals API for dynamic control of the <VideoPlayer>. + */ +export type LandingSignals = { + props: { + ready: t.Signal<boolean>; + canvasPosition: t.Signal<t.LandingCanvasPosition>; + sidebarVisible: t.Signal<boolean>; + }; +}; + +/** + * <Component>: + */ +export type Landing1Props = { + theme?: t.CommonTheme; + style?: t.CssInput; + signals?: t.LandingSignals; +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-1/ui.Layout.tsx b/deploy/@tdb.slc/src/ui/ui.Landing-1/ui.Layout.tsx new file mode 100644 index 0000000000..a779947df6 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-1/ui.Layout.tsx @@ -0,0 +1,91 @@ +import React from 'react'; +import { type t, Color, css, DEFAULTS, Motion } from './common.ts'; + +export type LayoutProps = { + canvas: { element?: JSX.Element; position?: t.LandingCanvasPosition }; + video: { element?: JSX.Element }; + theme?: t.CommonTheme; + style?: t.CssInput; +}; + +type P = LayoutProps; + +/** + * Component: + */ +export const Layout: React.FC<P> = (props) => { + const { canvas, video } = props; + const position = canvas.position ?? DEFAULTS.canvasPosition; + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ position: 'relative', color: theme.fg }), + canvas: css({ + position: 'absolute', + top: '50%', + left: '50%', + transform: 'translate(-50%, -50%) scale(1)', + }), + + video: css({ + position: 'absolute', + top: '50%', + left: '50%', + transform: 'translate(-50%, -50%)', + width: '85%', + aspectRatio: '16/9', + backgroundColor: 'white', + borderRadius: 15, + overflow: 'hidden', + display: position === 'Center:Bottom' ? 'grid' : 'none', + }), + }; + + /** + * Animation state: + */ + const canvasState = + position === 'Center:Bottom' + ? { top: 'calc(100% - 10px)', transform: 'translate(-50%, -100%) scale(0.65)' } + : { top: '50%', transform: 'translate(-50%, -50%) scale(1.1)' }; + + const backgroundAnimate = + position === 'Center:Bottom' + ? { opacity: 1, top: '50%', transform: 'translate(-50%, calc(-50% - 80px))' } + : { opacity: 0, top: '50%', transform: 'translate(-50%, calc(-50% - 80px))' }; + + /** + * Elements: + */ + const elVideo = ( + <Motion.div + className={styles.video.class} + initial={false} // Prevent initial animation on mount. + animate={backgroundAnimate} + transition={{ type: 'spring', stiffness: 200, damping: 15, mass: 0.8 }} + > + {video.element} + </Motion.div> + ); + + const elCanvas = canvas.element && ( + <Motion.div + className={styles.canvas.class} + initial={false} + animate={canvasState} + transition={{ type: 'spring', stiffness: 200, damping: 10, mass: 0.5 }} + > + {canvas.element} + </Motion.div> + ); + + return ( + <div className={css(styles.base, props.style).class}> + {elVideo} + {elCanvas} + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-1/ui.Sidebar.tsx b/deploy/@tdb.slc/src/ui/ui.Landing-1/ui.Sidebar.tsx new file mode 100644 index 0000000000..5fbdf7a243 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-1/ui.Sidebar.tsx @@ -0,0 +1,42 @@ +import React from 'react'; +import { type t, Color, css } from './common.ts'; + +export type SidebarProps = { + width?: t.Pixels; + theme?: t.CommonTheme; + style?: t.CssInput; +}; + +type P = SidebarProps; + +/** + * Component: + */ +export const Sidebar: React.FC<P> = (props) => { + const { width } = props; + const bgBlur = 20; + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ + backgroundColor: Color.alpha(theme.fg, 0.06), + color: theme.fg, + overflow: 'hidden', + backdropFilter: `blur(${bgBlur}px)`, + display: 'grid', + }), + body: css({ + borderLeft: `solid 1px ${Color.alpha(theme.fg, 0.1)}`, + padding: 10, + }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={styles.body.class}>{`š· Sidebar`}</div> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-1/ui.tsx b/deploy/@tdb.slc/src/ui/ui.Landing-1/ui.tsx new file mode 100644 index 0000000000..676cb6d13a --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-1/ui.tsx @@ -0,0 +1,127 @@ +import React, { useRef } from 'react'; +import { Player, VimeoBackground } from '../common.ts'; +import { LogoCanvas } from '../ui.Logo.Canvas/mod.ts'; + +import { type t, Color, css, DEFAULTS, LogoWordmark, Signal } from './common.ts'; +import { Layout } from './ui.Layout.tsx'; +import { Sidebar } from './ui.Sidebar.tsx'; +import { useKeyboard } from './use.Keyboard.ts'; +import { useSelectedPanel } from './use.SelectedPanel.ts'; + +type P = t.Landing1Props; +const D = DEFAULTS; + +const signalsFactory = () => + Player.Video.signals({ + // src: 'vimeo/727951677', // Rowan: "group scale", + src: 'vimeo/1068502644', // Trailer + }); + +/** + * Component: + */ +export const Landing: React.FC<P> = (props) => { + const { signals } = props; + const p = signals?.props; + const canvasPosition = p?.canvasPosition.value ?? D.canvasPosition; + const sidebarVisible = p?.sidebarVisible.value ?? D.sidebarVisible; + + /** + * Hooks: + */ + useKeyboard(); + const selectedPanel = useSelectedPanel(); + const playerSignalsRef = useRef(signalsFactory()); + const player = playerSignalsRef.current; + + /** + * Effects: + */ + Signal.useRedrawEffect(() => { + if (!p) return; + p.ready.value; + p.canvasPosition.value; + selectedPanel.value; + }); + + Signal.useEffect(() => { + const canvasPosition = p?.canvasPosition.value ?? DEFAULTS.canvasPosition; + if (!player) return; + if (canvasPosition === 'Center:Bottom') { + player.play(); + } else { + player.pause(); + } + }); + + /** + * Render. + */ + const sidebarRightWidth = sidebarVisible ? 360 : 0; + const theme = Color.theme(props.theme ?? 'Dark'); + const speed = 100; + + const styles = { + base: css({ + position: 'relative', + color: theme.fg, + backgroundColor: theme.bg, + fontFamily: 'sans-serif', + }), + + fill: css({ Absolute: 0 }), + logo: css({ + Absolute: [null, 15, 12, null], + width: 110, + }), + + body: css({ Absolute: 0, display: 'grid' }), + layout: css({ + Absolute: [0, sidebarRightWidth, 0, 0], + transition: `right ${speed}ms ease-in-out`, + }), + sidebar: { + right: css({ + Absolute: [0, 0, 0, null], + width: sidebarRightWidth, + transition: `width ${speed}ms ease-in-out`, + }), + }, + }; + + const elLayout = ( + <Layout + style={styles.layout} + theme={theme.name} + canvas={{ + element: <LogoCanvas theme={theme.name} selected={selectedPanel.value} />, + position: p?.canvasPosition.value, + }} + video={{ + element: <Player.Video.View signals={playerSignalsRef.current} />, + }} + /> + ); + + const elSidebarRight = ( + <Sidebar style={styles.sidebar.right} theme={props.theme} width={sidebarRightWidth} /> + ); + + return ( + <div className={css(styles.base, props.style).class}> + <LogoWordmark style={styles.logo} theme={theme.name} /> + + <VimeoBackground + video={499921561} // Tubes. + opacity={canvasPosition === 'Center' ? 0.3 : 0.15} + theme={theme.name} + style={styles.fill} + /> + + <div className={styles.body.class}> + {elLayout} + {elSidebarRight} + </div> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-1/use.Keyboard.ts b/deploy/@tdb.slc/src/ui/ui.Landing-1/use.Keyboard.ts new file mode 100644 index 0000000000..92702a18e2 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-1/use.Keyboard.ts @@ -0,0 +1,24 @@ +import { useEffect } from 'react'; +import { Keyboard, rx } from './common.ts'; + +export function useKeyboard() { + useEffect(() => { + const life = rx.disposable(); + const keyboard = Keyboard.until(life.dispose$); + + keyboard.on('Enter', () => { + const s = window.location.search; + const isDev = s.includes('dev=') || s.includes('d='); + if (!isDev) window.location.search = '?d'; + }); + + keyboard.on('Space', () => { + /** + * TODO š· + */ + console.log('š· START/STOP player'); + }); + + return life.dispose; + }); +} diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-1/use.SelectedPanel.ts b/deploy/@tdb.slc/src/ui/ui.Landing-1/use.SelectedPanel.ts new file mode 100644 index 0000000000..27b1f61aa0 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-1/use.SelectedPanel.ts @@ -0,0 +1,28 @@ +import React, { useEffect, useState, useRef, useCallback } from 'react'; +import { type t, Signal, rx, CanvasPanel, Time } from './common.ts'; + +export function useSelectedPanel() { + const panel = Signal.useSignal<t.CanvasPanel>('purpose'); + + /** + * Effect: Cycle the selected SLC panel. + */ + useEffect(() => { + const delay = 2_000; + const life = rx.lifecycle(); + + const next = async () => { + if (life.disposed) return; + Signal.cycle(panel, CanvasPanel.list); + Time.delay(delay, next); + }; + + Time.delay(delay, next); + return life.dispose; + }, []); + + /** + * API + */ + return panel; +} diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-2/-SPEC.Debug.tsx b/deploy/@tdb.slc/src/ui/ui.Landing-2/-SPEC.Debug.tsx new file mode 100644 index 0000000000..79b9df4483 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-2/-SPEC.Debug.tsx @@ -0,0 +1,78 @@ +import React from 'react'; +import { type t, Button, Color, css, Signal } from './common.ts'; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + type P = t.Landing2Props; + const s = Signal.create; + const props = { + debug: s<P['debug']>(false), + theme: s<P['theme']>('Dark'), + backgroundVideo: s<P['backgroundVideo']>(0.15), + }; + const api = { + props, + listen() { + const p = props; + p.theme.value; + p.debug.value; + p.backgroundVideo.value; + }, + }; + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => { + debug.listen(); + }); + + /** + * Render: + */ + const theme = Color.theme(p.theme.value); + const styles = { + base: css({}), + title: css({ fontWeight: 'bold', marginBottom: 10 }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={styles.title.class}>{'Landing-2'}</div> + <hr /> + + <Button block label={`debug: ${p.debug}`} onClick={() => Signal.toggle(p.debug)} /> + + <Button + block + label={`theme: "${p.theme}"`} + onClick={() => Signal.cycle<t.CommonTheme>(p.theme, ['Light', 'Dark'])} + /> + + <hr /> + + <Button + block + label={`backgroundVideo: ${p.backgroundVideo}`} + onClick={() => Signal.cycle(p.backgroundVideo, [undefined, 0, 0.15, 0.3, 0.5])} + /> + + <hr /> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-2/-SPEC.tsx b/deploy/@tdb.slc/src/ui/ui.Landing-2/-SPEC.tsx new file mode 100644 index 0000000000..9977744c58 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-2/-SPEC.tsx @@ -0,0 +1,36 @@ +import React from 'react'; +import { Dev, Signal, Spec } from '../-test.ui.ts'; +import { createDebugSignals, Debug } from './-SPEC.Debug.tsx'; +import { Landing } from './mod.ts'; + +export default Spec.describe('Landing-2', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + Dev.Theme.signalEffect(ctx, p.theme, 1); + Signal.effect(() => { + debug.listen(); + ctx.host.tracelineColor(p.theme.value === 'Dark' ? 0.15 : -0.06); + ctx.redraw(); + }); + + ctx.subject + .size('fill') + .display('grid') + .render((e) => ( + <Landing + theme={p.theme.value} + debug={p.debug.value} + backgroundVideo={p.backgroundVideo.value} + /> + )); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-2/common.ts b/deploy/@tdb.slc/src/ui/ui.Landing-2/common.ts new file mode 100644 index 0000000000..630470ac2c --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-2/common.ts @@ -0,0 +1,6 @@ +export * from '../common.ts'; + +/** + * Constants: + */ +export const DEFAULTS = {} as const; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-2/mod.ts b/deploy/@tdb.slc/src/ui/ui.Landing-2/mod.ts new file mode 100644 index 0000000000..f817f74f44 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-2/mod.ts @@ -0,0 +1,4 @@ +/** + * @module + */ +export { Landing } from './ui.tsx'; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-2/t.ts b/deploy/@tdb.slc/src/ui/ui.Landing-2/t.ts new file mode 100644 index 0000000000..041a6b5b70 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-2/t.ts @@ -0,0 +1,11 @@ +import type { t } from './common.ts'; + +/** + * <Component>: + */ +export type Landing2Props = { + debug?: boolean; + backgroundVideo?: t.Percent; + theme?: t.CommonTheme; + style?: t.CssInput; +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-2/ui.tsx b/deploy/@tdb.slc/src/ui/ui.Landing-2/ui.tsx new file mode 100644 index 0000000000..bcf86202f8 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-2/ui.tsx @@ -0,0 +1,82 @@ +import React from 'react'; +import { type t, Color, css, VimeoBackground } from './common.ts'; + +import { LogoCanvas } from '../ui.Logo.Canvas/mod.ts'; +import { useKeyboard } from '../ui.Landing-1/use.Keyboard.ts'; +import { LogoWordmark } from '../ui.Logo.Wordmark/mod.ts'; + +export const Landing: React.FC<t.Landing2Props> = (props) => { + const { debug = false, backgroundVideo = 0 } = props; + + useKeyboard(); + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ + position: 'relative', + containerType: 'size', + color: theme.fg, + backgroundColor: theme.bg, + }), + background: css({ Absolute: 0 }), + body: css({ + Absolute: 0, + overflow: 'hidden', + backgroundColor: debug ? 'rgba(255, 0, 0, 0.1)' /* RED:DEBUG */ : '', + display: 'grid', + }), + + canvas: { + base: css({ + backgroundColor: debug ? 'rgba(255, 0, 0, 0.1)' /* RED:DEBUG */ : '', + display: 'grid', + placeItems: 'center', + }) + .container('max-width: 550px', { PaddingX: 75 }) + .container('max-width: 320px', { PaddingX: 50 }).done, + subject: css({ + width: 400, + marginBottom: '6%', + }).container('max-width: 550px', { width: '100%' }).done, + }, + + logo: css({ + Absolute: [null, 10, 15, null], + width: 150, + transition: 'width 200ms', + }) + .container('max-width: 640px', { width: 110 }) + .container('max-height: 470px', { display: 'none' }).done, + }; + + const elBackground = backgroundVideo !== undefined && ( + <VimeoBackground + video={499921561} // Tubes. + opacity={backgroundVideo} + theme={theme.name} + style={styles.background} + /> + ); + + const elCanvas = ( + <div className={styles.canvas.base.class}> + <div className={styles.canvas.subject.class}> + <LogoCanvas theme={theme.name} /> + </div> + </div> + ); + + const elLogo = <LogoWordmark theme={theme.name} style={styles.logo} />; + const elBody = <div className={styles.body.class}>{elCanvas}</div>; + + return ( + <div className={css(styles.base, props.style).class}> + {elLogo} + {elBackground} + {elBody} + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-3/-SPEC.Debug.tsx b/deploy/@tdb.slc/src/ui/ui.Landing-3/-SPEC.Debug.tsx new file mode 100644 index 0000000000..a549a4ea3e --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-3/-SPEC.Debug.tsx @@ -0,0 +1,83 @@ +import React from 'react'; +import { Content } from '../../ui.Content/mod.ts'; +import { pushStackContentButtons, screenBreakpointButton } from '../ui.Layout/-SPEC.tsx'; +import { type t, App, Button, css, ObjectView, Signal, Str } from './common.ts'; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = Awaited<ReturnType<typeof createDebugSignals>>; + +/** + * Signals: + */ +export async function createDebugSignals(init?: (e: DebugSignals) => void) { + type P = t.Landing3Props; + const s = Signal.create; + + const app = App.signals(); + const props = { debug: s<P['debug']>(true) }; + const api = { app, props }; + app.stack.push(await Content.factory('Entry')); + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const app = debug.app; + const p = app.props; + const d = debug.props; + + Signal.useRedrawEffect(() => { + d.debug.value; + app.listen(); + }); + + /** + * Render: + */ + const styles = { + base: css({}), + title: css({ fontWeight: 'bold', marginBottom: 10 }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={styles.title.class}>{'Landing-3'}</div> + <Button block label={`debug: ${d.debug}`} onClick={() => Signal.toggle(d.debug)} /> + <hr /> + {screenBreakpointButton(app)} + <Button + block + label={`background.video.opacity: ${p.background.video.opacity}`} + onClick={() => Signal.cycle(p.background.video.opacity, [undefined, 0.15, 0.3, 1])} + /> + + <hr /> + {pushStackContentButtons(app)} + + <hr /> + <ObjectView block name={'dist'} data={wrangle.dist(app)} expand={1} /> + <ObjectView name={'stack'} data={app.stack.items} expand={1} style={{ marginTop: 10 }} /> + </div> + ); +}; + +/** + * Helpers + */ +const wrangle = { + dist(app: t.AppSignals) { + const dist = app.props.dist.value; + if (!dist) return { err: '[app.props.dist] not found' }; + return { + 'dist:size': Str.bytes(dist.size.bytes), + 'dist:hash:sha256': `#${dist.hash.digest.slice(-5)}`, + }; + }, +} as const; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-3/-SPEC.tsx b/deploy/@tdb.slc/src/ui/ui.Landing-3/-SPEC.tsx new file mode 100644 index 0000000000..38ea9893b0 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-3/-SPEC.tsx @@ -0,0 +1,51 @@ +import React from 'react'; +import { Signal, Spec } from '../-test.ui.ts'; +import { createDebugSignals, Debug } from './-SPEC.Debug.tsx'; +import { css, App } from './common.ts'; +import { Landing } from './mod.ts'; +import { updateForBreakpointSize } from '../ui.Layout/-SPEC.tsx'; + +export default Spec.describe('Landing-3', async (e) => { + const debug = await createDebugSignals(); + const app = debug.app; + const d = debug.props; + const p = app.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + const update = { size: () => updateForBreakpointSize(ctx, app) }; + + Signal.effect(() => { + d.debug.value; + app.listen(); + update.size(); + ctx.redraw(); + }); + + ctx.subject + .size('fill') + .display('grid') + .render((e) => { + const styles = { + base: css({ + position: 'relative', + display: 'grid', + overflow: 'hidden', + }), + }; + + return ( + <div className={styles.base.class}> + <Landing state={app} debug={d.debug.value} /> + </div> + ); + }); + + update.size(); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-3/common.ts b/deploy/@tdb.slc/src/ui/ui.Landing-3/common.ts new file mode 100644 index 0000000000..45d99b6304 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-3/common.ts @@ -0,0 +1,16 @@ +export * from '../common.ts'; + +/** + * Libs: + */ +export { App } from '../App/mod.ts'; +export { Layout } from '../ui.Layout/mod.ts'; +export { VideoBackground } from '../ui.Video.Background/mod.ts'; +export { useKeyboard } from '../use/mod.ts'; + +export { Content } from '../../ui.Content/mod.ts'; + +/** + * Constants: + */ +export const DEFAULTS = {} as const; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-3/mod.ts b/deploy/@tdb.slc/src/ui/ui.Landing-3/mod.ts new file mode 100644 index 0000000000..b5d5b9266b --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-3/mod.ts @@ -0,0 +1,5 @@ +/** + * @module + */ +export { App, Content } from './common.ts'; +export { Landing } from './ui.tsx'; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-3/t.ts b/deploy/@tdb.slc/src/ui/ui.Landing-3/t.ts new file mode 100644 index 0000000000..c82ef31fd0 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-3/t.ts @@ -0,0 +1,10 @@ +import type { t } from './common.ts'; + +/** + * <Component>: + */ +export type Landing3Props = { + state?: t.AppSignals; + debug?: boolean; + style?: t.CssInput; +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-3/ui.Content.tsx b/deploy/@tdb.slc/src/ui/ui.Landing-3/ui.Content.tsx new file mode 100644 index 0000000000..7cec6ce743 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-3/ui.Content.tsx @@ -0,0 +1,28 @@ +import React from 'react'; +import { type t, css, Layout } from './common.ts'; + +export type ContentProps = { + state?: t.AppSignals; + breakpoint: t.Breakpoint; + style?: t.CssInput; +}; + +type P = ContentProps; + +/** + * Component: + */ +export const Content: React.FC<P> = (props) => { + const { breakpoint, state } = props; + + /** + * Render: + */ + const styles = { + base: css({ display: 'grid' }), + }; + + const className = css(styles.base, props.style).class; + const el = Layout.render(breakpoint, state); + return <div className={className}>{el}</div>; +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Landing-3/ui.tsx b/deploy/@tdb.slc/src/ui/ui.Landing-3/ui.tsx new file mode 100644 index 0000000000..3bc6c9ad37 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Landing-3/ui.tsx @@ -0,0 +1,105 @@ +import React, { useEffect } from 'react'; +import { + type t, + App, + Color, + css, + Layout, + pkg, + rx, + useDist, + useSizeObserver, + VideoBackground, +} from './common.ts'; +import { Content } from './ui.Content.tsx'; + +type P = t.Landing3Props; + +/** + * Component: + */ +export const Landing: React.FC<P> = (props) => { + const { debug, state } = props; + const p = state?.props; + + const size = useSizeObserver(); + const width = size.rect?.width ?? -1; + const breakpoint = Layout.Breakpoint.fromWidth(width); + + const isReady = size.ready && breakpoint.is.ready; + + /** + * Hooks: + */ + const dist = useDist({ + useSampleFallback: wrangle.showSampleDist(props), + }); + + /** + * Effect: + */ + useEffect(() => { + const life = rx.lifecycle(); + if (state) App.Signals.Controllers.start(state, life); + return life.dispose; + }, [state]); + + /** + * Effects: + */ + useEffect(() => { + console.info(`š¦ ${pkg.name}@${pkg.version}: dist.json ā`, dist.json); + if (state) state.props.dist.value = dist.json; + }, [dist.count, !!state]); + + useEffect(() => { + if (!p) return; + p.screen.breakpoint.value = breakpoint.name; + }, [breakpoint.name]); + + if (!p) return; + const bg = p.background; + const backgroundVideoOpacity = bg.video.opacity.value; + + /** + * Render: + */ + const theme = Color.theme('Dark'); + const styles = { + base: css({ + backgroundColor: theme.bg, + color: theme.fg, + fontFamily: 'sans-serif', + opacity: isReady ? 1 : 0, + }), + fill: css({ Absolute: 0, display: 'grid' }), + }; + + const elBackground = typeof backgroundVideoOpacity === 'number' && ( + <VideoBackground state={state} style={styles.fill} /> + ); + + const elBody = ( + <div ref={size.ref} className={styles.fill.class}> + <Content breakpoint={breakpoint} state={state} /> + </div> + ); + + return ( + <div className={css(styles.base, props.style).class}> + {elBackground} + {elBody} + </div> + ); +}; + +/** + * Helpers + */ +const wrangle = { + showSampleDist(props: P) { + const { debug } = props; + const isLocalhost = location.hostname === 'localhost'; + return debug ?? (isLocalhost && location.port !== '8080'); + }, +} as const; diff --git a/deploy/@tdb.slc/src/ui/ui.Layout/-SPEC.Debug.tsx b/deploy/@tdb.slc/src/ui/ui.Layout/-SPEC.Debug.tsx new file mode 100644 index 0000000000..5f914d4d29 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Layout/-SPEC.Debug.tsx @@ -0,0 +1,123 @@ +import React from 'react'; +import { Sample } from './-SPEC.u.sample.tsx'; +import { + Content, + layerVideoPlayerButtons, + pushStackContentButtons, + screenBreakpointButton, +} from './-SPEC.u.tsx'; +import { type t, App, Button, css, Signal, Str, ObjectView } from './common.ts'; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = Awaited<ReturnType<typeof createDebugSignals>>; + +/** + * Signals: + */ +export async function createDebugSignals(init?: (e: DebugSignals) => void) { + const s = Signal.create; + const app = App.signals(); + + const props = { theme: s<t.CommonTheme>('Dark') }; + const api = { + app, + props, + listen() { + app.listen(); + props.theme.value; + }, + }; + + // app.props.screen.breakpoint.value = 'Mobile'; + app.props.screen.breakpoint.value = 'Desktop'; + + app.stack.push(Sample.sample0()); + app.stack.push(await Content.factory('Entry')); + // app.stack.push(await Content.factory('Overview')); + + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const app = debug.app; + const p = app.props; + const d = debug.props; + + Signal.useRedrawEffect(() => debug.listen()); + + /** + * Render: + */ + const styles = { + base: css({}), + title: css({ fontWeight: 'bold', marginBottom: 10 }), + cols: css({ display: 'grid', gridTemplateColumns: 'auto 1fr auto' }), + }; + + const pushSample = (name: string, fn: () => t.Content) => { + return <Button block label={`stack.push:( "${name}" )`} onClick={() => app.stack.push(fn())} />; + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={styles.title.class}>{`${p.screen.breakpoint.value} Layout`}</div> + + <Button + block + label={`theme: ${d.theme}`} + onClick={() => Signal.cycle<t.CommonTheme>(d.theme, ['Light', 'Dark'])} + /> + + <hr /> + {screenBreakpointButton(app)} + + <hr /> + <div className={css(styles.title, styles.cols).class}> + <div>{`Stack:`}</div> + <div /> + <div>{`${app.stack.length} ${Str.plural(app.stack.length, 'Layer', 'Layers')}`}</div> + </div> + + {pushSample('sample-layer-0', Sample.sample0)} + {pushSample('sample-layer-1', Sample.sample1)} + {pushSample('sample: top-down', Sample.sample2)} + + <hr /> + {pushStackContentButtons(app)} + + <hr /> + {layerVideoPlayerButtons(app)} + + <hr /> + <div className={styles.title.class}>{`Sample Configurations:`}</div> + + <Button + block + label={() => `- debug samples`} + onClick={() => { + app.stack.clear(); + app.stack.push(Sample.sample0()); + }} + /> + <Button + block + label={() => `- application ("SLC Product System")`} + onClick={async () => { + app.stack.clear(); + app.stack.push(await Content.factory('Entry')); + }} + /> + + <hr /> + <ObjectView name={'stack'} data={app.stack.items} expand={1} /> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Layout/-SPEC.tsx b/deploy/@tdb.slc/src/ui/ui.Layout/-SPEC.tsx new file mode 100644 index 0000000000..7073a38db5 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Layout/-SPEC.tsx @@ -0,0 +1,46 @@ +import { type t, Dev, Signal, Spec } from '../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { updateForBreakpointSize, Content } from './-SPEC.u.tsx'; +import { Color, css, App } from './common.ts'; +import { Layout } from './m.Layout.tsx'; + +export * from './-SPEC.u.tsx'; + +export default Spec.describe('MobileLayout', async (e) => { + const debug = await createDebugSignals(); + const app = debug.app; + const p = app.props; + + e.it('init', async (e) => { + const ctx = Spec.ctx(e); + const update = { size: () => updateForBreakpointSize(ctx, app) }; + + Dev.Theme.signalEffect(ctx, debug.props.theme, 1); + Signal.effect(() => { + debug.listen(); + update.size(); + ctx.redraw(); + }); + + ctx.host.tracelineColor(Color.alpha(Color.CYAN, 0.3)); + ctx.subject + .size() + .display('grid') + .render((e) => { + const style = css({ display: 'grid', overflow: 'hidden' }); + const el = Layout.render(p.screen.breakpoint.value, debug.app); + return <div className={style.class}>{el}</div>; + }); + + /** + * Initialize environment. + */ + update.size(); + await App.Render.preload(debug.app, Content.factory, 'Entry', 'Trailer'); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/deploy/@tdb.slc/src/ui/ui.Layout/-SPEC.u.sample.tsx b/deploy/@tdb.slc/src/ui/ui.Layout/-SPEC.u.sample.tsx new file mode 100644 index 0000000000..4a58e88266 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Layout/-SPEC.u.sample.tsx @@ -0,0 +1,84 @@ +import { type t, css, SheetBase } from './common.ts'; + +export const Sample = { + sample0(): t.Content { + const id = 'sample-0'; + return { + id, + render(props) { + const { state } = props; + const styles = { + base: css({ + pointerEvents: 'auto', + cursor: 'default', + userSelect: 'none', + opacity: 0.3, + padding: 6, + }), + }; + + return ( + <div + className={styles.base.class} + onDoubleClick={() => state.stack.clear(1)} + onClick={() => { + if (!props.is.top) state.stack.pop(1); + }} + > + <div>{`id: "${id}"`}</div> + <div>{'props.children š·'}</div> + </div> + ); + }, + }; + }, + + sample1(): t.Content { + const id = 'sample-1'; + return { + id, + render(props) { + const { state } = props; + const edge: t.SheetMarginInput = + state.breakpoint.name === 'Desktop' ? ['1fr', 390, '1fr'] : 10; + + const onClick = () => { + if (!props.is.top) props.state.stack.pop(); + }; + + const styles = { base: css({ padding: 10 }) }; + + return ( + <SheetBase.View edgeMargin={edge} onClick={onClick}> + <div className={styles.base.class}> + <div>{`š Hello: "${id}" [${props.index}]`}</div> + <div>{'props.children š·'}</div> + </div> + </SheetBase.View> + ); + }, + }; + }, + + sample2(): t.Content { + return { + id: 'sample-2', + render(props) { + const { state } = props; + const breakpoint = state.breakpoint; + const orientation: t.SheetOrientation = 'Top:Down'; + const edge: t.SheetMarginInput = breakpoint.name === 'Desktop' ? [30, '1fr', 30] : 10; + const styles = { base: css({ padding: 10 }) }; + + return ( + <SheetBase.View orientation={orientation} edgeMargin={edge}> + <div className={styles.base.class}> + <div>{`š Hello: "${props.content.id}" [${props.index}]`}</div> + <div>{'props.children š·'}</div> + </div> + </SheetBase.View> + ); + }, + }; + }, +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Layout/-SPEC.u.tsx b/deploy/@tdb.slc/src/ui/ui.Layout/-SPEC.u.tsx new file mode 100644 index 0000000000..3347f9df48 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Layout/-SPEC.u.tsx @@ -0,0 +1,96 @@ +import React from 'react'; +import { Content } from '../../ui.Content/mod.ts'; +import { type t, Button, Signal } from './common.ts'; + +export { Content }; + +/** + * Used to update a size of a DevHost subject based on the current size-breakpoint. + */ +export const updateForBreakpointSize = (dev: t.DevCtx, app: t.AppSignals) => { + const breakpoint = app.props.screen.breakpoint.value; + if (breakpoint === 'Mobile') dev.subject.size([390, 844]); + if (breakpoint === 'Intermediate') dev.subject.size([600, 650]); + if (breakpoint === 'Desktop') dev.subject.size('fill'); +}; + +/** + * Button: push content onto the stack. + */ +export const pushStackButton = (app: t.AppSignals, stage: t.ContentStage) => { + return ( + <Button + key={`stack.${stage}`} + block + label={`stack.push:( "${stage}" )`} + onClick={async () => app.stack.push(await Content.factory(stage))} + /> + ); +}; + +/** + * Button Set: content pushing and clearing from the stack. + */ +export const pushStackContentButtons = (app: t.AppSignals) => { + const clear = (leave: number) => { + return ( + <Button + block + key={`stack.clear(${leave ?? 0})`} + label={`stack.clear${leave > 0 ? `( leave: ${leave} )` : ''}`} + onClick={() => app.stack.clear(leave)} + /> + ); + }; + + return [ + pushStackButton(app, 'Entry'), + pushStackButton(app, 'Trailer'), + pushStackButton(app, 'Overview'), + pushStackButton(app, 'Programme'), + <React.Fragment key={'hr:pop'}> + <hr /> + <Button block label={`stack.pop`} onClick={() => app.stack.pop()} /> + </React.Fragment>, + clear(1), + clear(0), + ]; +}; + +/** + * Button: cycle through screen breakpoints. + */ +export const screenBreakpointButton = (app: t.AppSignals) => { + type T = t.BreakpointName; + const p = app.props; + const list: T[] = ['Desktop', 'Intermediate', 'Mobile']; + return ( + <Button + block + label={`screen.breakpoint: ${p.screen.breakpoint ?? '<undefined>'}`} + onClick={() => Signal.cycle<T>(p.screen.breakpoint, list)} + /> + ); +}; + +/** + * Buttons: play/pause controls for media-player signals-API on the stack. + */ +export function layerVideoPlayerButtons(app: t.AppSignals) { + const videoLayers = app.stack.items.filter((layer) => Content.Is.video(layer)); + if (videoLayers.length === 0) return <div>{`(no video layers)`}</div>; + + return videoLayers.map((layer, index) => { + const p = layer.video?.props; + if (!p) return null; + return ( + <Button + block + key={`${layer.id}.${index}`} + label={() => `[ ${layer.id} ]: playing: ${p.playing.value}`} + onClick={() => Signal.toggle(p.playing)} + subscribe={() => p.playing.value} + /> + ); + }); +} diff --git a/deploy/@tdb.slc/src/ui/ui.Layout/common.ts b/deploy/@tdb.slc/src/ui/ui.Layout/common.ts new file mode 100644 index 0000000000..601c981ebf --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Layout/common.ts @@ -0,0 +1,12 @@ +export { Breakpoint } from '../App.Layout/mod.ts'; +export { App } from '../App/mod.ts'; +export { LogoWordmark } from '../ui.Logo.Wordmark/mod.ts'; +export { Sheet } from '../ui.Sheet/mod.ts'; +export { TooSmall } from '../ui.TooSmall/mod.ts'; + +export * from '../common.ts'; + +/** + * Constants: + */ +export const DEFAULTS = {} as const; diff --git a/deploy/@tdb.slc/src/ui/ui.Layout/m.Layout.tsx b/deploy/@tdb.slc/src/ui/ui.Layout/m.Layout.tsx new file mode 100644 index 0000000000..a4b31c2b50 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Layout/m.Layout.tsx @@ -0,0 +1,28 @@ +import { type t, Breakpoint } from './common.ts'; + +import { Layout as Base } from '../App.Layout/mod.ts'; +import { LayoutDesktop } from './ui.Desktop.tsx'; +import { LayoutIntermediate } from './ui.Intermediate.tsx'; +import { LayoutMobile } from './ui.Mobile.tsx'; + +/** + * Main Layout API (with UI components). + */ +export const Layout = { + ...Base, + + /** + * Render factory for the <Layout> component that matches the current size-breakpoint. + */ + render(size: t.BreakpointSizeInput, state?: t.AppSignals) { + const theme: t.CommonTheme = 'Dark'; + const breakpoint = Breakpoint.from(size); + const is = breakpoint.is; + + if (is.mobile) return <LayoutMobile state={state} theme={theme} />; + if (is.intermediate) return <LayoutIntermediate state={state} theme={theme} />; + if (is.desktop) return <LayoutDesktop state={state} theme={theme} />; + + return <div>{`Unsupported supported breakpoint: "${breakpoint.name}"`}</div>; + }, +} as const; diff --git a/deploy/@tdb.slc/src/ui/ui.Layout/mod.ts b/deploy/@tdb.slc/src/ui/ui.Layout/mod.ts new file mode 100644 index 0000000000..055d3e1b99 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Layout/mod.ts @@ -0,0 +1,8 @@ +/** + * @module + */ +export { Layout } from './m.Layout.tsx'; + +export { LayoutDesktop } from './ui.Desktop.tsx'; +export { LayoutIntermediate } from './ui.Intermediate.tsx'; +export { LayoutMobile } from './ui.Mobile.tsx'; diff --git a/deploy/@tdb.slc/src/ui/ui.Layout/t.ts b/deploy/@tdb.slc/src/ui/ui.Layout/t.ts new file mode 100644 index 0000000000..e9e41a68e3 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Layout/t.ts @@ -0,0 +1,28 @@ +import type { t } from './common.ts'; + +/** + * <Component>: Mobile + */ +export type LayoutMobileProps = { + state?: t.AppSignals; + style?: t.CssInput; + theme?: t.CommonTheme; +}; + +/** + * <Component>: Intermediate + */ +export type LayoutIntermediateProps = { + state?: t.AppSignals; + style?: t.CssInput; + theme?: t.CommonTheme; +}; + +/** + * <Component>: Desktop + */ +export type LayoutDesktopProps = { + state?: t.AppSignals; + style?: t.CssInput; + theme?: t.CommonTheme; +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Layout/ui.Desktop.Footer.tsx b/deploy/@tdb.slc/src/ui/ui.Layout/ui.Desktop.Footer.tsx new file mode 100644 index 0000000000..6576727902 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Layout/ui.Desktop.Footer.tsx @@ -0,0 +1,87 @@ +import React from 'react'; +import { type t, Button, Color, css, LogoWordmark } from './common.ts'; + +export type DesktopFooterProps = { + state?: t.AppSignals; + theme?: t.CommonTheme; + style?: t.CssInput; +}; + +type P = DesktopFooterProps; + +/** + * Component: + */ +export const DesktopFooter: React.FC<P> = (props) => { + const { state } = props; + const p = state?.props; + if (!p) return null; + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ + fontSize: 11, + padding: 10, + color: theme.fg, + display: 'grid', + gridTemplateColumns: `auto 1fr auto`, + pointerEvents: 'none', + }), + left: css({ + pointerEvents: 'auto', + display: 'grid', + gridAutoFlow: 'column', + alignContent: 'end', + columnGap: '10px', + }), + right: css({ pointerEvents: 'auto' }), + logo: { cc: { width: 100, marginRight: 4 } }, + }; + + const elDiv = <div>{'ā¢'}</div>; + + const dist = p.dist.value; + const elDist = dist && ( + <Button + theme={theme.name} + label={() => `version: #${wrangle.versionHash(state)}`} + onClick={() => window.open('./dist.json', '_blank')} + /> + ); + + const elPdfDownload = ( + <Button + theme={theme.name} + label={() => `pdf worksheet`} + onClick={() => window.open('./pdf/slc.pdf', '_blank')} + /> + ); + + const elLogos = <LogoWordmark logo={'CC'} theme={theme.name} style={styles.logo.cc} />; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={styles.left.class}> + {elDist} + {elDiv} + {elPdfDownload} + </div> + <div /> + <div className={styles.right.class}>{elLogos}</div> + </div> + ); +}; + +/** + * Helpers + */ +const wrangle = { + versionHash(state: t.AppSignals) { + const dist = state.props.dist.value; + const hx = dist?.hash.digest ?? '000000'; + return hx.slice(-5); + }, +} as const; diff --git a/deploy/@tdb.slc/src/ui/ui.Layout/ui.Desktop.tsx b/deploy/@tdb.slc/src/ui/ui.Layout/ui.Desktop.tsx new file mode 100644 index 0000000000..e415f4d31b --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Layout/ui.Desktop.tsx @@ -0,0 +1,89 @@ +import React from 'react'; +import { type t, App, Cropmarks, css, Signal, TooSmall, useSizeObserver } from './common.ts'; +import { DesktopFooter } from './ui.Desktop.Footer.tsx'; + +type P = t.LayoutDesktopProps; + +/** + * Component: + */ +export const LayoutDesktop: React.FC<P> = (props) => { + const { state } = props; + const p = state?.props; + if (!p) return null; + + const size = useSizeObserver(); + const isReady = size.ready; + let isTooSmall = wrangle.tooSmall(size.rect); + + /** + * Effects: + */ + Signal.useRedrawEffect(() => state.listen()); + + /** + * Render: + */ + const styles = { + base: css({ + opacity: isReady ? 1 : 0, + position: 'relative', + display: 'grid', + }), + bg: css({ Absolute: 0, display: 'grid' }), + body: css({ Absolute: 0, display: 'grid' }), + stack: css({ Absolute: 0, display: 'grid', pointerEvents: 'none' }), + cropmarks: { + base: css({ Absolute: 0, display: 'grid', pointerEvents: 'none' }), + body: css({ width: 390, pointerEvents: 'auto' }), + }, + footer: css({ Absolute: [null, 0, 0, 0] }), + }; + + const elCropmarks = ( + <div className={styles.cropmarks.base.class}> + <Cropmarks + theme={'Dark'} + borderOpacity={0.05} + size={{ mode: 'fill', x: false, y: true, margin: [0, 40, 0, 40] }} + > + <div className={styles.cropmarks.body.class}></div> + </Cropmarks> + </div> + ); + + const elStackItems = App.Render.stack(state); + const elStack = <div className={styles.stack.class}>{elStackItems}</div>; + const elFooter = <DesktopFooter theme={props.theme} style={styles.footer} state={state} />; + + const elBody = ( + <React.Fragment> + <div className={styles.bg.class} /> + <div className={styles.body.class}> + {elCropmarks} + {elStack} + </div> + {elFooter} + </React.Fragment> + ); + + const elTooSmall = <TooSmall theme={props.theme} />; + + return ( + <div ref={size.ref} className={css(styles.base, props.style).class}> + {isTooSmall ? elTooSmall : elBody} + </div> + ); +}; + +/** + * Helpers + */ +const wrangle = { + tooSmall(size?: t.DomRect) { + if (!size) return null; + if (size.height < 520) return true; + if (size.width < 630) return true; + return false; + }, +} as const; diff --git a/deploy/@tdb.slc/src/ui/ui.Layout/ui.Intermediate.tsx b/deploy/@tdb.slc/src/ui/ui.Layout/ui.Intermediate.tsx new file mode 100644 index 0000000000..ad4f7f7f41 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Layout/ui.Intermediate.tsx @@ -0,0 +1,36 @@ +import React from 'react'; +import { type t, Cropmarks, css, TooSmall } from './common.ts'; + +type P = t.LayoutIntermediateProps; + +/** + * Component: + */ +export const LayoutIntermediate: React.FC<P> = (props) => { + const { state } = props; + const p = state?.props; + if (!p) return null; + + /** + * Render: + */ + const styles = { + base: css({ position: 'relative', userSelect: 'none', display: 'grid' }), + body: css({}), + }; + + const msg = ` + Please make your window bigger, or + move over to your mobile device. + `; + + return ( + <div className={css(styles.base, props.style).class}> + <Cropmarks theme={'Dark'} borderOpacity={0.05}> + <div className={styles.body.class}> + <TooSmall theme={props.theme}>{msg}</TooSmall> + </div> + </Cropmarks> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Layout/ui.Mobile.tsx b/deploy/@tdb.slc/src/ui/ui.Layout/ui.Mobile.tsx new file mode 100644 index 0000000000..4e440e93e6 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Layout/ui.Mobile.tsx @@ -0,0 +1,27 @@ +import React from 'react'; +import { type t, App, css, Signal } from './common.ts'; + +type P = t.LayoutMobileProps; + +export const LayoutMobile: React.FC<P> = (props) => { + const { state } = props; + const p = state?.props; + + Signal.useRedrawEffect(() => { + state?.listen(); + }); + if (!p) return null; + + /** + * Render: + */ + const styles = { + base: css({ position: 'relative', display: 'grid' }), + }; + + /** + * The stack of sheets. + */ + const elStack = App.Render.stack(state); + return <div className={css(styles.base, props.style).class}>{elStack}</div>; +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/-SPEC.Debug.tsx b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/-SPEC.Debug.tsx new file mode 100644 index 0000000000..230ba6a9d2 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/-SPEC.Debug.tsx @@ -0,0 +1,101 @@ +import React from 'react'; +import { type t, Button, CanvasPanel, Color, css, Signal, D } from './common.ts'; +import { Selection } from './m.Selection.ts'; + +/** + * Types + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssValue }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; +type P = t.LogoCanvasProps; + +/** + * Signals + */ +export function createDebugSignals() { + const s = Signal.create; + const props = { + theme: s<P['theme']>('Dark'), + width: s<P['width']>(400), + over: s<P['over']>(), + selected: s<P['selected']>('purpose'), + selectionAnimation: s<P['selectionAnimation']>(), + }; + const api = { + props, + listen() { + const p = props; + p.width.value; + p.over.value; + p.theme.value; + p.selected.value; + p.selectionAnimation.value; + }, + }; + return api; +} + +/** + * Component + */ +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => debug.listen()); + + /** + * Render: + */ + const theme = Color.theme(p.theme.value); + const styles = { + base: css({ color: theme.fg }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <Button + label={`theme: ${p.theme}`} + onClick={() => Signal.cycle(p.theme, ['Light', 'Dark'])} + /> + <Button + block={true} + label={`width: ${p.width.value ?? '<undefined>'}`} + onClick={() => Signal.cycle(p.width, [undefined, 280, 400])} + /> + + <hr /> + {canvasSelectedButton(p.selected)} + + <Button + block={true} + label={() => { + const value = Selection.animation(p.selectionAnimation.value); + return `selectionAnimation.loop: ${value.loop ?? '<undefined>'}`; + }} + onClick={() => { + const next = Selection.animation(p.selectionAnimation.value); + next.loop = !(next.loop ?? D.selectionAnimation.loop); + p.selectionAnimation.value = next; + }} + /> + </div> + ); +}; + +/** + * Dev: selected panel(s) test button. + */ +export function canvasSelectedButton(signal: t.Signal<P['selected']>) { + return ( + <Button + block={true} + label={() => { + const value = signal.value; + const fmt = Array.isArray(value) ? `array[${value.length}]` : value ?? '<undefined>'; + return `selected: ${fmt}`; + }} + onClick={() => Signal.cycle(signal, [undefined, CanvasPanel.list, 'purpose'])} + /> + ); +} diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/-SPEC.tsx b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/-SPEC.tsx new file mode 100644 index 0000000000..58384e43eb --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/-SPEC.tsx @@ -0,0 +1,65 @@ +import { type t, Dev, Spec } from '../-test.ui.ts'; +import { createDebugSignals, Debug } from './-SPEC.Debug.tsx'; +import { css, Signal } from './common.ts'; +import { LogoCanvas } from './mod.ts'; + +export default Spec.describe('Logo.Canvas', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + const renderCanvas = (options: { theme?: t.CommonTheme } = {}) => { + return ( + <LogoCanvas + theme={options.theme ?? p.theme.value} + width={p.width.value} + selected={p.selected.value} + selectionAnimation={p.selectionAnimation.value} + over={p.over.value} + onPanelEvent={(e) => { + if (e.type === 'leave' && p.over.value === e.panel) p.over.value = undefined; + if (e.type === 'enter') p.over.value = e.panel; + if (e.type === 'click') p.selected.value = e.panel; + }} + /> + ); + }; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + const updateSize = () => { + const width = p.width.value; + if (width === undefined) ctx.subject.size('fill-x', 180); + else ctx.subject.size([width, null]); + }; + + Dev.Theme.signalEffect(ctx, debug.props.theme, 1); + Signal.effect(() => { + debug.listen(); + updateSize(); + ctx.redraw(); + }); + + ctx.subject.display('grid').render((e) => renderCanvas()); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); + + e.it('ui:footer', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.footer.render((e) => { + const styles = { + base: css({ display: 'grid', placeItems: 'center', marginBottom: 55 }), + body: css({ width: 220 }), + }; + return ( + <div className={styles.base.class}> + <div className={styles.body.class}>{renderCanvas({ theme: 'Light' })}</div> + </div> + ); + }); + }); +}); diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/canvas.mini.svg b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/canvas.mini.svg new file mode 100644 index 0000000000..366d200007 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/canvas.mini.svg @@ -0,0 +1,37 @@ +<svg width="354" height="184" viewBox="0 0 354 184" fill="none" xmlns="http://www.w3.org/2000/svg"> +<g id="canvas.mini" clip-path="url(#clip0_1_15)"> +<g id="panels"> +<path id="panel.impact" opacity="0.3" d="M350.697 22.1289C350.697 11.6355 342.19 3.12891 331.697 3.12891H177V38.0391H350.697V22.1289Z" fill="white"/> +<rect id="panel.problem" opacity="0.3" x="2.39258" y="37.8516" width="73.4749" height="106.705" fill="white"/> +<rect id="panel.solution" opacity="0.3" x="75.8672" y="37.5206" width="67.7266" height="51.2861" fill="white"/> +<rect id="panel.metrics" opacity="0.3" x="75.8672" y="88.2812" width="67.7266" height="56.4634" fill="white"/> +<rect id="panel.uvp" opacity="0.3" x="143.594" y="37.8516" width="70.5621" height="106.705" fill="white"/> +<rect id="panel.advantage" opacity="0.3" x="214.156" y="37.8516" width="67.5486" height="50.9541" fill="white"/> +<rect id="panel.channels" opacity="0.3" x="214.156" y="88.4758" width="67.5482" height="56.0801" fill="white"/> +<rect id="panel.customers" opacity="0.3" x="281.704" y="38.0391" width="69.5714" height="106.068" fill="white"/> +<path id="panel.costs" opacity="0.3" d="M3.30273 161.718C3.30273 172.212 11.8093 180.718 22.3027 180.718H176.999V144.557H3.30273V161.718Z" fill="white"/> +<path id="panel.revenue" opacity="0.3" d="M351.276 161.719C351.276 172.212 342.77 180.719 332.276 180.719H176.999V144.745H351.276V161.719Z" fill="white"/> +<path id="panel.purpose" opacity="0.3" d="M3.30365 22.1289C3.30365 11.6355 11.8102 3.12891 22.3036 3.12891H177V38.0391H3.30365V22.1289Z" fill="white"/> +</g> +<g id="outline"> +<rect id="border" x="3" y="3" width="348.248" height="177.711" rx="17" fill="white" fill-opacity="0.1" stroke="white" stroke-width="6"/> +<g id="grid-lines"> +<line id="Line 445" x1="5.28125" y1="38.0703" x2="348.718" y2="38.0703" stroke="white" stroke-width="6"/> +<line id="Line 446" x1="5.28125" y1="144.795" x2="348.718" y2="144.795" stroke="white" stroke-width="6"/> +<line id="Line 447" x1="75.9106" y1="37.8496" x2="75.9106" y2="144.555" stroke="white" stroke-width="6"/> +<line id="Line 448" x1="143.303" y1="37.8496" x2="143.303" y2="144.555" stroke="white" stroke-width="6"/> +<line id="Line 451" x1="78.5269" y1="88.5293" x2="142.72" y2="88.5293" stroke="white" stroke-width="6"/> +<line id="Line 447_2" x1="213.896" y1="37.8496" x2="213.896" y2="144.555" stroke="white" stroke-width="6"/> +<line id="Line 448_2" x1="281.288" y1="37.8496" x2="281.288" y2="144.555" stroke="white" stroke-width="6"/> +<line id="Line 451_2" x1="216.512" y1="88.5293" x2="280.704" y2="88.5293" stroke="white" stroke-width="6"/> +<line id="Line 454" x1="176.904" y1="144.795" x2="176.904" y2="180.716" stroke="white" stroke-width="6"/> +<line id="Line 455" x1="176.904" y1="3.28125" x2="176.904" y2="39.2026" stroke="white" stroke-width="6"/> +</g> +</g> +</g> +<defs> +<clipPath id="clip0_1_15"> +<rect width="354" height="184" fill="white"/> +</clipPath> +</defs> +</svg> diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/common.ts b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/common.ts new file mode 100644 index 0000000000..15284cc8e2 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/common.ts @@ -0,0 +1,12 @@ +import type { t } from '../common.ts'; +export * from '../common.ts'; + +const selectionAnimation: Required<t.LogoCanvasSelectionAnimation> = { + loop: false, + delay: 200, +}; + +export const DEFAULTS = { + selectionAnimation, +} as const; +export const D = DEFAULTS; diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/m.Selection.ts b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/m.Selection.ts new file mode 100644 index 0000000000..50931dae43 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/m.Selection.ts @@ -0,0 +1,18 @@ +import { type t, D } from './common.ts'; + +type P = t.LogoCanvasProps; + +/** + * Helpers for working with the component selection properties. + */ +export const Selection = { + selected(value: P['selected']): t.CanvasPanel[] { + if (!value) return []; + return Array.isArray(value) ? value : [value]; + }, + + animation(value: P['selectionAnimation']): t.LogoCanvasSelectionAnimation { + const DEFAULT = D.selectionAnimation; + return value ? { ...DEFAULT, ...value } : DEFAULT; + }, +} as const; diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/mod.ts b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/mod.ts new file mode 100644 index 0000000000..2297de7c69 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/mod.ts @@ -0,0 +1,6 @@ +/** + * @module + */ +export { Theme } from './u.ts'; +export { LogoCanvas } from './ui.tsx'; +export { Selection } from './m.Selection.ts'; diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/t.ts b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/t.ts new file mode 100644 index 0000000000..48065de60d --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/t.ts @@ -0,0 +1,30 @@ +import type { t } from './common.ts'; + +/** + * <Component>: Mini Canvas + */ +export type LogoCanvasProps = { + selected?: t.CanvasPanel | t.CanvasPanel[]; + selectionAnimation?: LogoCanvasSelectionAnimation; + over?: t.CanvasPanel; + width?: number; + theme?: t.CommonTheme; + style?: t.CssInput; + onPanelEvent?: t.LogoCanvasPanelHandler; +}; + +/** Selection animation options */ +export type LogoCanvasSelectionAnimation = { + /** Cycle back to the beginning when animation completes. */ + loop?: boolean; + delay?: t.Msecs; +}; + +/** + * Events + */ +export type LogoCanvasPanelHandler = (e: LogoCanvasPanelHandlerArgs) => void; +export type LogoCanvasPanelHandlerArgs = { + type: 'enter' | 'leave' | 'click'; + panel: t.CanvasPanel; +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/u.Theme.ts b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/u.Theme.ts new file mode 100644 index 0000000000..fddd40477b --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/u.Theme.ts @@ -0,0 +1,7 @@ +import { type t, Color } from './common.ts'; + +export const Theme = { + color(theme: t.CommonTheme = 'Light') { + return theme === 'Dark' ? Color.WHITE : Color.DARK; + }, +} as const; diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/u.ts b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/u.ts new file mode 100644 index 0000000000..8bc5696ae3 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/u.ts @@ -0,0 +1 @@ +export * from './u.Theme.ts'; diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/ui.Svg.tsx b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/ui.Svg.tsx new file mode 100644 index 0000000000..01a840ab3a --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/ui.Svg.tsx @@ -0,0 +1,52 @@ +import React from 'react'; +import { type t, Color, css, Svg } from './common.ts'; +import { useMouse } from './use.Mouse.tsx'; +import { useTheme } from './use.Theme.ts'; + +export type SvgImageProps = { + width?: number; + selected?: t.CanvasPanel; + over?: t.CanvasPanel; + bgBlur?: number; + + theme?: t.CommonTheme; + style?: t.CssInput; + + onPanelEvent?: t.LogoCanvasPanelHandler; +}; + +/** + * Component. + * + */ +export const SvgImage: React.FC<SvgImageProps> = (props) => { + const { width, bgBlur = 20, over, selected, onPanelEvent } = props; + const theme = Color.theme(props.theme); + + /** + * Source design, search Figma: "canvas.mini" + */ + const svg = Svg.useSvg<HTMLDivElement>(() => import('./canvas.mini.svg'), [354, 184]); + + useTheme(svg, theme.name); + useMouse(svg, { theme: theme.name, over, selected, onPanelEvent }); + + /** + * Render: + */ + const styles = { + base: css({ + position: 'relative', + cursor: 'default', + color: theme.fg, + backdropFilter: `blur(${bgBlur}px)`, + lineHeight: 0, // NB: ensure no "baseline" gap below the <svg>. + }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div ref={svg.ref} /> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/ui.tsx b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/ui.tsx new file mode 100644 index 0000000000..cc5fd3a199 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/ui.tsx @@ -0,0 +1,37 @@ +import React from 'react'; +import { type t, css } from './common.ts'; +import { SvgImage } from './ui.Svg.tsx'; +import { useSelection } from './use.Selection.ts'; + +type P = t.LogoCanvasProps; + +/** + * Component. + */ +export const LogoCanvas: React.FC<P> = (props) => { + const { over, onPanelEvent, width } = props; + const { selected } = useSelection(props); + + /** + * Render: + */ + const styles = { + base: css({ + position: 'relative', + cursor: 'default', + userSelect: 'none', + }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <SvgImage + theme={props.theme} + width={width} + selected={selected} + over={over} + onPanelEvent={onPanelEvent} + /> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/use.Mouse.tsx b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/use.Mouse.tsx new file mode 100644 index 0000000000..8bf8a70eab --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/use.Mouse.tsx @@ -0,0 +1,70 @@ +import type { t } from './common.ts'; + +import { useEffect } from 'react'; +import { Theme } from './u.ts'; + +type Options = { + selected?: t.CanvasPanel; + over?: t.CanvasPanel; + theme?: t.CommonTheme; + onPanelEvent?: t.LogoCanvasPanelHandler; +}; + +/** + * Manages mouse behavior on a Canvas. + */ +export function useMouse<T extends HTMLElement>(svg: t.SvgInstance<T>, options: Options = {}) { + useCanvasPanelMouse('purpose', svg, options); + useCanvasPanelMouse('customers', svg, options); + useCanvasPanelMouse('problem', svg, options); + useCanvasPanelMouse('uvp', svg, options); + useCanvasPanelMouse('solution', svg, options); + useCanvasPanelMouse('channels', svg, options); + useCanvasPanelMouse('revenue', svg, options); + useCanvasPanelMouse('costs', svg, options); + useCanvasPanelMouse('metrics', svg, options); + useCanvasPanelMouse('advantage', svg, options); + useCanvasPanelMouse('impact', svg, options); +} + +/** + * Manages mouse behavior on an individual Canvas panel. + */ +export function useCanvasPanelMouse<T extends HTMLElement>( + panel: t.CanvasPanel, + svg: t.SvgInstance<T>, + options: Options = {}, +) { + const { selected, over, theme, onPanelEvent } = options; + const isSelected = selected === panel; + + useEffect(() => { + if (!svg.ready) return; + + const query = `#panel\\.${panel}`; + const svgPanel = svg.query(query); + const color = Theme.color(theme); + + const updateOpacity = () => { + const isOver = over === panel; + const opacity = isSelected ? 1 : isOver ? 0.2 : 0; + svgPanel?.opacity(opacity); + svgPanel?.fill(color); + }; + + svg.query('#outline')?.css('pointer-events' as any, 'none'); // NB: Allow click-through of the grid lines that sit above each panel. + updateOpacity(); // Set default opacity. + + /** + * Event Handlers. + */ + const onOver = (isOver: boolean) => { + updateOpacity(); + onPanelEvent?.({ panel, type: isOver ? 'enter' : 'leave' }); + }; + svgPanel?.off(); + svgPanel?.on('mouseover', () => onOver(true)); + svgPanel?.on('mouseleave', () => onOver(false)); + svgPanel?.on('mousedown', () => onPanelEvent?.({ panel, type: 'click' })); + }, [svg.ready, panel, selected, over, theme, isSelected]); +} diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/use.Selection.ts b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/use.Selection.ts new file mode 100644 index 0000000000..4ac0aca876 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/use.Selection.ts @@ -0,0 +1,45 @@ +import { useEffect, useState } from 'react'; +import { type t, Time } from './common.ts'; +import { Selection } from './m.Selection.ts'; + +type P = t.LogoCanvasProps; + +export function useSelection(props: t.LogoCanvasProps) { + const animation = Selection.animation(props.selectionAnimation); + const [selected, setSeleted] = useState<t.CanvasPanel>(); + + /** + * Effect: keep selected state up-to-date. + */ + useEffect(() => { + const time = Time.until(); + + async function animateSelected(list: t.CanvasPanel[]) { + if (time.disposed) return; + for (const value of list) { + if (time.disposed) break; + setSeleted(value); + await time.wait(animation.delay); + } + if (animation.loop) animateSelected(list); // ā š³ RECURSION. + } + + if (Array.isArray(props.selected)) { + animateSelected(props.selected); + } else { + setSeleted(props.selected); + } + + return time.dispose; + }, [ + // Deps: + Selection.selected(props.selected).join(':'), + animation.delay, + animation.loop, + ]); + + /** + * API: + */ + return { selected } as const; +} diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/use.Theme.ts b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/use.Theme.ts new file mode 100644 index 0000000000..19f984a575 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Canvas/use.Theme.ts @@ -0,0 +1,21 @@ +import { useEffect } from 'react'; +import { type t } from './common.ts'; +import { Theme } from './u.ts'; + +/** + * Handle updating the canvas theme. + */ +export function useTheme(svg: t.SvgInstance<HTMLDivElement>, theme?: t.CommonTheme) { + useEffect(() => { + const color = Theme.color(theme); + const setColor = (color: string, elements: t.SvgElement[]) => { + elements.forEach((el) => { + el?.fill(color); + el?.stroke(color); + }); + }; + setColor(color, svg.queryAll('#border')); + setColor(color, svg.queryAll('#outline')); + setColor(color, svg.queryAll('#grid-lines line')); + }, [svg.draw, theme]); +} diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/-SPEC.Debug.tsx b/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/-SPEC.Debug.tsx new file mode 100644 index 0000000000..e971e25b2a --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/-SPEC.Debug.tsx @@ -0,0 +1,77 @@ +import React from 'react'; +import { type t, Button, Color, css, DEFAULTS, Signal } from './common.ts'; + +const D = DEFAULTS; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; +type P = DebugProps; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + const s = Signal.create; + type P = t.LogoWordmarkProps; + const props = { + width: s<number | undefined>(300), + theme: s<P['theme']>('Dark'), + logo: s<P['logo']>(D.logo), + }; + const api = { + props, + listen() { + const p = props; + p.width.value; + p.theme.value; + p.logo.value; + }, + }; + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<P> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => debug.listen()); + + /** + * Render: + */ + const theme = Color.theme(p.theme.value); + const styles = { + base: css({ color: theme.fg }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <Button + block + label={`theme: ${p.theme}`} + onClick={() => Signal.cycle<t.LogoWordmarkProps['theme']>(p.theme, ['Light', 'Dark'])} + /> + <Button + block + label={`width: ${p.width.value ?? '<undefined>'}`} + onClick={() => Signal.cycle(p.width, [undefined, 90, 150, 300])} + /> + + <hr /> + <Button + block + label={`logo: "${p.logo}"`} + onClick={() => Signal.cycle<t.LogoKind>(p.logo, ['SLC', 'CC'])} + /> + + <hr /> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/-SPEC.tsx b/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/-SPEC.tsx new file mode 100644 index 0000000000..0fab3af041 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/-SPEC.tsx @@ -0,0 +1,37 @@ +import { Dev, Signal, Spec } from '../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { LogoWordmark } from './mod.ts'; + +export default Spec.describe('Logo.Wordmark', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + const updateSize = () => { + const width = p.width.value; + if (width === undefined) ctx.subject.size('fill-x', 180); + else ctx.subject.size([width, null]); + }; + + Dev.Theme.signalEffect(ctx, p.theme); + Signal.effect(() => { + debug.listen(); + updateSize(); + ctx.redraw(); + }); + + ctx.subject + .size() + .display('grid') + .render((e) => <LogoWordmark theme={p.theme.value} logo={p.logo.value} />); + + updateSize(); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/common.ts b/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/common.ts new file mode 100644 index 0000000000..f26e229f95 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/common.ts @@ -0,0 +1,15 @@ +import { type t } from '../common.ts'; + +export * from '../common.ts'; +export { Theme } from '../ui.Logo.Canvas/mod.ts'; + +/** + * Constants: + */ +export const DEFAULTS = { + width: 90, + get logo(): t.LogoKind { + return 'SLC'; + }, +} as const; +export const D = DEFAULTS; diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/images/logo.cc.svg b/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/images/logo.cc.svg new file mode 100644 index 0000000000..445c14b0c1 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/images/logo.cc.svg @@ -0,0 +1,20 @@ +<svg width="345" height="84" viewBox="0 0 345 84" fill="none" xmlns="http://www.w3.org/2000/svg"> +<path fill-rule="evenodd" clip-rule="evenodd" d="M34.9299 81.8755C54.2212 81.8755 69.8598 66.2368 69.8598 46.9456C69.8598 27.6543 54.2212 12.0157 34.9299 12.0157C15.6386 12.0157 0 27.6543 0 46.9456C0 66.2368 15.6386 81.8755 34.9299 81.8755ZM34.6505 75.7278C50.7008 75.7278 63.7121 62.7165 63.7121 46.6661C63.7121 30.6158 50.7008 17.6045 34.6505 17.6045C18.6001 17.6045 5.58878 30.6158 5.58878 46.6661C5.58878 62.7165 18.6001 75.7278 34.6505 75.7278Z" fill="black"/> +<path d="M272.967 45.9334C267.765 45.9334 263.592 44.3393 260.446 41.1513C257.341 37.9212 255.789 33.8103 255.789 28.8185C255.789 23.9525 257.32 19.9045 260.383 16.6745C263.487 13.4444 267.472 11.8294 272.338 11.8294C276.742 11.8294 280.413 13.1718 283.349 15.8565C287.209 19.3801 289.096 24.6027 289.012 31.5241H265.857C266.192 33.7474 266.968 35.4883 268.185 36.7467C269.401 37.9632 271.037 38.5714 273.093 38.5714C275.736 38.5714 277.518 37.5018 278.441 35.3624H288.446C287.691 38.4666 285.887 41.0044 283.035 42.976C280.224 44.9476 276.868 45.9334 272.967 45.9334ZM265.92 25.4207H278.63C278.462 23.4911 277.812 21.9599 276.679 20.8273C275.589 19.6947 274.205 19.1284 272.527 19.1284C268.877 19.1284 266.675 21.2258 265.92 25.4207Z" fill="black"/> +<path d="M235.85 44.9895L224.713 12.7733H235.284L239.689 27.3083L241.45 33.7893H241.576C242.122 31.608 242.709 29.4477 243.338 27.3083L247.806 12.7733H258.062L246.925 44.9895H235.85Z" fill="black"/> +<path d="M214.066 44.9895V12.7732H224.322V44.9895H214.066ZM214.066 8.30575V0H224.322V8.30575H214.066Z" fill="black"/> +<path d="M206.667 45.367C203.395 45.367 200.857 44.7168 199.053 43.4165C197.292 42.0741 196.411 39.767 196.411 36.495V19.1913H192.132V12.7733H196.411V2.57983H206.415V12.7733H212.267V19.1913H206.415V34.2927C206.415 35.4673 206.709 36.2853 207.296 36.7467C207.926 37.2081 208.827 37.4388 210.002 37.4388C210.338 37.4388 210.778 37.4388 211.323 37.4388C211.869 37.3969 212.183 37.3759 212.267 37.3759V44.8637C211.093 45.1993 209.226 45.367 206.667 45.367Z" fill="black"/> +<path d="M171.855 45.8075C168.541 45.8075 165.878 44.9476 163.864 43.2277C161.851 41.5078 160.844 39.0958 160.844 35.9916C160.844 32.7196 161.934 30.3496 164.116 28.8814C166.297 27.3712 169.317 26.3645 173.177 25.8611C176.323 25.4836 178.462 25.0641 179.595 24.6027C180.769 24.0993 181.357 23.3232 181.357 22.2745C181.357 19.9674 179.888 18.8138 176.952 18.8138C173.806 18.8138 172.065 20.1981 171.729 22.9667H162.291C162.417 19.7786 163.717 17.1359 166.192 15.0385C168.709 12.941 172.275 11.8923 176.889 11.8923C181.503 11.8923 184.985 12.7732 187.334 14.5351C190.061 16.5066 191.424 19.5689 191.424 23.7217V39.9557C191.424 42.5146 191.802 44.0457 192.557 44.549V44.9895H182.615C182.196 44.4022 181.839 43.2067 181.545 41.4029H181.419C179.364 44.3393 176.176 45.8075 171.855 45.8075ZM175.127 39.2636C177.057 39.2636 178.609 38.7392 179.783 37.6905C181 36.5999 181.608 35.1946 181.608 33.4747V29.6364C180.601 30.1398 178.84 30.7061 176.323 31.3353C174.267 31.7968 172.82 32.3421 171.981 32.9713C171.142 33.6006 170.723 34.5024 170.723 35.677C170.723 38.068 172.191 39.2636 175.127 39.2636Z" fill="black"/> +<path d="M144.763 45.9334C139.561 45.9334 135.387 44.3393 132.241 41.1513C129.137 37.9212 127.585 33.8103 127.585 28.8185C127.585 23.9525 129.116 19.9045 132.178 16.6745C135.282 13.4444 139.268 11.8294 144.134 11.8294C148.538 11.8294 152.209 13.1718 155.145 15.8565C159.004 19.3801 160.892 24.6027 160.808 31.5241H137.653C137.988 33.7474 138.764 35.4883 139.981 36.7467C141.197 37.9632 142.833 38.5714 144.889 38.5714C147.531 38.5714 149.314 37.5018 150.237 35.3624H160.242C159.487 38.4666 157.683 41.0044 154.83 42.976C152.02 44.9476 148.664 45.9334 144.763 45.9334ZM137.715 25.4207H150.426C150.258 23.4911 149.608 21.9599 148.475 20.8273C147.385 19.6947 146 19.1284 144.322 19.1284C140.673 19.1284 138.471 21.2258 137.715 25.4207Z" fill="black"/> +<path d="M107.447 44.9895V12.7732H117.263V17.807H117.451C119.633 14.1156 122.569 12.2699 126.261 12.2699C127.225 12.2699 127.897 12.3538 128.274 12.5216V21.3307H128.022C124.834 20.8693 122.317 21.4356 120.472 23.0296C118.626 24.5817 117.703 27.0776 117.703 30.5174V44.9895H107.447Z" fill="black"/> +<path d="M90.7759 45.9963C85.6582 45.9963 81.5263 44.3813 78.3801 41.1512C75.234 37.9212 73.661 33.8522 73.661 28.9443C73.661 24.0364 75.2131 19.9674 78.3172 16.7374C81.4214 13.5073 85.4904 11.8923 90.5242 11.8923C94.8029 11.8923 98.3056 13.0249 101.032 15.2901C103.759 17.5553 105.353 20.5337 105.814 24.2251H95.9355C95.2224 21.0371 93.4605 19.443 90.65 19.443C88.4687 19.443 86.8117 20.282 85.6791 21.9599C84.5885 23.6378 84.0432 25.966 84.0432 28.9443C84.0432 31.8807 84.6095 34.1878 85.7421 35.8658C86.8747 37.5017 88.5106 38.3197 90.65 38.3197C93.922 38.3197 95.7887 36.474 96.2501 32.7826H106.066C105.94 36.5999 104.472 39.7669 101.661 42.2838C98.8928 44.7588 95.2643 45.9963 90.7759 45.9963Z" fill="black"/> +<path d="M329.865 84C325.166 84 321.412 82.9932 318.602 80.9797C315.833 78.9243 314.365 76.1137 314.197 72.5481H323.887C324.265 75.7362 326.236 77.3302 329.802 77.3302C331.354 77.3302 332.57 77.0576 333.451 76.5122C334.374 75.925 334.836 75.128 334.836 74.1212C334.836 73.7856 334.794 73.492 334.71 73.2403C334.626 72.9886 334.479 72.7579 334.269 72.5481C334.059 72.3384 333.85 72.1496 333.64 71.9818C333.472 71.814 333.2 71.6672 332.822 71.5414C332.444 71.3736 332.109 71.2477 331.815 71.1638C331.564 71.08 331.165 70.9961 330.62 70.9122C330.116 70.7863 329.718 70.7024 329.424 70.6605C329.131 70.5766 328.669 70.4927 328.04 70.4088C327.411 70.2829 326.949 70.199 326.656 70.1571C325.313 69.9473 324.181 69.7376 323.258 69.5279C322.335 69.2762 321.307 68.8986 320.175 68.3953C319.084 67.8919 318.182 67.3256 317.469 66.6964C316.798 66.0252 316.231 65.1443 315.77 64.0536C315.309 62.921 315.078 61.6206 315.078 60.1524C315.078 56.8805 316.399 54.3426 319.042 52.5388C321.685 50.735 324.999 49.8332 328.984 49.8332C333.472 49.8332 336.954 50.756 339.429 52.6017C341.904 54.4055 343.267 56.9224 343.519 60.1524H334.08C333.745 57.6355 332.025 56.3771 328.921 56.3771C327.578 56.3771 326.488 56.6288 325.649 57.1322C324.852 57.6355 324.453 58.3696 324.453 59.3344C324.453 59.5861 324.495 59.8168 324.579 60.0266C324.663 60.2363 324.789 60.4251 324.957 60.5929C325.166 60.7607 325.355 60.9075 325.523 61.0333C325.733 61.1592 326.005 61.285 326.341 61.4109C326.677 61.5367 326.97 61.6416 327.222 61.7255C327.516 61.8094 327.893 61.8933 328.354 61.9772C328.858 62.0611 329.256 62.145 329.55 62.2289C329.886 62.2708 330.326 62.3547 330.871 62.4806C331.417 62.5645 331.857 62.6274 332.193 62.6693C333.619 62.921 334.794 63.1517 335.716 63.3615C336.639 63.5712 337.709 63.9487 338.925 64.4941C340.184 64.9975 341.17 65.6057 341.883 66.3188C342.596 66.99 343.204 67.9338 343.708 69.1503C344.253 70.3249 344.526 71.7092 344.526 73.3032C344.526 76.743 343.162 79.3857 340.436 81.2314C337.751 83.0771 334.227 84 329.865 84Z" fill="black"/> +<path d="M282.15 82.9933V50.777H292.029V55.1816H292.217C294.692 51.616 298.006 49.8332 302.159 49.8332C305.641 49.8332 308.367 50.9658 310.339 53.231C312.353 55.4542 313.359 58.3906 313.359 62.0401V82.9933H303.103V64.1165C303.103 62.3547 302.663 60.9704 301.782 59.9637C300.943 58.915 299.705 58.3906 298.069 58.3906C296.391 58.3906 295.028 59.0198 293.979 60.2783C292.931 61.5367 292.406 63.1727 292.406 65.1862V82.9933H282.15Z" fill="black"/> +<path d="M258.296 74.0582C259.47 75.8201 261.148 76.701 263.33 76.701C265.511 76.701 267.189 75.8201 268.363 74.0582C269.58 72.2964 270.188 69.9263 270.188 66.948C270.188 63.9697 269.58 61.5996 268.363 59.8378C267.189 58.034 265.511 57.1321 263.33 57.1321C261.148 57.1321 259.47 58.013 258.296 59.7749C257.121 61.5367 256.534 63.9277 256.534 66.948C256.534 69.9263 257.121 72.2964 258.296 74.0582ZM275.788 79.155C272.642 82.385 268.51 84 263.393 84C258.275 84 254.122 82.385 250.934 79.155C247.746 75.9249 246.152 71.856 246.152 66.948C246.152 62.0401 247.746 57.9711 250.934 54.7411C254.122 51.5111 258.275 49.8961 263.393 49.8961C268.51 49.8961 272.642 51.5111 275.788 54.7411C278.976 57.9711 280.57 62.0401 280.57 66.948C280.57 71.856 278.976 75.9249 275.788 79.155Z" fill="black"/> +<path d="M194.798 82.9933V50.777H204.614V55.3703H204.803C207.026 51.6789 210.277 49.8332 214.556 49.8332C218.793 49.8332 221.813 51.7208 223.617 55.4962H223.743C226.176 51.7208 229.552 49.8332 233.873 49.8332C237.439 49.8332 240.144 50.9448 241.99 53.168C243.878 55.3494 244.822 58.3067 244.822 62.0401V82.9933H234.565V63.8648C234.565 60.2154 233.055 58.3906 230.035 58.3906C228.441 58.3906 227.182 58.9989 226.259 60.2154C225.379 61.3899 224.938 62.963 224.938 64.9345V82.9933H214.682V63.8648C214.682 60.2154 213.172 58.3906 210.151 58.3906C208.557 58.3906 207.299 58.9989 206.376 60.2154C205.495 61.3899 205.055 62.963 205.055 64.9345V82.9933H194.798Z" fill="black"/> +<path d="M141.998 82.9933V50.777H151.813V55.3703H152.002C154.226 51.6789 157.477 49.8332 161.755 49.8332C165.992 49.8332 169.012 51.7208 170.816 55.4962H170.942C173.375 51.7208 176.752 49.8332 181.072 49.8332C184.638 49.8332 187.344 50.9448 189.189 53.168C191.077 55.3494 192.021 58.3067 192.021 62.0401V82.9933H181.765V63.8648C181.765 60.2154 180.254 58.3906 177.234 58.3906C175.64 58.3906 174.382 58.9989 173.459 60.2154C172.578 61.3899 172.137 62.963 172.137 64.9345V82.9933H161.881V63.8648C161.881 60.2154 160.371 58.3906 157.351 58.3906C155.757 58.3906 154.498 58.9989 153.575 60.2154C152.694 61.3899 152.254 62.963 152.254 64.9345V82.9933H141.998Z" fill="black"/> +<path d="M118.144 74.0582C119.318 75.8201 120.996 76.701 123.177 76.701C125.359 76.701 127.037 75.8201 128.211 74.0582C129.428 72.2964 130.036 69.9263 130.036 66.948C130.036 63.9697 129.428 61.5996 128.211 59.8378C127.037 58.034 125.359 57.1321 123.177 57.1321C120.996 57.1321 119.318 58.013 118.144 59.7749C116.969 61.5367 116.382 63.9277 116.382 66.948C116.382 69.9263 116.969 72.2964 118.144 74.0582ZM135.636 79.155C132.49 82.385 128.358 84 123.24 84C118.123 84 113.97 82.385 110.782 79.155C107.594 75.9249 106 71.856 106 66.948C106 62.0401 107.594 57.9711 110.782 54.7411C113.97 51.5111 118.123 49.8961 123.24 49.8961C128.358 49.8961 132.49 51.5111 135.636 54.7411C138.824 57.9711 140.418 62.0401 140.418 66.948C140.418 71.856 138.824 75.9249 135.636 79.155Z" fill="black"/> +<path d="M90.7759 84C85.6582 84 81.5263 82.385 78.3801 79.155C75.234 75.9249 73.661 71.856 73.661 66.948C73.661 62.0401 75.2131 57.9711 78.3172 54.7411C81.4214 51.5111 85.4904 49.8961 90.5242 49.8961C94.8029 49.8961 98.3056 51.0287 101.032 53.2939C103.759 55.5591 105.353 58.5374 105.814 62.2288H95.9355C95.2224 59.0408 93.4605 57.4467 90.65 57.4467C88.4687 57.4467 86.8117 58.2857 85.6791 59.9636C84.5885 61.6416 84.0432 63.9697 84.0432 66.948C84.0432 69.8844 84.6095 72.1916 85.7421 73.8695C86.8747 75.5055 88.5106 76.3235 90.65 76.3235C93.922 76.3235 95.7887 74.4777 96.2501 70.7863H106.066C105.94 74.6036 104.472 77.7707 101.661 80.2876C98.8928 82.7625 95.2643 84 90.7759 84Z" fill="black"/> +<path d="M45.1031 57.4654C41.8891 57.4654 39.2942 56.4511 37.3184 54.4226C35.3426 52.3941 34.3546 49.8387 34.3546 46.7564C34.3546 43.6741 35.3294 41.1187 37.2789 39.0902C39.2283 37.0617 41.7837 36.0474 44.9451 36.0474C47.6322 36.0474 49.8319 36.7587 51.5443 38.1813C53.2567 39.6039 54.2578 41.4744 54.5476 43.7927H48.3435C47.8956 41.7905 46.7892 40.7894 45.0241 40.7894C43.6542 40.7894 42.6136 41.3163 41.9023 42.3701C41.2173 43.4238 40.8749 44.8859 40.8749 46.7564C40.8749 48.6005 41.2305 50.0494 41.9418 51.1032C42.6531 52.1306 43.6805 52.6443 45.0241 52.6443C47.0789 52.6443 48.2513 51.4852 48.5411 49.1669H54.7056C54.6266 51.5642 53.7045 53.5532 51.9395 55.1339C50.2007 56.6882 47.922 57.4654 45.1031 57.4654Z" fill="black"/> +<path d="M24.7938 57.4654C21.5798 57.4654 18.9849 56.4511 17.0091 54.4226C15.0333 52.3941 14.0453 49.8387 14.0453 46.7564C14.0453 43.6741 15.0201 41.1187 16.9696 39.0902C18.919 37.0617 21.4744 36.0475 24.6358 36.0475C27.3229 36.0475 29.5226 36.7588 31.235 38.1813C32.9474 39.6039 33.9485 41.4744 34.2383 43.7927H28.0342C27.5863 41.7905 26.4799 40.7894 24.7148 40.7894C23.3449 40.7894 22.3043 41.3163 21.593 42.3701C20.908 43.4239 20.5656 44.886 20.5656 46.7564C20.5656 48.6005 20.9212 50.0495 21.6325 51.1032C22.3438 52.1307 23.3712 52.6444 24.7148 52.6444C26.7697 52.6444 27.942 51.4852 28.2318 49.1669H34.3963C34.3173 51.5643 33.3952 53.5532 31.6302 55.1339C29.8915 56.6882 27.6127 57.4654 24.7938 57.4654Z" fill="black"/> +</svg> diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/images/logo.slc.svg b/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/images/logo.slc.svg new file mode 100644 index 0000000000..3670945fa2 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/images/logo.slc.svg @@ -0,0 +1,26 @@ +<svg width="112" height="81" viewBox="0 0 112 81" fill="none" xmlns="http://www.w3.org/2000/svg"> +<g clip-path="url(#clip0_722_5)"> +<path d="M82.3339 26.6294V1.64714H87.5356V26.6294H82.3339Z" fill="white"/> +<path d="M69.0968 26.9716C65.6404 26.9716 62.7999 24.9867 62.7999 21.3591V21.2907C62.7999 17.2867 65.8457 15.4387 70.1919 15.4387C72.0399 15.4387 73.3746 15.7467 74.675 16.1916V15.8836C74.675 13.7276 73.3404 12.5298 70.7395 12.5298C68.7546 12.5298 67.3515 12.9063 65.6746 13.5223L64.3742 9.55247C66.3933 8.6627 68.3781 8.08092 71.4924 8.08092C74.3328 8.08092 76.3862 8.83381 77.6866 10.1343C79.0555 11.5031 79.6715 13.5223 79.6715 15.9863V26.6294H74.6408V24.6445C73.3746 26.0476 71.6293 26.9716 69.0968 26.9716ZM70.671 23.3783C73.1008 23.3783 74.7435 22.0436 74.7435 20.1614V19.2374C73.8537 18.8267 72.6902 18.5529 71.4239 18.5529C69.1995 18.5529 67.8306 19.4427 67.8306 21.0854V21.1538C67.8306 22.5569 68.9941 23.3783 70.671 23.3783Z" fill="white"/> +<path d="M55.8611 26.6294V8.28625H61.0629V26.6294H55.8611Z" fill="white"/> +<path d="M46.7823 27.04C41.2726 27.04 37.2343 22.7965 37.2343 17.5605V17.492C37.2343 12.256 41.2383 7.94403 46.8508 7.94403C50.3072 7.94403 52.4632 9.10759 54.1743 11.024L50.9917 14.4463C49.8281 13.2143 48.6646 12.4271 46.8166 12.4271C44.2157 12.4271 42.3677 14.72 42.3677 17.4236V17.492C42.3677 20.2983 44.1814 22.5569 47.0219 22.5569C48.7672 22.5569 49.965 21.804 51.2312 20.6063L54.277 23.6863C52.4974 25.6369 50.4441 27.04 46.7823 27.04Z" fill="white"/> +<path d="M26.4522 27.04C20.7713 27.04 16.562 22.8307 16.562 17.5605V17.492C16.562 12.2218 20.8055 7.94403 26.5206 7.94403C32.2015 7.94403 36.4108 12.1534 36.4108 17.4236V17.492C36.4108 22.7623 32.1673 27.04 26.4522 27.04ZM26.5206 22.5569C29.4979 22.5569 31.2775 20.264 31.2775 17.5605V17.492C31.2775 14.7885 29.3268 12.4271 26.4522 12.4271C23.4748 12.4271 21.6953 14.72 21.6953 17.4236V17.492C21.6953 20.1956 23.646 22.5569 26.5206 22.5569Z" fill="white"/> +<path d="M8.7609 26.9716C6.19423 26.9716 3.31956 26.116 0.889786 24.1996L3.11423 20.7774C5.09912 22.2147 7.18667 22.9676 8.89779 22.9676C10.4036 22.9676 11.088 22.42 11.088 21.5987V21.5302C11.088 20.4009 9.30845 20.0245 7.28934 19.4085C4.72267 18.6556 1.81379 17.4578 1.81379 13.8987V13.8302C1.81379 10.1 4.82534 8.01247 8.52134 8.01247C10.8485 8.01247 13.3809 8.79958 15.3658 10.1342L13.3809 13.7276C11.5671 12.6667 9.75334 12.0165 8.41867 12.0165C7.15245 12.0165 6.50223 12.564 6.50223 13.2827V13.3511C6.50223 14.3778 8.24756 14.8569 10.2325 15.5414C12.7991 16.3969 15.7765 17.6289 15.7765 20.9827V21.0511C15.7765 25.1236 12.7307 26.9716 8.7609 26.9716Z" fill="white"/> +<path d="M47.7344 54.6294V36.2863H52.9362V38.8871C54.134 37.3471 55.674 35.944 58.3091 35.944C62.2446 35.944 64.5375 38.5449 64.5375 42.7543V54.6294H59.3357V44.3969C59.3357 41.9329 58.1722 40.6667 56.1873 40.6667C54.2024 40.6667 52.9362 41.9329 52.9362 44.3969V54.6294H47.7344Z" fill="white"/> +<path d="M34.7369 54.9716C31.2804 54.9716 28.44 52.9867 28.44 49.3592V49.2907C28.44 45.2867 31.4858 43.4387 35.832 43.4387C37.68 43.4387 39.0147 43.7467 40.3151 44.1916V43.8836C40.3151 41.7276 38.9804 40.5298 36.3796 40.5298C34.3947 40.5298 32.9916 40.9063 31.3147 41.5223L30.0142 37.5525C32.0333 36.6627 34.0182 36.0809 37.1324 36.0809C39.9729 36.0809 42.0262 36.8338 43.3267 38.1343C44.6956 39.5032 45.3116 41.5223 45.3116 43.9863V54.6294H40.2809V52.6445C39.0147 54.0476 37.2693 54.9716 34.7369 54.9716ZM36.3111 51.3783C38.7409 51.3783 40.3836 50.0436 40.3836 48.1614V47.2374C39.4938 46.8267 38.3302 46.5529 37.064 46.5529C34.8396 46.5529 33.4707 47.4427 33.4707 49.0854V49.1538C33.4707 50.5569 34.6342 51.3783 36.3111 51.3783Z" fill="white"/> +<path d="M19.2931 55.04C13.7833 55.04 9.71083 51.1729 9.71083 45.5605V45.492C9.71083 40.256 13.4411 35.944 18.7797 35.944C24.9055 35.944 27.7117 40.7009 27.7117 45.9027C27.7117 46.3134 27.6775 46.7925 27.6433 47.2716H14.8784C15.3917 49.6329 17.0344 50.8649 19.3615 50.8649C21.1068 50.8649 22.3731 50.3174 23.8104 48.9827L26.7877 51.6178C25.0766 53.7396 22.6126 55.04 19.2931 55.04ZM14.8099 44.0205H22.6468C22.3388 41.6934 20.9699 40.1191 18.7797 40.1191C16.6237 40.1191 15.2206 41.6591 14.8099 44.0205Z" fill="white"/> +<path d="M2.56667 54.6294V29.6472H7.76844V54.6294H2.56667Z" fill="white"/> +<path d="M98.8549 78.2614C96.2882 78.2614 93.4135 77.4059 90.9838 75.4894L93.2082 72.0672C95.1931 73.5045 97.2807 74.2574 98.9918 74.2574C100.498 74.2574 101.182 73.7099 101.182 72.8885V72.8201C101.182 71.6907 99.4024 71.3143 97.3833 70.6983C94.8167 69.9454 91.9078 68.7476 91.9078 65.1885V65.1201C91.9078 61.3899 94.9193 59.3023 98.6153 59.3023C100.942 59.3023 103.475 60.0894 105.46 61.4241L103.475 65.0174C101.661 63.9565 99.8473 63.3063 98.5127 63.3063C97.2464 63.3063 96.5962 63.8539 96.5962 64.5725V64.641C96.5962 65.6676 98.3415 66.1467 100.326 66.8312C102.893 67.6867 105.87 68.9187 105.87 72.2725V72.341C105.87 76.4134 102.825 78.2614 98.8549 78.2614Z" fill="white"/> +<path d="M79.4236 78.2614C75.9672 78.2614 73.1267 76.2765 73.1267 72.649V72.5805C73.1267 68.5765 76.1725 66.7285 80.5187 66.7285C82.3667 66.7285 83.7014 67.0365 85.0018 67.4814V67.1734C85.0018 65.0174 83.6672 63.8196 81.0663 63.8196C79.0814 63.8196 77.6783 64.1961 76.0014 64.8121L74.7009 60.8423C76.72 59.9525 78.7049 59.3707 81.8192 59.3707C84.6596 59.3707 86.7129 60.1236 88.0134 61.4241C89.3823 62.793 89.9983 64.8121 89.9983 67.2761V77.9192H84.9676V75.9343C83.7014 77.3374 81.956 78.2614 79.4236 78.2614ZM80.9978 74.6681C83.4276 74.6681 85.0703 73.3334 85.0703 71.4512V70.5272C84.1805 70.1165 83.0169 69.8427 81.7507 69.8427C79.5263 69.8427 78.1574 70.7325 78.1574 72.3752V72.4436C78.1574 73.8467 79.3209 74.6681 80.9978 74.6681Z" fill="white"/> +<path d="M62.2024 78.0561L54.9815 59.5761H60.4913L64.5979 71.8619L68.7388 59.5761H74.1459L66.925 78.0561H62.2024Z" fill="white"/> +<path d="M38.0238 77.9192V59.5761H43.2256V62.177C44.4234 60.637 45.9634 59.2339 48.5985 59.2339C52.5341 59.2339 54.8269 61.8347 54.8269 66.0441V77.9192H49.6252V67.6867C49.6252 65.2227 48.4616 63.9565 46.4767 63.9565C44.4918 63.9565 43.2256 65.2227 43.2256 67.6867V77.9192H38.0238Z" fill="white"/> +<path d="M25.0263 78.2614C21.5699 78.2614 18.7294 76.2765 18.7294 72.649V72.5805C18.7294 68.5765 21.7752 66.7285 26.1214 66.7285C27.9694 66.7285 29.3041 67.0365 30.6046 67.4814V67.1734C30.6046 65.0174 29.2699 63.8196 26.669 63.8196C24.6841 63.8196 23.281 64.1961 21.6041 64.8121L20.3037 60.8423C22.3228 59.9525 24.3077 59.3707 27.4219 59.3707C30.2623 59.3707 32.3157 60.1236 33.6161 61.4241C34.985 62.793 35.601 64.8121 35.601 67.2761V77.9192H30.5703V75.9343C29.3041 77.3374 27.5588 78.2614 25.0263 78.2614ZM26.6006 74.6681C29.0303 74.6681 30.673 73.3334 30.673 71.4512V70.5272C29.7832 70.1165 28.6197 69.8427 27.3534 69.8427C25.129 69.8427 23.7601 70.7325 23.7601 72.3752V72.4436C23.7601 73.8467 24.9237 74.6681 26.6006 74.6681Z" fill="white"/> +<path d="M10.9853 78.3299C5.47555 78.3299 1.43733 74.0863 1.43733 68.8503V68.7819C1.43733 63.5459 5.44133 59.2339 11.0538 59.2339C14.5102 59.2339 16.6662 60.3974 18.3773 62.3139L15.1947 65.7361C14.0311 64.5041 12.8676 63.717 11.0196 63.717C8.41866 63.717 6.57066 66.0099 6.57066 68.7134V68.7819C6.57066 71.5881 8.38444 73.8467 11.2249 73.8467C12.9702 73.8467 14.168 73.0939 15.4342 71.8961L18.48 74.9761C16.7004 76.9267 14.6471 78.3299 10.9853 78.3299Z" fill="white"/> +<ellipse cx="58.7434" cy="3.13287" rx="3.135" ry="3.13287" fill="white"/> +</g> +<defs> +<clipPath id="clip0_722_5"> +<rect width="112" height="80.6713" fill="white"/> +</clipPath> +</defs> +</svg> diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/mod.ts b/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/mod.ts new file mode 100644 index 0000000000..9262c188b0 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/mod.ts @@ -0,0 +1,4 @@ +/** + * @module + */ +export { LogoWordmark } from './ui.tsx'; diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/t.ts b/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/t.ts new file mode 100644 index 0000000000..8a9b1a415a --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/t.ts @@ -0,0 +1,13 @@ +import type { t } from './common.ts'; + +/** + * <Component>: + */ +export type LogoWordmarkProps = { + logo?: t.LogoKind; + theme?: t.CommonTheme; + style?: t.CssInput; +}; + +/** List of supported logos. */ +export type LogoKind = 'SLC' | 'CC'; diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/ui.tsx b/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/ui.tsx new file mode 100644 index 0000000000..e51a3a3ceb --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/ui.tsx @@ -0,0 +1,46 @@ +import React from 'react'; +import { type t, css, Svg, D } from './common.ts'; +import { useTheme, setColors } from './use.Theme.ts'; + +const Images = { + slc: { import: () => import('./images/logo.slc.svg'), size: [112, 81] }, + cc: { import: () => import('./images/logo.cc.svg'), size: [345, 84] }, +}; + +export const LogoWordmark: React.FC<t.LogoWordmarkProps> = (props) => { + const { theme } = props; + const kind = props.logo ?? D.logo; + + /** + * Source design, search Figma: "logo.slc" + */ + const src = wrangle.svg(kind); + const svg = Svg.useSvg<HTMLDivElement>(src.import, src.size); + useTheme(svg, theme); + + /** + * Keep SVG colors in sync with the current props. + */ + React.useEffect(() => setColors(svg, theme), [kind, svg, theme]); + + /** + * Render: + */ + const styles = { base: css({}) }; + return ( + <div className={css(styles.base, props.style).class}> + <div ref={svg.ref} /> + </div> + ); +}; + +/** + * Helpers: + */ +const wrangle = { + svg(kind: 'SLC' | 'CC') { + if (kind === 'SLC') return Images.slc; + if (kind === 'CC') return Images.cc; + throw new Error(`Not supported: "${kind}"`); + }, +} as const; diff --git a/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/use.Theme.ts b/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/use.Theme.ts new file mode 100644 index 0000000000..e26e2c6579 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Logo.Wordmark/use.Theme.ts @@ -0,0 +1,18 @@ +import { useState, useEffect } from 'react'; +import { type t, Theme } from './common.ts'; + +/** + * Handle updating the canvas theme. + */ +export function useTheme(svg: t.SvgInstance<HTMLDivElement>, theme?: t.CommonTheme) { + useEffect(() => setColors(svg, theme), [svg.draw, theme]); +} + +export function setColors(svg: t.SvgInstance<HTMLDivElement>, theme?: t.CommonTheme) { + const setColor = (color: string, selector: string) => { + svg.queryAll(selector).forEach((el) => el.fill(color)); + }; + const color = Theme.color(theme); + setColor(color, 'path'); + setColor(color, 'ellipse'); +} diff --git a/deploy/@tdb.slc/src/ui/ui.Sheet/common.ts b/deploy/@tdb.slc/src/ui/ui.Sheet/common.ts new file mode 100644 index 0000000000..b6fd03f641 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Sheet/common.ts @@ -0,0 +1,2 @@ +export { Layout } from '../App.Layout/mod.ts'; +export * from '../common.ts'; diff --git a/deploy/@tdb.slc/src/ui/ui.Sheet/mod.ts b/deploy/@tdb.slc/src/ui/ui.Sheet/mod.ts new file mode 100644 index 0000000000..8a882f83e7 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Sheet/mod.ts @@ -0,0 +1 @@ +export { Sheet } from './ui.tsx'; diff --git a/deploy/@tdb.slc/src/ui/ui.Sheet/t.ts b/deploy/@tdb.slc/src/ui/ui.Sheet/t.ts new file mode 100644 index 0000000000..f11b679830 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Sheet/t.ts @@ -0,0 +1,8 @@ +import type { SheetProps as BaseSheetProps } from '@sys/ui-react-components/t'; +import type { t } from './common.ts'; + +/** + * <Sheet> component props. + */ +export type SheetProps = t.ContentProps & BaseProps & { orientation?: t.SheetOrientationY }; +type BaseProps = Pick<BaseSheetProps, 'children' | 'edgeMargin'>; diff --git a/deploy/@tdb.slc/src/ui/ui.Sheet/ui.tsx b/deploy/@tdb.slc/src/ui/ui.Sheet/ui.tsx new file mode 100644 index 0000000000..536f4db6b8 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Sheet/ui.tsx @@ -0,0 +1,29 @@ +import React from 'react'; +import { type t, Color, css, SheetBase } from './common.ts'; + +/** + * Component: + */ +export const Sheet: React.FC<t.SheetProps> = (props) => { + const { state, is, index, orientation } = props; + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({}), + body: css({ display: 'grid', pointerEvents: 'auto' }), + }; + + return ( + <SheetBase.View + style={css(styles.base, props.style)} + theme={theme.name} + edgeMargin={props.edgeMargin} + orientation={props.orientation} + > + <div className={styles.body.class}>{props.children}</div> + </SheetBase.View> + ); +}; diff --git a/deploy/@tdb.slc/src/ui/ui.TooSmall/-SPEC.Debug.tsx b/deploy/@tdb.slc/src/ui/ui.TooSmall/-SPEC.Debug.tsx new file mode 100644 index 0000000000..370082ca32 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.TooSmall/-SPEC.Debug.tsx @@ -0,0 +1,85 @@ +import React from 'react'; +import { type t, Button, css, Signal, Str } from './common.ts'; + +type P = t.TooSmallProps; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + const s = Signal.create; + const props = { + theme: s<P['theme']>('Light'), + children: s<P['children']>(), + }; + const api = { + props, + listen() { + const p = props; + p.theme.value; + p.children.value; + }, + }; + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const p = debug.props; + + Signal.useRedrawEffect(() => debug.listen()); + + /** + * Render: + */ + const styles = { + base: css({}), + title: css({ fontWeight: 'bold', marginBottom: 10 }), + cols: css({ display: 'grid', gridTemplateColumns: 'auto 1fr auto' }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={css(styles.title, styles.cols).class}> + <div>{'TooSmall'}</div> + </div> + <Button + block + label={() => `theme: ${p.theme}`} + onClick={() => Signal.cycle<P['theme']>(p.theme, ['Light', 'Dark'])} + /> + <Button + block + label={() => { + const value = p.children.value; + const fmt = value ? Str.truncate(String(value), 40) : '<undefined>'; + return `children: ${fmt}`; + }} + onClick={() => { + const multiline = ` + Please make your window bigger, or + move over to your mobile device. + `; + + Signal.cycle<P['children']>(p.children, [ + undefined, + 'š Hello', + multiline, + Str.Lorem.text, + ]); + }} + /> + <hr /> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui/ui.TooSmall/-SPEC.tsx b/deploy/@tdb.slc/src/ui/ui.TooSmall/-SPEC.tsx new file mode 100644 index 0000000000..4db0001b84 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.TooSmall/-SPEC.tsx @@ -0,0 +1,28 @@ +import { Dev, Signal, Spec } from '../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { TooSmall } from './mod.ts'; + +export default Spec.describe('TooSmall', (e) => { + const debug = createDebugSignals(); + const p = debug.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + Dev.Theme.signalEffect(ctx, p.theme, 1); + Signal.effect(() => { + debug.listen(); + ctx.redraw(); + }); + + ctx.subject + .size([390, null]) + .display('grid') + .render((e) => <TooSmall theme={p.theme.value}>{p.children.value}</TooSmall>); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/deploy/@tdb.slc/src/ui/ui.TooSmall/common.ts b/deploy/@tdb.slc/src/ui/ui.TooSmall/common.ts new file mode 100644 index 0000000000..a45006948c --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.TooSmall/common.ts @@ -0,0 +1,7 @@ +export * from '../common.ts'; + +/** + * Constants: + */ +export const DEFAULTS = {} as const; +export const D = DEFAULTS; diff --git a/deploy/@tdb.slc/src/ui/ui.TooSmall/mod.ts b/deploy/@tdb.slc/src/ui/ui.TooSmall/mod.ts new file mode 100644 index 0000000000..6b25093f95 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.TooSmall/mod.ts @@ -0,0 +1,4 @@ +/** + * @module + */ +export { TooSmall } from './ui.tsx'; diff --git a/deploy/@tdb.slc/src/ui/ui.TooSmall/t.ts b/deploy/@tdb.slc/src/ui/ui.TooSmall/t.ts new file mode 100644 index 0000000000..9161f2588e --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.TooSmall/t.ts @@ -0,0 +1,10 @@ +import type { t } from './common.ts'; + +/** + * <Component>: + */ +export type TooSmallProps = { + children?: t.ReactNode; + theme?: t.CommonTheme; + style?: t.CssInput; +}; diff --git a/deploy/@tdb.slc/src/ui/ui.TooSmall/ui.tsx b/deploy/@tdb.slc/src/ui/ui.TooSmall/ui.tsx new file mode 100644 index 0000000000..b30a5e5d0b --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.TooSmall/ui.tsx @@ -0,0 +1,55 @@ +import React from 'react'; +import { type t, Color, css, Icons, ReactString } from './common.ts'; + +type P = t.TooSmallProps; + +export const TooSmall: React.FC<P> = (props) => { + const {} = props; + + /** + * Render: + */ + const theme = Color.theme(props.theme); + const styles = { + base: css({ + position: 'relative', + color: theme.fg, + userSelect: 'none', + Padding: [20, 40], + fontSize: 16, + display: 'grid', + placeItems: 'center', + }), + body: css({ display: 'grid', gridTemplateRows: 'auto auto', rowGap: '1em', lineHeight: 1.5 }), + header: css({ display: 'grid', placeItems: 'center' }), + children: css({ textAlign: 'center' }), + }; + + const elHeader = ( + <div className={styles.header.class}> + <Icons.ProjectorScreen size={50} /> + </div> + ); + + const elBody = <div className={styles.children.class}>{wrangle.body(props.children)}</div>; + + return ( + <div className={css(styles.base, props.style).class}> + <div className={styles.body.class}> + {elHeader} + {elBody} + </div> + </div> + ); +}; + +/** + * Helpers: + */ +const wrangle = { + body(children?: t.ReactNode): t.ReactNode { + if (!children) return wrangle.body('Please make your window bigger.'); + if (typeof children === 'string') return ReactString.break(children); + return children; + }, +} as const; diff --git a/deploy/@tdb.slc/src/ui/ui.Video.Background/-SPEC.Debug.tsx b/deploy/@tdb.slc/src/ui/ui.Video.Background/-SPEC.Debug.tsx new file mode 100644 index 0000000000..8560b3a752 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Video.Background/-SPEC.Debug.tsx @@ -0,0 +1,83 @@ +import React from 'react'; +import { App } from '../App/mod.ts'; +import { type t, Button, css, Signal, D } from './common.ts'; + +type P = t.VideoBackgroundProps; + +/** + * Types: + */ +export type DebugProps = { debug: DebugSignals; style?: t.CssInput }; +export type DebugSignals = ReturnType<typeof createDebugSignals>; + +/** + * Signals: + */ +export function createDebugSignals(init?: (e: DebugSignals) => void) { + const s = Signal.create; + const app = App.signals(); + const props = { + theme: s<t.CommonTheme>('Dark'), + }; + const api = { + props, + app, + listen() { + app.listen(); + const p = props; + p.theme.value; + }, + }; + + app.props.background.video.opacity.value = 0.6; + init?.(api); + return api; +} + +/** + * Component: + */ +export const Debug: React.FC<DebugProps> = (props) => { + const { debug } = props; + const app = debug.app; + const p = app.props; + const d = debug.props; + const bg = p.background; + + Signal.useRedrawEffect(() => debug.listen()); + + /** + * Render: + */ + const styles = { + base: css({}), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <Button + block + label={`theme: ${d.theme}`} + onClick={() => Signal.cycle<t.CommonTheme>(d.theme, ['Light', 'Dark'])} + /> + + <hr /> + <Button + block + label={`background.video.playing: ${bg.video.playing.value}`} + onClick={() => { + Signal.toggle(bg.video.playing); + }} + /> + <Button + block + label={`background.video.opacity: ${bg.video.opacity.value ?? '<undefined> (100%)'}`} + onClick={() => { + Signal.cycle<number | undefined>(bg.video.opacity, [0, 0.3, 0.6, undefined]); + }} + /> + + <hr /> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Video.Background/-SPEC.tsx b/deploy/@tdb.slc/src/ui/ui.Video.Background/-SPEC.tsx new file mode 100644 index 0000000000..bdc86057ee --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Video.Background/-SPEC.tsx @@ -0,0 +1,29 @@ +import { Dev, Signal, Spec } from '../-test.ui.ts'; +import { Debug, createDebugSignals } from './-SPEC.Debug.tsx'; +import { VideoBackground } from './mod.ts'; + +export default Spec.describe('VideoBackground', (e) => { + const debug = createDebugSignals(); + const app = debug.app; + const p = app.props; + + e.it('init', (e) => { + const ctx = Spec.ctx(e); + + Dev.Theme.signalEffect(ctx, debug.props.theme, 1); + Signal.effect(() => { + debug.listen(); + ctx.redraw(); + }); + + ctx.subject + .size('fill') + .display('grid') + .render((e) => <VideoBackground state={app} />); + }); + + e.it('ui:debug', (e) => { + const ctx = Spec.ctx(e); + ctx.debug.row(<Debug debug={debug} />); + }); +}); diff --git a/deploy/@tdb.slc/src/ui/ui.Video.Background/common.ts b/deploy/@tdb.slc/src/ui/ui.Video.Background/common.ts new file mode 100644 index 0000000000..d75a28d192 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Video.Background/common.ts @@ -0,0 +1,9 @@ +export * from '../common.ts'; + +/** + * Constants: + */ +export const DEFAULTS = { + playing: true, +} as const; +export const D = DEFAULTS; diff --git a/deploy/@tdb.slc/src/ui/ui.Video.Background/mod.ts b/deploy/@tdb.slc/src/ui/ui.Video.Background/mod.ts new file mode 100644 index 0000000000..0b560fe60c --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Video.Background/mod.ts @@ -0,0 +1,4 @@ +/** + * @module + */ +export { VideoBackground } from './ui.tsx'; diff --git a/deploy/@tdb.slc/src/ui/ui.Video.Background/t.ts b/deploy/@tdb.slc/src/ui/ui.Video.Background/t.ts new file mode 100644 index 0000000000..f6f0b77ee3 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Video.Background/t.ts @@ -0,0 +1,9 @@ +import type { t } from './common.ts'; + +/** + * <Component>: + */ +export type VideoBackgroundProps = { + state: t.AppSignals; + style?: t.CssInput; +}; diff --git a/deploy/@tdb.slc/src/ui/ui.Video.Background/ui.tsx b/deploy/@tdb.slc/src/ui/ui.Video.Background/ui.tsx new file mode 100644 index 0000000000..bc74e21835 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/ui.Video.Background/ui.tsx @@ -0,0 +1,47 @@ +import React from 'react'; +import { type t, css, D, VimeoBackground, Signal } from './common.ts'; + +type P = t.VideoBackgroundProps; + +export const VideoBackground: React.FC<P> = (props) => { + const { state } = props; + + const p = state.props.background.video; + const src = p.src.value; + const opacity = p.opacity.value; + const blur = p.blur.value; + const playing = p.playing.value ?? D.playing; + + const playerRef = React.useRef<t.VimeoIFrame>(); + + /** + * Effect: redraw (watch). + */ + Signal.useRedrawEffect(() => { + p.src.value; + p.opacity.value; + p.blur.value; + p.playing.value; + }); + + /** + * Render: + */ + const styles = { + base: css({ position: 'relative', pointerEvents: 'none' }), + video: css({ Absolute: 0 }), + }; + + return ( + <div className={css(styles.base, props.style).class}> + <VimeoBackground + video={src} + playing={playing} + blur={blur} + opacity={opacity} + onReady={(api) => (playerRef.current = api)} + style={styles.video} + /> + </div> + ); +}; diff --git a/deploy/@tdb.slc/src/ui/use/-.test.ts b/deploy/@tdb.slc/src/ui/use/-.test.ts new file mode 100644 index 0000000000..10fa9d00a8 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/use/-.test.ts @@ -0,0 +1,11 @@ +import { type t, describe, it, expect, c } from '../../-test.ts'; + +describe('Hooks', () => { + it('imports', async () => { + const m = await import('./mod.ts'); + console.info(c.brightGreen('(Hooks) Module:')); + console.info({ ...m }); + + expect(typeof m.useKeyboard == 'function').to.be.true; + }); +}); diff --git a/deploy/@tdb.slc/src/ui/use/common.ts b/deploy/@tdb.slc/src/ui/use/common.ts new file mode 100644 index 0000000000..8cae67176a --- /dev/null +++ b/deploy/@tdb.slc/src/ui/use/common.ts @@ -0,0 +1 @@ +export * from '../common.ts'; diff --git a/deploy/@tdb.slc/src/ui/use/mod.ts b/deploy/@tdb.slc/src/ui/use/mod.ts new file mode 100644 index 0000000000..780afa37e4 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/use/mod.ts @@ -0,0 +1 @@ +export * from './use.Keyboard.ts'; diff --git a/deploy/@tdb.slc/src/ui/use/t.ts b/deploy/@tdb.slc/src/ui/use/t.ts new file mode 100644 index 0000000000..6377ac4805 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/use/t.ts @@ -0,0 +1,6 @@ +import { type t } from './common.ts'; + +/** + * Hook: Keyboard controller. + */ +export type UseKeyboardFactory = (state?: t.AppSignals) => void; diff --git a/deploy/@tdb.slc/src/ui/use/use.Keyboard.ts b/deploy/@tdb.slc/src/ui/use/use.Keyboard.ts new file mode 100644 index 0000000000..a713f10bd4 --- /dev/null +++ b/deploy/@tdb.slc/src/ui/use/use.Keyboard.ts @@ -0,0 +1,26 @@ +import { useKeyboard as useDevKeyboard } from '@sys/ui-react-devharness'; +import { useEffect } from 'react'; +import { type t, Keyboard } from './common.ts'; + +/** + * Hook: Keyboard controller. + */ +export const useKeyboard: t.UseKeyboardFactory = (state) => { + useDevKeyboard(); + + useEffect(() => { + const keyboard = Keyboard.until(); + + /** + * š· START/STOP player + */ + keyboard.on('Space', () => { + /** + * TODO š· + */ + console.log('š· START/STOP player'); + }); + + return keyboard.dispose; + }, [state]); +}; diff --git a/deploy/@tdb.slc/vite.config.ts b/deploy/@tdb.slc/vite.config.ts new file mode 100644 index 0000000000..dd19226f00 --- /dev/null +++ b/deploy/@tdb.slc/vite.config.ts @@ -0,0 +1,16 @@ +import { Vite } from 'jsr:@sys/driver-vite'; +import { defineConfig } from 'npm:vite'; + +export default defineConfig(() => { + const entry = './src/-test/index.html'; + const paths = Vite.Config.paths({ app: { entry } }); + return Vite.Config.app({ + paths, + chunks(e) { + e.chunk('react', 'react'); + e.chunk('react.dom', 'react-dom'); + e.chunk('sys', ['@sys/std']); + e.chunk('css', ['@sys/ui-css']); + }, + }); +}); diff --git a/deploy/api.db.team/deno.json b/deploy/api.db.team/deno.json index aba5e127ba..d7d1d05d3c 100644 --- a/deploy/api.db.team/deno.json +++ b/deploy/api.db.team/deno.json @@ -1,6 +1,6 @@ { "name": "@tdb/api", - "version": "0.0.72", + "version": "0.0.82", "tasks": { "dev": "deno run -RNE --watch ./main.ts", "check": "deno check ./main.ts", diff --git a/deploy/slc.db.team/-design/NOTES.md b/deploy/slc.db.team/-design/NOTES.md deleted file mode 100644 index b354f6e3b8..0000000000 --- a/deploy/slc.db.team/-design/NOTES.md +++ /dev/null @@ -1,9 +0,0 @@ -```yaml notes -- Sample JSON/Timestamps for Video Player - - - -- Sample (historical) - https://socialleancanvas.com/ember-slc/ - -``` diff --git a/deploy/slc.db.team/-design/public/data/index.json b/deploy/slc.db.team/-design/public/data/index.json deleted file mode 100644 index aa89ae9fed..0000000000 --- a/deploy/slc.db.team/-design/public/data/index.json +++ /dev/null @@ -1,27 +0,0 @@ -{ - "kind": "dir", - "dir": { - "indexedAt": 1695690211367 - }, - "hash": { - "files": "sha256-5944ac2fa7a41caa297b32ef5b8bdec623f547b5ef1a976c65f536ef8ae16282", - "dir": "sha256-84280c1be94bef2b763966ec11d78df6232a390115fc6ca0b70ec7377906db77" - }, - "files": [ - { - "path": "index.json", - "bytes": 578, - "filehash": "sha256-38056e83a896e48bd2cd5cdd56d31cc14195f6781c476eabe96643f0bb6b0d19" - }, - { - "path": "log.json", - "bytes": 63, - "filehash": "sha256-7cbb17b6f2216f43d33eac4c3ee0979f8fabe48153ec93250201e65b6a9d6356" - }, - { - "path": "md/README.md", - "bytes": 57, - "filehash": "sha256-7a48826da2af737a86eada318b8bdf2e0e1db02e1592b44295068c25c4625d7f" - } - ] -} diff --git a/deploy/slc.db.team/-design/public/data/log.json b/deploy/slc.db.team/-design/public/data/log.json deleted file mode 100644 index 814122db7b..0000000000 --- a/deploy/slc.db.team/-design/public/data/log.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "latest": { - "version": "0.0.64" - }, - "history": [] -} diff --git a/deploy/slc.db.team/-design/public/data/md/README.md b/deploy/slc.db.team/-design/public/data/md/README.md deleted file mode 100644 index 6cb6fda7b5..0000000000 --- a/deploy/slc.db.team/-design/public/data/md/README.md +++ /dev/null @@ -1,5 +0,0 @@ -# Hello - -```yaml project.props -version: 0.0.0-draft.0 -``` \ No newline at end of file diff --git a/deploy/slc.db.team/-design/public/json/ember-pitch.json b/deploy/slc.db.team/-design/public/json/ember-pitch.json deleted file mode 100644 index 1110cf5847..0000000000 --- a/deploy/slc.db.team/-design/public/json/ember-pitch.json +++ /dev/null @@ -1,109 +0,0 @@ -{ - "slugs": [ - { - "kind": "slug:namespace", - "namespace": "ember.pitch", - "title": "Pitching" - }, - { - "id": "ember.pitch.intro", - "kind": "slug:VideoDiagram", - "split": 0.5, - "title": "Introduction", - "video": { - "innerScale": 1.1, - "src": { - "id": "847196819", - "kind": "Vimeo" - }, - "timestamps": [ - { - "start": 0, - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258939645-3d0a92fc-a173-4e2e-a7a9-b9608d8c0d27.png" - } - ] - } - }, - { - "id": "ember.pitch.structure", - "kind": "slug:VideoDiagram", - "split": 0.5, - "title": "Simple Structure", - "video": { - "innerScale": 1.1, - "src": { - "id": "846848747", - "kind": "Vimeo" - }, - "timestamps": [ - { - "start": 0, - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258939657-514453bd-a1fa-4463-bdc3-8ba853393400.png" - } - ] - } - }, - { - "id": "ember.pitch.example", - "kind": "slug:VideoDiagram", - "split": 0.5, - "title": "An Example", - "video": { - "innerScale": 1.1, - "src": { - "id": "846848687", - "kind": "Vimeo" - }, - "timestamps": [ - { - "start": 0, - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258939648-6cecd1ac-9e59-4512-8767-37cbe0e9ea86.png" - } - ] - } - }, - { - "id": "ember.pitch.succes", - "kind": "slug:VideoDiagram", - "split": 0.653, - "title": "Making it Memorable", - "video": { - "innerScale": 1.1, - "src": { - "id": "847199651", - "kind": "Vimeo" - }, - "timestamps": [ - { - "start": 0, - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258939654-212918dc-b58a-4b9e-b448-ed221b4142f7.png" - } - ] - } - }, - { - "id": "ember.pitch.conclusion", - "kind": "slug:VideoDiagram", - "split": 0.5, - "title": "Tie it all Together", - "video": { - "innerScale": 1.1, - "src": { - "id": "847193255", - "kind": "Vimeo" - }, - "timestamps": [ - { - "start": 0, - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258939662-4305ff3c-a432-4193-a1ba-fb1f3537af03.png" - } - ] - } - } - ] -} diff --git a/deploy/slc.db.team/-design/public/json/ember-slc.json b/deploy/slc.db.team/-design/public/json/ember-slc.json deleted file mode 100644 index 3a4a3d7430..0000000000 --- a/deploy/slc.db.team/-design/public/json/ember-slc.json +++ /dev/null @@ -1,2020 +0,0 @@ -{ - "slugs": [ - { - "kind": "slug:namespace", - "namespace": "intro", - "title": "Introduction" - }, - { - "id": "intro", - "kind": "slug:VideoDiagram", - "split": 0.653, - "title": "SLC Innovation Programme", - "video": { - "innerScale": 1.1, - "src": { - "id": "851209192", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258671853-81ef82e9-c2c0-4d59-a0e4-806c3a889cca.png", - "start": 0 - }, - { - "src": "https://user-images.githubusercontent.com/185555/258659458-ca448cab-383c-4170-80ba-d55e3c83a312.png", - "start": 64 - } - ] - } - }, - { - "id": "intro.programme", - "kind": "slug:VideoDiagram", - "split": 0.717, - "title": "About This Programme", - "video": { - "innerScale": 1.1, - "src": { - "id": "851211170", - "kind": "Vimeo" - }, - "timestamps": [ - { - "src": "https://user-images.githubusercontent.com/185555/258672044-788b5fda-85af-447c-8651-ce8b1030eda3.png", - "start": 0 - }, - { - "src": "https://user-images.githubusercontent.com/185555/258672166-576b41d7-dfa8-4b3f-9960-7bedbb188d54.png", - "start": 94 - }, - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258673262-d1314e2f-ed07-4ec1-a746-457cfe2ccff8.png", - "start": 116 - } - ] - } - }, - { - "id": "intro.canvas", - "kind": "slug:VideoDiagram", - "split": 0.653, - "title": "How to Use The Canvas", - "video": { - "innerScale": 1.05, - "src": { - "id": "851212258", - "kind": "Vimeo" - }, - "timestamps": [ - { - "src": "https://user-images.githubusercontent.com/185555/258673999-724e767c-560e-46fc-9868-ec9763cbb6b7.png", - "start": 0 - }, - { - "src": "https://user-images.githubusercontent.com/185555/258673947-c35bbfd9-e1a4-4974-bcf1-be970365b69d.png", - "start": 14 - }, - { - "src": "https://user-images.githubusercontent.com/185555/258673948-93447543-e794-42da-95e4-e47d5d1aa4ad.png", - "start": 21 - }, - { - "src": "https://user-images.githubusercontent.com/185555/258673999-724e767c-560e-46fc-9868-ec9763cbb6b7.png", - "start": 45 - }, - { - "src": "https://user-images.githubusercontent.com/185555/258672166-576b41d7-dfa8-4b3f-9960-7bedbb188d54.png", - "start": 122 - } - ] - } - }, - { - "id": "purpose", - "kind": "slug:VideoDiagram", - "split": 0.68, - "title": "Purpose", - "video": { - "innerScale": 1.01, - "src": { - "id": "577945255", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.6, - "src": "https://user-images.githubusercontent.com/185555/258676537-1f72bdfd-2c67-4286-aac8-b489204ba79e.png", - "start": 0 - } - ] - } - }, - { - "kind": "slug:namespace", - "namespace": "customermodel", - "title": "Customer Model" - }, - { - "id": "customermodel.overview", - "kind": "slug:VideoDiagram", - "split": 0.64, - "title": "Customers", - "video": { - "innerScale": 1.02, - "src": { - "id": "577933592", - "kind": "Vimeo" - }, - "timestamps": [ - { - "start": 0, - "scale": 0.6, - "src": "https://user-images.githubusercontent.com/185555/259265555-11f11af7-d993-4e81-bac6-d88fdcd315ad.png" - }, - { - "start": 0, - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/259265623-95af98f0-e9b0-470f-b3df-d67075b79369.png" - } - ] - } - }, - { - "id": "customermodel.customers.1", - "kind": "slug:VideoDiagram", - "split": 0.693, - "title": "Customers (One)", - "video": { - "innerScale": 1.02, - "src": { - "id": "577928799", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258678843-72027c74-17f4-450c-806f-d4aeae158931.png", - "start": 0 - } - ] - } - }, - { - "id": "customermodel.customers.2", - "kind": "slug:VideoDiagram", - "title": "Customers (Two)", - "video": { - "innerScale": 1.02, - "src": { - "id": "577928965", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258678843-72027c74-17f4-450c-806f-d4aeae158931.png", - "start": 0 - } - ] - } - }, - { - "id": "customermodel.customers.earlyadopters", - "kind": "slug:VideoDiagram", - "title": "Early Adopters", - "video": { - "innerScale": 1.02, - "src": { - "id": "577929214", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258678843-72027c74-17f4-450c-806f-d4aeae158931.png", - "start": 0 - } - ] - } - }, - { - "id": "customermodel.jobs", - "kind": "slug:VideoDiagram", - "title": "Jobs to be Done", - "video": { - "innerScale": 1.02, - "src": { - "id": "577933180", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258679138-f1f61d51-9803-4928-bbe5-790153e578eb.png", - "start": 0 - } - ] - } - }, - { - "id": "customermodel.jobs.existingalternatives", - "kind": "slug:VideoDiagram", - "title": "Existing Alternatives ", - "video": { - "innerScale": 1.02, - "src": { - "id": "577933435", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258679138-f1f61d51-9803-4928-bbe5-790153e578eb.png", - "start": 0 - } - ] - } - }, - { - "id": "customermodel.uvp", - "kind": "slug:VideoDiagram", - "title": "Unique Value Proposition", - "video": { - "innerScale": 1.02, - "src": { - "id": "577937937", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258679207-83301cb2-c07b-44a5-9178-0485f3679b8a.png", - "start": 0 - } - ] - } - }, - { - "id": "customermodel.solution", - "kind": "slug:VideoDiagram", - "title": "Solution", - "video": { - "innerScale": 1.02, - "src": { - "id": "577937777", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258679288-5920e11b-331c-4158-87d4-b8bb8af5516b.png", - "start": 0 - } - ] - } - }, - { - "id": "customermodel.example.FB.intro", - "kind": "slug:VideoDiagram", - "split": 0.677, - "title": "Example", - "video": { - "innerScale": 1.01, - "src": { - "id": "577929636", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258687717-9d34a221-1610-4a55-9157-a5a943fcfc9a.png", - "start": 0 - } - ] - } - }, - { - "id": "customermodel.example.FB.customers", - "kind": "slug:VideoDiagram", - "split": 0.683, - "title": "Example: Customers", - "video": { - "innerScale": 1.03, - "src": { - "id": "577929426", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258688907-9c1fd4ba-4727-41b0-9ce7-5407debed3c0.png", - "start": 0 - }, - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258688921-347fbf6d-ad69-4157-855f-7457b38f8cc1.png", - "start": 8 - }, - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258688923-88d5ac08-5788-4734-8e46-78bf5b931f13.png", - "start": 35 - } - ] - } - }, - { - "id": "customermodel.example.FB.jobs", - "kind": "slug:VideoDiagram", - "split": 0.687, - "title": "Example: Jobs to be Done ", - "video": { - "innerScale": 1.03, - "src": { - "id": "577929737", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258690425-71eb6113-1541-48ae-b094-1f262fac6ac7.png", - "start": 0 - }, - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258690432-e19aabbb-d9e1-4260-8cd0-0e483b52a2e4.png", - "start": 5 - } - ] - } - }, - { - "id": "customermodel.example.FB.UVP", - "kind": "slug:VideoDiagram", - "split": 0.71, - "title": "Example: UVP", - "video": { - "innerScale": 1.03, - "src": { - "id": "577933078", - "kind": "Vimeo" - }, - "timestamps": [ - { - "start": 0, - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258691202-19b6d32a-f225-4802-8b2c-2665c929fe46.png" - }, - { - "start": 5, - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258691209-4d84be14-65c9-4d7e-a49f-eb81657f2e45.png" - } - ] - } - }, - { - "id": "customermodel.example.FB.solution", - "kind": "slug:VideoDiagram", - "split": 0.713, - "title": "Example: Solution", - "video": { - "innerScale": 1.03, - "src": { - "id": "577932985", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258691728-1e92ad06-cfd1-4eb1-b9cb-6d740485a09a.png", - "start": 0 - }, - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258691726-7e690b29-77d4-47f1-bc9f-65cef6f4860b.png", - "start": 17 - } - ] - } - }, - { - "id": "customermodel.example.FB.conclusion", - "kind": "slug:VideoDiagram", - "split": 0.707, - "title": "Example: Conclusion", - "video": { - "innerScale": 1.02, - "src": { - "id": "577929355", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/258693837-92bb4005-ff1d-4654-85f3-b51feedebdeb.png", - "start": 0 - } - ] - } - }, - { - "kind": "slug:namespace", - "namespace": "impactmodel", - "title": "Impact Model" - }, - { - "id": "impactmodel.overview", - "kind": "slug:VideoDiagram", - "split": 0.687, - "title": "Impact Model", - "video": { - "innerScale": 1.02, - "src": { - "id": "577955565", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.7, - "src": "https://user-images.githubusercontent.com/185555/258683136-72156ccf-9cf7-404c-9251-9e6ca40cb646.png", - "start": 0 - }, - { - "scale": 0.7, - "src": "https://user-images.githubusercontent.com/185555/258683177-16ad8903-e263-4f5b-b7d2-7604fcadbe70.png", - "start": 15 - } - ] - } - }, - { - "id": "impactmodel.overview.context", - "kind": "slug:VideoDiagram", - "title": "Context", - "video": { - "innerScale": 1.02, - "src": { - "id": "577955697", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.7, - "src": "https://user-images.githubusercontent.com/185555/258683323-e5282930-acbd-44f5-ae4c-b787da4a4202.png", - "start": 0 - } - ] - } - }, - { - "id": "impactmodel.overview.disclaimer", - "kind": "slug:VideoDiagram", - "title": "Just One Approach ", - "video": { - "innerScale": 1.02, - "src": { - "id": "577955864", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.7, - "src": "https://user-images.githubusercontent.com/185555/258683323-e5282930-acbd-44f5-ae4c-b787da4a4202.png", - "start": 0 - } - ] - } - }, - { - "id": "impactmodel.issue", - "kind": "slug:VideoDiagram", - "split": 0.723, - "title": "Issue", - "video": { - "innerScale": 1.02, - "src": { - "id": "577955305", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.7, - "src": "https://user-images.githubusercontent.com/185555/258683421-76d6631a-78a0-47b1-8743-f6d19382a2ce.png", - "start": 0 - } - ] - } - }, - { - "id": "impactmodel.participants", - "kind": "slug:VideoDiagram", - "split": 0.733, - "title": "Participants", - "video": { - "innerScale": 1.02, - "src": { - "id": "577955976", - "kind": "Vimeo" - }, - "timestamps": [ - { - "start": 0, - "scale": 0.7, - "src": "https://user-images.githubusercontent.com/185555/259269324-ef25251c-b135-41f9-9fe9-520c9383ae30.png" - } - ] - } - }, - { - "id": "impactmodel.activities", - "kind": "slug:VideoDiagram", - "split": 0.69, - "title": "Activities", - "video": { - "innerScale": 1.02, - "src": { - "id": "577943744", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.7, - "src": "https://user-images.githubusercontent.com/185555/258683501-35f87ff3-f808-4c3b-9ce2-46b24f061ce1.png", - "start": 0 - } - ] - } - }, - { - "id": "impactmodel.shorttermoutcomes", - "kind": "slug:VideoDiagram", - "split": 0.74, - "title": "Short Term Outcomes", - "video": { - "innerScale": 1.02, - "src": { - "id": "577956096", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.7, - "src": "https://user-images.githubusercontent.com/185555/258683528-dc5670e3-d1c9-45dc-ad2d-845e7dcd51e9.png", - "start": 0 - } - ] - } - }, - { - "id": "impactmodel.mediumtermoutcomes", - "kind": "slug:VideoDiagram", - "split": 0.71, - "title": "Medium Term Outcomes", - "video": { - "innerScale": 1.02, - "src": { - "id": "577955482", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.7, - "src": "https://user-images.githubusercontent.com/185555/258683755-32d9afc4-6f62-483f-b0ef-41dd1f838d84.png", - "start": 0 - } - ] - } - }, - { - "id": "impactmodel.longtermoutcomes", - "kind": "slug:VideoDiagram", - "split": 0.733, - "title": "Long Term Outcomes", - "video": { - "innerScale": 1.02, - "src": { - "id": "577955408", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.7, - "src": "https://user-images.githubusercontent.com/185555/258683790-74900d97-7d28-4729-817b-686ef9932db3.png", - "start": 0 - } - ] - } - }, - { - "id": "impactmodel.impact", - "kind": "slug:VideoDiagram", - "title": "Impact ", - "video": { - "innerScale": 1.02, - "src": { - "id": "577955192", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.7, - "src": "https://user-images.githubusercontent.com/185555/258683884-a2649ae6-1205-425c-b082-b3e0d4d9335d.png", - "start": 0 - } - ] - } - }, - { - "id": "impactmodel.example.intro", - "kind": "slug:VideoDiagram", - "title": "Example", - "video": { - "innerScale": 1.02, - "src": { - "id": "577944077", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258694530-1ad84605-ef8a-4d46-b2a9-8e96b1f4c9b1.png", - "start": 0 - } - ] - } - }, - { - "id": "impactmodel.example.issue", - "kind": "slug:VideoDiagram", - "split": 0.733, - "title": "Example: Issue", - "video": { - "innerScale": 1.02, - "src": { - "id": "577944185", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258694807-54af1d7e-1198-40e4-a524-5a4d051351c0.png", - "start": 0 - } - ] - } - }, - { - "id": "impactmodel.example.participants", - "kind": "slug:VideoDiagram", - "split": 0.753, - "title": "Example: Participants", - "video": { - "innerScale": 1.02, - "src": { - "id": "577944806", - "kind": "Vimeo" - }, - "timestamps": [ - { - "start": 0, - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/259268593-2651e155-e1c2-484b-aa9b-6783971180f9.png" - } - ] - } - }, - { - "id": "impactmodel.example.activities", - "kind": "slug:VideoDiagram", - "split": 0.763, - "title": "Example: Activities", - "video": { - "innerScale": 1.02, - "src": { - "id": "577943848", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258695530-ed1df6d6-4b79-48bd-b0c7-ff37b4fab1bb.png", - "start": 0 - } - ] - } - }, - { - "id": "impactmodel.example.shorttermoutcomesI", - "kind": "slug:VideoDiagram", - "split": 0.75, - "title": "Example: Short-term Outcomes", - "video": { - "innerScale": 1.02, - "src": { - "id": "577944940", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258697747-108503dd-d81b-495e-b750-66bf68709120.png", - "start": 0 - } - ] - } - }, - { - "id": "impactmodel.example.mediumtermoutcomes", - "kind": "slug:VideoDiagram", - "split": 0.75, - "title": "Example: Medium-term Outcomes", - "video": { - "innerScale": 1.02, - "src": { - "id": "577944731", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258697741-0eacf674-633d-42e7-943c-713fcfb668fb.png", - "start": 0 - } - ] - } - }, - { - "id": "impactmodel.example.longtermoutcomes", - "kind": "slug:VideoDiagram", - "split": 0.75, - "title": "Example: Long-term Outcomes", - "video": { - "innerScale": 1.02, - "src": { - "id": "577944642", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258697731-77673802-b620-4869-bc95-9c18c769125d.png", - "start": 0 - } - ] - } - }, - { - "id": "impactmodel.example.impact", - "kind": "slug:VideoDiagram", - "split": 0.627, - "title": "Example: Impact", - "video": { - "innerScale": 1.02, - "src": { - "id": "577943950", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258697719-01e41c37-65f6-4db2-8b46-2882a0bf094c.png", - "start": 0 - } - ] - } - }, - { - "kind": "slug:namespace", - "namespace": "metrics", - "title": "Economic Model" - }, - { - "id": "economicmodel.overview", - "kind": "slug:VideoDiagram", - "split": 0.543, - "title": "Economic Model", - "video": { - "innerScale": 1.02, - "src": { - "id": "577954558", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.6, - "src": "https://user-images.githubusercontent.com/185555/258698575-65aa349d-430a-4870-ae58-4a1635848511.png", - "start": 0 - }, - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258698783-37d60998-478d-4709-8c73-a736e22e36b1.png", - "start": 18 - } - ] - } - }, - { - "id": "economicmodel.channels", - "kind": "slug:VideoDiagram", - "split": 0.703, - "title": "Channels", - "video": { - "innerScale": 1.02, - "src": { - "id": "577938360", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258698984-df1da57d-f508-41d8-964a-c8537e0c6a91.png", - "start": 0 - } - ] - } - }, - { - "id": "economicmodel.financialmodel.intro", - "kind": "slug:VideoDiagram", - "split": 0.69, - "title": "Financial Model (Intro)", - "video": { - "innerScale": 1.02, - "src": { - "id": "577954346", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258699030-ead93fa0-83a9-4586-bb18-9b51c13690c5.png", - "start": 0 - } - ] - } - }, - { - "id": "economicmodel.financialmodel.revenue", - "kind": "slug:VideoDiagram", - "split": 0.69, - "title": "Revenue", - "video": { - "innerScale": 1.02, - "src": { - "id": "577954492", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258699083-0fc5bd70-2cbf-4a56-98cc-f8fd0c63d38e.png", - "start": 0 - } - ] - } - }, - { - "id": "economicmodel.financialmodel.costs", - "kind": "slug:VideoDiagram", - "split": 0.687, - "title": "Costs", - "video": { - "innerScale": 1.02, - "src": { - "id": "577954228", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258699124-0fe80d38-db0d-43d3-ab6a-507108e57869.png", - "start": 0 - } - ] - } - }, - { - "id": "economicmodel.advantage.1leverage", - "kind": "slug:VideoDiagram", - "split": 0.683, - "title": "Advantage: Leverage", - "video": { - "innerScale": 1.02, - "src": { - "id": "577938084", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258699180-af0c5f15-b019-4a45-8a02-2960899387db.png", - "start": 0 - } - ] - } - }, - { - "id": "economicmodel.advantage.2innovate", - "kind": "slug:VideoDiagram", - "split": 0.687, - "title": "Advantage: Innovate", - "video": { - "innerScale": 1.02, - "src": { - "id": "577938331", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258699180-af0c5f15-b019-4a45-8a02-2960899387db.png", - "start": 0 - } - ] - } - }, - { - "id": "economicmodel.FB.example.overview", - "kind": "slug:VideoDiagram", - "split": 0.48, - "title": "Example", - "video": { - "innerScale": 1.02, - "src": { - "id": "577938899", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.7, - "src": "https://user-images.githubusercontent.com/185555/258701584-bc064632-7492-4f1e-b333-4d111af63126.png", - "start": 0 - } - ] - } - }, - { - "id": "economicmodel.FB.example.channels", - "kind": "slug:VideoDiagram", - "split": 0.673, - "title": "Example: Channels", - "video": { - "innerScale": 1.02, - "src": { - "id": "577938645", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258701634-3e8b3437-2964-4427-b750-11d1ee4110e3.png", - "start": 0 - }, - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258701742-ad0e2ad1-c5cf-4921-a830-bb97810fbe72.png", - "start": 0 - } - ] - } - }, - { - "id": "economicmodel.FB.example.financialmodel", - "kind": "slug:VideoDiagram", - "split": 0.667, - "title": "Example: Financial Model", - "video": { - "innerScale": 1.02, - "src": { - "id": "577938753", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258702272-b45f3292-fee0-4305-8574-8cd838e7d86a.png", - "start": 0 - }, - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258702294-b3c8b9c2-50b0-4538-9d1e-7301dfe44e4f.png", - "start": 6 - }, - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258702298-2f342a99-2b5b-4b88-8544-aba3234000c6.png", - "start": 25 - } - ] - } - }, - { - "id": "economicmodel.FB.example.advantage", - "kind": "slug:VideoDiagram", - "split": 0.667, - "title": "Example: Advantage", - "video": { - "innerScale": 1.02, - "src": { - "id": "577938493", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258702762-cc2c78dc-8f66-4dd7-9f25-ba503f040a6c.png", - "start": 0 - }, - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258702765-53af8ba0-9955-4299-a5d6-1b73edaeafce.png", - "start": 14 - } - ] - } - }, - { - "kind": "slug:namespace", - "namespace": "metrics", - "title": "Metrics" - }, - { - "id": "metrics.keymetrics", - "kind": "slug:VideoDiagram", - "title": "Key Metrics", - "video": { - "innerScale": 1.02, - "src": { - "id": "577945041", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.6, - "src": "https://user-images.githubusercontent.com/185555/258703729-9d2e7906-3079-4299-8ffe-546caf77f941.png", - "start": 0 - } - ] - } - }, - { - "kind": "slug:namespace", - "namespace": "conclusion", - "title": "Conclusion" - }, - { - "id": "canvas.conclusion", - "kind": "slug:VideoDiagram", - "title": "When you've finished", - "video": { - "innerScale": 1.02, - "src": { - "id": "851213132", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258672044-788b5fda-85af-447c-8651-ce8b1030eda3.png", - "start": 0 - } - ] - } - }, - { - "kind": "slug:namespace", - "namespace": "strategy", - "title": "Strategy" - }, - { - "id": "strategy.intro", - "kind": "slug:VideoDiagram", - "title": "Introduction to Strategy", - "video": { - "innerScale": 1.02, - "src": { - "id": "851213828", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258672166-576b41d7-dfa8-4b3f-9960-7bedbb188d54.png", - "start": 0 - } - ] - } - }, - { - "id": "strategy.intro.2", - "kind": "slug:VideoDiagram", - "split": 0.693, - "title": "Getting Started", - "video": { - "innerScale": 1.02, - "src": { - "id": "598791412", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258672166-576b41d7-dfa8-4b3f-9960-7bedbb188d54.png", - "start": 0 - }, - { - "scale": 0.85, - "src": "https://user-images.githubusercontent.com/185555/258704864-6cf19bfd-6bfe-4148-aac8-e9bc6f327962.png", - "start": 54 - } - ] - } - }, - { - "kind": "slug:namespace", - "namespace": "startupjourney", - "title": "Startup Journey" - }, - { - "id": "startupjourney.intro", - "kind": "slug:VideoDiagram", - "split": 0.41, - "title": "The Startup Journey", - "video": { - "innerScale": 1.02, - "src": { - "id": "598792779", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258706852-98db5f16-c2ba-4e4b-bb3f-eef4a151121d.png", - "start": 0 - } - ] - } - }, - { - "id": "startupjourney.2", - "kind": "slug:VideoDiagram", - "split": 0.657, - "title": "Definition of a Startup", - "video": { - "innerScale": 1.02, - "src": { - "id": "598791681", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258707039-92bfe94b-a63e-4bf4-8eb1-df87be2eb4c9.png", - "start": 0 - }, - { - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/258707140-b2df4a87-8bae-4589-a2cd-5a89cf9155bd.png", - "start": 12 - }, - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258707039-92bfe94b-a63e-4bf4-8eb1-df87be2eb4c9.png", - "start": 36 - } - ] - } - }, - { - "id": "startupjourney.3", - "kind": "slug:VideoDiagram", - "split": 0.643, - "title": "From Startup to Business", - "video": { - "innerScale": 1.02, - "src": { - "id": "598791979", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/258707436-fd39d04a-effa-4d88-a43d-16560f676abb.png", - "start": 0 - }, - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258707464-0fc4d6c6-0bc1-4d15-9fe0-18e46a3644e3.png", - "start": 6 - }, - { - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/258707493-310febee-fc1f-48e1-bfec-51fa05585dfc.png", - "start": 40 - }, - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258707524-3b05f66f-8ac4-44be-a9f6-862862b05611.png", - "start": 56 - } - ] - } - }, - { - "id": "startupjourney.4", - "kind": "slug:VideoDiagram", - "split": 0.74, - "title": "From Founders to Employees", - "video": { - "innerScale": 1.02, - "src": { - "id": "598792314", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/258708016-bf798904-635a-44a9-a6d0-2ac9015e4f96.png", - "start": 0 - } - ] - } - }, - { - "id": "startupjourney.5", - "kind": "slug:VideoDiagram", - "split": 0.65, - "title": "The Three Circles", - "video": { - "innerScale": 1.02, - "src": { - "id": "598792522", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258711740-d4b70a85-5486-4512-b89c-75debe92ca3d.png", - "start": 0 - }, - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258711782-62e3a768-68d5-406c-85e2-78f9e95eeb39.png", - "start": 41 - }, - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258711802-58242189-bd60-4789-9e71-94d6cad515af.png", - "start": 51 - }, - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258711759-f9171352-30b6-40f6-87bd-f43963aa85e5.png", - "start": 75 - } - ] - } - }, - { - "id": "startupjourney5.eg", - "kind": "slug:VideoDiagram", - "split": 0.643, - "title": "Example", - "video": { - "innerScale": 1.02, - "src": { - "id": "598793055", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258711740-d4b70a85-5486-4512-b89c-75debe92ca3d.png", - "start": 0 - } - ] - } - }, - { - "kind": "slug:namespace", - "namespace": "goodbm", - "title": "Good Business Model" - }, - { - "id": "goodbm.intro", - "kind": "slug:VideoDiagram", - "split": 0.767, - "title": "A Good Business Model", - "video": { - "innerScale": 1.02, - "src": { - "id": "598789079", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258715942-01c9eb71-1325-4211-9b06-d12ba7e63fe7.png", - "start": 0 - }, - { - "scale": 0.95, - "src": "https://user-images.githubusercontent.com/185555/258715968-6d478f21-f15d-4f5e-8a1d-dffdd11400ae.png", - "start": 68 - } - ] - } - }, - { - "id": "goodbm.2.customermodel", - "kind": "slug:VideoDiagram", - "title": "Product Market Fit", - "video": { - "innerScale": 1.02, - "src": { - "id": "598757457", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.95, - "src": "https://user-images.githubusercontent.com/185555/258716362-22cb4a33-5321-4c6d-95ec-26b90a81eb70.png", - "start": 0 - }, - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258716385-d5ff0a7b-cbb9-45bf-9401-efc247e71ff6.png", - "start": 8 - } - ] - } - }, - { - "id": "goodbm.3.impactmodel", - "kind": "slug:VideoDiagram", - "title": "Impact Micro / Macro", - "video": { - "innerScale": 1.02, - "src": { - "id": "598757720", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.95, - "src": "https://user-images.githubusercontent.com/185555/258716752-50ef5a31-5c5b-4973-9d20-7026ef348e87.png", - "start": 0 - }, - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258716776-213794cc-6087-4052-a9a0-96b9703ba645.png", - "start": 8 - } - ] - } - }, - { - "id": "goodbm.4.economicmodel.intro", - "kind": "slug:VideoDiagram", - "split": 0.683, - "title": "Economic Model", - "video": { - "innerScale": 1.02, - "src": { - "id": "598758279", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.7, - "src": "https://user-images.githubusercontent.com/185555/258717247-558d0b0f-3825-4e8f-b910-f9c2acd0d86b.png", - "start": 0 - }, - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258717383-dc18622b-77d4-4cc2-bfc2-dccf339d817f.png", - "start": 18 - } - ] - } - }, - { - "id": "goodbm.4.economicmodel.uniteconomics.1", - "kind": "slug:VideoDiagram", - "split": 0.753, - "title": "Positive Unit Economics ", - "video": { - "innerScale": 1.02, - "src": { - "id": "598758327", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/258718196-3ab075d8-59cb-46f7-9da3-7ae26aa8251c.png", - "start": 0 - }, - { - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/258718248-578d9842-4c16-46b1-88fa-70e474373ced.png", - "start": 15 - } - ] - } - }, - { - "id": "goodbm.4.economicmodel.uniteconomics.2", - "kind": "slug:VideoDiagram", - "split": 0.783, - "title": "Calculating", - "video": { - "innerScale": 1.02, - "src": { - "id": "598758555", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/258718409-aba235d8-2008-4dff-ad3a-bba247be8702.png", - "start": 0 - }, - { - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/258718479-d9b84723-b9dd-4351-b67a-042dd4d86e05.png", - "start": 62 - } - ] - } - }, - { - "id": "goodbm.4.economicmodel.growth", - "kind": "slug:VideoDiagram", - "title": "Can You Grow?", - "video": { - "innerScale": 1.02, - "src": { - "id": "598757995", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/258718724-3b12ff14-e4f1-4f25-a190-76bc53e686bf.png", - "start": 0 - }, - { - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/258718738-170d388d-de31-4afe-bd70-f613352e6af8.png", - "start": 4 - }, - { - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/258718767-410b3c52-416a-41bb-9004-32297fad4296.png", - "start": 25 - } - ] - } - }, - { - "id": "goodbm.conclusion", - "kind": "slug:VideoDiagram", - "split": 0.387, - "title": "Conclusion", - "video": { - "innerScale": 1.02, - "src": { - "id": "598758962", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/258722197-38fbb2cb-c3d5-4f75-80c9-aeef0200a6ba.png", - "start": 0 - } - ] - } - }, - { - "kind": "slug:namespace", - "namespace": "goodbm.improve", - "title": "Improvement" - }, - { - "id": "goodbm.improve.jeff", - "kind": "slug:VideoDiagram", - "title": "Improve Your Business Model", - "video": { - "innerScale": 1.02, - "src": { - "id": "598759314", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258722963-f8da0a30-2477-4d7f-aadf-ec2097d63b3f.png", - "start": 0 - }, - { - "scale": 0.74, - "src": "https://user-images.githubusercontent.com/185555/258723084-170d2d17-5d56-494e-9ac4-68d5176de157.png", - "start": 12 - }, - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258723112-642cb060-c2db-413f-97ed-67ca0eb273e1.png", - "start": 55 - } - ] - } - }, - { - "id": "goodbm.improve.leverage.intro", - "kind": "slug:VideoDiagram", - "split": 0.727, - "title": "Leverage", - "video": { - "innerScale": 1.02, - "src": { - "id": "598788777", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258723704-9852d071-5eb1-44d4-b3ec-7ff3e386fb1a.png", - "start": 0 - } - ] - } - }, - { - "id": "goodbm.improve.leverage.2types.intro", - "kind": "slug:VideoDiagram", - "split": 0.667, - "title": "Types of Leverage", - "video": { - "innerScale": 1.02, - "src": { - "id": "598786562", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.95, - "src": "https://user-images.githubusercontent.com/185555/258724142-506416bd-3e64-41d9-b85e-8315d2254daa.png", - "start": 0 - } - ] - } - }, - { - "id": "goodbm.improve.leverage.2types.labour", - "kind": "slug:VideoDiagram", - "title": "Labour", - "video": { - "innerScale": 1.02, - "src": { - "id": "598786787", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.95, - "src": "https://user-images.githubusercontent.com/185555/258724235-ad03d995-eee1-486a-8f13-b8658cd81c5b.png", - "start": 0 - } - ] - } - }, - { - "id": "goodbm.improve.leverage.2types.technology", - "kind": "slug:VideoDiagram", - "title": "Technology", - "video": { - "innerScale": 1.15, - "src": { - "id": "598787988", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.95, - "src": "https://user-images.githubusercontent.com/185555/258724317-a78a3bc6-fcd5-4cb0-b1d0-113f0d94ae74.png", - "start": 0 - } - ] - } - }, - { - "id": "goodbm.improve.leverage.2types.capital", - "kind": "slug:VideoDiagram", - "title": "Capital", - "video": { - "innerScale": 1.02, - "src": { - "id": "598759864", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.95, - "src": "https://user-images.githubusercontent.com/185555/258724389-d57fc9e2-bb59-4d2d-a646-3703493f79aa.png", - "start": 0 - } - ] - } - }, - { - "id": "goodbm.improve.leverage.2types.Product", - "kind": "slug:VideoDiagram", - "title": "Product", - "video": { - "innerScale": 1.02, - "src": { - "id": "598787543", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.95, - "src": "https://user-images.githubusercontent.com/185555/258724467-273fabf7-4e97-4a4a-bae0-882ec8b5da47.png", - "start": 0 - } - ] - } - }, - { - "id": "goodbm.improve.leverage.2types.media", - "kind": "slug:VideoDiagram", - "title": "Media", - "video": { - "innerScale": 1.02, - "src": { - "id": "598786893", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.95, - "src": "https://user-images.githubusercontent.com/185555/258724550-96195499-9f2c-4cc5-a618-1870d525defa.png", - "start": 0 - } - ] - } - }, - { - "id": "goodbm.improve.leverage.2types.perceptual", - "kind": "slug:VideoDiagram", - "title": "Perceptual", - "video": { - "innerScale": 1.02, - "src": { - "id": "598787297", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.95, - "src": "https://user-images.githubusercontent.com/185555/258724616-d1a76df8-a4e0-4482-9d9d-85f6cade59fe.png", - "start": 0 - } - ] - } - }, - { - "id": "goodbm.improve.leverage.2types.network", - "kind": "slug:VideoDiagram", - "title": "Network", - "video": { - "innerScale": 1.02, - "src": { - "id": "598787119", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.95, - "src": "https://user-images.githubusercontent.com/185555/258724693-57184816-71bb-43df-b0dd-494327d72e57.png", - "start": 0 - } - ] - } - }, - { - "id": "goodbm.improve.leverage.2types.strategic", - "kind": "slug:VideoDiagram", - "split": 0.753, - "title": "Strategic", - "video": { - "innerScale": 1.02, - "src": { - "id": "598787671", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.95, - "src": "https://user-images.githubusercontent.com/185555/258724758-314e94b2-b9bc-4da9-9ac3-44763fd67ceb.png", - "start": 0 - } - ] - } - }, - { - "id": "goodbm.improve.leverage.3size.1", - "kind": "slug:VideoDiagram", - "split": 0.663, - "title": "Enterprise Size", - "video": { - "innerScale": 1.02, - "src": { - "id": "598788082", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/258724873-6813f52a-01de-48bc-bcfe-49d7fcb79e81.png", - "start": 0 - } - ] - } - }, - { - "id": "goodbm.improve.leverage.3size.2", - "kind": "slug:VideoDiagram", - "title": "Enterprise Complexity", - "video": { - "innerScale": 1.02, - "src": { - "id": "598788467", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.95, - "src": "https://user-images.githubusercontent.com/185555/258724944-7a99c045-65e0-4e0b-81a0-d2363b6cca39.png", - "start": 0 - } - ] - } - }, - { - "id": "goodbm.improve.conclusion", - "kind": "slug:VideoDiagram", - "split": 0.773, - "title": "Conclusion ", - "video": { - "innerScale": 1.02, - "src": { - "id": "598759076", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/258725150-f1a44a40-21fc-406a-94aa-7c330dd24ede.png", - "start": 0 - } - ] - } - }, - { - "kind": "slug:namespace", - "namespace": "strategy.stage", - "title": "Strategy Stage" - }, - { - "id": "strategy.stage.intro", - "kind": "slug:VideoDiagram", - "title": "Stage", - "video": { - "innerScale": 1.02, - "src": { - "id": "598793514", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258729336-52097361-322e-4bb2-a0e6-9f2b39aee9c9.png", - "start": 0 - } - ] - } - }, - { - "id": "strategy.stage.1", - "kind": "slug:VideoDiagram", - "split": 0.72, - "title": "Assessing Your Stage", - "video": { - "innerScale": 1.02, - "src": { - "id": "598795704", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.97, - "src": "https://user-images.githubusercontent.com/185555/258729479-036ae33e-47d0-4533-b08d-b0388484d5da.png", - "start": 0 - } - ] - } - }, - { - "id": "strategy.stage.2", - "kind": "slug:VideoDiagram", - "split": 0.72, - "title": "The Venture Map", - "video": { - "innerScale": 1.02, - "src": { - "id": "598795918", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258729516-fa75631c-32ef-45f1-91dc-9370a67aaddf.png", - "start": 0 - } - ] - } - }, - { - "id": "strategy.stage.3", - "kind": "slug:VideoDiagram", - "split": 0.717, - "title": "Conclusions", - "video": { - "innerScale": 1.02, - "src": { - "id": "598796095", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.85, - "src": "https://user-images.githubusercontent.com/185555/258729578-3c37e55e-4aa1-45c5-9c74-78d01dc42962.png", - "start": 0 - } - ] - } - }, - { - "kind": "slug:namespace", - "namespace": "strategy.models", - "title": "Strategy Models" - }, - { - "id": "strategy.models.intro", - "kind": "slug:VideoDiagram", - "split": 0.62, - "title": "Strategy", - "video": { - "innerScale": 1.02, - "src": { - "id": "598794082", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.8, - "src": "https://user-images.githubusercontent.com/185555/258731304-0c36ff18-5adb-4ca7-84c6-679f024a6fc2.png", - "start": 0 - } - ] - } - }, - { - "id": "strategy.models.2", - "kind": "slug:VideoDiagram", - "split": 0.743, - "title": "Canvas Models", - "video": { - "innerScale": 1.02, - "src": { - "id": "598793732", - "kind": "Vimeo" - }, - "timestamps": [ - { - "start": 0, - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/258731717-f2698344-9b96-46d7-8237-069d1cb9b3a8.png" - }, - { - "start": 60, - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/259271445-e0a699f2-5782-4c40-ba5d-f77f803b7090.png" - }, - { - "start": 67, - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/259271453-bd6b4125-498d-4a56-848e-1db0737a4fa1.png" - }, - { - "start": 76, - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/259271461-ebd7d480-178e-4e5c-a9b1-6f98e19fd192.png" - } - ] - } - }, - { - "id": "strategy.models.validation", - "kind": "slug:VideoDiagram", - "split": 0.727, - "title": "Validation", - "video": { - "innerScale": 1.02, - "src": { - "id": "598795327", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.95, - "src": "https://user-images.githubusercontent.com/185555/258732088-883064dc-60be-44f3-b59e-430e66bb05f5.png", - "start": 0 - }, - { - "scale": 0.9, - "src": "https://user-images.githubusercontent.com/185555/258732112-cdd6881b-539c-4ce2-8ad5-0026b5bede20.png", - "start": 0 - } - ] - } - }, - { - "id": "strategy.models.together", - "kind": "slug:VideoDiagram", - "split": 0.72, - "title": "Pulling Your Strategy Together", - "video": { - "innerScale": 1.02, - "src": { - "id": "598796367", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.95, - "src": "https://user-images.githubusercontent.com/185555/258732446-070a44a1-3c7b-4415-a03d-2e641f526f9d.png", - "start": 0 - } - ] - } - }, - { - "id": "strategy.timeline", - "kind": "slug:VideoDiagram", - "split": 0.723, - "title": "Strategy Timeline", - "video": { - "innerScale": 1.02, - "src": { - "id": "851214652", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 0.97, - "src": "https://user-images.githubusercontent.com/185555/258733175-6c16ec25-0395-4629-a267-f7bb537656c0.png", - "start": 0 - } - ] - } - }, - { - "id": "strategy.conclusion", - "kind": "slug:VideoDiagram", - "split": 0.717, - "title": "Conclusion", - "video": { - "innerScale": 1.02, - "src": { - "id": "598793191", - "kind": "Vimeo" - }, - "timestamps": [ - { - "scale": 1, - "src": "https://user-images.githubusercontent.com/185555/258735372-664939ab-6574-4aca-8d05-0901cab728fc.png", - "start": 0 - } - ] - } - } - ] -} diff --git a/deploy/slc.db.team/deno.json b/deploy/slc.db.team/deno.json deleted file mode 100644 index 468a77373c..0000000000 --- a/deploy/slc.db.team/deno.json +++ /dev/null @@ -1,29 +0,0 @@ -{ - "name": "@tdb/slc", - "version": "0.0.68", - "license": "MIT", - "tasks": { - "start": "deno run -RNE --watch ./src/s.main/http-server.ts", - "lint": "deno lint", - "dry": "deno publish --allow-dirty --dry-run", - "clean": "deno run -RWE ./scripts/-clean.ts", - "test": "deno test -RWNE --allow-run --allow-ffi", - "prep": "deno run -RWE ./scripts/-prep.ts", - "dev": "deno run -RWNE --allow-run --allow-ffi ./scripts/-dev.ts", - "build": "deno run -RWNE --allow-run --allow-ffi ./scripts/-build.ts", - "serve": "deno run -RNE --allow-run jsr:@sys/http/server/start", - "deploy": "deno task dry && deployctl deploy --prod --org=tdb --project=tdb-slc" - }, - "exports": { - ".": "./src/mod.ts", - "./t": "./src/types.ts", - "./types": "./src/types.ts" - }, - "deploy": { - "exclude": [ - "**/node_modules" - ], - "include": [], - "entrypoint": "src/s.main/http-server.ts" - } -} diff --git a/deploy/slc.db.team/scripts/-build.ts b/deploy/slc.db.team/scripts/-build.ts deleted file mode 100644 index ae681c7ddf..0000000000 --- a/deploy/slc.db.team/scripts/-build.ts +++ /dev/null @@ -1,14 +0,0 @@ -import { Fs } from '@sys/fs'; -import { Vite } from '@sys/driver-vite'; -import { pkg } from '../src/pkg.ts'; - -const input = './src/-test/index.html'; -const bundle = await Vite.build({ pkg, input }); -console.info(bundle.toString({ pad: true })); - -/** - * Ensure the {manifest.json} file exists. - */ -const from = Fs.resolve('./src/manifest.json'); -const to = Fs.resolve('./dist/manifest.json'); -await Fs.copy(from, to, { throw: true }); diff --git a/deploy/slc.db.team/scripts/-clean.ts b/deploy/slc.db.team/scripts/-clean.ts deleted file mode 100644 index 30ae5e6470..0000000000 --- a/deploy/slc.db.team/scripts/-clean.ts +++ /dev/null @@ -1,4 +0,0 @@ -import { Fs } from '@sys/fs'; - -const removeDir = (path: string) => Fs.remove(Fs.resolve(path), { log: true }); -await removeDir('./.tmp'); diff --git a/deploy/slc.db.team/scripts/-dev.ts b/deploy/slc.db.team/scripts/-dev.ts deleted file mode 100644 index 530ecf6199..0000000000 --- a/deploy/slc.db.team/scripts/-dev.ts +++ /dev/null @@ -1,6 +0,0 @@ -import { Vite } from '@sys/driver-vite'; -import { pkg } from '../src/pkg.ts'; - -const input = './src/-test/index.html'; -const server = await Vite.dev({ pkg, input }); -await server.listen(); diff --git a/deploy/slc.db.team/scripts/-prep.ts b/deploy/slc.db.team/scripts/-prep.ts deleted file mode 100644 index 50e80eb1ba..0000000000 --- a/deploy/slc.db.team/scripts/-prep.ts +++ /dev/null @@ -1,9 +0,0 @@ -import { Fs } from '@sys/fs'; - -const PATH = { - from: '/Users/phil/code/samples/vitepress-slc/docs/.vitepress/dist', - to: Fs.resolve('./dist.docs'), -}; - -const res = await Fs.copy(PATH.from, PATH.to, { force: true }); -console.log('res', res); diff --git a/deploy/slc.db.team/src/-mod.test.ts b/deploy/slc.db.team/src/-mod.test.ts deleted file mode 100644 index 174dbeeac3..0000000000 --- a/deploy/slc.db.team/src/-mod.test.ts +++ /dev/null @@ -1,7 +0,0 @@ -import { describe, expect, it, pkg } from './-test.ts'; - -describe(`Pkg: ${pkg.name}@${pkg.version}`, () => { - it('š· placeholder', () => { - expect(123).to.equal(123); - }); -}); diff --git a/deploy/slc.db.team/src/-test.ts b/deploy/slc.db.team/src/-test.ts deleted file mode 100644 index ad27e650df..0000000000 --- a/deploy/slc.db.team/src/-test.ts +++ /dev/null @@ -1 +0,0 @@ -export * from './-test/mod.ts'; diff --git a/deploy/slc.db.team/src/-test/entry.tsx b/deploy/slc.db.team/src/-test/entry.tsx deleted file mode 100644 index b253d5cf25..0000000000 --- a/deploy/slc.db.team/src/-test/entry.tsx +++ /dev/null @@ -1,19 +0,0 @@ -import { StrictMode } from 'react'; -import { createRoot } from 'react-dom/client'; - -import { pkg } from '../common.ts'; -import { Tmp } from '../mod.ts'; - -/** - * Render UI. - */ -const document = globalThis.document; -document.title = pkg.name; -console.log('š· entry.tsx', pkg); - -const root = createRoot(document.getElementById('root')); -root.render( - <StrictMode> - <Tmp /> - </StrictMode>, -); diff --git a/deploy/slc.db.team/src/-test/mod.ts b/deploy/slc.db.team/src/-test/mod.ts deleted file mode 100644 index 7539a23ea4..0000000000 --- a/deploy/slc.db.team/src/-test/mod.ts +++ /dev/null @@ -1,2 +0,0 @@ -export { Testing, describe, expect, it } from '@sys/testing/server'; -export * from '../common.ts'; diff --git a/deploy/slc.db.team/src/common/libs.ts b/deploy/slc.db.team/src/common/libs.ts deleted file mode 100644 index 25445ccbbf..0000000000 --- a/deploy/slc.db.team/src/common/libs.ts +++ /dev/null @@ -1,5 +0,0 @@ -export { Err, Hash, Path, Pkg, Time, rx } from '@sys/std'; - -export { COLORS, Str } from '@sys/std'; -export { Color, css } from '@sys/ui-css/react'; -export { FC } from '@sys/ui-react'; diff --git a/deploy/slc.db.team/src/common/t.ts b/deploy/slc.db.team/src/common/t.ts deleted file mode 100644 index e83af329a1..0000000000 --- a/deploy/slc.db.team/src/common/t.ts +++ /dev/null @@ -1,14 +0,0 @@ -export type { - CommonTheme, - Disposable, - DistPkg, - Lifecycle, - Msecs, - StdError, - StringHash, - UntilObservable, -} from '@sys/types'; - -export type { CssValue } from '@sys/ui-css/t'; - -export type * from '../types.ts'; diff --git a/deploy/slc.db.team/src/manifest.json b/deploy/slc.db.team/src/manifest.json deleted file mode 100644 index 093787acf1..0000000000 --- a/deploy/slc.db.team/src/manifest.json +++ /dev/null @@ -1,8 +0,0 @@ -{ - "name": "@tdb/slc", - "short_name": "@tdb/slc", - "start_url": "/", - "display": "standalone", - "background_color": "#FE0064", - "theme_color": "#FE0064" -} diff --git a/deploy/slc.db.team/src/mod.ts b/deploy/slc.db.team/src/mod.ts deleted file mode 100644 index e147b64cee..0000000000 --- a/deploy/slc.db.team/src/mod.ts +++ /dev/null @@ -1,13 +0,0 @@ -/** - * @module - * Tools for ... - * - * @example - * ```ts - * // sample - * ``` - */ -export { pkg } from './pkg.ts'; -export type * as t from './types.ts'; - -export { Tmp } from './ui/Tmp.tsx'; diff --git a/deploy/slc.db.team/src/pkg.ts b/deploy/slc.db.team/src/pkg.ts deleted file mode 100644 index 79cb3f9c80..0000000000 --- a/deploy/slc.db.team/src/pkg.ts +++ /dev/null @@ -1,8 +0,0 @@ -import { Pkg, type t } from '@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - - -/** - * Package meta-data. - */ -export const pkg: t.Pkg = Pkg.fromJson(deno); diff --git a/deploy/slc.db.team/src/s.main/http-server.ts b/deploy/slc.db.team/src/s.main/http-server.ts deleted file mode 100644 index d717427604..0000000000 --- a/deploy/slc.db.team/src/s.main/http-server.ts +++ /dev/null @@ -1,80 +0,0 @@ -import { Pkg } from 'jsr:@sys/fs'; -import { HttpServer } from 'jsr:@sys/http/server'; -import { pkg } from './pkg.ts'; - -const dist = (await Pkg.Dist.load('./dist')).dist; -const hash = dist?.hash.digest ?? ''; - -/** - * Define HTTP Web Server. - */ -const app = HttpServer.create({ pkg, hash }); - -/** - * Docs - */ -app.get('/docs', HttpServer.static({ root: './dist.docs', path: '/index.html' })); -app.use( - '/docs/*', - HttpServer.static({ - root: './dist.docs', - rewriteRequestPath: (path) => path.replace(/^\/docs/, ''), - }), -); - -// Serve static files from /dist for all other paths. -app.use('/*', HttpServer.static({ root: './dist' })); - -/** - * Start Server. - */ -const config = HttpServer.options(8080, pkg, hash); -Deno.serve(config, app.fetch); - -/** - * Sample static server middleware. - */ - -// const staticMiddleware = (rootDir: string) => { -// return async (c: Context, next: Next) => { -// const url = new URL(c.req.url); -// let filepath = decodeURIComponent(url.pathname); -// -// // Prevent directory traversal attacks -// if (filepath.includes('..')) { -// return c.text('Forbidden', 403); -// } -// -// // Map URL path to filesystem path -// filepath = `${rootDir}${filepath}`; -// -// try { -// let fileInfo = await Deno.stat(filepath); -// -// // If it's a directory, try to serve index.html -// if (fileInfo.isDirectory) { -// filepath = `${filepath}/index.html`; -// fileInfo = await Deno.stat(filepath); // Re-stat the index.html file -// } -// -// // Read the file content -// const file = await Deno.readFile(filepath); -// -// // Determine the content type -// const contentType = lookup(filepath) || 'application/octet-stream'; -// -// // Return the file content -// return c.body(file, 200, { -// 'Content-Type': contentType, -// }); -// } catch (e) { -// if (e instanceof Deno.errors.NotFound) { -// // File not found, proceed to the next middleware or route -// await next(); -// } else { -// // Other errors (e.g., permission issues) -// return c.text('Internal Server Error', 500); -// } -// } -// }; -// }; diff --git a/deploy/slc.db.team/src/s.main/pkg.ts b/deploy/slc.db.team/src/s.main/pkg.ts deleted file mode 100644 index 9e44b0ad96..0000000000 --- a/deploy/slc.db.team/src/s.main/pkg.ts +++ /dev/null @@ -1,7 +0,0 @@ -import { Pkg, type t } from 'jsr:@sys/std'; -import { default as deno } from '../../deno.json' with { type: 'json' }; - -/** - * Package meta-data. - */ -export const pkg: t.Pkg = Pkg.fromJson(deno); diff --git a/deploy/slc.db.team/src/types.ts b/deploy/slc.db.team/src/types.ts deleted file mode 100644 index 0f2f882807..0000000000 --- a/deploy/slc.db.team/src/types.ts +++ /dev/null @@ -1,5 +0,0 @@ -/** - * @module - * Module types. - */ -export {}; diff --git a/deploy/slc.db.team/src/ui/Tmp.tsx b/deploy/slc.db.team/src/ui/Tmp.tsx deleted file mode 100644 index 86fdf97611..0000000000 --- a/deploy/slc.db.team/src/ui/Tmp.tsx +++ /dev/null @@ -1,127 +0,0 @@ -import { useEffect, useState } from 'react'; -import { type t, Color, COLORS, css, Hash, Pkg, pkg, rx, Str } from './common.ts'; - -export type TmpProps = { - digest?: t.StringHash; - theme?: t.CommonTheme; - style?: t.CssInput; -}; - -export const Tmp: React.FC<TmpProps> = (props) => { - const [_dist, setDist] = useState<t.DistPkg>(); - const dist: t.DistPkg | undefined = _dist; - const digest = props.digest ? wrangle.fmtHash(props.digest) : wrangle.digest(dist); - - /** - * Lifecycle. - */ - useEffect(() => { - const { dispose, dispose$ } = rx.disposable(); - (async () => { - /** - * GET fetch - */ - const res = await Pkg.Dist.fetch({ dispose$, disposeReason: 'react:useEffect:dispose' }); - setDist(res.dist); - })(); - return dispose; - }, []); - - /** - * Render - */ - const theme = Color.theme(props.theme ?? 'Dark'); - const styles = { - base: css({ - Absolute: 0, - backgroundColor: COLORS.MAGENTA, - color: theme.fg, - fontFamily: 'sans-serif', - }), - body: { - base: css({ Absolute: 0, display: 'grid', placeItems: 'center', userSelect: 'none' }), - inner: css({ marginBottom: 25 }), - pig: css({ fontSize: 50 }), - title: css({ fontSize: 28 }), - }, - pkg: { - base: css({ - Absolute: [null, null, 16, digest.display ? 18 : 15], - fontFamily: 'monospace', - cursor: 'pointer', - display: 'grid', - rowGap: '0.5em', - }), - at: css({ MarginX: '0.6em', opacity: 0.5 }), - version: css({ opacity: 1 }), - hash: css({ opacity: 0.5 }), - name: css({}), - }, - a: css({ - textDecoration: 'none', - color: theme.fg, - }), - }; - - const elBody = ( - <div {...styles.body.base}> - <div {...styles.body.inner}> - <a {...styles.a} href="./docs"> - <div {...styles.body.pig}>{`š·`}</div> - <div {...styles.body.title}>{`Social Lean Canvas`}</div> - </a> - </div> - </div> - ); - - const elHash = digest.display && ( - <div {...styles.pkg.hash} title={digest.tooltip}> - {digest.display} - </div> - ); - - const elNameVersion = ( - <div> - <div> - <span {...styles.pkg.name}>{pkg.name}</span> - <span {...styles.pkg.at}>{'@'}</span> - <span {...styles.pkg.version}>{pkg.version}</span> - </div> - </div> - ); - - const elPkg = ( - <div {...styles.pkg.base}> - {elNameVersion} - {elHash} - </div> - ); - - return ( - <div style={css(styles.base, props.style)}> - {elBody} - {elPkg} - </div> - ); -}; - -/** - * Helpers - */ -const wrangle = { - digest(dist?: t.DistPkg) { - const digest = wrangle.fmtHash(dist?.hash.digest); - const b = dist?.size.bytes; - const bytes = Str.bytes(b); - const display = `${digest.display} (${bytes})`; - return { ...digest, display }; - }, - - fmtHash(hash?: t.StringHash) { - const long = hash ?? ''; - const short = Hash.shorten(long, [0, 4], true); - const tooltip = `pkg:digest:${long}`; - const display = `pkg:sha256:#${short}`; - return { long, short, display, tooltip }; - }, -} as const; diff --git a/deploy/slc.db.team/vite.config.ts b/deploy/slc.db.team/vite.config.ts deleted file mode 100644 index 25af1be683..0000000000 --- a/deploy/slc.db.team/vite.config.ts +++ /dev/null @@ -1,14 +0,0 @@ -import { Vite } from '@sys/driver-vite'; -import { defineConfig } from 'vite'; -import { pkg } from './src/pkg.ts'; - -export default defineConfig(() => { - return Vite.Plugin.common({ - pkg, - chunks(e) { - e.chunk('react', 'react'); - e.chunk('react.dom', 'react-dom'); - e.chunk('sys', ['@sys/std']); - }, - }); -}); diff --git a/deploy/tmp.db.team/deno.json b/deploy/tmp.db.team/deno.json index 182f6d7ec1..f121f85398 100644 --- a/deploy/tmp.db.team/deno.json +++ b/deploy/tmp.db.team/deno.json @@ -1,6 +1,6 @@ { "name": "@tdb/tmp", - "version": "0.0.72", + "version": "0.0.82", "tasks": { "dev": "deno run -RNE --watch ./src/main.ts", "check": "deno check ./src/main.ts", diff --git a/deploy/tmp.db.team/src/pkg.ts b/deploy/tmp.db.team/src/pkg.ts index 094f9fce61..d265bd13a2 100644 --- a/deploy/tmp.db.team/src/pkg.ts +++ b/deploy/tmp.db.team/src/pkg.ts @@ -1,8 +1,16 @@ -import { Pkg, type t } from 'jsr:@sys/std'; -import { default as deno } from '../deno.json' with { type: 'json' }; - +import type { Pkg } from '@sys/types'; /** * Package meta-data. + * + * AUTO-GENERATED: + * This file is generated via the `prep` command across the + * @system monorepo. See command: + * + * cd ./<system-repo-root> + * deno task prep + * + * - DO check this file in to source-control. + * - Do NOT manually alter the file (as your work will be lost). */ -export const pkg: t.Pkg = Pkg.fromJson(deno); +export const pkg: Pkg = { name: '@tdb/tmp', version: '0.0.82' }; diff --git a/deps.yaml b/deps.yaml index 9bd188f371..14d4b33669 100644 --- a/deps.yaml +++ b/deps.yaml @@ -3,7 +3,7 @@ # # ./š¦ # | deno.json -# |(write) ā deno.imports.json +# |(write) ā imports.json # |(write) ā package.json # # This is the "single-source-of-truth" with regards to dependencies and versioning. @@ -44,6 +44,7 @@ groups: build/tools: - import: npm:@vitejs/plugin-react-swc@3.8.0 + - import: npm:@deno/vite-plugin@1.0.4 - import: npm:rollup@4.34.8 - import: npm:vite@6.1.1 - import: npm:vite-plugin-wasm@3.4.1 @@ -64,6 +65,9 @@ deno.json: - import: jsr:@cliffy/prompt@1.0.0-rc.7 - import: jsr:@cliffy/table@1.0.0-rc.7 + # Sundry: JSR + - import: jsr:@deno/dnt@0.41.3 + # Sundry: NPM - import: npm:@types/diff@7.0.1 - import: npm:chai@5 @@ -89,15 +93,20 @@ deno.json: - import: npm:ts-essentials@10.0.4 - import: npm:valibot@1.0.0-rc.1 - import: npm:yaml@2.7.0 + - import: npm:@preact/signals-core@1.8.0 # Browser - import: npm:csstype@3 - import: npm:ua-parser-js@2.0.2 # UI + - import: npm:@svgdotjs/svg.js@3.2.4 + # UI:React - import: npm:react-error-boundary@5 - import: npm:react-inspector@6 - import: npm:react-spinners@0.15.0 + - import: npm:@preact/signals-react@3.0.1 + - import: npm:react-inspector@6.0.2 package.json: - group: std/deno @@ -111,6 +120,7 @@ package.json: - group: ui/react - import: npm:react-icons@5.5.0 - import: npm:@vidstack/react@1.12.12 + - import: npm:motion@12.5.0 # UI:Frameworks - import: npm:vitepress@1.6.3 diff --git a/deno.imports.json b/imports.json similarity index 89% rename from deno.imports.json rename to imports.json index 8303ae8b14..4ecdadc967 100644 --- a/deno.imports.json +++ b/imports.json @@ -8,9 +8,12 @@ "@cliffy/keypress": "jsr:@cliffy/keypress@1.0.0-rc.7", "@cliffy/prompt": "jsr:@cliffy/prompt@1.0.0-rc.7", "@cliffy/table": "jsr:@cliffy/table@1.0.0-rc.7", + "@deno/dnt": "jsr:@deno/dnt@0.41.3", "@noble/hashes": "npm:@noble/hashes@1.7.1", "@noble/hashes/*": "npm:@noble/hashes@1.7.1/*", "@onsetsoftware/automerge-patcher": "npm:@onsetsoftware/automerge-patcher@0.14.0", + "@preact/signals-core": "npm:@preact/signals-core@1.8.0", + "@preact/signals-react": "npm:@preact/signals-react@3.0.1", "@std/async": "jsr:@std/async@1.0.10", "@std/datetime": "jsr:@std/datetime@0.225.3", "@std/dotenv": "jsr:@std/dotenv@0.225.3", @@ -20,6 +23,7 @@ "@std/semver": "jsr:@std/semver@1.0.4", "@std/testing": "jsr:@std/testing@1.0.9", "@std/uuid": "jsr:@std/uuid@1.0.4", + "@svgdotjs/svg.js": "npm:@svgdotjs/svg.js@3.2.4", "@types/diff": "npm:@types/diff@7.0.1", "approx-string-match": "npm:approx-string-match@2", "chai": "npm:chai@5", @@ -38,7 +42,7 @@ "rambda": "npm:rambda@9.4.2", "ramda": "npm:ramda@0.30.1", "react-error-boundary": "npm:react-error-boundary@5", - "react-inspector": "npm:react-inspector@6", + "react-inspector": "npm:react-inspector@6.0.2", "react-spinners": "npm:react-spinners@0.15.0", "rxjs": "npm:rxjs@7.8.2", "strip-ansi": "npm:strip-ansi@7", diff --git a/package.json b/package.json index 73019e3192..5b7c817292 100644 --- a/package.json +++ b/package.json @@ -14,6 +14,7 @@ "@types/react-dom": "18.3.5", "@vidstack/react": "1.12.12", "hono": "4.7.2", + "motion": "12.5.0", "react": "18.3.1", "react-dom": "18.3.1", "react-icons": "5.5.0", @@ -21,6 +22,7 @@ "vue": "3.5.13" }, "devDependencies": { + "@deno/vite-plugin": "1.0.4", "@vitejs/plugin-react-swc": "3.8.0", "rollup": "4.34.8", "vite": "6.1.1", diff --git a/scripts/-tmp.ts b/scripts/-tmp.ts index 5ff92b9f19..5451d7103c 100644 --- a/scripts/-tmp.ts +++ b/scripts/-tmp.ts @@ -1,25 +1,79 @@ -import { Jsr } from '@sys/jsr'; -import { Cli } from '@sys/cli'; -import { Paths } from './u.paths.ts'; - -// console.log('Jsr', Jsr); -// const info = await Jsr.Fetch.Pkg.info('@sys/std'); -// console.log('info', info); -// console.log(`-------------------------------------------`); -// console.log('moduleGraph1', info.data?.moduleGraph1); -// console.log('moduleGraph2', info.data?.moduleGraph2); - -const options = Paths.modules.map((path) => { - return { name: path, value: path }; -}); - -const res = await Cli.Prompt.Checkbox.prompt({ - message: '(Workflow) JSR Publish Modules:', - options, -}); -console.log('res', res); +import { Args, c, Cli } from '@sys/cli'; +import { Fs } from '@sys/fs'; +import { rx, Str } from '@sys/std'; + +const { italic: i } = c; + +type TArgs = { watch?: boolean }; +const args = Args.parse<TArgs>(Deno.args, { boolean: ['watch'], alias: { w: 'watch' } }); +console.info(c.cyan('args:'), args); + +let copyCount = 0; /** - * Finish up. + * Copy the SLC project content to the VitePress (driver) development sample directory + * from it's external content authoring location. */ -Deno.exit(0); +const dir = { + source: '/Users/phil/Documents/Notes/tdb/slc/slc-public', + target: Fs.resolve('code/sys.driver/driver-vitepress/.tmp/sample'), +} as const; + +export async function copyDocs() { + const Fmt = { + path: (path: string) => `${c.gray(Fs.dirname(path))}/${c.white(Fs.basename(path))}`, + } as const; + + copyCount++; + + const title = c.bold(c.brightGreen('Copy')); + const table = Cli.table([title]); + const push = (label: string, value: string) => table.push([c.gray(label), value]); + const blankLine = () => table.push([]); + + const copy = async (from: string, to: string) => { + from = Fs.join(dir.source, from); + to = Fs.join(dir.target, to); + await Fs.copy(from, to, { force: true }); + push(` ⢠from`, c.gray(Fs.trimCwd(from))); + push(` ⢠${c.cyan('to')}`, Fmt.path(Fs.trimCwd(to))); + blankLine(); + }; + + blankLine(); + await copy('docs', 'docs'); + await copy('src', 'src'); + + // Ouput. + console.info(); + console.info(table.toString().trim()); + console.info(); + console.info(i(`copied ${c.green(String(copyCount))} ${Str.plural(copyCount, 'time', 'times')}`)); +} + +await copyDocs(); + +/** + * Watcher. + */ +if (args.watch) { + const watcher = await Fs.watch(dir.source); + watcher.$.pipe(rx.debounceTime(1000)).subscribe(copyDocs); + console.info(); + console.info(i(c.gray(` (watching for file changes)`))); + console.info(i(c.gray(` ${c.yellow('Enter')} to force copy`))); + console.info(); + + for await (const e of Cli.keypress()) { + if (e.key === 'return') copyDocs(); + if ((e.ctrlKey && e.key === 'c') || e.key === 'q') Deno.exit(0); + } +} + +// Finish up. +if (!args.watch) { + const y = c.yellow; + console.info(c.italic(c.gray(`(pass ${y('--watch')} (${y('-w')}) to re-run on file changes)`))); + console.info(); + Deno.exit(0); +} diff --git a/scripts/Task.-bump.ts b/scripts/Task.-bump.ts index 55d212dc60..c2670a7d26 100644 --- a/scripts/Task.-bump.ts +++ b/scripts/Task.-bump.ts @@ -36,13 +36,13 @@ export async function main(options: Options = {}) { * Retrieve the child modules within the workspace. */ const children = ws.children - .filter((child) => !exclude(child.path)) - .filter((child) => !!child.file.version) - .filter((child) => typeof child.file.version === 'string') - .filter((child) => typeof child.file.name === 'string') + .filter((child) => !exclude(child.path.denofile)) + .filter((child) => !!child.denofile.version) + .filter((child) => typeof child.denofile.version === 'string') + .filter((child) => typeof child.denofile.name === 'string') .map((child) => { - const json = child.file; - const path = child.path; + const json = child.denofile; + const path = child.path.denofile; const { name = '', version = '' } = json; const current = Semver.parse(version).version; const next = wrangle.increment(current, release); @@ -59,9 +59,9 @@ export async function main(options: Options = {}) { const pkg = `${c.gray(modScope)}/${c.white(c.bold(modName))}`; const vCurrent = Semver.toString(version.current); - const vNext = Semver.Fmt.colorize(version.next, { highlight: release, baseColor: c.green }); + const vNext = Semver.Fmt.colorize(version.next, { highlight: release }); - const title = `${c.green(' ā¢')} ${pkg}`; + const title = `${c.cyan(' ā¢')} ${pkg}`; table.push([title, vCurrent, 'ā', vNext]); }); diff --git a/scripts/Task.-dry.ts b/scripts/Task.-dry.ts index 1066efaf9e..6f7e4fd460 100644 --- a/scripts/Task.-dry.ts +++ b/scripts/Task.-dry.ts @@ -6,11 +6,14 @@ export async function main() { const results: CmdResult[] = []; const run = async (path: string, index: number, total: number) => { - const command = `deno task dry`; + const cmd = 'dry'; + const command = `deno task ${cmd}`; + const commandFmt = c.green(`deno task ${c.bold(c.cyan(cmd))}`); + const title = c.gray(`${c.white('Type Checks')} (${c.white(String(index + 1))} of ${total})`); const moduleList = Log.moduleList({ index, indent: 3 }); - const text = `${title}\n ${c.cyan(command)}\n${moduleList}`; + const text = `${title}\n ${commandFmt}\n${moduleList}`; spinner.text = c.gray(text); const output = await Process.sh(path).run(command); diff --git a/scripts/Task.-info.ts b/scripts/Task.-info.ts index 2aa5f8261b..600639204b 100644 --- a/scripts/Task.-info.ts +++ b/scripts/Task.-info.ts @@ -11,7 +11,7 @@ export async function main() { const process = async (path: string) => { const fileInfo = await Deno.stat(path); if (fileInfo.isFile) { - const file = (await Fs.readText(path)).data; + const file = (await Fs.readText(path)).data ?? ''; const lines = file.split('\n'); files.push({ path, total: { lines: lines.length } }); } diff --git a/scripts/Task.-prep.ts b/scripts/Task.-prep.ts index 632af83a77..46316a1d09 100644 --- a/scripts/Task.-prep.ts +++ b/scripts/Task.-prep.ts @@ -1,18 +1,20 @@ -import { c, DenoDeps, Fs, Process } from './common.ts'; +import { type t, c, DenoDeps, DenoFile, Err, Fs, Process, Tmpl } from './common.ts'; const i = c.italic; /** - * Prepare the [deno.json | package.json] files from - * definitions within the monorepo's `deps.yaml` configuration. + * Proecss the dependencies into a`deno.json` and `package.json` files. */ -export async function main() { +async function processDeps() { const res = await DenoDeps.from('./deps.yaml'); - if (res.error) return console.error(res.error); + if (res.error) { + console.error(res.error); + return; + } const PATH = { package: './package.json', - deno: './deno.imports.json', - }; + deno: './imports.json', + } as const; /** * Write to file-system: [deno.json | package.json]. @@ -21,16 +23,6 @@ export async function main() { await Fs.writeJson(PATH.package, DenoDeps.toJson('package.json', deps)); await Fs.writeJson(PATH.deno, DenoDeps.toJson('deno.json', deps)); - /** - * Run `prep` ā `init` commands on sub-modules. - */ - const sh = (path: string) => Process.sh({ path }); - const module = (...parts: string[]) => sh(Fs.resolve('./code', ...parts)); - - const cmd = 'deno task prep && deno task init'; - await module('sys.driver/driver-vite').run(cmd); - await module('sys.driver/driver-vitepress').run(cmd); - /** * Output: console. */ @@ -43,3 +35,65 @@ export async function main() { console.info(DenoDeps.Fmt.deps(deps, { indent: 1 })); console.info(); } + +/** + * Write all {pkg}.ts files with name/version values synced + * to their corresponding current `deno.json` file values. + */ +async function updatePackages() { + const errors = Err.errors(); + const ws = await DenoFile.workspace(); + + const tmpl = Tmpl.create('./code/-tmpl/pkg', async (e) => { + const pkg = e.ctx?.pkg as t.Pkg; + if (typeof pkg !== 'object') { + const err = `[UpdatePackages] Template expected a {pkg} on the context. Module: ${e.tmpl.absolute}`; + errors.push(err); + return; + } + + if (e.target.file.name === 'pkg.ts') { + const text = e.text.tmpl.replace(/<NAME>/, pkg.name).replace(/<VERSION>/, pkg.version); + e.modify(text); + } + }); + + for (const item of ws.children) { + const targetDir = Fs.join(item.path.dir, 'src'); + const exists = await Fs.exists(Fs.join(targetDir, 'pkg.ts')); + if (exists) { + const pkg = item.pkg; + const ctx = { pkg }; + await tmpl.write(targetDir, { ctx }); + } + } + + const error = errors.toError(); + if (error) console.error(error); + return { error }; +} + +/** + * Run `prep` ā `init` commands on sub-modules. + */ +async function prepSubmodules() { + const ws = await DenoFile.workspace(); + for (const item of ws.children) { + const tasks = item.denofile.tasks; + if (tasks) { + const sh = Process.sh(item.path.dir); + if (tasks.prep) await sh.run('deno task prep'); + if (tasks.init) await sh.run('deno task init'); + } + } +} + +/** + * Prepare the [deno.json | package.json] files from + * definitions within the monorepo's `deps.yaml` configuration. + */ +export async function main() { + await processDeps(); + await updatePackages(); + await prepSubmodules(); +} diff --git a/scripts/Task.-test.ts b/scripts/Task.-test.ts index 44a5545dca..89a1438141 100644 --- a/scripts/Task.-test.ts +++ b/scripts/Task.-test.ts @@ -9,11 +9,14 @@ export async function main() { */ const results: CmdResult[] = []; const run = async (path: string, index: number, total: number) => { - const command = `deno task test`; + const cmd = 'test'; + const command = `deno task ${cmd}`; + const commandFmt = c.green(`deno task ${c.bold(c.cyan(cmd))}`); + const title = c.gray(`${c.white('Tests')} (${c.white(String(index + 1))} of ${total})`); const moduleList = Log.moduleList({ index, indent: 3 }); - spinner.text = c.gray(`${title}\n ${c.cyan(command)}\n${moduleList}`); - const output = await Process.sh({ silent: true, path }).run(command); + spinner.text = c.gray(`${title}\n ${commandFmt}\n${moduleList}`); + const output = await Process.sh({ path, silent: true }).run(command); results.push({ output, path }); }; diff --git a/scripts/Task.-tmpl.ts b/scripts/Task.-tmpl.ts new file mode 100644 index 0000000000..4e615832a8 --- /dev/null +++ b/scripts/Task.-tmpl.ts @@ -0,0 +1,65 @@ +import { c, Cli, Fs, Tmpl } from './common.ts'; + +type Options = { + argv?: string[]; +}; + +const Templates = { + 'm.mod': 'code/-tmpl/m.mod/', + 'm.mod.ui': 'code/-tmpl/m.mod.ui/', + 'pkg.deno': 'code/-tmpl/deno/', +} as const; + +type TArgs = { + tmpl?: string | boolean; +}; + +/** + * COMMAND š³ Create from template action. + */ +export async function main(options: Options = {}) { + const args = Cli.args<TArgs>(options.argv ?? Deno.args); + console.info(c.gray('args:'), args); + console.info(); + + let name = typeof args.tmpl === 'string' ? args.tmpl : ''; + const templates = Object.keys(Templates); + + if (!name) { + name = await Cli.Prompt.Select.prompt({ + message: 'Select Template:', + options: templates.map((name: string) => ({ name, value: name })), + }); + } + + if (!templates.includes(name)) { + const msg = `${c.yellow('Failed:')} A template named "${c.white(name)}" does not exist.`; + console.info(); + console.warn(c.gray(msg)); + console.info(c.gray(c.italic('(pass nothing for interactive list)'))); + console.info(); + return; + } + + const dirname = await Cli.Prompt.Input.prompt('Directory Name:'); + const targetDir = Fs.join(Fs.cwd('init'), dirname); + + if (await Fs.exists(targetDir)) { + const noChange = c.green('No Change'); + const msg = `${c.yellow('Warning:')} Something already exists at that location (${noChange}).`; + console.info(); + console.warn(c.gray(msg)); + console.warn(c.gray(targetDir)); + console.info(); + return; + } + + const sourceDir = Fs.resolve(Templates[name as keyof typeof Templates]); + const tmpl = Tmpl.create(sourceDir); + const res = await tmpl.write(targetDir); + + console.info(c.gray(`Target: ${Fs.trimCwd(targetDir)}`)); + console.info(); + console.info(Tmpl.Log.table(res.ops)); + console.info(); +} diff --git a/scripts/common.ts b/scripts/common.ts index b78185f6d9..55e4577d6e 100644 --- a/scripts/common.ts +++ b/scripts/common.ts @@ -1,10 +1,11 @@ -export { R, Value } from '@sys/std'; +export { Err, R, Value } from '@sys/std'; export { Semver } from '@sys/std/semver/server'; export { c, Cli } from '@sys/cli'; +export { DenoDeps, DenoFile } from '@sys/driver-deno/runtime'; export { Fs } from '@sys/fs'; export { Process } from '@sys/process'; -export { DenoDeps } from '@sys/driver-deno/runtime'; +export { Tmpl } from '@sys/tmpl/fs'; export * as t from './t.ts'; export { Path, Paths } from './u.paths.ts'; diff --git a/scripts/main.ts b/scripts/main.ts index e17b8172bb..ea625b3063 100644 --- a/scripts/main.ts +++ b/scripts/main.ts @@ -7,28 +7,35 @@ import { main as info } from './Task.-info.ts'; import { main as lint } from './Task.-lint.ts'; import { main as prep } from './Task.-prep.ts'; import { main as test } from './Task.-test.ts'; +import { main as tmpl } from './Task.-tmpl.ts'; type T = { dry?: boolean; test?: boolean; - clean?: boolean; info?: boolean; + + clean?: boolean; lint?: boolean; bump?: boolean; prep?: boolean; + + tmpl?: boolean; }; const args = Cli.args<T>(Deno.args); -// Maintenance. +// CI: +if (args.dry) await dry(); +if (args.test) await test(); +if (args.info) await info(); + +// Maintenance: if (args.clean) await clean(); if (args.lint) await lint(); if (args.bump) await bump(); if (args.prep) await prep(); -// CI. -if (args.dry) await dry(); -if (args.test) await test(); -if (args.info) await info(); +// Development: +if (args.tmpl) await tmpl(); // Finish up. Deno.exit(0); diff --git a/scripts/t.ts b/scripts/t.ts index 7c9f8f93d4..2dc734b4f2 100644 --- a/scripts/t.ts +++ b/scripts/t.ts @@ -1,3 +1,3 @@ export type * from '@sys/types/t'; export type { ProcOutput } from '@sys/process/t'; -export type { DenoFileJson } from '@sys/driver-deno/t'; +export type { DenoFileJson, DenoWorkspace } from '@sys/driver-deno/t'; diff --git a/scripts/u.paths.ts b/scripts/u.paths.ts index 13191b843e..8216613e90 100644 --- a/scripts/u.paths.ts +++ b/scripts/u.paths.ts @@ -5,7 +5,7 @@ export const Paths = { workspace: denojson.workspace, modules: [ /** - * @sys: standard libs. + * @sys: standard-libs: */ 'code/sys/types', 'code/sys/std', @@ -29,7 +29,7 @@ export const Paths = { 'code/sys.tmp', /** - * UI + * UI: */ 'code/sys.ui/ui-css', 'code/sys.ui/ui-dom', @@ -38,7 +38,7 @@ export const Paths = { 'code/sys.ui/ui-react-components', /** - * Drivers + * Drivers: */ // 'code/sys.driver/driver-automerge', 'code/sys.driver/driver-deno', @@ -53,9 +53,14 @@ export const Paths = { 'code/sys.driver/driver-quilibrium', /** - * Barrels + * Barrels: */ 'code/sys/sys', 'code/sys/main', + + /** + * Instance Apps: + */ + 'deploy/@tdb.slc', ], } as const;