@indexeddb-orm/idb-orm 0.0.1

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.
Files changed (216) hide show
  1. package/.vscode/extensions.json +5 -0
  2. package/README.md +1280 -0
  3. package/angular-demo-app/README.md +84 -0
  4. package/angular-demo-app/angular.json +109 -0
  5. package/angular-demo-app/package-lock.json +14215 -0
  6. package/angular-demo-app/package.json +41 -0
  7. package/angular-demo-app/src/app/app.component.ts +481 -0
  8. package/angular-demo-app/src/app/app.routes.ts +8 -0
  9. package/angular-demo-app/src/app/components/actions.component.ts +202 -0
  10. package/angular-demo-app/src/app/components/cloud-sync-demo.component.ts +296 -0
  11. package/angular-demo-app/src/app/components/live-query-demo.component.ts +307 -0
  12. package/angular-demo-app/src/app/components/main-info.component.ts +148 -0
  13. package/angular-demo-app/src/app/components/posts-live-query-demo.component.ts +336 -0
  14. package/angular-demo-app/src/app/components/typescript-demo.component.ts +268 -0
  15. package/angular-demo-app/src/entities/post-tag.entity.ts +25 -0
  16. package/angular-demo-app/src/entities/post.entity.ts +49 -0
  17. package/angular-demo-app/src/entities/profile.entity.ts +42 -0
  18. package/angular-demo-app/src/entities/tag.entity.ts +36 -0
  19. package/angular-demo-app/src/entities/user.entity.ts +59 -0
  20. package/angular-demo-app/src/favicon.ico +1 -0
  21. package/angular-demo-app/src/index.html +16 -0
  22. package/angular-demo-app/src/main.ts +13 -0
  23. package/angular-demo-app/src/services/app-logic.service.ts +449 -0
  24. package/angular-demo-app/src/services/cloud-sync.service.ts +95 -0
  25. package/angular-demo-app/src/services/database.service.ts +26 -0
  26. package/angular-demo-app/src/services/live-query.service.ts +63 -0
  27. package/angular-demo-app/src/services/posts-live-query.service.ts +86 -0
  28. package/angular-demo-app/src/services/typescript-demo.service.ts +59 -0
  29. package/angular-demo-app/src/styles.scss +50 -0
  30. package/angular-demo-app/tsconfig.app.json +13 -0
  31. package/angular-demo-app/tsconfig.json +34 -0
  32. package/angular-demo-app/tsconfig.spec.json +13 -0
  33. package/dist/Database.d.ts +206 -0
  34. package/dist/Database.js +288 -0
  35. package/dist/decorators/Column.d.ts +79 -0
  36. package/dist/decorators/Column.js +236 -0
  37. package/dist/decorators/Entity.d.ts +32 -0
  38. package/dist/decorators/Entity.js +44 -0
  39. package/dist/decorators/Relation.d.ts +70 -0
  40. package/dist/decorators/Relation.js +120 -0
  41. package/dist/decorators/index.d.ts +3 -0
  42. package/dist/decorators/index.js +3 -0
  43. package/dist/errors/ValidationError.d.ts +4 -0
  44. package/dist/errors/ValidationError.js +8 -0
  45. package/dist/index.d.ts +8 -0
  46. package/dist/index.js +7 -0
  47. package/dist/metadata/Column.d.ts +8 -0
  48. package/dist/metadata/Column.js +44 -0
  49. package/dist/metadata/Entity.d.ts +11 -0
  50. package/dist/metadata/Entity.js +21 -0
  51. package/dist/metadata/Relation.d.ts +20 -0
  52. package/dist/metadata/Relation.js +74 -0
  53. package/dist/metadata/index.d.ts +3 -0
  54. package/dist/metadata/index.js +3 -0
  55. package/dist/services/AggregationService.d.ts +38 -0
  56. package/dist/services/AggregationService.js +229 -0
  57. package/dist/services/BaseEntity.d.ts +32 -0
  58. package/dist/services/BaseEntity.js +62 -0
  59. package/dist/services/CloudSyncService.d.ts +100 -0
  60. package/dist/services/CloudSyncService.js +196 -0
  61. package/dist/services/DecoratorUtils.d.ts +12 -0
  62. package/dist/services/DecoratorUtils.js +10 -0
  63. package/dist/services/EntityFactory.d.ts +25 -0
  64. package/dist/services/EntityFactory.js +27 -0
  65. package/dist/services/EntityRegistry.d.ts +61 -0
  66. package/dist/services/EntityRegistry.js +56 -0
  67. package/dist/services/EntitySchema.d.ts +56 -0
  68. package/dist/services/EntitySchema.js +125 -0
  69. package/dist/services/MigrationManager.d.ts +70 -0
  70. package/dist/services/MigrationManager.js +181 -0
  71. package/dist/services/RelationLoader.d.ts +66 -0
  72. package/dist/services/RelationLoader.js +310 -0
  73. package/dist/services/SchemaBuilder.d.ts +68 -0
  74. package/dist/services/SchemaBuilder.js +191 -0
  75. package/dist/services/index.d.ts +7 -0
  76. package/dist/services/index.js +7 -0
  77. package/dist/types.d.ts +152 -0
  78. package/dist/types.js +1 -0
  79. package/dist/utils/logger.d.ts +12 -0
  80. package/dist/utils/logger.js +16 -0
  81. package/eslint.config.js +49 -0
  82. package/homepage/favicon.svg +36 -0
  83. package/homepage/index.html +1725 -0
  84. package/package.json +78 -0
  85. package/react-demo-app/README.md +61 -0
  86. package/react-demo-app/eslint.config.js +60 -0
  87. package/react-demo-app/index.html +13 -0
  88. package/react-demo-app/package-lock.json +4955 -0
  89. package/react-demo-app/package.json +39 -0
  90. package/react-demo-app/src/App.tsx +172 -0
  91. package/react-demo-app/src/assets/react.svg +1 -0
  92. package/react-demo-app/src/components/Actions.tsx +171 -0
  93. package/react-demo-app/src/components/CloudSyncDemo.tsx +191 -0
  94. package/react-demo-app/src/components/LiveQueryDemo.tsx +122 -0
  95. package/react-demo-app/src/components/MainInfo.tsx +75 -0
  96. package/react-demo-app/src/components/PostsLiveQueryDemo.tsx +185 -0
  97. package/react-demo-app/src/components/TypeScriptDemo.tsx +190 -0
  98. package/react-demo-app/src/database/Database.ts +30 -0
  99. package/react-demo-app/src/entities/Post.ts +48 -0
  100. package/react-demo-app/src/entities/PostTag.ts +26 -0
  101. package/react-demo-app/src/entities/Profile.ts +41 -0
  102. package/react-demo-app/src/entities/Tag.ts +35 -0
  103. package/react-demo-app/src/entities/User.ts +61 -0
  104. package/react-demo-app/src/hooks/useAppLogic.ts +565 -0
  105. package/react-demo-app/src/hooks/useCloudSyncDemo.ts +84 -0
  106. package/react-demo-app/src/hooks/useLiveQueryDemo.ts +68 -0
  107. package/react-demo-app/src/hooks/usePostsLiveQueryDemo.ts +64 -0
  108. package/react-demo-app/src/hooks/useTypeScriptDemo.ts +43 -0
  109. package/react-demo-app/src/index.css +26 -0
  110. package/react-demo-app/src/main.tsx +18 -0
  111. package/react-demo-app/src/migrations/001-add-user-email-index.ts +17 -0
  112. package/react-demo-app/src/migrations/002-add-post-category.ts +37 -0
  113. package/react-demo-app/src/migrations/index.ts +8 -0
  114. package/react-demo-app/src/vite-env.d.ts +1 -0
  115. package/react-demo-app/tsconfig.app.json +22 -0
  116. package/react-demo-app/tsconfig.json +6 -0
  117. package/react-demo-app/vite.config.ts +10 -0
  118. package/src/Database.ts +405 -0
  119. package/src/errors/ValidationError.ts +9 -0
  120. package/src/index.ts +13 -0
  121. package/src/metadata/Column.ts +74 -0
  122. package/src/metadata/Entity.ts +42 -0
  123. package/src/metadata/Relation.ts +121 -0
  124. package/src/metadata/index.ts +5 -0
  125. package/src/services/AggregationService.ts +348 -0
  126. package/src/services/BaseEntity.ts +77 -0
  127. package/src/services/CloudSyncService.ts +248 -0
  128. package/src/services/EntityFactory.ts +35 -0
  129. package/src/services/EntityRegistry.ts +109 -0
  130. package/src/services/EntitySchema.ts +154 -0
  131. package/src/services/MigrationManager.ts +276 -0
  132. package/src/services/RelationLoader.ts +532 -0
  133. package/src/services/SchemaBuilder.ts +237 -0
  134. package/src/services/index.ts +7 -0
  135. package/src/types.d.ts +1 -0
  136. package/src/types.ts +169 -0
  137. package/src/utils/logger.ts +40 -0
  138. package/svelte-demo-app/README.md +61 -0
  139. package/svelte-demo-app/package-lock.json +3000 -0
  140. package/svelte-demo-app/package.json +30 -0
  141. package/svelte-demo-app/src/app.d.ts +12 -0
  142. package/svelte-demo-app/src/app.html +13 -0
  143. package/svelte-demo-app/src/components/Actions.svelte +121 -0
  144. package/svelte-demo-app/src/components/CloudSyncDemo.svelte +333 -0
  145. package/svelte-demo-app/src/components/LiveQueryDemo.svelte +191 -0
  146. package/svelte-demo-app/src/components/MainInfo.svelte +133 -0
  147. package/svelte-demo-app/src/components/PostsLiveQueryDemo.svelte +330 -0
  148. package/svelte-demo-app/src/components/TypeScriptDemo.svelte +251 -0
  149. package/svelte-demo-app/src/database/Database.ts +29 -0
  150. package/svelte-demo-app/src/entities/Post.ts +46 -0
  151. package/svelte-demo-app/src/entities/PostTag.ts +22 -0
  152. package/svelte-demo-app/src/entities/Profile.ts +39 -0
  153. package/svelte-demo-app/src/entities/Tag.ts +33 -0
  154. package/svelte-demo-app/src/entities/User.ts +62 -0
  155. package/svelte-demo-app/src/lib/database/Database.ts +30 -0
  156. package/svelte-demo-app/src/lib/entities/Post.ts +47 -0
  157. package/svelte-demo-app/src/lib/entities/PostTag.ts +23 -0
  158. package/svelte-demo-app/src/lib/entities/Profile.ts +40 -0
  159. package/svelte-demo-app/src/lib/entities/Tag.ts +34 -0
  160. package/svelte-demo-app/src/lib/entities/User.ts +59 -0
  161. package/svelte-demo-app/src/lib/index.ts +7 -0
  162. package/svelte-demo-app/src/lib/migrations/001-add-user-email-index.ts +17 -0
  163. package/svelte-demo-app/src/lib/migrations/002-add-post-category.ts +37 -0
  164. package/svelte-demo-app/src/lib/migrations/index.ts +8 -0
  165. package/svelte-demo-app/src/migrations/001-add-user-email-index.ts +17 -0
  166. package/svelte-demo-app/src/migrations/002-add-post-category.ts +37 -0
  167. package/svelte-demo-app/src/migrations/index.ts +8 -0
  168. package/svelte-demo-app/src/routes/+layout.js +3 -0
  169. package/svelte-demo-app/src/routes/+layout.svelte +228 -0
  170. package/svelte-demo-app/src/routes/+page.js +3 -0
  171. package/svelte-demo-app/src/routes/+page.svelte +1305 -0
  172. package/svelte-demo-app/src/stores/appStore.js +603 -0
  173. package/svelte-demo-app/svelte.config.js +18 -0
  174. package/svelte-demo-app/tsconfig.json +14 -0
  175. package/svelte-demo-app/vite.config.ts +6 -0
  176. package/tests/aggregation.e2e.test.ts +87 -0
  177. package/tests/base-entity.e2e.test.ts +47 -0
  178. package/tests/database-api.e2e.test.ts +177 -0
  179. package/tests/decorators.e2e.test.ts +40 -0
  180. package/tests/entity-schema.e2e.test.ts +58 -0
  181. package/tests/relation-loader-table-names.test.ts +192 -0
  182. package/tests/relations.e2e.test.ts +178 -0
  183. package/tests/zod-runtime.e2e.test.ts +69 -0
  184. package/tsconfig.json +21 -0
  185. package/vitest.config.ts +21 -0
  186. package/vitest.setup.ts +27 -0
  187. package/vue-demo-app/README.md +61 -0
  188. package/vue-demo-app/index.html +13 -0
  189. package/vue-demo-app/package-lock.json +1537 -0
  190. package/vue-demo-app/package.json +27 -0
  191. package/vue-demo-app/src/App.vue +100 -0
  192. package/vue-demo-app/src/components/Actions.vue +135 -0
  193. package/vue-demo-app/src/components/CloudSyncDemo.vue +139 -0
  194. package/vue-demo-app/src/components/LiveQueryDemo.vue +122 -0
  195. package/vue-demo-app/src/components/MainInfo.vue +80 -0
  196. package/vue-demo-app/src/components/PostsLiveQueryDemo.vue +136 -0
  197. package/vue-demo-app/src/components/TypeScriptDemo.vue +133 -0
  198. package/vue-demo-app/src/database/Database.ts +29 -0
  199. package/vue-demo-app/src/entities/Post.ts +48 -0
  200. package/vue-demo-app/src/entities/PostTag.ts +24 -0
  201. package/vue-demo-app/src/entities/Profile.ts +41 -0
  202. package/vue-demo-app/src/entities/Tag.ts +35 -0
  203. package/vue-demo-app/src/entities/User.ts +61 -0
  204. package/vue-demo-app/src/main.ts +29 -0
  205. package/vue-demo-app/src/migrations/001-add-user-email-index.ts +23 -0
  206. package/vue-demo-app/src/migrations/002-add-post-category.ts +46 -0
  207. package/vue-demo-app/src/migrations/index.ts +14 -0
  208. package/vue-demo-app/src/services/useAppLogic.ts +565 -0
  209. package/vue-demo-app/src/services/useCloudSyncDemo.ts +84 -0
  210. package/vue-demo-app/src/services/useLiveQueryDemo.ts +82 -0
  211. package/vue-demo-app/src/services/usePostsLiveQueryDemo.ts +77 -0
  212. package/vue-demo-app/src/services/useTypeScriptDemo.ts +56 -0
  213. package/vue-demo-app/src/vite-env.d.ts +1 -0
  214. package/vue-demo-app/tsconfig.json +25 -0
  215. package/vue-demo-app/tsconfig.node.json +10 -0
  216. package/vue-demo-app/vite.config.ts +16 -0
@@ -0,0 +1,336 @@
1
+ import { CommonModule } from '@angular/common';
2
+ import {
3
+ Component, Input,
4
+ } from '@angular/core';
5
+ import { MatButtonModule } from '@angular/material/button';
6
+ import { MatCardModule } from '@angular/material/card';
7
+ import { MatIconModule } from '@angular/material/icon';
8
+ import { MatListModule } from '@angular/material/list';
9
+
10
+ import { PostEntity } from '../../entities/post.entity';
11
+
12
+ @Component({
13
+ selector: 'app-posts-live-query-demo',
14
+ standalone: true,
15
+ imports: [
16
+ CommonModule, MatButtonModule, MatCardModule, MatIconModule, MatListModule,
17
+ ],
18
+ template: `
19
+ <div class="posts-paper">
20
+ <h2 class="main-title">
21
+ <mat-icon color="primary" class="main-icon">data_object</mat-icon>
22
+ Posts useLiveQuery Demo
23
+ </h2>
24
+
25
+ <div class="add-button-section">
26
+ <button mat-stroked-button
27
+ (click)="addSamplePost()"
28
+ class="add-button">
29
+ Add sample post
30
+ </button>
31
+ </div>
32
+
33
+ <div class="demo-grid">
34
+ <div class="demo-section">
35
+ <h3 class="section-title">
36
+ <mat-icon color="primary" class="section-icon">bar_chart</mat-icon>
37
+ Statistics
38
+ </h3>
39
+ <div class="stats-content" *ngIf="postStats">
40
+ <p><strong>Total posts:</strong> {{ postStats.total }}</p>
41
+ <p><strong>Published:</strong> {{ postStats.published }}</p>
42
+ <p><strong>Total likes:</strong> {{ postStats.totalLikes }}</p>
43
+ </div>
44
+ <p *ngIf="!postStats" class="loading-text">Loading statistics...</p>
45
+ </div>
46
+
47
+ <div class="demo-section">
48
+ <h3 class="section-title">
49
+ <mat-icon color="primary" class="section-icon">trending_up</mat-icon>
50
+ Top 3 posts
51
+ </h3>
52
+ <div class="posts-content" *ngIf="topPosts && topPosts.length > 0">
53
+ <div *ngFor="let post of topPosts; let i = index" class="post-card">
54
+ <strong>#{{ i + 1 }}</strong> {{ post.title }}
55
+ <span class="likes-count">❤️ {{ post.likes }}</span>
56
+ </div>
57
+ </div>
58
+ </div>
59
+
60
+ <div class="demo-section">
61
+ <h3 class="section-title">Published posts</h3>
62
+ <div
63
+ class="posts-content" *ngIf="publishedPosts && publishedPosts.length > 0">
64
+ <div class="published-scroll">
65
+ <div *ngFor="let post of publishedPosts" class="published-item">
66
+ <strong>{{ post.title }}</strong> ❤️ {{ post.likes }}
67
+ </div>
68
+ </div>
69
+ </div>
70
+ </div>
71
+ </div>
72
+
73
+ <div class="all-posts-section" *ngIf="allPosts && allPosts.length > 0">
74
+ <h3 class="all-posts-title">
75
+ <mat-icon class="section-icon">list</mat-icon>
76
+ All posts (newest first)
77
+ </h3>
78
+ <div class="all-posts-content">
79
+ <div *ngFor="let post of allPosts" class="all-post-item">
80
+ <div class="post-header">
81
+ <div class="post-title-section">
82
+ <strong>{{ post.title }}</strong>
83
+ <span
84
+ class="post-status"
85
+ [class.published]="post.published" [class.draft]="!post.published">
86
+ {{ post.published ? 'Published' : 'Draft' }}
87
+ </span>
88
+ </div>
89
+ <div class="post-likes">❤️ {{ post.likes }}</div>
90
+ </div>
91
+ <div class="post-content">{{ post.content?.substring(0, 100) }}...</div>
92
+ </div>
93
+ </div>
94
+ </div>
95
+
96
+ <div class="note-section">
97
+ <p class="note-text">
98
+ <mat-icon class="note-icon">lightbulb</mat-icon>
99
+ <strong>
100
+ Note:
101
+ </strong>
102
+ All data updates automatically!
103
+ Add a post and see how all sections refresh in real time.
104
+ </p>
105
+ </div>
106
+ </div>
107
+ `,
108
+ styles: [`
109
+ .posts-paper {
110
+ border: 2px solid #9c27b0;
111
+ padding: 24px;
112
+ margin: 24px 0;
113
+ border-radius: 8px;
114
+ background-color: #faf5ff;
115
+ }
116
+
117
+ .main-title {
118
+ display: flex;
119
+ align-items: center;
120
+ gap: 8px;
121
+ margin: 0 0 16px 0;
122
+ font-size: 2rem;
123
+ font-weight: 400;
124
+ color: rgba(0, 0, 0, 0.87);
125
+ }
126
+
127
+ .main-icon {
128
+ font-size: 24px;
129
+ width: 24px;
130
+ height: 24px;
131
+ }
132
+
133
+ .add-button-section {
134
+ margin-bottom: 16px;
135
+ }
136
+
137
+ .add-button {
138
+ background-color: #9c27b0;
139
+ color: white;
140
+ border: none;
141
+ padding: 10px 15px;
142
+ border-radius: 4px;
143
+ cursor: pointer;
144
+ font-size: 0.875rem;
145
+ text-transform: none !important;
146
+ }
147
+
148
+ .add-button:hover {
149
+ background-color: #7b1fa2;
150
+ }
151
+
152
+ .demo-grid {
153
+ display: grid;
154
+ grid-template-columns: 1fr;
155
+ gap: 20px;
156
+ margin-bottom: 20px;
157
+ align-items: stretch;
158
+ }
159
+
160
+ @media (min-width: 960px) {
161
+ .demo-grid {
162
+ grid-template-columns: repeat(3, 1fr);
163
+ }
164
+ }
165
+
166
+ .demo-section {
167
+ display: flex;
168
+ flex-direction: column;
169
+ height: 100%;
170
+ }
171
+
172
+ .section-title {
173
+ display: flex;
174
+ align-items: center;
175
+ gap: 8px;
176
+ margin: 0 0 12px 0;
177
+ font-size: 1rem;
178
+ font-weight: 500;
179
+ color: rgba(0, 0, 0, 0.87);
180
+ }
181
+
182
+ .section-icon {
183
+ font-size: 20px;
184
+ width: 20px;
185
+ height: 20px;
186
+ }
187
+
188
+ .stats-content {
189
+ flex: 1;
190
+ }
191
+
192
+ .stats-content p {
193
+ margin: 0 0 8px 0;
194
+ font-size: 0.875rem;
195
+ line-height: 1.5;
196
+ }
197
+
198
+ .loading-text {
199
+ flex: 1;
200
+ font-size: 0.875rem;
201
+ color: rgba(0, 0, 0, 0.6);
202
+ }
203
+
204
+ .posts-content {
205
+ flex: 1;
206
+ }
207
+
208
+ .post-card {
209
+ padding: 8px;
210
+ margin: 4px 0;
211
+ background-color: white;
212
+ border-radius: 4px;
213
+ border: 1px solid #ddd;
214
+ font-size: 0.875rem;
215
+ }
216
+
217
+ .likes-count {
218
+ color: #9c27b0;
219
+ }
220
+
221
+ .published-scroll {
222
+ max-height: 150px;
223
+ overflow-y: auto;
224
+ flex: 1;
225
+ }
226
+
227
+ .published-item {
228
+ padding: 4px 0;
229
+ font-size: 0.9em;
230
+ border-bottom: 1px solid #eee;
231
+ font-size: 0.875rem;
232
+ }
233
+
234
+ .all-posts-section {
235
+ margin-top: 16px;
236
+ }
237
+
238
+ .all-posts-title {
239
+ display: flex;
240
+ align-items: center;
241
+ gap: 8px;
242
+ margin: 0 0 16px 0;
243
+ font-size: 1.1rem;
244
+ font-weight: 500;
245
+ color: rgba(0, 0, 0, 0.87);
246
+ }
247
+
248
+ .all-posts-content {
249
+ max-height: 200px;
250
+ overflow-y: auto;
251
+ background-color: white;
252
+ padding: 10px;
253
+ border-radius: 4px;
254
+ border: 1px solid #ddd;
255
+ }
256
+
257
+ .all-post-item {
258
+ padding: 8px 0;
259
+ border-bottom: 1px solid #eee;
260
+ font-size: 0.9em;
261
+ }
262
+
263
+ .all-post-item:last-child {
264
+ border-bottom: none;
265
+ }
266
+
267
+ .post-header {
268
+ display: flex;
269
+ justify-content: space-between;
270
+ align-items: center;
271
+ margin-bottom: 2px;
272
+ }
273
+
274
+ .post-title-section {
275
+ display: flex;
276
+ align-items: center;
277
+ gap: 10px;
278
+ }
279
+
280
+ .post-status {
281
+ margin-left: 10px;
282
+ padding: 2px 6px;
283
+ color: white;
284
+ border-radius: 3px;
285
+ font-size: 0.8em;
286
+ }
287
+
288
+ .post-status.published {
289
+ background-color: #4caf50;
290
+ }
291
+
292
+ .post-status.draft {
293
+ background-color: #ff9800;
294
+ }
295
+
296
+ .post-likes {
297
+ color: #9c27b0;
298
+ }
299
+
300
+ .post-content {
301
+ color: #666;
302
+ font-size: 0.8em;
303
+ margin-top: 2px;
304
+ }
305
+
306
+ .note-section {
307
+ margin-top: 16px;
308
+ }
309
+
310
+ .note-text {
311
+ display: flex;
312
+ align-items: center;
313
+ gap: 4px;
314
+ margin: 0;
315
+ font-size: 0.875rem;
316
+ color: rgba(0, 0, 0, 0.6);
317
+ }
318
+
319
+ .note-icon {
320
+ font-size: 16px;
321
+ width: 16px;
322
+ height: 16px;
323
+ }
324
+ `],
325
+ })
326
+ export class PostsLiveQueryDemoComponent {
327
+ @Input() allPosts: PostEntity[] = [];
328
+ @Input() topPosts: PostEntity[] = [];
329
+ @Input() publishedPosts: PostEntity[] = [];
330
+ @Input() postStats: {
331
+ total: number;
332
+ published: number;
333
+ totalLikes: number;
334
+ } = { total: 0, published: 0, totalLikes: 0 };
335
+ @Input() addSamplePost!: () => void;
336
+ }
@@ -0,0 +1,268 @@
1
+ import { CommonModule } from '@angular/common';
2
+ import {
3
+ Component, Input,
4
+ } from '@angular/core';
5
+ import { MatButtonModule } from '@angular/material/button';
6
+ import { MatCardModule } from '@angular/material/card';
7
+ import { MatIconModule } from '@angular/material/icon';
8
+ import { MatListModule } from '@angular/material/list';
9
+
10
+ import { PostEntity } from '../../entities/post.entity';
11
+ import { ProfileEntity } from '../../entities/profile.entity';
12
+ import { UserEntity } from '../../entities/user.entity';
13
+
14
+ @Component({
15
+ selector: 'app-typescript-demo',
16
+ standalone: true,
17
+ imports: [
18
+ CommonModule, MatButtonModule, MatCardModule, MatIconModule, MatListModule,
19
+ ],
20
+ template: `
21
+ <div class="typescript-paper">
22
+ <h2 class="main-title">
23
+ <mat-icon color="primary" class="main-icon">code</mat-icon>
24
+ TypeScript Typing Test
25
+ </h2>
26
+ <p class="main-description">
27
+ <strong>Instructions:</strong>
28
+ Hover over the variables below in your IDE and
29
+ verify TypeScript shows full types! Click items to
30
+ see auto-complete in the console.
31
+ </p>
32
+
33
+ <div class="demo-grid">
34
+ <div class="demo-section">
35
+ <h3 class="section-title">
36
+ <mat-icon color="primary" class="section-icon">people</mat-icon>
37
+ Users (UserEntity[])
38
+ </h3>
39
+ <div class="type-info">Typ: <code>UserEntity[] | undefined</code></div>
40
+ <div class="items-content" *ngIf="users && users.length > 0">
41
+ <div class="items-scroll">
42
+ <div *ngFor="let user of users"
43
+ class="item-card"
44
+ (click)="onUserClick(user)"
45
+ title="Click to see auto-complete in the console">
46
+ <div class="item-text">
47
+ <strong>{{ user.name }}</strong> - {{ user.email }}
48
+ </div>
49
+ </div>
50
+ </div>
51
+ </div>
52
+ </div>
53
+
54
+ <div class="demo-section">
55
+ <h3 class="section-title">
56
+ <mat-icon color="primary" class="section-icon">description</mat-icon>
57
+ Posts (PostEntity[])
58
+ </h3>
59
+ <div class="type-info">Typ: <code>PostEntity[] | undefined</code></div>
60
+ <div class="items-content" *ngIf="posts && posts.length > 0">
61
+ <div class="items-scroll">
62
+ <div *ngFor="let post of posts"
63
+ class="item-card"
64
+ (click)="onPostClick(post)"
65
+ title="Click to see auto-complete in the console">
66
+ <div class="item-text">
67
+ <strong>
68
+ {{ post.title }}
69
+ </strong> - {{ post.published ? 'Published' : 'Draft' }}
70
+ </div>
71
+ </div>
72
+ </div>
73
+ </div>
74
+ </div>
75
+
76
+ <div class="demo-section">
77
+ <h3 class="section-title">
78
+ <mat-icon color="primary" class="section-icon">account_circle</mat-icon>
79
+ Profiles (ProfileEntity[])
80
+ </h3>
81
+ <div class="type-info">Type: <code>ProfileEntity[] | undefined</code></div>
82
+ <div class="items-content" *ngIf="profiles && profiles.length > 0">
83
+ <div class="items-scroll">
84
+ <div *ngFor="let profile of profiles"
85
+ class="item-card"
86
+ (click)="onProfileClick(profile)"
87
+ title="Click to see auto-complete in the console">
88
+ <div class="item-text">
89
+ <strong>
90
+ User {{ profile.userId }}
91
+ </strong> - {{ profile.bio?.substring(0, 30) }}...
92
+ </div>
93
+ </div>
94
+ </div>
95
+ </div>
96
+ </div>
97
+ </div>
98
+
99
+ <div class="ide-test-section">
100
+ <div class="ide-test-content">
101
+ <p><strong>IDE Test:</strong></p>
102
+ <p>
103
+ 1. Hover over variables
104
+ <code>users</code>, <code>posts</code>, <code>profiles</code></p>
105
+ <p>
106
+ 2. Verify TypeScript shows full types:
107
+ (UserEntity[], PostEntity[], ProfileEntity[])</p>
108
+ <p>
109
+ 3. Click items to see auto-complete inside handle* functions
110
+ </p>
111
+ <p>
112
+ 4. Check that IDE suggests properties like
113
+ <code>user.name</code>, <code>post.title</code>, <code>profile.bio</code
114
+ ></p>
115
+ </div>
116
+ </div>
117
+ </div>
118
+ `,
119
+ styles: [`
120
+ .typescript-paper {
121
+ border: 2px solid #ff9800;
122
+ padding: 24px;
123
+ margin: 24px 0;
124
+ border-radius: 8px;
125
+ background-color: #fff8e1;
126
+ }
127
+
128
+ .main-title {
129
+ display: flex;
130
+ align-items: center;
131
+ gap: 8px;
132
+ margin: 0 0 16px 0;
133
+ font-size: 2rem;
134
+ font-weight: 400;
135
+ color: rgba(0, 0, 0, 0.87);
136
+ }
137
+
138
+ .main-icon {
139
+ font-size: 24px;
140
+ width: 24px;
141
+ height: 24px;
142
+ }
143
+
144
+ .main-description {
145
+ color: rgba(0, 0, 0, 0.6);
146
+ margin: 0 0 24px 0;
147
+ font-size: 0.875rem;
148
+ line-height: 1.5;
149
+ }
150
+
151
+ .demo-grid {
152
+ display: grid;
153
+ grid-template-columns: 1fr;
154
+ gap: 24px;
155
+ align-items: stretch;
156
+ }
157
+
158
+ @media (min-width: 960px) {
159
+ .demo-grid {
160
+ grid-template-columns: repeat(3, 1fr);
161
+ grid-template-rows: 1fr;
162
+ }
163
+ }
164
+
165
+ .demo-section {
166
+ display: flex;
167
+ flex-direction: column;
168
+ flex: 1;
169
+ }
170
+
171
+ .section-title {
172
+ display: flex;
173
+ align-items: center;
174
+ gap: 8px;
175
+ margin: 0 0 16px 0;
176
+ font-size: 1.25rem;
177
+ font-weight: 500;
178
+ color: rgba(0, 0, 0, 0.87);
179
+ }
180
+
181
+ .section-icon {
182
+ font-size: 20px;
183
+ width: 20px;
184
+ height: 20px;
185
+ }
186
+
187
+ .type-info {
188
+ font-size: 0.875rem;
189
+ color: rgba(0, 0, 0, 0.6);
190
+ margin-bottom: 16px;
191
+ }
192
+
193
+ .type-info code {
194
+ background-color: #f5f5f5;
195
+ padding: 2px 4px;
196
+ border-radius: 2px;
197
+ font-family: 'Courier New', monospace;
198
+ font-size: 0.8rem;
199
+ }
200
+
201
+ .items-content {
202
+ flex: 1;
203
+ display: flex;
204
+ flex-direction: column;
205
+ }
206
+
207
+ .items-scroll {
208
+ max-height: 150px;
209
+ overflow-y: auto;
210
+ padding: 4px;
211
+ }
212
+
213
+ .item-card {
214
+ padding: 4px 0;
215
+ border-bottom: 1px solid #eee;
216
+ cursor: pointer;
217
+ }
218
+
219
+ .item-card:hover {
220
+ background-color: rgba(0, 0, 0, 0.04);
221
+ }
222
+
223
+ .item-card:last-child {
224
+ border-bottom: none;
225
+ }
226
+
227
+ .item-text {
228
+ font-size: 0.875rem;
229
+ line-height: 1.5;
230
+ }
231
+
232
+ .ide-test-section {
233
+ margin-top: 16px;
234
+ }
235
+
236
+ .ide-test-content {
237
+ padding: 16px;
238
+ background-color: #e3f2fd;
239
+ border-radius: 4px;
240
+ }
241
+
242
+ .ide-test-content p {
243
+ margin: 0 0 8px 0;
244
+ font-size: 0.875rem;
245
+ line-height: 1.5;
246
+ }
247
+
248
+ .ide-test-content p:last-child {
249
+ margin-bottom: 0;
250
+ }
251
+
252
+ .ide-test-content code {
253
+ background-color: #f5f5f5;
254
+ padding: 2px 4px;
255
+ border-radius: 2px;
256
+ font-family: 'Courier New', monospace;
257
+ font-size: 0.8rem;
258
+ }
259
+ `],
260
+ })
261
+ export class TypeScriptDemoComponent {
262
+ @Input() users: UserEntity[] = [];
263
+ @Input() posts: PostEntity[] = [];
264
+ @Input() profiles: ProfileEntity[] = [];
265
+ @Input() onUserClick!: (_user: UserEntity) => void;
266
+ @Input() onPostClick!: (_post: PostEntity) => void;
267
+ @Input() onProfileClick!: (_profile: ProfileEntity) => void;
268
+ }
@@ -0,0 +1,25 @@
1
+ import {
2
+ BaseEntity,
3
+ defineEntity,
4
+ } from 'idb-orm';
5
+ import { z } from 'zod';
6
+
7
+ const PostTagSchema = z.object({
8
+ id: z.number().optional(),
9
+ sourceId: z.number(),
10
+ targetId: z.number(),
11
+ });
12
+
13
+ export class PostTagEntity extends BaseEntity<number> {
14
+ sourceId!: number;
15
+ targetId!: number;
16
+ }
17
+
18
+ defineEntity(PostTagEntity, {
19
+ tableName: 'post_tags',
20
+ schema: PostTagSchema,
21
+ columns: {
22
+ sourceId: { indexed: true },
23
+ targetId: { indexed: true },
24
+ },
25
+ });
@@ -0,0 +1,49 @@
1
+ import {
2
+ BaseEntity,
3
+ defineEntity,
4
+ } from 'idb-orm';
5
+ import { z } from 'zod';
6
+
7
+ import type { TagEntity } from './tag.entity';
8
+
9
+ const PostSchema = z.object({
10
+ id: z.number().optional(),
11
+ title: z.string().min(1, 'Title is required'),
12
+ content: z.string().min(1, 'Content is required'),
13
+ authorId: z.number().positive('Author ID must be positive'),
14
+ published: z.boolean().optional(),
15
+ postTags: z.array(z.string()).optional(),
16
+ likes: z.number().min(0, 'Likes must be non-negative').optional(),
17
+ createdAt: z.number().optional(),
18
+ updatedAt: z.number().optional(),
19
+ });
20
+
21
+ export class PostEntity extends BaseEntity<number> {
22
+ title!: string;
23
+ content!: string;
24
+ authorId!: number;
25
+ published?: boolean;
26
+ postTags?: string[];
27
+ likes?: number;
28
+ createdAt?: number;
29
+ updatedAt?: number;
30
+ tags?: TagEntity[];
31
+ }
32
+
33
+ defineEntity(PostEntity, {
34
+ tableName: 'posts',
35
+ schema: PostSchema,
36
+ columns: {
37
+ title: { required: true, indexed: true },
38
+ content: { required: true },
39
+ authorId: { required: true, indexed: true },
40
+ published: { default: false, indexed: true },
41
+ postTags: {},
42
+ likes: { default: 0, indexed: true },
43
+ createdAt: { default: () => Date.now(), indexed: true },
44
+ updatedAt: { default: () => Date.now() },
45
+ },
46
+ relations: {
47
+ tags: { type: 'many-to-many', target: 'tags', joinTable: 'post_tags' },
48
+ },
49
+ });