fontastic 0.1.3 → 0.2.0

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 (452) hide show
  1. package/.editorconfig +16 -16
  2. package/.github/FUNDING.yml +3 -12
  3. package/.github/ISSUE_TEMPLATE/bug_report.yml +41 -0
  4. package/.github/ISSUE_TEMPLATE/feature_request.yml +29 -0
  5. package/.github/dependabot.yml +6 -6
  6. package/.github/pull_request_template.md +21 -21
  7. package/.github/workflows/claude-code-review.yml +45 -0
  8. package/.github/workflows/claude.yml +46 -0
  9. package/.github/workflows/macos.yml +77 -58
  10. package/.github/workflows/release.yml +165 -0
  11. package/.github/workflows/ubuntu.yml +87 -70
  12. package/.github/workflows/windows.yml +81 -64
  13. package/.husky/pre-commit +1 -0
  14. package/.lintstagedrc.json +6 -0
  15. package/.node-version +1 -1
  16. package/.postcssrc.json +5 -0
  17. package/.prettierignore +8 -0
  18. package/.prettierrc +18 -0
  19. package/.vscode/launch.json +42 -45
  20. package/.vscode/tasks.json +48 -48
  21. package/CODE_OF_CONDUCT.md +128 -45
  22. package/HOW_TO.md +39 -0
  23. package/LICENSE.md +7 -0
  24. package/README.md +103 -111
  25. package/angular.json +166 -194
  26. package/app/Application.js +39 -39
  27. package/app/Application.ts +38 -38
  28. package/app/config/alert.ts +31 -31
  29. package/app/config/database.js +194 -202
  30. package/app/config/database.ts +194 -202
  31. package/app/config/index.js +19 -19
  32. package/app/config/index.ts +2 -2
  33. package/app/config/mimes.js +48 -48
  34. package/app/config/mimes.ts +53 -53
  35. package/app/config/system.js +578 -496
  36. package/app/config/system.ts +580 -498
  37. package/app/config/themes.ts +38 -38
  38. package/app/core/AppLogger.js +30 -30
  39. package/app/core/AppLogger.ts +37 -37
  40. package/app/core/ConfigManager.js +129 -113
  41. package/app/core/ConfigManager.ts +150 -137
  42. package/app/core/ConnectionManager.js +93 -82
  43. package/app/core/ConnectionManager.ts +107 -95
  44. package/app/core/FontCatalog.js +32 -50
  45. package/app/core/FontCatalog.ts +20 -42
  46. package/app/core/FontFinder.js +72 -85
  47. package/app/core/FontFinder.ts +75 -96
  48. package/app/core/FontManager.js +128 -164
  49. package/app/core/FontManager.ts +136 -172
  50. package/app/core/FontObject.js +53 -53
  51. package/app/core/FontObject.ts +62 -62
  52. package/app/core/MessageHandler.js +270 -271
  53. package/app/core/MessageHandler.ts +330 -337
  54. package/app/core/SystemManager.js +132 -139
  55. package/app/core/SystemManager.ts +160 -171
  56. package/app/core/menu/Example.ts +279 -279
  57. package/app/core/menu/MenuBuilder.js +43 -44
  58. package/app/core/menu/MenuBuilder.ts +51 -53
  59. package/app/core/menu/templates/DarwinTemplate.js +124 -126
  60. package/app/core/menu/templates/DarwinTemplate.ts +144 -142
  61. package/app/core/menu/templates/SystemTemplate.js +121 -123
  62. package/app/core/menu/templates/SystemTemplate.ts +136 -139
  63. package/app/database/entity/Collection.schema.js +98 -98
  64. package/app/database/entity/Collection.schema.ts +68 -68
  65. package/app/database/entity/Logger.schema.js +49 -49
  66. package/app/database/entity/Logger.schema.ts +26 -26
  67. package/app/database/entity/Store.schema.js +159 -164
  68. package/app/database/entity/Store.schema.ts +110 -116
  69. package/app/database/entity/index.js +19 -19
  70. package/app/database/entity/index.ts +2 -2
  71. package/app/database/repository/Collection.repository.js +277 -180
  72. package/app/database/repository/Collection.repository.ts +291 -190
  73. package/app/database/repository/Logger.repository.js +24 -24
  74. package/app/database/repository/Logger.repository.ts +10 -10
  75. package/app/database/repository/Store.repository.js +324 -327
  76. package/app/database/repository/Store.repository.ts +349 -358
  77. package/app/database/repository/User.repository.js +11 -11
  78. package/app/database/repository/User.repository.ts +8 -8
  79. package/app/database/repository/index.js +19 -19
  80. package/app/database/repository/index.ts +2 -2
  81. package/app/enums/ChannelType.js +62 -61
  82. package/app/enums/ChannelType.ts +67 -65
  83. package/app/enums/StorageType.js +19 -18
  84. package/app/enums/{storageType.ts → StorageType.ts} +15 -14
  85. package/app/enums/index.js +18 -18
  86. package/app/enums/index.ts +2 -2
  87. package/app/helpers/command.js +27 -27
  88. package/app/helpers/command.ts +19 -19
  89. package/app/helpers/random.js +15 -15
  90. package/app/helpers/random.ts +12 -12
  91. package/app/main.js +97 -81
  92. package/app/main.ts +114 -94
  93. package/app/package-lock.json +2240 -2649
  94. package/app/package.json +23 -35
  95. package/app/types/AppAlert.js +2 -2
  96. package/app/types/AppAlert.ts +8 -8
  97. package/app/types/AuthUser.js +6 -6
  98. package/app/types/AuthUser.ts +5 -5
  99. package/app/types/Breadcrumb.js +2 -2
  100. package/app/types/Breadcrumb.ts +4 -4
  101. package/app/types/FontMetrics.ts +7 -0
  102. package/app/types/ImportOptions.js +6 -6
  103. package/app/types/ImportOptions.ts +3 -3
  104. package/app/types/QueryOptions.js +2 -2
  105. package/app/types/QueryOptions.ts +12 -12
  106. package/app/types/SystemConfig.js +2 -2
  107. package/app/types/SystemConfig.ts +127 -132
  108. package/app/types/SystemStats.js +2 -2
  109. package/app/types/SystemStats.ts +6 -7
  110. package/app/types/SystemTheme.js +2 -2
  111. package/app/types/SystemTheme.ts +5 -5
  112. package/app/types/index.js +24 -24
  113. package/app/types/index.ts +9 -8
  114. package/commitlint.config.mjs +3 -0
  115. package/e2e/main.spec.ts +57 -59
  116. package/e2e/playwright.config.ts +20 -19
  117. package/e2e/tsconfig.e2e.json +13 -13
  118. package/electron-builder.json +34 -71
  119. package/eslint.config.mjs +111 -0
  120. package/package.json +122 -121
  121. package/src/app/app.component.html +1 -1
  122. package/src/app/app.component.spec.ts +21 -0
  123. package/src/app/app.component.ts +40 -37
  124. package/src/app/core/core.module.ts +8 -11
  125. package/src/app/core/services/database/database.service.ts +410 -286
  126. package/src/app/core/services/electron/electron.service.spec.ts +12 -12
  127. package/src/app/core/services/electron/electron.service.ts +62 -62
  128. package/src/app/core/services/index.ts +5 -14
  129. package/src/app/core/services/message/message.service.ts +291 -289
  130. package/src/app/core/services/news/news.service.ts +52 -94
  131. package/src/app/core/services/presentation/presentation.service.ts +253 -341
  132. package/src/app/home/home.component.html +7 -0
  133. package/src/app/home/home.component.spec.ts +31 -0
  134. package/src/app/home/home.component.ts +15 -0
  135. package/src/app/layout/aside/aside.component.html +35 -9
  136. package/src/app/layout/aside/aside.component.ts +60 -26
  137. package/src/app/layout/footer/footer.component.html +5 -11
  138. package/src/app/layout/footer/footer.component.ts +13 -32
  139. package/src/app/layout/header/header.component.html +118 -35
  140. package/src/app/layout/header/header.component.ts +52 -229
  141. package/src/app/layout/layout.component.html +32 -14
  142. package/src/app/layout/layout.component.ts +19 -23
  143. package/src/app/layout/main/main.component.html +46 -0
  144. package/src/app/layout/main/main.component.ts +20 -0
  145. package/src/app/layout/navigation/library/library.component.ts +51 -0
  146. package/src/app/layout/navigation/navigation.component.html +126 -142
  147. package/src/app/layout/navigation/navigation.component.ts +372 -302
  148. package/src/app/layout/navigation/stats/stats.component.ts +56 -0
  149. package/src/app/settings/ai-keys/ai-keys.component.html +56 -0
  150. package/src/app/settings/ai-keys/ai-keys.component.ts +44 -0
  151. package/src/app/settings/danger-zone/danger-zone.component.html +22 -0
  152. package/src/app/settings/danger-zone/danger-zone.component.ts +37 -0
  153. package/src/app/settings/logs/logs.component.html +41 -0
  154. package/src/app/settings/logs/logs.component.ts +43 -0
  155. package/src/app/settings/news-api/news-api.component.html +40 -0
  156. package/src/app/settings/news-api/news-api.component.ts +57 -0
  157. package/src/app/settings/pages/general/general.component.html +7 -0
  158. package/src/app/settings/pages/general/general.component.ts +20 -0
  159. package/src/app/settings/pages/logs/logs.component.html +1 -0
  160. package/src/app/settings/pages/logs/logs.component.ts +10 -0
  161. package/src/app/settings/pages/system/system.component.html +1 -0
  162. package/src/app/settings/pages/system/system.component.ts +10 -0
  163. package/src/app/settings/quick-actions/quick-actions.component.html +49 -0
  164. package/src/app/settings/quick-actions/quick-actions.component.ts +40 -0
  165. package/src/app/settings/settings.component.html +47 -0
  166. package/src/app/settings/settings.component.ts +10 -0
  167. package/src/app/settings/system-info/system-info.component.html +21 -0
  168. package/src/app/settings/system-info/system-info.component.ts +42 -0
  169. package/src/app/settings/theme/theme.component.html +15 -0
  170. package/src/app/settings/theme/theme.component.ts +22 -0
  171. package/src/app/shared/components/collapsible-panel/collapsible-panel.component.ts +43 -0
  172. package/src/app/shared/components/context-menu/context-menu.component.ts +67 -0
  173. package/src/app/shared/components/datagrid/datagrid.component.html +96 -0
  174. package/src/app/shared/components/datagrid/datagrid.component.ts +49 -0
  175. package/src/app/shared/components/index.ts +11 -16
  176. package/src/app/shared/components/inspector/inspector.component.html +140 -0
  177. package/src/app/shared/components/inspector/inspector.component.ts +41 -0
  178. package/src/app/shared/components/page-not-found/page-not-found.component.html +1 -3
  179. package/src/app/shared/components/page-not-found/page-not-found.component.spec.ts +22 -0
  180. package/src/app/shared/components/page-not-found/page-not-found.component.ts +14 -12
  181. package/src/app/shared/components/panel/panel.component.html +5 -0
  182. package/src/app/shared/components/panel/panel.component.ts +8 -0
  183. package/src/app/shared/components/preview/preview.component.html +36 -0
  184. package/src/app/shared/components/preview/preview.component.ts +68 -0
  185. package/src/app/shared/components/prompt-dialog/prompt-dialog.component.html +36 -0
  186. package/src/app/shared/components/prompt-dialog/prompt-dialog.component.ts +39 -0
  187. package/src/app/shared/components/search/search.component.html +329 -54
  188. package/src/app/shared/components/search/search.component.ts +221 -77
  189. package/src/app/shared/components/spinner/spinner.component.ts +10 -26
  190. package/src/app/shared/components/toolbar/toolbar.component.html +207 -0
  191. package/src/app/shared/components/toolbar/toolbar.component.ts +99 -0
  192. package/src/app/shared/components/waterfall/waterfall.component.html +119 -0
  193. package/src/app/shared/components/waterfall/waterfall.component.ts +119 -0
  194. package/src/app/shared/directives/autofocus/autofocus.directive.ts +14 -0
  195. package/src/app/shared/directives/index.ts +2 -2
  196. package/src/app/shared/directives/webview/webview.directive.spec.ts +8 -0
  197. package/src/app/shared/directives/webview/webview.directive.ts +9 -8
  198. package/src/app/shared/shared.module.ts +13 -92
  199. package/src/assets/background.jpg +0 -0
  200. package/src/assets/fonts/md/MaterialIcons-Regular.woff2 +0 -0
  201. package/src/assets/fonts/md/MaterialSymbolsOutlined.woff2 +0 -0
  202. package/src/assets/i18n/en.json +12 -12
  203. package/src/assets/icons/electron.bmp +0 -0
  204. package/src/assets/icons/favicon.256x256.png +0 -0
  205. package/src/assets/icons/favicon.512x512.png +0 -0
  206. package/src/assets/icons/favicon.icns +0 -0
  207. package/src/assets/icons/favicon.ico +0 -0
  208. package/src/assets/icons/favicon.png +0 -0
  209. package/src/environments/environment.dev.ts +4 -4
  210. package/src/environments/environment.prod.ts +4 -4
  211. package/src/environments/environment.ts +4 -4
  212. package/src/favicon.ico +0 -0
  213. package/src/index.html +19 -55
  214. package/src/main.ts +65 -15
  215. package/src/styles/base/reset.css +41 -0
  216. package/src/styles/base/variables.css +356 -0
  217. package/src/styles/components/buttons.css +108 -0
  218. package/src/styles/components/cards.css +60 -0
  219. package/src/styles/components/forms.css +70 -0
  220. package/src/styles/components/navigation.css +24 -0
  221. package/src/styles/components/scrollbox.css +50 -0
  222. package/src/styles/components/spinner.css +7 -0
  223. package/src/styles/components/splash.css +47 -0
  224. package/src/styles/components/toolbar.css +92 -0
  225. package/src/styles/fonts/md.css +39 -0
  226. package/src/styles/themes/dashboard.scss +293 -294
  227. package/src/styles/themes/euphoria.scss +284 -283
  228. package/src/styles/themes/mellow.scss +281 -280
  229. package/src/styles/themes/midnight.scss +284 -283
  230. package/src/styles/themes/passion.scss +281 -280
  231. package/src/styles/themes/swiss.scss +284 -283
  232. package/src/styles.css +17 -0
  233. package/src/tsconfig.app.json +11 -20
  234. package/src/tsconfig.spec.json +10 -23
  235. package/src/typings.d.ts +9 -9
  236. package/src/vitest.d.ts +1 -0
  237. package/tsconfig.json +38 -42
  238. package/tsconfig.serve.json +27 -27
  239. package/.eslintrc.json +0 -54
  240. package/.github/ISSUE_TEMPLATE/bug_report.md +0 -29
  241. package/.github/ISSUE_TEMPLATE/feature_request.md +0 -20
  242. package/.github/stale.yml +0 -17
  243. package/CONTRIBUTING.md +0 -72
  244. package/LICENSE +0 -21
  245. package/angular.webpack.js +0 -32
  246. package/app/core/FontInstaller.js +0 -67
  247. package/app/core/FontInstaller.ts +0 -63
  248. package/docs/logo.png +0 -0
  249. package/docs/screenshots/readme.md +0 -7
  250. package/docs/screenshots/screen-grab1.png +0 -0
  251. package/docs/screenshots/screen-grab2.png +0 -0
  252. package/docs/screenshots/screen-grab3.png +0 -0
  253. package/src/app/app-routing.module.ts +0 -15
  254. package/src/app/app.component.scss +0 -0
  255. package/src/app/app.component.spec.ts.dist +0 -33
  256. package/src/app/app.module.ts +0 -63
  257. package/src/app/core/model/AuthUserModel.ts +0 -7
  258. package/src/app/core/model/CustomThemeModel.ts +0 -7
  259. package/src/app/core/model/DbConnectionModel.ts +0 -16
  260. package/src/app/core/model/ImportOptionsModel.ts +0 -5
  261. package/src/app/core/model/LatestNewsModel.ts +0 -7
  262. package/src/app/core/model/SearchFormModel.ts +0 -10
  263. package/src/app/core/model/index.ts +0 -6
  264. package/src/app/core/services/alert/alert.service.spec.ts +0 -22
  265. package/src/app/core/services/alert/alert.service.ts +0 -71
  266. package/src/app/core/services/auth/auth.service.ts +0 -30
  267. package/src/app/core/services/boot/boot.service.ts +0 -33
  268. package/src/app/core/services/breadcrumb/breadcrumb.service.spec.ts +0 -22
  269. package/src/app/core/services/breadcrumb/breadcrumb.service.ts +0 -30
  270. package/src/app/core/services/config/config.service.ts +0 -63
  271. package/src/app/core/services/font/font.service.ts +0 -79
  272. package/src/app/core/services/gravatar/gravatar.service.spec.ts +0 -16
  273. package/src/app/core/services/gravatar/gravatar.service.ts +0 -18
  274. package/src/app/core/services/modal/modal.service.spec.ts +0 -16
  275. package/src/app/core/services/modal/modal.service.ts +0 -34
  276. package/src/app/core/services/utils/utils.service.spec.ts +0 -12
  277. package/src/app/core/services/utils/utils.service.ts +0 -144
  278. package/src/app/layout/aside/aside.component.scss +0 -1
  279. package/src/app/layout/footer/footer.component.scss +0 -27
  280. package/src/app/layout/header/header.component.scss +0 -60
  281. package/src/app/layout/layout.component.scss +0 -119
  282. package/src/app/layout/layout.module.ts +0 -32
  283. package/src/app/layout/layout.service.ts +0 -13
  284. package/src/app/layout/navigation/navigation.component.scss +0 -0
  285. package/src/app/pages/main/grid/grid.component.html +0 -160
  286. package/src/app/pages/main/grid/grid.component.scss +0 -3
  287. package/src/app/pages/main/grid/grid.component.ts +0 -98
  288. package/src/app/pages/main/inspect/inspect.component.html +0 -7
  289. package/src/app/pages/main/inspect/inspect.component.scss +0 -3
  290. package/src/app/pages/main/inspect/inspect.component.ts +0 -66
  291. package/src/app/pages/main/main-routing.module.ts +0 -30
  292. package/src/app/pages/main/main.component.html +0 -6
  293. package/src/app/pages/main/main.component.scss +0 -191
  294. package/src/app/pages/main/main.component.ts +0 -35
  295. package/src/app/pages/main/main.module.ts +0 -28
  296. package/src/app/pages/main/preview/preview.component.html +0 -46
  297. package/src/app/pages/main/preview/preview.component.scss +0 -0
  298. package/src/app/pages/main/preview/preview.component.ts +0 -239
  299. package/src/app/pages/main/toolbar/toolbar.component.html +0 -78
  300. package/src/app/pages/main/toolbar/toolbar.component.scss +0 -3
  301. package/src/app/pages/main/toolbar/toolbar.component.ts +0 -132
  302. package/src/app/shared/components/alert/alert.component.html +0 -9
  303. package/src/app/shared/components/alert/alert.component.scss +0 -0
  304. package/src/app/shared/components/alert/alert.component.ts +0 -37
  305. package/src/app/shared/components/breadcrumbs/breadcrumbs.component.html +0 -14
  306. package/src/app/shared/components/breadcrumbs/breadcrumbs.component.scss +0 -0
  307. package/src/app/shared/components/breadcrumbs/breadcrumbs.component.ts +0 -36
  308. package/src/app/shared/components/button/button.component.html +0 -3
  309. package/src/app/shared/components/button/button.component.scss +0 -2
  310. package/src/app/shared/components/button/button.component.ts +0 -22
  311. package/src/app/shared/components/collection/create-collection/create-collection.component.html +0 -41
  312. package/src/app/shared/components/collection/create-collection/create-collection.component.scss +0 -0
  313. package/src/app/shared/components/collection/create-collection/create-collection.component.ts +0 -68
  314. package/src/app/shared/components/collection/form-collection/form-collection.component.html +0 -1
  315. package/src/app/shared/components/collection/form-collection/form-collection.component.scss +0 -0
  316. package/src/app/shared/components/collection/form-collection/form-collection.component.ts +0 -15
  317. package/src/app/shared/components/collection/update-collection/update-collection.component.html +0 -40
  318. package/src/app/shared/components/collection/update-collection/update-collection.component.scss +0 -0
  319. package/src/app/shared/components/collection/update-collection/update-collection.component.ts +0 -71
  320. package/src/app/shared/components/glyph-list/glyph-list.component.html +0 -40
  321. package/src/app/shared/components/glyph-list/glyph-list.component.scss +0 -1
  322. package/src/app/shared/components/glyph-list/glyph-list.component.ts +0 -165
  323. package/src/app/shared/components/glyph-view/glyph-view.component.html +0 -23
  324. package/src/app/shared/components/glyph-view/glyph-view.component.scss +0 -0
  325. package/src/app/shared/components/glyph-view/glyph-view.component.ts +0 -228
  326. package/src/app/shared/components/loading/loading.component.html +0 -1
  327. package/src/app/shared/components/loading/loading.component.scss +0 -0
  328. package/src/app/shared/components/loading/loading.component.ts +0 -26
  329. package/src/app/shared/components/modal/modal.component.html +0 -31
  330. package/src/app/shared/components/modal/modal.component.scss +0 -0
  331. package/src/app/shared/components/modal/modal.component.ts +0 -51
  332. package/src/app/shared/components/page-not-found/page-not-found.component.scss +0 -0
  333. package/src/app/shared/components/paginator/paginator.component.html +0 -21
  334. package/src/app/shared/components/paginator/paginator.component.scss +0 -3
  335. package/src/app/shared/components/paginator/paginator.component.ts +0 -133
  336. package/src/app/shared/components/pairing/pairing.component.html +0 -20
  337. package/src/app/shared/components/pairing/pairing.component.scss +0 -0
  338. package/src/app/shared/components/pairing/pairing.component.ts +0 -33
  339. package/src/app/shared/components/search/search.component.scss +0 -10
  340. package/src/app/shared/components/settings/database/database.component.html +0 -151
  341. package/src/app/shared/components/settings/database/database.component.scss +0 -0
  342. package/src/app/shared/components/settings/database/database.component.ts +0 -145
  343. package/src/app/shared/components/settings/general/general.component.html +0 -170
  344. package/src/app/shared/components/settings/general/general.component.scss +0 -0
  345. package/src/app/shared/components/settings/general/general.component.ts +0 -150
  346. package/src/app/shared/components/settings/logs/logs.component.html +0 -44
  347. package/src/app/shared/components/settings/logs/logs.component.scss +0 -0
  348. package/src/app/shared/components/settings/logs/logs.component.ts +0 -37
  349. package/src/app/shared/components/settings/news/news.component.html +0 -41
  350. package/src/app/shared/components/settings/news/news.component.scss +0 -0
  351. package/src/app/shared/components/settings/news/news.component.ts +0 -52
  352. package/src/app/shared/components/settings/settings.component.html +0 -46
  353. package/src/app/shared/components/settings/settings.component.scss +0 -0
  354. package/src/app/shared/components/settings/settings.component.ts +0 -23
  355. package/src/app/shared/components/settings/system/system.component.html +0 -30
  356. package/src/app/shared/components/settings/system/system.component.scss +0 -0
  357. package/src/app/shared/components/settings/system/system.component.ts +0 -43
  358. package/src/app/shared/components/settings/theme/theme.component.html +0 -22
  359. package/src/app/shared/components/settings/theme/theme.component.scss +0 -0
  360. package/src/app/shared/components/settings/theme/theme.component.ts +0 -50
  361. package/src/app/shared/components/spinner/spinner.component.html +0 -1
  362. package/src/app/shared/components/spinner/spinner.component.scss +0 -0
  363. package/src/app/shared/components/store/store.component.html +0 -22
  364. package/src/app/shared/components/store/store.component.scss +0 -10
  365. package/src/app/shared/components/store/store.component.ts +0 -58
  366. package/src/app/shared/components/tables/tables.component.html +0 -47
  367. package/src/app/shared/components/tables/tables.component.scss +0 -10
  368. package/src/app/shared/components/tables/tables.component.ts +0 -81
  369. package/src/app/shared/components/typescale/typescale.component.html +0 -53
  370. package/src/app/shared/components/typescale/typescale.component.scss +0 -0
  371. package/src/app/shared/components/typescale/typescale.component.ts +0 -168
  372. package/src/app/shared/directives/gravatar/gravatar.directive.ts +0 -29
  373. package/src/app/shared/pipes/installable.pipe.ts +0 -10
  374. package/src/app/shared/pipes/prettybytes.pipe.ts +0 -10
  375. package/src/app/shared/pipes/safehtml.pipe.ts +0 -10
  376. package/src/assets/fonts/fa/fa-brands-400.eot +0 -0
  377. package/src/assets/fonts/fa/fa-brands-400.svg +0 -3535
  378. package/src/assets/fonts/fa/fa-brands-400.ttf +0 -0
  379. package/src/assets/fonts/fa/fa-brands-400.woff +0 -0
  380. package/src/assets/fonts/fa/fa-brands-400.woff2 +0 -0
  381. package/src/assets/fonts/fa/fa-regular-400.eot +0 -0
  382. package/src/assets/fonts/fa/fa-regular-400.svg +0 -803
  383. package/src/assets/fonts/fa/fa-regular-400.ttf +0 -0
  384. package/src/assets/fonts/fa/fa-regular-400.woff +0 -0
  385. package/src/assets/fonts/fa/fa-regular-400.woff2 +0 -0
  386. package/src/assets/fonts/fa/fa-solid-900.eot +0 -0
  387. package/src/assets/fonts/fa/fa-solid-900.svg +0 -4700
  388. package/src/assets/fonts/fa/fa-solid-900.ttf +0 -0
  389. package/src/assets/fonts/fa/fa-solid-900.woff +0 -0
  390. package/src/assets/fonts/fa/fa-solid-900.woff2 +0 -0
  391. package/src/assets/fonts/md/MaterialIcons-Regular.eot +0 -0
  392. package/src/assets/fonts/md/MaterialIcons-Regular.ijmap +0 -1
  393. package/src/assets/fonts/md/MaterialIcons-Regular.svg +0 -2373
  394. package/src/assets/fonts/md/MaterialIcons-Regular.ttf +0 -0
  395. package/src/assets/fonts/md/MaterialIcons-Regular.woff +0 -0
  396. package/src/assets/fonts/md/README.md +0 -9
  397. package/src/assets/fonts/md/codepoints +0 -932
  398. package/src/assets/fonts/md/material-icons.css +0 -36
  399. package/src/assets/fonts/octicon/octicon.eot +0 -0
  400. package/src/assets/fonts/octicon/octicon.svg +0 -378
  401. package/src/assets/fonts/octicon/octicon.ttf +0 -0
  402. package/src/assets/fonts/octicon/octicon.woff +0 -0
  403. package/src/assets/fonts/octicon/octicon.woff2 +0 -0
  404. package/src/assets/fonts/roboto/Roboto-Bold-webfont.eot +0 -0
  405. package/src/assets/fonts/roboto/Roboto-Bold-webfont.svg +0 -607
  406. package/src/assets/fonts/roboto/Roboto-Bold-webfont.ttf +0 -0
  407. package/src/assets/fonts/roboto/Roboto-Bold-webfont.woff +0 -0
  408. package/src/assets/fonts/roboto/Roboto-Medium-webfont.eot +0 -0
  409. package/src/assets/fonts/roboto/Roboto-Medium-webfont.svg +0 -607
  410. package/src/assets/fonts/roboto/Roboto-Medium-webfont.ttf +0 -0
  411. package/src/assets/fonts/roboto/Roboto-Medium-webfont.woff +0 -0
  412. package/src/assets/fonts/roboto/Roboto-Regular-webfont.eot +0 -0
  413. package/src/assets/fonts/roboto/Roboto-Regular-webfont.svg +0 -635
  414. package/src/assets/fonts/roboto/Roboto-Regular-webfont.ttf +0 -0
  415. package/src/assets/fonts/roboto/Roboto-Regular-webfont.woff +0 -0
  416. package/src/assets/icons/android-chrome-192x192.png +0 -0
  417. package/src/assets/icons/android-chrome-512x512.png +0 -0
  418. package/src/assets/icons/apple-touch-icon.png +0 -0
  419. package/src/assets/icons/favicon-16x16.png +0 -0
  420. package/src/assets/icons/favicon-32x32.png +0 -0
  421. package/src/assets/icons/site.webmanifest +0 -1
  422. package/src/assets/images/electron.bmp +0 -0
  423. package/src/bin/activator +0 -0
  424. package/src/bin/activator.exe +0 -0
  425. package/src/environments/environment.web.prod.ts +0 -4
  426. package/src/environments/environment.web.ts +0 -4
  427. package/src/karma.conf.js +0 -50
  428. package/src/polyfills-test.ts +0 -1
  429. package/src/polyfills.ts +0 -53
  430. package/src/styles/base/loading.scss +0 -87
  431. package/src/styles/base/reset.scss +0 -408
  432. package/src/styles/components/alert.scss +0 -54
  433. package/src/styles/components/buttons.scss +0 -93
  434. package/src/styles/components/forms.scss +0 -96
  435. package/src/styles/components/inputs.scss +0 -113
  436. package/src/styles/components/modal.scss +0 -52
  437. package/src/styles/components/paginator.scss +0 -108
  438. package/src/styles/components/scrollbox.scss +0 -71
  439. package/src/styles/components/spinner.scss +0 -11
  440. package/src/styles/components/tables.scss +0 -42
  441. package/src/styles/components/tabs.scss +0 -44
  442. package/src/styles/components/toolbar.scss +0 -130
  443. package/src/styles/fonts/md.scss +0 -54
  444. package/src/styles/fonts/octicons.scss +0 -1351
  445. package/src/styles/fonts/roboto.scss +0 -32
  446. package/src/styles/tools/_functions.scss +0 -5
  447. package/src/styles/tools/_mixins.scss +0 -159
  448. package/src/styles/tools/_placeholders.scss +0 -17
  449. package/src/styles/tools/_variables.scss +0 -67
  450. package/src/styles.scss +0 -37
  451. package/src/test.ts +0 -22
  452. package/tailwind.config.js +0 -67
@@ -1,77 +1,221 @@
1
- import { Component, OnInit, ViewChild } from '@angular/core';
2
- import { NgForm } from '@angular/forms';
3
- import { BreadcrumbService, DatabaseService, PresentationService } from '@app/core/services';
4
- import { SearchFormModel } from '@app/core/model';
5
- import { fontMimeTypes } from '@main/config/mimes';
6
- import { dbColumns } from '@main/config/database';
7
-
8
- @Component({
9
- selector: 'app-search',
10
- templateUrl: './search.component.html',
11
- styleUrls: ['./search.component.scss']
12
- })
13
- export class SearchComponent implements OnInit {
14
-
15
- @ViewChild(NgForm) form: NgForm;
16
-
17
- model: SearchFormModel = new SearchFormModel('', [], 'id', 'ASC', true);
18
-
19
- fontTypes = fontMimeTypes;
20
- dbColumns = dbColumns;
21
- orderBy = ['ASC', 'DESC'];
22
-
23
- constructor(
24
- private databaseService: DatabaseService,
25
- private breadcrumbService: BreadcrumbService,
26
- private presentationService: PresentationService
27
- ) { }
28
-
29
- ngOnInit(): void {
30
- }
31
-
32
- onSubmit(): void {
33
- const value = this.form.value;
34
-
35
- this.databaseService.resetWhere();
36
-
37
- this.databaseService.setSearch(true);
38
- this.databaseService.stats = false;
39
-
40
- if (value?.term) {
41
- this.databaseService.setWhere('term', value.term);
42
- this.addBreadcrumb(value.term);
43
- }
44
-
45
- if (value?.mimes?.length) {
46
- this.databaseService.setWhere('file_type', value.mimes);
47
- }
48
-
49
- if (value?.sort && value?.order) {
50
- this.databaseService.setOrder(value.sort, value.order);
51
- }
52
-
53
- this.databaseService.run();
54
- }
55
-
56
- addBreadcrumb(searchTerm: string): void {
57
- this.breadcrumbService.set([{
58
- title: 'System Search',
59
- link: '/main',
60
- type: 'collection'
61
- }, {
62
- title: searchTerm,
63
- link: '',
64
- type: 'collection'
65
- }]);
66
- }
67
-
68
- onReset(): void {
69
- this.databaseService.setSearch(false);
70
- this.databaseService.resetWhere().run();
71
- this.breadcrumbService.setNavigation(this.databaseService.getCollectionId(), this.databaseService.getCollectionResultSet());
72
- }
73
-
74
- onComponentSwitch() {
75
- this.presentationService.setAsideComponent('tables');
76
- }
77
- }
1
+ import { Component, inject, signal } from '@angular/core';
2
+ import { FormsModule } from '@angular/forms';
3
+ import { DatabaseService, PresentationService } from '../../../core/services';
4
+ import type { Collection } from '@main/database/entity/Collection.schema';
5
+
6
+ const FILE_TYPES = [
7
+ { label: 'TrueType (.ttf)', value: 'font/ttf' },
8
+ { label: 'OpenType (.otf)', value: 'font/otf' },
9
+ { label: 'WOFF (.woff)', value: 'font/woff' },
10
+ { label: 'WOFF2 (.woff2)', value: 'font/woff2' },
11
+ ];
12
+
13
+ const SEARCH_FIELDS = [
14
+ { label: 'Font Name', value: 'full_name' },
15
+ { label: 'Font Family', value: 'font_family' },
16
+ { label: 'Font Subfamily', value: 'font_subfamily' },
17
+ { label: 'PostScript Name', value: 'post_script_name' },
18
+ { label: 'Designer', value: 'designer' },
19
+ { label: 'Manufacturer', value: 'manufacturer' },
20
+ { label: 'Copyright', value: 'copyright' },
21
+ { label: 'License', value: 'license' },
22
+ { label: 'Description', value: 'description' },
23
+ { label: 'Trademark', value: 'trademark' },
24
+ { label: 'File Name', value: 'file_name' },
25
+ { label: 'Version', value: 'version' },
26
+ ];
27
+
28
+ const SORT_OPTIONS = [
29
+ { label: 'Name (A-Z)', column: 'full_name', direction: 'ASC' },
30
+ { label: 'Name (Z-A)', column: 'full_name', direction: 'DESC' },
31
+ { label: 'Family (A-Z)', column: 'font_family', direction: 'ASC' },
32
+ { label: 'Family (Z-A)', column: 'font_family', direction: 'DESC' },
33
+ { label: 'File Name (A-Z)', column: 'file_name', direction: 'ASC' },
34
+ { label: 'File Name (Z-A)', column: 'file_name', direction: 'DESC' },
35
+ { label: 'File Size (Smallest)', column: 'file_size', direction: 'ASC' },
36
+ { label: 'File Size (Largest)', column: 'file_size', direction: 'DESC' },
37
+ { label: 'Date Added (Newest)', column: 'created', direction: 'DESC' },
38
+ { label: 'Date Added (Oldest)', column: 'created', direction: 'ASC' },
39
+ { label: 'Date Updated (Newest)', column: 'updated', direction: 'DESC' },
40
+ { label: 'Date Updated (Oldest)', column: 'updated', direction: 'ASC' },
41
+ { label: 'Designer (A-Z)', column: 'designer', direction: 'ASC' },
42
+ { label: 'Designer (Z-A)', column: 'designer', direction: 'DESC' },
43
+ ];
44
+
45
+ @Component({
46
+ selector: 'app-search',
47
+ standalone: true,
48
+ imports: [FormsModule],
49
+ templateUrl: './search.component.html',
50
+ })
51
+ export class SearchComponent {
52
+ readonly db = inject(DatabaseService);
53
+ readonly presentation = inject(PresentationService);
54
+
55
+ readonly fileTypes = FILE_TYPES;
56
+ readonly searchFields = SEARCH_FIELDS;
57
+ readonly sortOptions = SORT_OPTIONS;
58
+
59
+ searchTerm = '';
60
+ selectedFileTypes: string[] = [];
61
+ selectedSearchFields: string[] = [];
62
+
63
+ // Status filters
64
+ favoritesOnly = false;
65
+ systemOnly = false;
66
+ installedOnly = false;
67
+
68
+ // Collection filter
69
+ selectedCollectionId: number | null = null;
70
+
71
+ // Designer/Manufacturer filter
72
+ designerFilter = '';
73
+ manufacturerFilter = '';
74
+
75
+ // Font subfamily filter
76
+ subfamilyFilter = '';
77
+
78
+ // File size range (KB)
79
+ fileSizeMin: number | null = null;
80
+ fileSizeMax: number | null = null;
81
+
82
+ // Date range
83
+ dateFrom = '';
84
+ dateTo = '';
85
+
86
+ // Sort
87
+ selectedSortIndex: number | null = null;
88
+
89
+ readonly hasSearched = signal(false);
90
+
91
+ get collections(): Collection[] {
92
+ return this.db.collections();
93
+ }
94
+
95
+ toggleFileType(value: string) {
96
+ const idx = this.selectedFileTypes.indexOf(value);
97
+ if (idx >= 0) {
98
+ this.selectedFileTypes.splice(idx, 1);
99
+ } else {
100
+ this.selectedFileTypes.push(value);
101
+ }
102
+ }
103
+
104
+ isFileTypeSelected(value: string): boolean {
105
+ return this.selectedFileTypes.includes(value);
106
+ }
107
+
108
+ toggleSearchField(value: string) {
109
+ const idx = this.selectedSearchFields.indexOf(value);
110
+ if (idx >= 0) {
111
+ this.selectedSearchFields.splice(idx, 1);
112
+ } else {
113
+ this.selectedSearchFields.push(value);
114
+ }
115
+ }
116
+
117
+ isSearchFieldSelected(value: string): boolean {
118
+ return this.selectedSearchFields.includes(value);
119
+ }
120
+
121
+ onSearch() {
122
+ const where: { key: string; value: any }[] = [];
123
+
124
+ if (this.searchTerm.trim()) {
125
+ where.push({ key: 'term', value: this.searchTerm.trim() });
126
+ }
127
+
128
+ if (this.selectedSearchFields.length > 0) {
129
+ where.push({ key: 'search_fields', value: [...this.selectedSearchFields] });
130
+ }
131
+
132
+ if (this.selectedFileTypes.length > 0) {
133
+ where.push({ key: 'file_type', value: [...this.selectedFileTypes] });
134
+ }
135
+
136
+ if (this.favoritesOnly) {
137
+ where.push({ key: 'favorite', value: 1 });
138
+ }
139
+
140
+ if (this.systemOnly) {
141
+ where.push({ key: 'system', value: 1 });
142
+ }
143
+
144
+ if (this.installedOnly) {
145
+ where.push({ key: 'installable', value: 1 });
146
+ }
147
+
148
+ if (this.selectedCollectionId !== null) {
149
+ where.push({ key: 'collection_id', value: this.selectedCollectionId });
150
+ }
151
+
152
+ if (this.designerFilter.trim()) {
153
+ where.push({ key: 'designer', value: this.designerFilter.trim() });
154
+ }
155
+
156
+ if (this.manufacturerFilter.trim()) {
157
+ where.push({ key: 'manufacturer', value: this.manufacturerFilter.trim() });
158
+ }
159
+
160
+ if (this.subfamilyFilter.trim()) {
161
+ where.push({ key: 'font_subfamily', value: this.subfamilyFilter.trim() });
162
+ }
163
+
164
+ if (this.fileSizeMin !== null && this.fileSizeMin > 0) {
165
+ where.push({ key: 'file_size_min', value: this.fileSizeMin * 1024 });
166
+ }
167
+
168
+ if (this.fileSizeMax !== null && this.fileSizeMax > 0) {
169
+ where.push({ key: 'file_size_max', value: this.fileSizeMax * 1024 });
170
+ }
171
+
172
+ if (this.dateFrom) {
173
+ where.push({ key: 'date_from', value: this.dateFrom });
174
+ }
175
+
176
+ if (this.dateTo) {
177
+ where.push({ key: 'date_to', value: this.dateTo });
178
+ }
179
+
180
+ if (where.length === 0) {
181
+ return;
182
+ }
183
+
184
+ const order =
185
+ this.selectedSortIndex !== null
186
+ ? {
187
+ column: this.sortOptions[this.selectedSortIndex].column,
188
+ direction: this.sortOptions[this.selectedSortIndex].direction,
189
+ }
190
+ : undefined;
191
+
192
+ this.hasSearched.set(true);
193
+ this.db.selectSearch(where, order);
194
+ }
195
+
196
+ onReset() {
197
+ this.searchTerm = '';
198
+ this.selectedFileTypes = [];
199
+ this.selectedSearchFields = [];
200
+ this.favoritesOnly = false;
201
+ this.systemOnly = false;
202
+ this.installedOnly = false;
203
+ this.selectedCollectionId = null;
204
+ this.designerFilter = '';
205
+ this.manufacturerFilter = '';
206
+ this.subfamilyFilter = '';
207
+ this.fileSizeMin = null;
208
+ this.fileSizeMax = null;
209
+ this.dateFrom = '';
210
+ this.dateTo = '';
211
+ this.selectedSortIndex = null;
212
+ this.hasSearched.set(false);
213
+ this.db.clearSearch();
214
+ }
215
+
216
+ onKeydown(event: KeyboardEvent) {
217
+ if (event.key === 'Enter') {
218
+ this.onSearch();
219
+ }
220
+ }
221
+ }
@@ -1,26 +1,10 @@
1
- import { Component, OnInit } from '@angular/core';
2
- import { PresentationService } from '@app/core/services';
3
-
4
- @Component({
5
- selector: 'app-spinner',
6
- templateUrl: './spinner.component.html',
7
- styleUrls: ['./spinner.component.scss']
8
- })
9
- export class SpinnerComponent implements OnInit {
10
-
11
- isActivated = false;
12
-
13
- constructor(
14
- private presentationService: PresentationService
15
- ) { }
16
-
17
- ngOnInit() {
18
- this.presentationService._loadingSpinner.subscribe((status: boolean) => {
19
- if (status) {
20
- this.isActivated = status;
21
- } else {
22
- setTimeout(() => this.isActivated = status, 1e3/3);
23
- }
24
- });
25
- }
26
- }
1
+ import { Component, input } from '@angular/core';
2
+
3
+ @Component({
4
+ selector: 'app-spinner',
5
+ standalone: true,
6
+ template: `<span class="app-spinner" [class.is-animating]="animating()"></span>`,
7
+ })
8
+ export class SpinnerComponent {
9
+ readonly animating = input(false);
10
+ }
@@ -0,0 +1,207 @@
1
+ <div class="panel">
2
+ <div class="h-full flex items-center justify-around px-3">
3
+ <div class="flex-1 flex items-center gap-3">
4
+ @if (db.totalPages() > 1) {
5
+ <div class="flex items-center gap-0.5 text-[11px]">
6
+ <button
7
+ class="px-1 py-0.5 rounded cursor-pointer disabled:cursor-default transition-colors"
8
+ [style.color]="'var(--text-secondary)'"
9
+ [disabled]="db.currentPage() === 1"
10
+ [class.opacity-30]="db.currentPage() === 1"
11
+ (click)="db.firstPage()"
12
+ >
13
+ <span
14
+ class="material-symbols-outlined text-sm"
15
+ style="
16
+ font-variation-settings:
17
+ 'opsz' 20,
18
+ 'wght' 300;
19
+ "
20
+ >first_page</span
21
+ >
22
+ </button>
23
+ <button
24
+ class="px-1 py-0.5 rounded cursor-pointer disabled:cursor-default transition-colors"
25
+ [style.color]="'var(--text-secondary)'"
26
+ [disabled]="db.currentPage() === 1"
27
+ [class.opacity-30]="db.currentPage() === 1"
28
+ (click)="db.prevPage()"
29
+ >
30
+ <span
31
+ class="material-symbols-outlined text-sm"
32
+ style="
33
+ font-variation-settings:
34
+ 'opsz' 20,
35
+ 'wght' 300;
36
+ "
37
+ >chevron_left</span
38
+ >
39
+ </button>
40
+ <span class="px-1.5 select-none" [style.color]="'var(--text-muted)'">{{ db.currentPage() }} / {{ db.totalPages() }}</span>
41
+ <button
42
+ class="px-1 py-0.5 rounded cursor-pointer disabled:cursor-default transition-colors"
43
+ [style.color]="'var(--text-secondary)'"
44
+ [disabled]="db.currentPage() === db.totalPages()"
45
+ [class.opacity-30]="db.currentPage() === db.totalPages()"
46
+ (click)="db.nextPage()"
47
+ >
48
+ <span
49
+ class="material-symbols-outlined text-sm"
50
+ style="
51
+ font-variation-settings:
52
+ 'opsz' 20,
53
+ 'wght' 300;
54
+ "
55
+ >chevron_right</span
56
+ >
57
+ </button>
58
+ <button
59
+ class="px-1 py-0.5 rounded cursor-pointer disabled:cursor-default transition-colors"
60
+ [style.color]="'var(--text-secondary)'"
61
+ [disabled]="db.currentPage() === db.totalPages()"
62
+ [class.opacity-30]="db.currentPage() === db.totalPages()"
63
+ (click)="db.lastPage()"
64
+ >
65
+ <span
66
+ class="material-symbols-outlined text-sm"
67
+ style="
68
+ font-variation-settings:
69
+ 'opsz' 20,
70
+ 'wght' 300;
71
+ "
72
+ >last_page</span
73
+ >
74
+ </button>
75
+ </div>
76
+ }
77
+ <span class="text-[11px]" [style.color]="'var(--text-muted)'">
78
+ @if (db.storeCount()) {
79
+ {{ db.storeCount() }} fonts
80
+ }
81
+ </span>
82
+ </div>
83
+
84
+ <div class="flex-1 flex items-center justify-center gap-3">
85
+ <input
86
+ type="text"
87
+ class="form-input w-52"
88
+ placeholder="Enter preview text..."
89
+ [value]="displayText"
90
+ (input)="handleDisplayInput($event)"
91
+ />
92
+ <select class="form-select" [value]="presentation.quickTextIndex()" (change)="handleQuickTextChange($event)">
93
+ @for (item of presentation.quickTexts; track item.title; let i = $index) {
94
+ <option [value]="i">{{ item.title }}</option>
95
+ }
96
+ </select>
97
+ @if (news.hasArticles()) {
98
+ <button class="flex items-center cursor-pointer" title="Use random news headline as preview text" (click)="setRandomHeadline()">
99
+ <span
100
+ class="material-symbols-outlined text-sm"
101
+ [style.color]="'var(--text-muted)'"
102
+ style="
103
+ font-variation-settings:
104
+ 'opsz' 20,
105
+ 'wght' 300;
106
+ "
107
+ >newspaper</span
108
+ >
109
+ </button>
110
+ }
111
+ </div>
112
+
113
+ <div class="flex items-center gap-2 mx-6">
114
+ <input type="color" class="color-picker" [value]="colorPickerValue()" (input)="handleFontColor($event)" />
115
+ <input type="color" class="color-picker" [value]="bgPickerValue()" (input)="handleBackgroundColor($event)" />
116
+ </div>
117
+
118
+ <div class="flex-1 flex items-center justify-end gap-3">
119
+ <div class="flex items-center gap-2">
120
+ <span
121
+ class="material-symbols-outlined text-sm"
122
+ [style.color]="'var(--text-muted)'"
123
+ style="
124
+ font-variation-settings:
125
+ 'opsz' 20,
126
+ 'wght' 300;
127
+ "
128
+ >text_decrease</span
129
+ >
130
+ <input
131
+ type="range"
132
+ class="preview-range"
133
+ min="8"
134
+ max="72"
135
+ step="1"
136
+ [value]="presentation.fontSize()"
137
+ (input)="handleFontSize($event)"
138
+ />
139
+ <span
140
+ class="material-symbols-outlined text-sm"
141
+ [style.color]="'var(--text-muted)'"
142
+ style="
143
+ font-variation-settings:
144
+ 'opsz' 20,
145
+ 'wght' 300;
146
+ "
147
+ >text_increase</span
148
+ >
149
+ <span class="text-[11px] w-8 text-center select-none" [style.color]="'var(--text-muted)'">{{ presentation.fontSize() }}px</span>
150
+ </div>
151
+ <span
152
+ class="material-symbols-outlined text-sm"
153
+ [style.color]="'var(--text-muted)'"
154
+ style="
155
+ font-variation-settings:
156
+ 'opsz' 20,
157
+ 'wght' 300;
158
+ "
159
+ >format_letter_spacing</span
160
+ >
161
+ <input
162
+ type="range"
163
+ class="preview-range"
164
+ min="-5"
165
+ max="20"
166
+ step="0.5"
167
+ [value]="presentation.letterSpacing()"
168
+ (input)="handleLetterSpacing($event)"
169
+ />
170
+ <span class="text-[11px] w-10 text-center select-none" [style.color]="'var(--text-muted)'">{{ presentation.letterSpacing() }}px</span>
171
+ <span
172
+ class="material-symbols-outlined text-sm"
173
+ [style.color]="'var(--text-muted)'"
174
+ style="
175
+ font-variation-settings:
176
+ 'opsz' 20,
177
+ 'wght' 300;
178
+ "
179
+ >width</span
180
+ >
181
+ <input
182
+ type="range"
183
+ class="preview-range"
184
+ min="-5"
185
+ max="20"
186
+ step="0.5"
187
+ [value]="presentation.wordSpacing()"
188
+ (input)="handleWordSpacing($event)"
189
+ />
190
+ <span class="text-[11px] w-10 text-center select-none" [style.color]="'var(--text-muted)'">{{ presentation.wordSpacing() }}px</span>
191
+ </div>
192
+
193
+ <div class="flex items-center justify-end">
194
+ <button class="toolbar-reset-btn" (click)="presentation.resetToolbarDefaults()">
195
+ <span
196
+ class="material-symbols-outlined"
197
+ style="
198
+ font-variation-settings:
199
+ 'opsz' 20,
200
+ 'wght' 300;
201
+ "
202
+ >restart_alt</span
203
+ >
204
+ </button>
205
+ </div>
206
+ </div>
207
+ </div>
@@ -0,0 +1,99 @@
1
+ import { Component, inject, computed, ElementRef } from '@angular/core';
2
+ import { DatabaseService, NewsService, PresentationService } from '../../../core/services';
3
+
4
+ @Component({
5
+ selector: 'app-toolbar',
6
+ standalone: true,
7
+ templateUrl: './toolbar.component.html',
8
+ })
9
+ export class ToolbarComponent {
10
+ readonly db = inject(DatabaseService);
11
+ readonly presentation = inject(PresentationService);
12
+ readonly news = inject(NewsService);
13
+ private el = inject(ElementRef);
14
+
15
+ setRandomHeadline() {
16
+ const article = this.news.randomArticle();
17
+ if (article?.title) {
18
+ this.presentation.setCustomText(article.title);
19
+ }
20
+ }
21
+
22
+ readonly colorPickerValue = computed(() => {
23
+ return this.presentation.fontColor() ?? this.getInheritedColor();
24
+ });
25
+
26
+ readonly bgPickerValue = computed(() => {
27
+ return this.presentation.backgroundColor() ?? this.getInheritedBgColor();
28
+ });
29
+
30
+ get displayText() {
31
+ return this.presentation.displayText();
32
+ }
33
+
34
+ handleDisplayInput(event: Event) {
35
+ const value = (event.target as HTMLInputElement).value;
36
+ this.presentation.setCustomText(value);
37
+ }
38
+
39
+ handleQuickTextChange(event: Event) {
40
+ const index = +(event.target as HTMLSelectElement).value;
41
+ this.presentation.customText.set(null);
42
+ this.presentation.quickTextIndex.set(index);
43
+ }
44
+
45
+ handleFontSize(event: Event) {
46
+ const value = +(event.target as HTMLInputElement).value;
47
+ this.presentation.fontSize.set(value);
48
+ }
49
+
50
+ handleFontColor(event: Event) {
51
+ const value = (event.target as HTMLInputElement).value;
52
+ this.presentation.fontColor.set(value);
53
+ }
54
+
55
+ handleBackgroundColor(event: Event) {
56
+ const value = (event.target as HTMLInputElement).value;
57
+ this.presentation.backgroundColor.set(value);
58
+ }
59
+
60
+ handleLetterSpacing(event: Event) {
61
+ const value = +(event.target as HTMLInputElement).value;
62
+ this.presentation.letterSpacing.set(value);
63
+ }
64
+
65
+ handleWordSpacing(event: Event) {
66
+ const value = +(event.target as HTMLInputElement).value;
67
+ this.presentation.wordSpacing.set(value);
68
+ }
69
+
70
+ private getInheritedColor(): string {
71
+ const rgb = getComputedStyle(this.el.nativeElement).color;
72
+ const match = rgb.match(/\d+/g);
73
+ if (match && match.length >= 3) {
74
+ return (
75
+ '#' +
76
+ match
77
+ .slice(0, 3)
78
+ .map((n) => (+n).toString(16).padStart(2, '0'))
79
+ .join('')
80
+ );
81
+ }
82
+ return '#000000';
83
+ }
84
+
85
+ private getInheritedBgColor(): string {
86
+ const rgb = getComputedStyle(this.el.nativeElement).backgroundColor;
87
+ const match = rgb.match(/\d+/g);
88
+ if (match && match.length >= 3) {
89
+ return (
90
+ '#' +
91
+ match
92
+ .slice(0, 3)
93
+ .map((n) => (+n).toString(16).padStart(2, '0'))
94
+ .join('')
95
+ );
96
+ }
97
+ return '#ffffff';
98
+ }
99
+ }