lego-dom 0.0.7 → 0.0.9
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/.github/workflows/deploy-docs.yml +56 -0
- package/LICENSE +21 -0
- package/README.md +52 -314
- package/docs/.vitepress/config.js +107 -0
- package/docs/.vitepress/dist/404.html +22 -0
- package/docs/.vitepress/dist/api/define.html +35 -0
- package/docs/.vitepress/dist/api/directives.html +32 -0
- package/docs/.vitepress/dist/api/globals.html +27 -0
- package/docs/.vitepress/dist/api/index.html +25 -0
- package/docs/.vitepress/dist/api/lifecycle.html +38 -0
- package/docs/.vitepress/dist/api/route.html +34 -0
- package/docs/.vitepress/dist/api/vite-plugin.html +37 -0
- package/docs/.vitepress/dist/assets/api_define.md.UA-ygUnQ.js +11 -0
- package/docs/.vitepress/dist/assets/api_define.md.UA-ygUnQ.lean.js +1 -0
- package/docs/.vitepress/dist/assets/api_directives.md.BV-D251p.js +8 -0
- package/docs/.vitepress/dist/assets/api_directives.md.BV-D251p.lean.js +1 -0
- package/docs/.vitepress/dist/assets/api_globals.md.CEznyRAY.js +3 -0
- package/docs/.vitepress/dist/assets/api_globals.md.CEznyRAY.lean.js +1 -0
- package/docs/.vitepress/dist/assets/api_index.md.IEYUxUIr.js +1 -0
- package/docs/.vitepress/dist/assets/api_index.md.IEYUxUIr.lean.js +1 -0
- package/docs/.vitepress/dist/assets/api_lifecycle.md.Ccm5xw6-.js +14 -0
- package/docs/.vitepress/dist/assets/api_lifecycle.md.Ccm5xw6-.lean.js +1 -0
- package/docs/.vitepress/dist/assets/api_route.md.CAHf_KNp.js +10 -0
- package/docs/.vitepress/dist/assets/api_route.md.CAHf_KNp.lean.js +1 -0
- package/docs/.vitepress/dist/assets/api_vite-plugin.md.DC8Li09k.js +13 -0
- package/docs/.vitepress/dist/assets/api_vite-plugin.md.DC8Li09k.lean.js +1 -0
- package/docs/.vitepress/dist/assets/app.BfblNDJy.js +1 -0
- package/docs/.vitepress/dist/assets/chunks/@localSearchIndexroot.Crdp7-Zp.js +1 -0
- package/docs/.vitepress/dist/assets/chunks/VPLocalSearchBox.C18E44rY.js +9 -0
- package/docs/.vitepress/dist/assets/chunks/framework.B7OFBR9X.js +19 -0
- package/docs/.vitepress/dist/assets/chunks/theme.VX3itTW6.js +2 -0
- package/docs/.vitepress/dist/assets/examples_form.md.DQoAgbLR.js +34 -0
- package/docs/.vitepress/dist/assets/examples_form.md.DQoAgbLR.lean.js +1 -0
- package/docs/.vitepress/dist/assets/examples_index.md.CVJJjXXE.js +28 -0
- package/docs/.vitepress/dist/assets/examples_index.md.CVJJjXXE.lean.js +1 -0
- package/docs/.vitepress/dist/assets/examples_routing.md.sRnA5RXw.js +338 -0
- package/docs/.vitepress/dist/assets/examples_routing.md.sRnA5RXw.lean.js +1 -0
- package/docs/.vitepress/dist/assets/examples_sfc-showcase.md.DPf9Wm99.js +13 -0
- package/docs/.vitepress/dist/assets/examples_sfc-showcase.md.DPf9Wm99.lean.js +1 -0
- package/docs/.vitepress/dist/assets/examples_todo-app.md.CqF4JaWn.js +297 -0
- package/docs/.vitepress/dist/assets/examples_todo-app.md.CqF4JaWn.lean.js +1 -0
- package/docs/.vitepress/dist/assets/guide_cdn-usage.md.CjIjusre.js +182 -0
- package/docs/.vitepress/dist/assets/guide_cdn-usage.md.CjIjusre.lean.js +1 -0
- package/docs/.vitepress/dist/assets/guide_components.md.CMU3iM6R.js +174 -0
- package/docs/.vitepress/dist/assets/guide_components.md.CMU3iM6R.lean.js +1 -0
- package/docs/.vitepress/dist/assets/guide_contributing.md.Crrv3T_0.js +1 -0
- package/docs/.vitepress/dist/assets/guide_contributing.md.Crrv3T_0.lean.js +1 -0
- package/docs/.vitepress/dist/assets/guide_directives.md.DFwqvqOv.js +140 -0
- package/docs/.vitepress/dist/assets/guide_directives.md.DFwqvqOv.lean.js +1 -0
- package/docs/.vitepress/dist/assets/guide_getting-started.md.DtaJPe0i.js +107 -0
- package/docs/.vitepress/dist/assets/guide_getting-started.md.DtaJPe0i.lean.js +1 -0
- package/docs/.vitepress/dist/assets/guide_index.md.DtJVpLI9.js +2 -0
- package/docs/.vitepress/dist/assets/guide_index.md.DtJVpLI9.lean.js +1 -0
- package/docs/.vitepress/dist/assets/guide_lifecycle.md.CfY3jlU1.js +304 -0
- package/docs/.vitepress/dist/assets/guide_lifecycle.md.CfY3jlU1.lean.js +1 -0
- package/docs/.vitepress/dist/assets/guide_quick-start.md.CwdNNA21.js +33 -0
- package/docs/.vitepress/dist/assets/guide_quick-start.md.CwdNNA21.lean.js +1 -0
- package/docs/.vitepress/dist/assets/guide_reactivity.md.DgTH0MTn.js +135 -0
- package/docs/.vitepress/dist/assets/guide_reactivity.md.DgTH0MTn.lean.js +1 -0
- package/docs/.vitepress/dist/assets/guide_routing.md.nMB0QOBR.js +193 -0
- package/docs/.vitepress/dist/assets/guide_routing.md.nMB0QOBR.lean.js +1 -0
- package/docs/.vitepress/dist/assets/guide_sfc.md.BUkWma1z.js +187 -0
- package/docs/.vitepress/dist/assets/guide_sfc.md.BUkWma1z.lean.js +1 -0
- package/docs/.vitepress/dist/assets/guide_templating.md.XI3uUlYI.js +119 -0
- package/docs/.vitepress/dist/assets/guide_templating.md.XI3uUlYI.lean.js +1 -0
- package/docs/.vitepress/dist/assets/index.md.M4_o26kF.js +23 -0
- package/docs/.vitepress/dist/assets/index.md.M4_o26kF.lean.js +1 -0
- package/docs/.vitepress/dist/assets/inter-italic-cyrillic-ext.r48I6akx.woff2 +0 -0
- package/docs/.vitepress/dist/assets/inter-italic-cyrillic.By2_1cv3.woff2 +0 -0
- package/docs/.vitepress/dist/assets/inter-italic-greek-ext.1u6EdAuj.woff2 +0 -0
- package/docs/.vitepress/dist/assets/inter-italic-greek.DJ8dCoTZ.woff2 +0 -0
- package/docs/.vitepress/dist/assets/inter-italic-latin-ext.CN1xVJS-.woff2 +0 -0
- package/docs/.vitepress/dist/assets/inter-italic-latin.C2AdPX0b.woff2 +0 -0
- package/docs/.vitepress/dist/assets/inter-italic-vietnamese.BSbpV94h.woff2 +0 -0
- package/docs/.vitepress/dist/assets/inter-roman-cyrillic-ext.BBPuwvHQ.woff2 +0 -0
- package/docs/.vitepress/dist/assets/inter-roman-cyrillic.C5lxZ8CY.woff2 +0 -0
- package/docs/.vitepress/dist/assets/inter-roman-greek-ext.CqjqNYQ-.woff2 +0 -0
- package/docs/.vitepress/dist/assets/inter-roman-greek.BBVDIX6e.woff2 +0 -0
- package/docs/.vitepress/dist/assets/inter-roman-latin-ext.4ZJIpNVo.woff2 +0 -0
- package/docs/.vitepress/dist/assets/inter-roman-latin.Di8DUHzh.woff2 +0 -0
- package/docs/.vitepress/dist/assets/inter-roman-vietnamese.BjW4sHH5.woff2 +0 -0
- package/docs/.vitepress/dist/assets/style.eycE2Jhw.css +1 -0
- package/docs/.vitepress/dist/examples/form.html +58 -0
- package/docs/.vitepress/dist/examples/index.html +52 -0
- package/docs/.vitepress/dist/examples/routing.html +362 -0
- package/docs/.vitepress/dist/examples/sfc-showcase.html +37 -0
- package/docs/.vitepress/dist/examples/todo-app.html +321 -0
- package/docs/.vitepress/dist/guide/cdn-usage.html +206 -0
- package/docs/.vitepress/dist/guide/components.html +198 -0
- package/docs/.vitepress/dist/guide/contributing.html +25 -0
- package/docs/.vitepress/dist/guide/directives.html +164 -0
- package/docs/.vitepress/dist/guide/getting-started.html +131 -0
- package/docs/.vitepress/dist/guide/index.html +26 -0
- package/docs/.vitepress/dist/guide/lifecycle.html +328 -0
- package/docs/.vitepress/dist/guide/quick-start.html +57 -0
- package/docs/.vitepress/dist/guide/reactivity.html +159 -0
- package/docs/.vitepress/dist/guide/routing.html +217 -0
- package/docs/.vitepress/dist/guide/sfc.html +211 -0
- package/docs/.vitepress/dist/guide/templating.html +143 -0
- package/docs/.vitepress/dist/hashmap.json +1 -0
- package/docs/.vitepress/dist/index.html +47 -0
- package/docs/.vitepress/dist/logo.svg +38 -0
- package/docs/.vitepress/dist/vp-icons.css +1 -0
- package/docs/api/define.md +31 -0
- package/docs/api/directives.md +42 -0
- package/docs/api/globals.md +29 -0
- package/docs/api/index.md +29 -0
- package/docs/api/lifecycle.md +40 -0
- package/docs/api/route.md +37 -0
- package/docs/api/vite-plugin.md +58 -0
- package/docs/examples/form.md +42 -0
- package/docs/examples/index.md +104 -0
- package/docs/examples/routing.md +409 -0
- package/docs/examples/sfc-showcase.md +34 -0
- package/docs/examples/todo-app.md +383 -0
- package/docs/guide/cdn-usage.md +320 -0
- package/docs/guide/components.md +394 -0
- package/docs/guide/contributing.md +32 -0
- package/docs/guide/directives.md +430 -0
- package/docs/guide/getting-started.md +233 -0
- package/docs/guide/index.md +88 -0
- package/docs/guide/lifecycle.md +493 -0
- package/docs/guide/quick-start.md +46 -0
- package/docs/guide/reactivity.md +394 -0
- package/docs/guide/routing.md +373 -0
- package/docs/guide/sfc.md +381 -0
- package/docs/guide/templating.md +383 -0
- package/docs/index.md +126 -0
- package/docs/public/logo.svg +17 -0
- package/examples/vite-app/README.md +71 -0
- package/examples/vite-app/index.html +49 -0
- package/examples/vite-app/package.json +16 -0
- package/examples/vite-app/src/components/greeting-card.lego +41 -0
- package/examples/vite-app/src/components/sample-component.lego +75 -0
- package/examples/vite-app/src/components/todo-list.lego +242 -0
- package/examples/vite-app/src/main.js +11 -0
- package/examples/vite-app/vite.config.js +17 -0
- package/examples.js +99 -0
- package/go.html +117 -0
- package/lego.js +2 -0
- package/main.js +41 -35
- package/package.json +39 -6
- package/parse-lego.js +119 -0
- package/parse-lego.test.js +107 -0
- package/vite-plugin.js +133 -0
- package/.ignore/auto.html +0 -135
- package/.ignore/test.html +0 -73
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
<style>
|
|
2
|
+
self {
|
|
3
|
+
display: block;
|
|
4
|
+
padding: 1rem;
|
|
5
|
+
background: #f3f4f6;
|
|
6
|
+
border-radius: 0.5rem;
|
|
7
|
+
margin: 0.5rem 0;
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
input {
|
|
11
|
+
padding: 0.5rem;
|
|
12
|
+
border: 1px solid #d1d5db;
|
|
13
|
+
border-radius: 0.25rem;
|
|
14
|
+
margin-right: 0.5rem;
|
|
15
|
+
font-size: 1rem;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
.greeting {
|
|
19
|
+
margin-top: 1rem;
|
|
20
|
+
padding: 1rem;
|
|
21
|
+
background: white;
|
|
22
|
+
border-radius: 0.25rem;
|
|
23
|
+
font-size: 1.25rem;
|
|
24
|
+
color: #1f2937;
|
|
25
|
+
}
|
|
26
|
+
</style>
|
|
27
|
+
|
|
28
|
+
<template>
|
|
29
|
+
<div>
|
|
30
|
+
<input b-sync="name" type="text" placeholder="Enter your name">
|
|
31
|
+
<div class="greeting" b-if="name">
|
|
32
|
+
Hello, {{ name }}! 👋
|
|
33
|
+
</div>
|
|
34
|
+
</div>
|
|
35
|
+
</template>
|
|
36
|
+
|
|
37
|
+
<script>
|
|
38
|
+
export default {
|
|
39
|
+
name: ''
|
|
40
|
+
}
|
|
41
|
+
</script>
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
<style>
|
|
2
|
+
self {
|
|
3
|
+
display: block;
|
|
4
|
+
padding: 1.5rem;
|
|
5
|
+
margin: 1rem 0;
|
|
6
|
+
border: 2px solid #4f46e5;
|
|
7
|
+
border-radius: 0.5rem;
|
|
8
|
+
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
|
|
9
|
+
color: white;
|
|
10
|
+
font-family: system-ui, -apple-system, sans-serif;
|
|
11
|
+
box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1);
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
.header {
|
|
15
|
+
font-size: 1.5rem;
|
|
16
|
+
font-weight: bold;
|
|
17
|
+
margin-bottom: 0.5rem;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
.content {
|
|
21
|
+
margin: 1rem 0;
|
|
22
|
+
line-height: 1.6;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
button {
|
|
26
|
+
background: white;
|
|
27
|
+
color: #4f46e5;
|
|
28
|
+
border: none;
|
|
29
|
+
padding: 0.5rem 1rem;
|
|
30
|
+
border-radius: 0.25rem;
|
|
31
|
+
font-weight: 600;
|
|
32
|
+
cursor: pointer;
|
|
33
|
+
transition: transform 0.2s;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
button:hover {
|
|
37
|
+
transform: scale(1.05);
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
.count {
|
|
41
|
+
display: inline-block;
|
|
42
|
+
background: rgba(255, 255, 255, 0.2);
|
|
43
|
+
padding: 0.25rem 0.75rem;
|
|
44
|
+
border-radius: 0.25rem;
|
|
45
|
+
margin-left: 0.5rem;
|
|
46
|
+
}
|
|
47
|
+
</style>
|
|
48
|
+
|
|
49
|
+
<template>
|
|
50
|
+
<div class="header">{{ title }}</div>
|
|
51
|
+
<div class="content">
|
|
52
|
+
<p>This is a sample Lego component loaded from a .lego file!</p>
|
|
53
|
+
<p>Message: {{ message }}</p>
|
|
54
|
+
</div>
|
|
55
|
+
<button @click="incrementCount()">
|
|
56
|
+
Click me!
|
|
57
|
+
<span class="count">{{ count }}</span>
|
|
58
|
+
</button>
|
|
59
|
+
</template>
|
|
60
|
+
|
|
61
|
+
<script>
|
|
62
|
+
export default {
|
|
63
|
+
title: 'Sample Component',
|
|
64
|
+
message: 'Hello from .lego file!',
|
|
65
|
+
count: 0,
|
|
66
|
+
|
|
67
|
+
incrementCount() {
|
|
68
|
+
this.count++;
|
|
69
|
+
},
|
|
70
|
+
|
|
71
|
+
mounted() {
|
|
72
|
+
console.log('Sample component mounted!');
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
</script>
|
|
@@ -0,0 +1,242 @@
|
|
|
1
|
+
<style>
|
|
2
|
+
self {
|
|
3
|
+
display: block;
|
|
4
|
+
background: white;
|
|
5
|
+
border-radius: 8px;
|
|
6
|
+
padding: 1.5rem;
|
|
7
|
+
box-shadow: 0 2px 8px rgba(0,0,0,0.1);
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
.input-group {
|
|
11
|
+
display: flex;
|
|
12
|
+
gap: 0.5rem;
|
|
13
|
+
margin-bottom: 1.5rem;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
input[type="text"] {
|
|
17
|
+
flex: 1;
|
|
18
|
+
padding: 0.75rem;
|
|
19
|
+
font-size: 1rem;
|
|
20
|
+
border: 2px solid #e0e0e0;
|
|
21
|
+
border-radius: 4px;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
input[type="text"]:focus {
|
|
25
|
+
outline: none;
|
|
26
|
+
border-color: #4CAF50;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
.btn {
|
|
30
|
+
padding: 0.75rem 1.5rem;
|
|
31
|
+
font-size: 1rem;
|
|
32
|
+
border: none;
|
|
33
|
+
border-radius: 4px;
|
|
34
|
+
cursor: pointer;
|
|
35
|
+
font-weight: 600;
|
|
36
|
+
transition: background 0.2s;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
.btn-primary {
|
|
40
|
+
background: #4CAF50;
|
|
41
|
+
color: white;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
.btn-primary:hover {
|
|
45
|
+
background: #45a049;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
.filters {
|
|
49
|
+
display: flex;
|
|
50
|
+
gap: 0.5rem;
|
|
51
|
+
margin-bottom: 1rem;
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
.filter-btn {
|
|
55
|
+
padding: 0.5rem 1rem;
|
|
56
|
+
background: #f0f0f0;
|
|
57
|
+
border: none;
|
|
58
|
+
border-radius: 4px;
|
|
59
|
+
cursor: pointer;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
.filter-btn.active {
|
|
63
|
+
background: #4CAF50;
|
|
64
|
+
color: white;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
ul {
|
|
68
|
+
list-style: none;
|
|
69
|
+
padding: 0;
|
|
70
|
+
margin: 0;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
li {
|
|
74
|
+
display: flex;
|
|
75
|
+
align-items: center;
|
|
76
|
+
gap: 0.75rem;
|
|
77
|
+
padding: 0.75rem;
|
|
78
|
+
border-bottom: 1px solid #f0f0f0;
|
|
79
|
+
transition: background 0.2s;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
li:hover {
|
|
83
|
+
background: #f9f9f9;
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
input[type="checkbox"] {
|
|
87
|
+
width: 20px;
|
|
88
|
+
height: 20px;
|
|
89
|
+
cursor: pointer;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
.todo-text {
|
|
93
|
+
flex: 1;
|
|
94
|
+
font-size: 1rem;
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
.todo-text.done {
|
|
98
|
+
text-decoration: line-through;
|
|
99
|
+
color: #999;
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
.delete-btn {
|
|
103
|
+
padding: 0.25rem 0.5rem;
|
|
104
|
+
background: #f44336;
|
|
105
|
+
color: white;
|
|
106
|
+
border: none;
|
|
107
|
+
border-radius: 4px;
|
|
108
|
+
cursor: pointer;
|
|
109
|
+
font-size: 0.875rem;
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
.delete-btn:hover {
|
|
113
|
+
background: #da190b;
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
.stats {
|
|
117
|
+
margin-top: 1rem;
|
|
118
|
+
padding-top: 1rem;
|
|
119
|
+
border-top: 2px solid #f0f0f0;
|
|
120
|
+
display: flex;
|
|
121
|
+
justify-content: space-between;
|
|
122
|
+
color: #666;
|
|
123
|
+
font-size: 0.875rem;
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
.clear-completed {
|
|
127
|
+
background: none;
|
|
128
|
+
border: none;
|
|
129
|
+
color: #f44336;
|
|
130
|
+
cursor: pointer;
|
|
131
|
+
text-decoration: underline;
|
|
132
|
+
}
|
|
133
|
+
</style>
|
|
134
|
+
|
|
135
|
+
<template>
|
|
136
|
+
<h1>📝 Todo App</h1>
|
|
137
|
+
|
|
138
|
+
<div class="input-group">
|
|
139
|
+
<input
|
|
140
|
+
type="text"
|
|
141
|
+
b-sync="newTodo"
|
|
142
|
+
placeholder="What needs to be done?"
|
|
143
|
+
@keyup="event.key === 'Enter' && addTodo()">
|
|
144
|
+
<button class="btn btn-primary" @click="addTodo()">Add</button>
|
|
145
|
+
</div>
|
|
146
|
+
|
|
147
|
+
<div class="filters">
|
|
148
|
+
<button
|
|
149
|
+
class="filter-btn {{ filter === 'all' ? 'active' : '' }}"
|
|
150
|
+
@click="filter = 'all'">
|
|
151
|
+
All
|
|
152
|
+
</button>
|
|
153
|
+
<button
|
|
154
|
+
class="filter-btn {{ filter === 'active' ? 'active' : '' }}"
|
|
155
|
+
@click="filter = 'active'">
|
|
156
|
+
Active
|
|
157
|
+
</button>
|
|
158
|
+
<button
|
|
159
|
+
class="filter-btn {{ filter === 'completed' ? 'active' : '' }}"
|
|
160
|
+
@click="filter = 'completed'">
|
|
161
|
+
Completed
|
|
162
|
+
</button>
|
|
163
|
+
</div>
|
|
164
|
+
|
|
165
|
+
<ul>
|
|
166
|
+
<li b-for="todo in filteredTodos()">
|
|
167
|
+
<input type="checkbox" b-sync="todo.done">
|
|
168
|
+
<span class="todo-text {{ todo.done ? 'done' : '' }}">
|
|
169
|
+
{{ todo.text }} - first
|
|
170
|
+
</span>
|
|
171
|
+
<button class="delete-btn" @click="deleteTodo(todo)">Delete</button>
|
|
172
|
+
</li>
|
|
173
|
+
</ul>
|
|
174
|
+
|
|
175
|
+
<div class="stats">
|
|
176
|
+
<span>{{ remaining() }} item{{ remaining() === 1 ? '' : 's' }} left</span>
|
|
177
|
+
<button
|
|
178
|
+
class="clear-completed"
|
|
179
|
+
b-if="completedCount() > 0"
|
|
180
|
+
@click="clearCompleted()">
|
|
181
|
+
Clear completed ({{ completedCount() }})
|
|
182
|
+
</button>
|
|
183
|
+
</div>
|
|
184
|
+
</template>
|
|
185
|
+
|
|
186
|
+
<script>
|
|
187
|
+
export default {
|
|
188
|
+
newTodo: '',
|
|
189
|
+
filter: 'all',
|
|
190
|
+
todos: [],
|
|
191
|
+
|
|
192
|
+
mounted() {
|
|
193
|
+
const saved = localStorage.getItem('legojs-todos');
|
|
194
|
+
if (saved) {
|
|
195
|
+
this.todos = JSON.parse(saved);
|
|
196
|
+
}
|
|
197
|
+
},
|
|
198
|
+
|
|
199
|
+
updated() {
|
|
200
|
+
localStorage.setItem('legojs-todos', JSON.stringify(this.todos));
|
|
201
|
+
},
|
|
202
|
+
|
|
203
|
+
addTodo() {
|
|
204
|
+
if (this.newTodo.trim()) {
|
|
205
|
+
this.todos.push({
|
|
206
|
+
id: Date.now(),
|
|
207
|
+
text: this.newTodo,
|
|
208
|
+
done: false
|
|
209
|
+
});
|
|
210
|
+
this.newTodo = '';
|
|
211
|
+
}
|
|
212
|
+
},
|
|
213
|
+
|
|
214
|
+
deleteTodo(todo) {
|
|
215
|
+
const index = this.todos.indexOf(todo);
|
|
216
|
+
if (index > -1) {
|
|
217
|
+
this.todos.splice(index, 1);
|
|
218
|
+
}
|
|
219
|
+
},
|
|
220
|
+
|
|
221
|
+
filteredTodos() {
|
|
222
|
+
if (this.filter === 'active') {
|
|
223
|
+
return this.todos.filter(t => !t.done);
|
|
224
|
+
} else if (this.filter === 'completed') {
|
|
225
|
+
return this.todos.filter(t => t.done);
|
|
226
|
+
}
|
|
227
|
+
return this.todos;
|
|
228
|
+
},
|
|
229
|
+
|
|
230
|
+
remaining() {
|
|
231
|
+
return this.todos.filter(t => !t.done).length;
|
|
232
|
+
},
|
|
233
|
+
|
|
234
|
+
completedCount() {
|
|
235
|
+
return this.todos.filter(t => t.done).length;
|
|
236
|
+
},
|
|
237
|
+
|
|
238
|
+
clearCompleted() {
|
|
239
|
+
this.todos = this.todos.filter(t => !t.done);
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
</script>
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
// Import Lego core
|
|
2
|
+
import { Lego } from 'lego-dom/main.js';
|
|
3
|
+
|
|
4
|
+
// Import virtual module that auto-discovers and registers all .lego components
|
|
5
|
+
import registerComponents from 'virtual:lego-components';
|
|
6
|
+
|
|
7
|
+
// Register all auto-discovered components
|
|
8
|
+
registerComponents();
|
|
9
|
+
|
|
10
|
+
// Initialize Lego
|
|
11
|
+
console.log('Lego initialized with auto-discovered components!');
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import { defineConfig } from 'vite';
|
|
2
|
+
import path from 'path';
|
|
3
|
+
import legoPlugin from 'lego-dom/vite-plugin';
|
|
4
|
+
|
|
5
|
+
export default defineConfig({
|
|
6
|
+
plugins: [
|
|
7
|
+
legoPlugin({
|
|
8
|
+
componentsDir: './src/components',
|
|
9
|
+
include: ['**/*.lego']
|
|
10
|
+
})
|
|
11
|
+
],
|
|
12
|
+
resolve: {
|
|
13
|
+
alias: {
|
|
14
|
+
'lego-dom/main.js': path.resolve(__dirname, '../../lego.js')
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
});
|
package/examples.js
ADDED
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
export const examples = {
|
|
2
|
+
counter: `<!DOCTYPE html>
|
|
3
|
+
<html>
|
|
4
|
+
<head>
|
|
5
|
+
<title>Lego Counter</title>
|
|
6
|
+
</head>
|
|
7
|
+
<body>
|
|
8
|
+
<template b-id="click-counter">
|
|
9
|
+
<style>
|
|
10
|
+
button { font-size: 1.2rem; padding: 0.5rem 1rem; }
|
|
11
|
+
</style>
|
|
12
|
+
<p>Count: {{ count }}</p>
|
|
13
|
+
<button @click="count++">Increment</button>
|
|
14
|
+
</template>
|
|
15
|
+
|
|
16
|
+
<click-counter b-data="{ count: 0 }"></click-counter>
|
|
17
|
+
|
|
18
|
+
<script src="https://unpkg.com/lego-dom/main.js"></script>
|
|
19
|
+
</body>
|
|
20
|
+
</html>`,
|
|
21
|
+
|
|
22
|
+
todo: `<!DOCTYPE html>
|
|
23
|
+
<html lang="en">
|
|
24
|
+
<head>
|
|
25
|
+
<meta charset="UTF-8">
|
|
26
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
27
|
+
<title>Todo App - Lego</title>
|
|
28
|
+
<style>
|
|
29
|
+
body {
|
|
30
|
+
font-family: system-ui, -apple-system, sans-serif;
|
|
31
|
+
max-width: 600px;
|
|
32
|
+
margin: 2rem auto;
|
|
33
|
+
padding: 0 1rem;
|
|
34
|
+
background: #f5f5f5;
|
|
35
|
+
}
|
|
36
|
+
h1 {
|
|
37
|
+
text-align: center;
|
|
38
|
+
color: #333;
|
|
39
|
+
}
|
|
40
|
+
</style>
|
|
41
|
+
</head>
|
|
42
|
+
<body>
|
|
43
|
+
|
|
44
|
+
<script src="https://unpkg.com/lego-dom/main.js"></script>
|
|
45
|
+
|
|
46
|
+
|
|
47
|
+
|
|
48
|
+
<script>
|
|
49
|
+
</script>
|
|
50
|
+
</body>
|
|
51
|
+
</html>`,
|
|
52
|
+
|
|
53
|
+
form: `<!DOCTYPE html>
|
|
54
|
+
<html>
|
|
55
|
+
<head>
|
|
56
|
+
<title>Lego Form</title>
|
|
57
|
+
</head>
|
|
58
|
+
<body>
|
|
59
|
+
<template b-id="login-form">
|
|
60
|
+
<form @submit.prevent="login()">
|
|
61
|
+
<div>
|
|
62
|
+
<label>Email:</label>
|
|
63
|
+
<input type="email" b-sync="email">
|
|
64
|
+
</div>
|
|
65
|
+
|
|
66
|
+
<div>
|
|
67
|
+
<label>Password:</label>
|
|
68
|
+
<input type="password" b-sync="password">
|
|
69
|
+
</div>
|
|
70
|
+
|
|
71
|
+
<p b-if="error" style="color: red">{{ error }}</p>
|
|
72
|
+
|
|
73
|
+
<button type="submit">Login</button>
|
|
74
|
+
</form>
|
|
75
|
+
</template>
|
|
76
|
+
|
|
77
|
+
<login-form></login-form>
|
|
78
|
+
|
|
79
|
+
<script src="https://unpkg.com/lego-dom/main.js"></script>
|
|
80
|
+
|
|
81
|
+
<script>
|
|
82
|
+
Lego.define('login-form', document.querySelector('template[b-id="login-form"]').innerHTML, {
|
|
83
|
+
email: '',
|
|
84
|
+
password: '',
|
|
85
|
+
error: '',
|
|
86
|
+
|
|
87
|
+
login() {
|
|
88
|
+
if (!this.email || !this.password) {
|
|
89
|
+
this.error = 'Please fill in all fields';
|
|
90
|
+
return;
|
|
91
|
+
}
|
|
92
|
+
alert(\`Logging in as \${this.email}\`);
|
|
93
|
+
this.error = '';
|
|
94
|
+
}
|
|
95
|
+
});
|
|
96
|
+
</script>
|
|
97
|
+
</body>
|
|
98
|
+
</html>`
|
|
99
|
+
};
|
package/go.html
ADDED
|
@@ -0,0 +1,117 @@
|
|
|
1
|
+
<html lang="en">
|
|
2
|
+
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="UTF-8">
|
|
5
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
6
|
+
<title>Lego Test</title>
|
|
7
|
+
<script src="./main.js"></script>
|
|
8
|
+
</head>
|
|
9
|
+
|
|
10
|
+
<body @todo-added="() => {console.log('Todo added!')}">
|
|
11
|
+
<template b-id="todo-list">
|
|
12
|
+
<style>
|
|
13
|
+
:host {
|
|
14
|
+
display: block;
|
|
15
|
+
font-family: system-ui, sans-serif;
|
|
16
|
+
padding: 20px;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
.done {
|
|
20
|
+
text-decoration: line-through;
|
|
21
|
+
color: #888;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
.input {
|
|
25
|
+
padding: .4em;
|
|
26
|
+
border: 1px solid #ccc;
|
|
27
|
+
border-radius: 0.1em;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
.item-row {
|
|
31
|
+
margin-bottom: 8px;
|
|
32
|
+
display: flex;
|
|
33
|
+
align-items: center;
|
|
34
|
+
gap: 10px;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
button {
|
|
38
|
+
cursor: pointer;
|
|
39
|
+
border: none;
|
|
40
|
+
padding: 5px 10px;
|
|
41
|
+
border-radius: 0px;
|
|
42
|
+
background-color: #404;
|
|
43
|
+
color: #fff;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
h3 {
|
|
47
|
+
color: #318d8d;
|
|
48
|
+
margin-top: 0;
|
|
49
|
+
}
|
|
50
|
+
</style>
|
|
51
|
+
|
|
52
|
+
<h3>{{title}}</h3>
|
|
53
|
+
<p>{{ items.filter(t => !t.done).length }} of {{ items.length }} items remaining</p>
|
|
54
|
+
<div style="margin-bottom: 20px;">
|
|
55
|
+
<input b-sync="newItem" placeholder="Add task..." class="input" @keydown="() => {
|
|
56
|
+
}">
|
|
57
|
+
<button @click="() => {
|
|
58
|
+
if(!newItem.trim()) return;
|
|
59
|
+
items.push({text: `${newItem} - ${items.length + 1} - ${$ancestors('user-card').name}`, done: false});
|
|
60
|
+
newItem = '';
|
|
61
|
+
$emit('todo-added', { item: newItem });
|
|
62
|
+
}">Add</button>
|
|
63
|
+
</div>
|
|
64
|
+
|
|
65
|
+
<div b-for="todo in items">
|
|
66
|
+
<div class="item-row {{todo.done ? 'done' : ''}}">
|
|
67
|
+
<input type="checkbox" b-sync="todo.done">
|
|
68
|
+
<span>{{todo.text}}</span>
|
|
69
|
+
</div>
|
|
70
|
+
</div>
|
|
71
|
+
</template>
|
|
72
|
+
|
|
73
|
+
<template b-id="user-card">
|
|
74
|
+
<style>
|
|
75
|
+
:host {
|
|
76
|
+
display: block;
|
|
77
|
+
font-family: system-ui, sans-serif;
|
|
78
|
+
padding: 20px;
|
|
79
|
+
border: 1px solid #ccc;
|
|
80
|
+
border-radius: 8px;
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
h3 {
|
|
84
|
+
color: #62318d;
|
|
85
|
+
margin-top: 0;
|
|
86
|
+
}
|
|
87
|
+
</style>
|
|
88
|
+
|
|
89
|
+
<h3>{{name}}</h3>
|
|
90
|
+
<p>{{bio}}</p>
|
|
91
|
+
<slot></slot>
|
|
92
|
+
</template>
|
|
93
|
+
|
|
94
|
+
<user-card b-data="{
|
|
95
|
+
name: 'John Doe',
|
|
96
|
+
bio: 'Software Engineer'
|
|
97
|
+
}">
|
|
98
|
+
<p>Additional content here</p>
|
|
99
|
+
<todo-list b-data="{
|
|
100
|
+
title: 'Enterprise Todo List',
|
|
101
|
+
items: [{text: 'Fix b-for strikethrough', done: true}],
|
|
102
|
+
newItem: '',
|
|
103
|
+
mounted() {
|
|
104
|
+
console.log('THis lego block has been mounted!');
|
|
105
|
+
},
|
|
106
|
+
updated() {
|
|
107
|
+
console.log('This lego block has been updated!');
|
|
108
|
+
},
|
|
109
|
+
unmounted() {
|
|
110
|
+
console.log('This lego block has been unmounted!');
|
|
111
|
+
}
|
|
112
|
+
}">
|
|
113
|
+
</todo-list>
|
|
114
|
+
</user-card>
|
|
115
|
+
</body>
|
|
116
|
+
|
|
117
|
+
</html>
|
package/lego.js
ADDED