@e280/sly 0.2.0-3 → 0.2.0-30
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +624 -97
- package/package.json +13 -6
- package/s/base/element.ts +76 -0
- package/s/base/index.ts +6 -0
- package/s/{views → base}/use.ts +25 -16
- package/s/base/utils/attr-watcher.ts +22 -0
- package/s/base/utils/reactor.ts +32 -0
- package/s/base/utils/states.ts +49 -0
- package/s/base/utils/use-attrs.ts +36 -0
- package/s/demo/demo.bundle.ts +9 -5
- package/s/demo/views/counter.ts +21 -24
- package/s/demo/views/demo.ts +10 -6
- package/s/demo/views/fastcount.ts +29 -0
- package/s/demo/views/loaders.ts +7 -7
- package/s/dom/attrs/attrs.ts +21 -0
- package/s/dom/attrs/parts/attr-fns.ts +68 -0
- package/s/dom/attrs/parts/attr-proxies.ts +35 -0
- package/s/dom/attrs/parts/attr-spec.ts +29 -0
- package/s/dom/attrs/parts/on-attrs.ts +8 -0
- package/s/dom/dom.ts +22 -38
- package/s/dom/index.ts +4 -0
- package/s/dom/parts/dom-scope.ts +46 -0
- package/s/dom/parts/el.ts +14 -0
- package/s/dom/parts/elmer.ts +38 -0
- package/s/dom/parts/eve.ts +24 -0
- package/s/dom/parts/mk.ts +9 -0
- package/s/dom/parts/queries.ts +26 -0
- package/s/dom/{register.ts → parts/register.ts} +2 -7
- package/s/dom/types.ts +42 -0
- package/s/index.html.ts +4 -2
- package/s/index.ts +7 -19
- package/s/loaders/index.barrel.ts +10 -0
- package/s/loaders/index.ts +4 -0
- package/s/loaders/make.ts +14 -0
- package/s/loaders/mock.ts +11 -0
- package/s/{ops/loaders → loaders}/parts/anims.ts +1 -1
- package/s/{ops/loaders → loaders}/parts/ascii-anim.ts +6 -5
- package/s/{ops/loaders → loaders}/parts/error-display.ts +2 -2
- package/s/loaders/types.ts +6 -0
- package/s/loot/drag-and-drops.ts +82 -0
- package/s/loot/{drop.ts → drops.ts} +8 -17
- package/s/loot/helpers.ts +3 -3
- package/s/loot/index.barrel.ts +5 -0
- package/s/loot/index.ts +2 -3
- package/s/ops/index.ts +5 -0
- package/s/ops/op.ts +1 -0
- package/s/spa/index.barrel.ts +6 -0
- package/s/spa/index.ts +4 -0
- package/s/spa/plumbing/braces.ts +76 -0
- package/s/spa/plumbing/primitives.ts +85 -0
- package/s/spa/plumbing/router-core.ts +49 -0
- package/s/spa/plumbing/types.ts +45 -0
- package/s/spa/router.ts +49 -0
- package/s/spa/spa.test.ts +91 -0
- package/s/tests.test.ts +4 -1
- package/s/view/index.ts +7 -0
- package/s/view/types.ts +39 -0
- package/s/view/utils/contextualize.ts +45 -0
- package/s/view/utils/make-component.ts +34 -0
- package/s/view/utils/make-view.ts +48 -0
- package/s/view/utils/parts/capsule.ts +67 -0
- package/s/view/utils/parts/chain.ts +40 -0
- package/s/view/utils/parts/context.ts +11 -0
- package/s/view/utils/parts/directive.ts +29 -0
- package/s/view/utils/parts/sly-view.ts +15 -0
- package/s/view/view.ts +24 -0
- package/x/base/css-reset.js.map +1 -0
- package/x/base/element.d.ts +19 -0
- package/x/base/element.js +52 -0
- package/x/base/element.js.map +1 -0
- package/x/base/index.d.ts +4 -0
- package/x/base/index.js +5 -0
- package/x/base/index.js.map +1 -0
- package/x/{views → base}/use.d.ts +8 -4
- package/x/{views → base}/use.js +15 -9
- package/x/base/use.js.map +1 -0
- package/x/base/utils/apply-styles.js.map +1 -0
- package/x/base/utils/attr-watcher.d.ts +8 -0
- package/x/base/utils/attr-watcher.js +20 -0
- package/x/base/utils/attr-watcher.js.map +1 -0
- package/x/base/utils/mounts.js.map +1 -0
- package/x/base/utils/reactor.d.ts +5 -0
- package/x/base/utils/reactor.js +25 -0
- package/x/base/utils/reactor.js.map +1 -0
- package/x/base/utils/states.d.ts +13 -0
- package/x/base/utils/states.js +41 -0
- package/x/base/utils/states.js.map +1 -0
- package/x/base/utils/use-attrs.d.ts +11 -0
- package/x/base/utils/use-attrs.js +18 -0
- package/x/base/utils/use-attrs.js.map +1 -0
- package/x/demo/demo.bundle.js +8 -4
- package/x/demo/demo.bundle.js.map +1 -1
- package/x/demo/demo.bundle.min.js +19 -22
- package/x/demo/demo.bundle.min.js.map +4 -4
- package/x/demo/views/counter.d.ts +374 -1
- package/x/demo/views/counter.js +19 -22
- package/x/demo/views/counter.js.map +1 -1
- package/x/demo/views/demo.d.ts +4 -1
- package/x/demo/views/demo.js +10 -5
- package/x/demo/views/demo.js.map +1 -1
- package/x/demo/views/fastcount.d.ts +12 -0
- package/x/demo/views/fastcount.js +21 -0
- package/x/demo/views/fastcount.js.map +1 -0
- package/x/demo/views/loaders.js +6 -6
- package/x/demo/views/loaders.js.map +1 -1
- package/x/dom/attrs/attrs.d.ts +23 -0
- package/x/dom/attrs/attrs.js +17 -0
- package/x/dom/attrs/attrs.js.map +1 -0
- package/x/dom/attrs/parts/attr-fns.d.ts +16 -0
- package/x/dom/attrs/parts/attr-fns.js +64 -0
- package/x/dom/attrs/parts/attr-fns.js.map +1 -0
- package/x/dom/attrs/parts/attr-proxies.d.ts +8 -0
- package/x/dom/attrs/parts/attr-proxies.js +21 -0
- package/x/dom/attrs/parts/attr-proxies.js.map +1 -0
- package/x/dom/attrs/parts/attr-spec.d.ts +3 -0
- package/x/dom/attrs/parts/attr-spec.js +21 -0
- package/x/dom/attrs/parts/attr-spec.js.map +1 -0
- package/x/dom/attrs/parts/on-attrs.d.ts +2 -0
- package/x/dom/attrs/parts/on-attrs.js +7 -0
- package/x/dom/attrs/parts/on-attrs.js.map +1 -0
- package/x/dom/dom.d.ts +15 -16
- package/x/dom/dom.js +21 -34
- package/x/dom/dom.js.map +1 -1
- package/x/dom/index.d.ts +2 -0
- package/x/dom/index.js +3 -0
- package/x/dom/index.js.map +1 -0
- package/x/dom/parts/dashify.js.map +1 -0
- package/x/dom/parts/dom-scope.d.ts +15 -0
- package/x/dom/parts/dom-scope.js +35 -0
- package/x/dom/parts/dom-scope.js.map +1 -0
- package/x/dom/parts/el.d.ts +2 -0
- package/x/dom/parts/el.js +7 -0
- package/x/dom/parts/el.js.map +1 -0
- package/x/dom/parts/elmer.d.ts +11 -0
- package/x/dom/parts/elmer.js +32 -0
- package/x/dom/parts/elmer.js.map +1 -0
- package/x/dom/parts/eve.d.ts +7 -0
- package/x/dom/parts/eve.js +16 -0
- package/x/dom/parts/eve.js.map +1 -0
- package/x/dom/parts/mk.d.ts +2 -0
- package/x/dom/parts/mk.js +7 -0
- package/x/dom/parts/mk.js.map +1 -0
- package/x/dom/parts/queries.d.ts +4 -0
- package/x/dom/parts/queries.js +13 -0
- package/x/dom/parts/queries.js.map +1 -0
- package/x/dom/{register.d.ts → parts/register.d.ts} +2 -6
- package/x/dom/parts/register.js.map +1 -0
- package/x/dom/types.d.ts +15 -0
- package/x/index.d.ts +7 -16
- package/x/index.html +6 -4
- package/x/index.html.js +4 -2
- package/x/index.html.js.map +1 -1
- package/x/index.js +7 -16
- package/x/index.js.map +1 -1
- package/x/loaders/index.barrel.d.ts +7 -0
- package/x/loaders/index.barrel.js +7 -0
- package/x/loaders/index.barrel.js.map +1 -0
- package/x/loaders/index.d.ts +2 -0
- package/x/loaders/index.js +2 -0
- package/x/loaders/index.js.map +1 -0
- package/x/loaders/make.d.ts +3 -0
- package/x/loaders/make.js +6 -0
- package/x/loaders/make.js.map +1 -0
- package/x/loaders/mock.d.ts +2 -0
- package/x/loaders/mock.js +8 -0
- package/x/loaders/mock.js.map +1 -0
- package/x/{ops/loaders → loaders}/parts/anims.d.ts +1 -1
- package/x/loaders/parts/anims.js.map +1 -0
- package/x/{ops/loaders → loaders}/parts/ascii-anim.d.ts +2 -2
- package/x/{ops/loaders → loaders}/parts/ascii-anim.js +4 -4
- package/x/loaders/parts/ascii-anim.js.map +1 -0
- package/x/loaders/parts/error-display.d.ts +1 -0
- package/x/{ops/loaders → loaders}/parts/error-display.js +2 -2
- package/x/loaders/parts/error-display.js.map +1 -0
- package/x/loaders/types.d.ts +3 -0
- package/x/loaders/types.js.map +1 -0
- package/x/loot/drag-and-drops.d.ts +30 -0
- package/x/loot/drag-and-drops.js +63 -0
- package/x/loot/drag-and-drops.js.map +1 -0
- package/x/loot/{drop.d.ts → drops.d.ts} +3 -5
- package/x/loot/drops.js +25 -0
- package/x/loot/drops.js.map +1 -0
- package/x/loot/helpers.d.ts +3 -3
- package/x/loot/helpers.js +3 -3
- package/x/loot/helpers.js.map +1 -1
- package/x/loot/index.barrel.d.ts +3 -0
- package/x/loot/index.barrel.js +4 -0
- package/x/loot/index.barrel.js.map +1 -0
- package/x/loot/index.d.ts +2 -3
- package/x/loot/index.js +1 -3
- package/x/loot/index.js.map +1 -1
- package/x/ops/index.d.ts +3 -0
- package/x/ops/index.js +4 -0
- package/x/ops/index.js.map +1 -0
- package/x/ops/op.js +1 -0
- package/x/ops/op.js.map +1 -1
- package/x/spa/index.barrel.d.ts +4 -0
- package/x/spa/index.barrel.js +3 -0
- package/x/spa/index.barrel.js.map +1 -0
- package/x/spa/index.d.ts +2 -0
- package/x/spa/index.js +2 -0
- package/x/spa/index.js.map +1 -0
- package/x/spa/plumbing/braces.d.ts +12 -0
- package/x/spa/plumbing/braces.js +55 -0
- package/x/spa/plumbing/braces.js.map +1 -0
- package/x/spa/plumbing/primitives.d.ts +22 -0
- package/x/spa/plumbing/primitives.js +65 -0
- package/x/spa/plumbing/primitives.js.map +1 -0
- package/x/spa/plumbing/router-core.d.ts +13 -0
- package/x/spa/plumbing/router-core.js +38 -0
- package/x/spa/plumbing/router-core.js.map +1 -0
- package/x/spa/plumbing/types.d.ts +35 -0
- package/x/spa/plumbing/types.js +2 -0
- package/x/spa/plumbing/types.js.map +1 -0
- package/x/spa/router.d.ts +16 -0
- package/x/spa/router.js +39 -0
- package/x/spa/router.js.map +1 -0
- package/x/spa/spa.test.d.ts +15 -0
- package/x/spa/spa.test.js +78 -0
- package/x/spa/spa.test.js.map +1 -0
- package/x/tests.test.js +4 -1
- package/x/tests.test.js.map +1 -1
- package/x/view/index.d.ts +5 -0
- package/x/view/index.js +6 -0
- package/x/view/index.js.map +1 -0
- package/x/view/types.d.ts +21 -0
- package/x/view/types.js +2 -0
- package/x/{views → view}/types.js.map +1 -1
- package/x/view/utils/contextualize.d.ts +13 -0
- package/x/view/utils/contextualize.js +18 -0
- package/x/view/utils/contextualize.js.map +1 -0
- package/x/view/utils/make-component.d.ts +5 -0
- package/x/view/utils/make-component.js +17 -0
- package/x/view/utils/make-component.js.map +1 -0
- package/x/view/utils/make-view.d.ts +2 -0
- package/x/view/utils/make-view.js +24 -0
- package/x/view/utils/make-view.js.map +1 -0
- package/x/view/utils/parts/capsule.d.ts +13 -0
- package/x/view/utils/parts/capsule.js +49 -0
- package/x/view/utils/parts/capsule.js.map +1 -0
- package/x/view/utils/parts/chain.d.ts +13 -0
- package/x/view/utils/parts/chain.js +26 -0
- package/x/view/utils/parts/chain.js.map +1 -0
- package/x/view/utils/parts/context.d.ts +9 -0
- package/x/view/utils/parts/context.js +10 -0
- package/x/view/utils/parts/context.js.map +1 -0
- package/x/view/utils/parts/directive.d.ts +5 -0
- package/x/view/utils/parts/directive.js +18 -0
- package/x/view/utils/parts/directive.js.map +1 -0
- package/x/view/utils/parts/sly-view.d.ts +5 -0
- package/x/view/utils/parts/sly-view.js +13 -0
- package/x/view/utils/parts/sly-view.js.map +1 -0
- package/x/view/view.d.ts +11 -0
- package/x/view/view.js +15 -0
- package/x/view/view.js.map +1 -0
- package/s/loot/drag-drop.ts +0 -76
- package/s/ops/loaders/make-loader.ts +0 -18
- package/s/views/attributes.ts +0 -89
- package/s/views/types.ts +0 -40
- package/s/views/utils/apply-attrs.ts +0 -33
- package/s/views/view.ts +0 -150
- package/x/dom/dashify.js.map +0 -1
- package/x/dom/register.js.map +0 -1
- package/x/loot/drag-drop.d.ts +0 -29
- package/x/loot/drag-drop.js +0 -54
- package/x/loot/drag-drop.js.map +0 -1
- package/x/loot/drop.js +0 -32
- package/x/loot/drop.js.map +0 -1
- package/x/ops/loaders/make-loader.d.ts +0 -5
- package/x/ops/loaders/make-loader.js +0 -7
- package/x/ops/loaders/make-loader.js.map +0 -1
- package/x/ops/loaders/parts/anims.js.map +0 -1
- package/x/ops/loaders/parts/ascii-anim.js.map +0 -1
- package/x/ops/loaders/parts/error-display.d.ts +0 -1
- package/x/ops/loaders/parts/error-display.js.map +0 -1
- package/x/views/attributes.d.ts +0 -10
- package/x/views/attributes.js +0 -46
- package/x/views/attributes.js.map +0 -1
- package/x/views/css-reset.js.map +0 -1
- package/x/views/types.d.ts +0 -31
- package/x/views/use.js.map +0 -1
- package/x/views/utils/apply-attrs.d.ts +0 -2
- package/x/views/utils/apply-attrs.js +0 -21
- package/x/views/utils/apply-attrs.js.map +0 -1
- package/x/views/utils/apply-styles.js.map +0 -1
- package/x/views/utils/mounts.js.map +0 -1
- package/x/views/view.d.ts +0 -9
- package/x/views/view.js +0 -116
- package/x/views/view.js.map +0 -1
- /package/s/{views → base}/css-reset.ts +0 -0
- /package/s/{views → base}/utils/apply-styles.ts +0 -0
- /package/s/{views → base}/utils/mounts.ts +0 -0
- /package/s/dom/{dashify.ts → parts/dashify.ts} +0 -0
- /package/x/{views → base}/css-reset.d.ts +0 -0
- /package/x/{views → base}/css-reset.js +0 -0
- /package/x/{views → base}/utils/apply-styles.d.ts +0 -0
- /package/x/{views → base}/utils/apply-styles.js +0 -0
- /package/x/{views → base}/utils/mounts.d.ts +0 -0
- /package/x/{views → base}/utils/mounts.js +0 -0
- /package/x/dom/{dashify.d.ts → parts/dashify.d.ts} +0 -0
- /package/x/dom/{dashify.js → parts/dashify.js} +0 -0
- /package/x/dom/{register.js → parts/register.js} +0 -0
- /package/x/{ops/loaders → loaders}/parts/anims.js +0 -0
- /package/x/{views → loaders}/types.js +0 -0
package/package.json
CHANGED
|
@@ -1,12 +1,19 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@e280/sly",
|
|
3
|
-
"version": "0.2.0-
|
|
3
|
+
"version": "0.2.0-30",
|
|
4
4
|
"description": "web shadow views",
|
|
5
5
|
"license": "MIT",
|
|
6
6
|
"type": "module",
|
|
7
7
|
"main": "./x/index.js",
|
|
8
8
|
"exports": {
|
|
9
|
-
".": "./x/index.js"
|
|
9
|
+
".": "./x/index.js",
|
|
10
|
+
"./base": "./x/base/index.js",
|
|
11
|
+
"./dom": "./x/dom/index.js",
|
|
12
|
+
"./loaders": "./x/loaders/index.js",
|
|
13
|
+
"./loot": "./x/loot/index.js",
|
|
14
|
+
"./ops": "./x/ops/index.js",
|
|
15
|
+
"./spa": "./x/spa/index.js",
|
|
16
|
+
"./view": "./x/view/index.js"
|
|
10
17
|
},
|
|
11
18
|
"files": [
|
|
12
19
|
"x",
|
|
@@ -16,15 +23,15 @@
|
|
|
16
23
|
"lit": "^3.3.1"
|
|
17
24
|
},
|
|
18
25
|
"dependencies": {
|
|
19
|
-
"@e280/strata": "^0.2.0
|
|
20
|
-
"@e280/stz": "^0.2.
|
|
26
|
+
"@e280/strata": "^0.2.0",
|
|
27
|
+
"@e280/stz": "^0.2.10"
|
|
21
28
|
},
|
|
22
29
|
"devDependencies": {
|
|
23
30
|
"@e280/science": "^0.1.2",
|
|
24
|
-
"@e280/scute": "^0.
|
|
31
|
+
"@e280/scute": "^0.1.1",
|
|
25
32
|
"http-server": "^14.1.1",
|
|
26
33
|
"npm-run-all": "^4.1.5",
|
|
27
|
-
"typescript": "^5.9.
|
|
34
|
+
"typescript": "^5.9.3"
|
|
28
35
|
},
|
|
29
36
|
"scripts": {
|
|
30
37
|
"build": "run-s _clean _ln _tsc _scute",
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
|
|
2
|
+
import {debounce} from "@e280/stz"
|
|
3
|
+
import {CSSResultGroup} from "lit"
|
|
4
|
+
|
|
5
|
+
import {dom} from "../dom/dom.js"
|
|
6
|
+
import {Content} from "../view/types.js"
|
|
7
|
+
import {Reactor} from "./utils/reactor.js"
|
|
8
|
+
import {AttrWatcher} from "./utils/attr-watcher.js"
|
|
9
|
+
import {applyStyles} from "./utils/apply-styles.js"
|
|
10
|
+
import {Use, _disconnect, _reconnect, _wrap} from "./use.js"
|
|
11
|
+
|
|
12
|
+
export class BaseElement extends HTMLElement {
|
|
13
|
+
static styles: CSSResultGroup | undefined
|
|
14
|
+
|
|
15
|
+
readonly shadow: ShadowRoot
|
|
16
|
+
|
|
17
|
+
#use: Use
|
|
18
|
+
#mountCount = 0
|
|
19
|
+
#reactor = new Reactor()
|
|
20
|
+
#attrWatcher = new AttrWatcher(this, () => this.update())
|
|
21
|
+
|
|
22
|
+
/** create the shadow root. override this if you want to change the shadow root settings. */
|
|
23
|
+
createShadow() {
|
|
24
|
+
return this.attachShadow({mode: "open"})
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
constructor() {
|
|
28
|
+
super()
|
|
29
|
+
this.shadow = this.createShadow()
|
|
30
|
+
this.#use = new Use(
|
|
31
|
+
this,
|
|
32
|
+
this.shadow,
|
|
33
|
+
this.updateNow,
|
|
34
|
+
this.update,
|
|
35
|
+
)
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
/** return some content to render. */
|
|
39
|
+
render(_use: Use): Content {}
|
|
40
|
+
|
|
41
|
+
/** immediately perform a fresh render into the shadow root. */
|
|
42
|
+
updateNow = () => {
|
|
43
|
+
this.#use[_wrap](() => {
|
|
44
|
+
dom.render(
|
|
45
|
+
this.shadow,
|
|
46
|
+
this.#reactor.effect(
|
|
47
|
+
() => this.render(this.#use),
|
|
48
|
+
this.update,
|
|
49
|
+
),
|
|
50
|
+
)
|
|
51
|
+
})
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
/** request a rerender which will happen soon (debounced). */
|
|
55
|
+
update = debounce(0, this.updateNow)
|
|
56
|
+
|
|
57
|
+
connectedCallback() {
|
|
58
|
+
if (this.#mountCount === 0) {
|
|
59
|
+
const styles = (this.constructor as any).styles
|
|
60
|
+
if (styles) applyStyles(this.shadow, styles)
|
|
61
|
+
this.updateNow()
|
|
62
|
+
}
|
|
63
|
+
else {
|
|
64
|
+
this.#use[_reconnect]()
|
|
65
|
+
}
|
|
66
|
+
this.#attrWatcher.start()
|
|
67
|
+
this.#mountCount++
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
disconnectedCallback() {
|
|
71
|
+
this.#use[_disconnect]()
|
|
72
|
+
this.#reactor.clear()
|
|
73
|
+
this.#attrWatcher.stop()
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
|
package/s/base/index.ts
ADDED
package/s/{views → base}/use.ts
RENAMED
|
@@ -1,18 +1,22 @@
|
|
|
1
1
|
|
|
2
2
|
import {CSSResultGroup} from "lit"
|
|
3
3
|
import {defer, MapG} from "@e280/stz"
|
|
4
|
-
import {
|
|
4
|
+
import {signal, SignalOptions} from "@e280/strata/signals"
|
|
5
5
|
|
|
6
6
|
import {Op} from "../ops/op.js"
|
|
7
7
|
import {Mounts} from "./utils/mounts.js"
|
|
8
|
+
import {eve, EveSpec} from "../dom/parts/eve.js"
|
|
8
9
|
import {applyStyles} from "./utils/apply-styles.js"
|
|
9
|
-
import {
|
|
10
|
+
import {useAttrs, UseAttrs} from "./utils/use-attrs.js"
|
|
11
|
+
import { States } from "./utils/states.js"
|
|
10
12
|
|
|
11
13
|
export const _wrap = Symbol()
|
|
12
14
|
export const _disconnect = Symbol()
|
|
13
15
|
export const _reconnect = Symbol()
|
|
14
16
|
|
|
15
17
|
export class Use {
|
|
18
|
+
readonly attrs: UseAttrs
|
|
19
|
+
|
|
16
20
|
#runs = 0
|
|
17
21
|
#position = 0
|
|
18
22
|
#values = new MapG<number, any>()
|
|
@@ -37,11 +41,13 @@ export class Use {
|
|
|
37
41
|
}
|
|
38
42
|
|
|
39
43
|
constructor(
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
44
|
+
public element: HTMLElement,
|
|
45
|
+
public shadow: ShadowRoot,
|
|
46
|
+
public renderNow: () => void,
|
|
47
|
+
public render: () => Promise<void>,
|
|
48
|
+
) {
|
|
49
|
+
this.attrs = useAttrs(this)
|
|
50
|
+
}
|
|
45
51
|
|
|
46
52
|
get renderCount() {
|
|
47
53
|
return this.#runs
|
|
@@ -64,11 +70,6 @@ export class Use {
|
|
|
64
70
|
return this.styles(...styles)
|
|
65
71
|
}
|
|
66
72
|
|
|
67
|
-
attrs<A extends AttrSpec>(spec: A) {
|
|
68
|
-
this.mount(() => onAttrChange(this.element, this.render))
|
|
69
|
-
return this.once(() => attributes(this.element, spec))
|
|
70
|
-
}
|
|
71
|
-
|
|
72
73
|
once<V>(fn: () => V) {
|
|
73
74
|
return this.#values.guarantee(this.#position++, fn) as V
|
|
74
75
|
}
|
|
@@ -91,6 +92,14 @@ export class Use {
|
|
|
91
92
|
return this.life(() => [fn(), () => {}])
|
|
92
93
|
}
|
|
93
94
|
|
|
95
|
+
events(spec: EveSpec) {
|
|
96
|
+
return this.mount(() => eve(this.element, spec))
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
states() {
|
|
100
|
+
return this.once(() => new States(this.element))
|
|
101
|
+
}
|
|
102
|
+
|
|
94
103
|
op = (() => {
|
|
95
104
|
const that = this
|
|
96
105
|
function op<V>(f: () => Promise<V>) {
|
|
@@ -106,8 +115,8 @@ export class Use {
|
|
|
106
115
|
function sig<V>(value: V, options?: Partial<SignalOptions>) {
|
|
107
116
|
return that.once(() => signal<V>(value, options))
|
|
108
117
|
}
|
|
109
|
-
sig.
|
|
110
|
-
return that.once(() => signal.
|
|
118
|
+
sig.derived = function derived<V>(formula: () => V, options?: Partial<SignalOptions>) {
|
|
119
|
+
return that.once(() => signal.derived<V>(formula, options))
|
|
111
120
|
}
|
|
112
121
|
sig.lazy = function lazy<V>(formula: () => V, options?: Partial<SignalOptions>) {
|
|
113
122
|
return that.once(() => signal.lazy<V>(formula, options))
|
|
@@ -115,8 +124,8 @@ export class Use {
|
|
|
115
124
|
return sig
|
|
116
125
|
})()
|
|
117
126
|
|
|
118
|
-
|
|
119
|
-
return this.once(() => signal.
|
|
127
|
+
derived<V>(formula: () => V, options?: Partial<SignalOptions>) {
|
|
128
|
+
return this.once(() => signal.derived<V>(formula, options))
|
|
120
129
|
}
|
|
121
130
|
|
|
122
131
|
lazy<V>(formula: () => V, options?: Partial<SignalOptions>) {
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
|
|
2
|
+
import {dom} from "../../dom/dom.js"
|
|
3
|
+
|
|
4
|
+
export class AttrWatcher {
|
|
5
|
+
#stopper: (() => void) | undefined
|
|
6
|
+
|
|
7
|
+
constructor(
|
|
8
|
+
private element: HTMLElement,
|
|
9
|
+
private response: () => void,
|
|
10
|
+
) {}
|
|
11
|
+
|
|
12
|
+
start() {
|
|
13
|
+
if (!this.#stopper)
|
|
14
|
+
this.#stopper = dom.attrs(this.element).on(this.response)
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
stop() {
|
|
18
|
+
if (this.#stopper) this.#stopper()
|
|
19
|
+
this.#stopper = undefined
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
|
|
2
|
+
import {MapG} from "@e280/stz"
|
|
3
|
+
import {tracker} from "@e280/strata"
|
|
4
|
+
|
|
5
|
+
export class Reactor {
|
|
6
|
+
#map = new MapG<any, () => void>()
|
|
7
|
+
|
|
8
|
+
effect<R>(collect: () => R, respond: () => Promise<void>) {
|
|
9
|
+
const {seen, result} = tracker.observe(collect)
|
|
10
|
+
|
|
11
|
+
// add seen items
|
|
12
|
+
for (const item of seen)
|
|
13
|
+
this.#map.guarantee(item, () => tracker.subscribe(item, respond))
|
|
14
|
+
|
|
15
|
+
// remove orphaned items
|
|
16
|
+
for (const [item, dispose] of this.#map) {
|
|
17
|
+
if (!seen.has(item)) {
|
|
18
|
+
dispose()
|
|
19
|
+
this.#map.delete(item)
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
return result
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
clear() {
|
|
27
|
+
for (const dispose of this.#map.values())
|
|
28
|
+
dispose()
|
|
29
|
+
this.#map.clear()
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
|
|
2
|
+
export class States<S extends string = string> {
|
|
3
|
+
#states: Set<S>
|
|
4
|
+
|
|
5
|
+
constructor(element: HTMLElement) {
|
|
6
|
+
this.#states = element.attachInternals().states as any
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
;[Symbol.iterator]() {
|
|
10
|
+
return this.#states.values()
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
values() {
|
|
14
|
+
return this.#states.values()
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
forEach(fn: (s: S) => void) {
|
|
18
|
+
this.#states.forEach(fn)
|
|
19
|
+
return this
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
get size() {
|
|
23
|
+
return this.#states.size
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
has(state: S) {
|
|
27
|
+
return this.#states.has(state)
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
add(...states: S[]) {
|
|
31
|
+
for (const s of states) this.#states.add(s)
|
|
32
|
+
return this
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
delete(...states: S[]) {
|
|
36
|
+
for (const s of states) this.#states.delete(s)
|
|
37
|
+
return this
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
clear() {
|
|
41
|
+
this.#states.clear()
|
|
42
|
+
return this
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
assign(...states: S[]) {
|
|
46
|
+
return this.clear().add(...states)
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
|
|
2
|
+
import {Use} from "../use.js"
|
|
3
|
+
import {dom} from "../../dom/dom.js"
|
|
4
|
+
import {AttrSpec, AttrTypes} from "../../dom/types.js"
|
|
5
|
+
|
|
6
|
+
export type UseAttrs = {
|
|
7
|
+
<A extends AttrSpec>(spec: A): AttrTypes<A>
|
|
8
|
+
strings: Record<string, undefined | string>
|
|
9
|
+
numbers: Record<string, undefined | number>
|
|
10
|
+
booleans: Record<string, undefined | boolean>
|
|
11
|
+
spec<A extends AttrSpec>(spec: A): AttrTypes<A>
|
|
12
|
+
on(fn: () => void): void
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export function useAttrs(use: Use): UseAttrs {
|
|
16
|
+
const attrs = dom.attrs(use.element)
|
|
17
|
+
|
|
18
|
+
function attrsFn<A extends AttrSpec>(spec: A) {
|
|
19
|
+
return use.once(() => attrs.spec<A>(spec))
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
attrsFn.strings = attrs.strings
|
|
23
|
+
attrsFn.numbers = attrs.numbers
|
|
24
|
+
attrsFn.booleans = attrs.booleans
|
|
25
|
+
|
|
26
|
+
attrsFn.spec = <A extends AttrSpec>(spec: A) => {
|
|
27
|
+
return use.once(() => attrs.spec<A>(spec))
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
attrsFn.on = (fn: () => void) => {
|
|
31
|
+
return use.mount(() => attrs.on(fn))
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
return attrsFn
|
|
35
|
+
}
|
|
36
|
+
|
package/s/demo/demo.bundle.ts
CHANGED
|
@@ -1,10 +1,14 @@
|
|
|
1
1
|
|
|
2
2
|
import {dom} from "../dom/dom.js"
|
|
3
|
-
import {
|
|
4
|
-
import {
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
dom.register({
|
|
3
|
+
import {CounterComponent} from "./views/counter.js"
|
|
4
|
+
import {DemoComponent} from "./views/demo.js"
|
|
5
|
+
import {FastcountElement} from "./views/fastcount.js"
|
|
6
|
+
|
|
7
|
+
dom.register({
|
|
8
|
+
DemoComponent,
|
|
9
|
+
CounterComponent,
|
|
10
|
+
FastcountElement,
|
|
11
|
+
})
|
|
8
12
|
|
|
9
13
|
console.log("🦝 sly")
|
|
10
14
|
|
package/s/demo/views/counter.ts
CHANGED
|
@@ -1,44 +1,41 @@
|
|
|
1
1
|
|
|
2
2
|
import {css, html} from "lit"
|
|
3
|
-
import {repeat} from "@e280/stz"
|
|
4
3
|
|
|
5
|
-
import {
|
|
6
|
-
import {
|
|
4
|
+
import {dom} from "../../dom/dom.js"
|
|
5
|
+
import {view} from "../../view/view.js"
|
|
6
|
+
import {cssReset} from "../../base/css-reset.js"
|
|
7
|
+
import {BaseElement} from "../../base/element.js"
|
|
7
8
|
|
|
8
|
-
export const CounterView = view(use => (
|
|
9
|
+
export const CounterView = view(use => (start: number, step: number) => {
|
|
9
10
|
use.name("counter")
|
|
10
11
|
use.styles(cssReset, styles)
|
|
11
12
|
|
|
12
|
-
const $
|
|
13
|
-
const
|
|
14
|
-
use.mount(() => repeat(async() => {
|
|
15
|
-
const since = Date.now() - start
|
|
16
|
-
$seconds.set(Math.floor(since / 1000))
|
|
17
|
-
}))
|
|
18
|
-
|
|
19
|
-
const $count = use.signal(initial)
|
|
20
|
-
const increment = () => $count.value++
|
|
21
|
-
|
|
22
|
-
const $product = use.signal
|
|
23
|
-
.derive(() => $count() * $seconds())
|
|
13
|
+
const $count = use.signal(start)
|
|
14
|
+
const increment = () => { $count.value += step }
|
|
24
15
|
|
|
25
16
|
return html`
|
|
26
17
|
<slot></slot>
|
|
27
18
|
<div>
|
|
28
|
-
<span>${$
|
|
29
|
-
</div>
|
|
30
|
-
<div>
|
|
31
|
-
<span>${$count.get()}</span>
|
|
19
|
+
<span>${$count()}</span>
|
|
32
20
|
</div>
|
|
33
21
|
<div>
|
|
34
|
-
<
|
|
35
|
-
</div>
|
|
36
|
-
<div>
|
|
37
|
-
<button @click="${increment}">+</button>
|
|
22
|
+
<button @click="${increment}">++</button>
|
|
38
23
|
</div>
|
|
39
24
|
`
|
|
40
25
|
})
|
|
41
26
|
|
|
27
|
+
// convert a view into a web component
|
|
28
|
+
export class CounterComponent extends (
|
|
29
|
+
CounterView
|
|
30
|
+
.component(class extends BaseElement {
|
|
31
|
+
attrs = dom.attrs(this).spec({
|
|
32
|
+
start: Number,
|
|
33
|
+
step: Number,
|
|
34
|
+
})
|
|
35
|
+
})
|
|
36
|
+
.props(c => [c.attrs.start ?? 0, c.attrs.step ?? 1])
|
|
37
|
+
) {}
|
|
38
|
+
|
|
42
39
|
const styles = css`
|
|
43
40
|
:host {
|
|
44
41
|
display: flex;
|
package/s/demo/views/demo.ts
CHANGED
|
@@ -1,24 +1,28 @@
|
|
|
1
1
|
|
|
2
2
|
import {css, html} from "lit"
|
|
3
|
-
|
|
3
|
+
|
|
4
|
+
import {view} from "../../view/view.js"
|
|
4
5
|
import {CounterView} from "./counter.js"
|
|
5
6
|
import {LoadersView} from "./loaders.js"
|
|
6
|
-
import {cssReset} from "../../
|
|
7
|
+
import {cssReset} from "../../base/css-reset.js"
|
|
7
8
|
|
|
8
|
-
export
|
|
9
|
+
export class DemoComponent extends (view.component(use => {
|
|
9
10
|
use.name("demo")
|
|
10
11
|
use.styles(cssReset, styles)
|
|
11
|
-
|
|
12
12
|
return html`
|
|
13
|
-
${CounterView
|
|
13
|
+
${CounterView
|
|
14
|
+
.props(768, 3)
|
|
15
|
+
.children("view")
|
|
16
|
+
.render()}
|
|
14
17
|
${LoadersView()}
|
|
15
18
|
`
|
|
16
|
-
})
|
|
19
|
+
})) {}
|
|
17
20
|
|
|
18
21
|
const styles = css`
|
|
19
22
|
:host {
|
|
20
23
|
display: flex;
|
|
21
24
|
flex-direction: column;
|
|
25
|
+
align-items: center;
|
|
22
26
|
gap: 1em;
|
|
23
27
|
}
|
|
24
28
|
`
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
|
|
2
|
+
import {css, html} from "lit"
|
|
3
|
+
import {nap, cycle} from "@e280/stz"
|
|
4
|
+
|
|
5
|
+
import {dom} from "../../dom/dom.js"
|
|
6
|
+
import {Use} from "../../base/use.js"
|
|
7
|
+
import {BaseElement} from "../../base/element.js"
|
|
8
|
+
|
|
9
|
+
export class FastcountElement extends BaseElement {
|
|
10
|
+
static styles = css`span{color:orange}`
|
|
11
|
+
|
|
12
|
+
attrs = dom.attrs(this).spec({value: Number})
|
|
13
|
+
something = {whatever: "rofl"}
|
|
14
|
+
|
|
15
|
+
render(use: Use) {
|
|
16
|
+
const {value = 1} = this.attrs
|
|
17
|
+
const $count = use.signal(0)
|
|
18
|
+
|
|
19
|
+
use.mount(() => cycle(async() => {
|
|
20
|
+
await nap(10)
|
|
21
|
+
await $count($count() + 1)
|
|
22
|
+
}))
|
|
23
|
+
|
|
24
|
+
return html`
|
|
25
|
+
<span>${$count() * value}</span>
|
|
26
|
+
`
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
|
package/s/demo/views/loaders.ts
CHANGED
|
@@ -1,9 +1,9 @@
|
|
|
1
1
|
|
|
2
2
|
import {css, html} from "lit"
|
|
3
3
|
import {Op} from "../../ops/op.js"
|
|
4
|
-
import {view} from "../../
|
|
5
|
-
import {
|
|
6
|
-
import {
|
|
4
|
+
import {view} from "../../view/view.js"
|
|
5
|
+
import {loaders} from "../../loaders/index.js"
|
|
6
|
+
import {cssReset} from "../../base/css-reset.js"
|
|
7
7
|
|
|
8
8
|
export const LoadersView = view(use => () => {
|
|
9
9
|
use.name("loaders")
|
|
@@ -11,14 +11,14 @@ export const LoadersView = view(use => () => {
|
|
|
11
11
|
|
|
12
12
|
const op = use.once(() => Op.loading())
|
|
13
13
|
|
|
14
|
-
const
|
|
15
|
-
Object.entries(anims).map(([key, anim]) => ({
|
|
14
|
+
const library = use.once(() =>
|
|
15
|
+
Object.entries(loaders.anims).map(([key, anim]) => ({
|
|
16
16
|
key,
|
|
17
|
-
loader:
|
|
17
|
+
loader: loaders.make(anim)
|
|
18
18
|
}))
|
|
19
19
|
)
|
|
20
20
|
|
|
21
|
-
return
|
|
21
|
+
return library.map(({key, loader}) => html`
|
|
22
22
|
<div data-anim="${key}">
|
|
23
23
|
<span>${key}</span>
|
|
24
24
|
<span>${loader(op, () => null)}</span>
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
|
|
2
|
+
import {AttrSpec} from "../types.js"
|
|
3
|
+
import {onAttrs} from "./parts/on-attrs.js"
|
|
4
|
+
import {attrSpec} from "./parts/attr-spec.js"
|
|
5
|
+
import {AttrProxies} from "./parts/attr-proxies.js"
|
|
6
|
+
import {attrGet, attrSet} from "./parts/attr-fns.js"
|
|
7
|
+
|
|
8
|
+
export function attrs(element: HTMLElement) {
|
|
9
|
+
const proxies = new AttrProxies(element)
|
|
10
|
+
return {
|
|
11
|
+
strings: proxies.strings,
|
|
12
|
+
numbers: proxies.numbers,
|
|
13
|
+
booleans: proxies.booleans,
|
|
14
|
+
on: (fn: () => void) => onAttrs(element, fn),
|
|
15
|
+
spec: <A extends AttrSpec>(spec: A) => attrSpec(element, spec),
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
attrs.get = attrGet
|
|
20
|
+
attrs.set = attrSet
|
|
21
|
+
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
|
|
2
|
+
import {AttrValue} from "../../types.js"
|
|
3
|
+
|
|
4
|
+
/** get html attributes */
|
|
5
|
+
export const attrGet = {
|
|
6
|
+
string: (e: HTMLElement, key: string) => {
|
|
7
|
+
return e.getAttribute(key) ?? undefined
|
|
8
|
+
},
|
|
9
|
+
number: (e: HTMLElement, key: string) => {
|
|
10
|
+
const raw = e.getAttribute(key)
|
|
11
|
+
return (raw === null || !raw)
|
|
12
|
+
? undefined
|
|
13
|
+
: Number(raw)
|
|
14
|
+
},
|
|
15
|
+
boolean: (e: HTMLElement, key: string) => {
|
|
16
|
+
const raw = e.getAttribute(key)
|
|
17
|
+
return raw !== null
|
|
18
|
+
},
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
/** set html attributes */
|
|
22
|
+
export const attrSet = {
|
|
23
|
+
string: (e: HTMLElement, key: string, value: string | undefined) => {
|
|
24
|
+
if (value === undefined) e.removeAttribute(key)
|
|
25
|
+
else e.setAttribute(key, value)
|
|
26
|
+
return true
|
|
27
|
+
},
|
|
28
|
+
number: (e: HTMLElement, key: string, value: number | undefined) => {
|
|
29
|
+
if (value === undefined) e.removeAttribute(key)
|
|
30
|
+
else e.setAttribute(key, value.toString())
|
|
31
|
+
return true
|
|
32
|
+
},
|
|
33
|
+
boolean: (e: HTMLElement, key: string, value: boolean | undefined) => {
|
|
34
|
+
if (value) e.setAttribute(key, "")
|
|
35
|
+
else e.removeAttribute(key)
|
|
36
|
+
return true
|
|
37
|
+
},
|
|
38
|
+
|
|
39
|
+
any: (element: HTMLElement, key: string, value: AttrValue) => {
|
|
40
|
+
if (value === undefined || value === null)
|
|
41
|
+
element.removeAttribute(key)
|
|
42
|
+
|
|
43
|
+
else if (typeof value === "string")
|
|
44
|
+
element.setAttribute(key, value)
|
|
45
|
+
|
|
46
|
+
else if (typeof value === "number")
|
|
47
|
+
element.setAttribute(key, value.toString())
|
|
48
|
+
|
|
49
|
+
else if (typeof value === "boolean") {
|
|
50
|
+
if (value === true)
|
|
51
|
+
element.setAttribute(key, "")
|
|
52
|
+
else
|
|
53
|
+
element.removeAttribute(key)
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
else console.warn(`invalid attribute "${key}" type is "${typeof value}"`)
|
|
57
|
+
},
|
|
58
|
+
|
|
59
|
+
entries: (element: HTMLElement, entries: Iterable<[key: string, value: AttrValue]>) => {
|
|
60
|
+
for (const [key, value] of entries)
|
|
61
|
+
attrSet.any(element, key, value)
|
|
62
|
+
},
|
|
63
|
+
|
|
64
|
+
record: (element: HTMLElement, record: Record<string, AttrValue>) => {
|
|
65
|
+
return attrSet.entries(element, Object.entries(record))
|
|
66
|
+
},
|
|
67
|
+
}
|
|
68
|
+
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
|
|
2
|
+
import {attrGet, attrSet} from "./attr-fns.js"
|
|
3
|
+
|
|
4
|
+
/** a typed proxy accessor for html attributes */
|
|
5
|
+
export class AttrProxies {
|
|
6
|
+
constructor(public element: HTMLElement) {}
|
|
7
|
+
|
|
8
|
+
strings = new Proxy({}, {
|
|
9
|
+
get: (_t, key: string) => (
|
|
10
|
+
attrGet.string(this.element, key)
|
|
11
|
+
),
|
|
12
|
+
set: (_t, key: string, value: string | undefined) => (
|
|
13
|
+
attrSet.string(this.element, key, value)
|
|
14
|
+
),
|
|
15
|
+
}) as Record<string, string | undefined>
|
|
16
|
+
|
|
17
|
+
numbers = new Proxy({}, {
|
|
18
|
+
get: (_t, key: string) => (
|
|
19
|
+
attrGet.number(this.element, key)
|
|
20
|
+
),
|
|
21
|
+
set: (_t, key: string, value: number | undefined) => (
|
|
22
|
+
attrSet.number(this.element, key, value)
|
|
23
|
+
),
|
|
24
|
+
}) as Record<string, number | undefined>
|
|
25
|
+
|
|
26
|
+
booleans = new Proxy({}, {
|
|
27
|
+
get: (_t, key: string) => (
|
|
28
|
+
attrGet.boolean(this.element, key)
|
|
29
|
+
),
|
|
30
|
+
set: (_t, key: string, value: boolean | undefined) => (
|
|
31
|
+
attrSet.boolean(this.element, key, value)
|
|
32
|
+
),
|
|
33
|
+
}) as Record<string, boolean | undefined>
|
|
34
|
+
}
|
|
35
|
+
|