fontastic 0.1.7 → 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.
- package/.editorconfig +16 -16
- package/.github/FUNDING.yml +3 -12
- package/.github/ISSUE_TEMPLATE/bug_report.yml +41 -0
- package/.github/ISSUE_TEMPLATE/feature_request.yml +29 -0
- package/.github/dependabot.yml +6 -6
- package/.github/pull_request_template.md +21 -21
- package/.github/workflows/claude-code-review.yml +45 -0
- package/.github/workflows/claude.yml +46 -0
- package/.github/workflows/macos.yml +77 -58
- package/.github/workflows/release.yml +165 -0
- package/.github/workflows/ubuntu.yml +87 -70
- package/.github/workflows/windows.yml +81 -64
- package/.husky/pre-commit +1 -0
- package/.lintstagedrc.json +6 -0
- package/.node-version +1 -1
- package/.postcssrc.json +5 -0
- package/.prettierignore +8 -0
- package/.prettierrc +18 -0
- package/.vscode/launch.json +42 -45
- package/.vscode/tasks.json +48 -48
- package/CODE_OF_CONDUCT.md +128 -45
- package/HOW_TO.md +39 -0
- package/LICENSE.md +7 -0
- package/README.md +103 -111
- package/angular.json +166 -202
- package/app/Application.js +39 -39
- package/app/Application.ts +38 -38
- package/app/config/alert.ts +31 -31
- package/app/config/database.js +194 -202
- package/app/config/database.ts +194 -202
- package/app/config/index.js +19 -19
- package/app/config/index.ts +2 -2
- package/app/config/mimes.js +48 -48
- package/app/config/mimes.ts +53 -53
- package/app/config/system.js +578 -496
- package/app/config/system.ts +580 -498
- package/app/config/themes.ts +38 -38
- package/app/core/AppLogger.js +30 -30
- package/app/core/AppLogger.ts +37 -37
- package/app/core/ConfigManager.js +129 -113
- package/app/core/ConfigManager.ts +150 -137
- package/app/core/ConnectionManager.js +93 -82
- package/app/core/ConnectionManager.ts +107 -95
- package/app/core/FontCatalog.js +32 -50
- package/app/core/FontCatalog.ts +20 -42
- package/app/core/FontFinder.js +72 -85
- package/app/core/FontFinder.ts +75 -96
- package/app/core/FontManager.js +128 -164
- package/app/core/FontManager.ts +136 -172
- package/app/core/FontObject.js +53 -53
- package/app/core/FontObject.ts +62 -62
- package/app/core/MessageHandler.js +270 -271
- package/app/core/MessageHandler.ts +330 -337
- package/app/core/SystemManager.js +132 -139
- package/app/core/SystemManager.ts +160 -171
- package/app/core/menu/Example.ts +279 -279
- package/app/core/menu/MenuBuilder.js +43 -44
- package/app/core/menu/MenuBuilder.ts +51 -53
- package/app/core/menu/templates/DarwinTemplate.js +124 -126
- package/app/core/menu/templates/DarwinTemplate.ts +144 -142
- package/app/core/menu/templates/SystemTemplate.js +121 -123
- package/app/core/menu/templates/SystemTemplate.ts +136 -139
- package/app/database/entity/Collection.schema.js +98 -98
- package/app/database/entity/Collection.schema.ts +68 -68
- package/app/database/entity/Logger.schema.js +49 -49
- package/app/database/entity/Logger.schema.ts +26 -26
- package/app/database/entity/Store.schema.js +159 -164
- package/app/database/entity/Store.schema.ts +110 -116
- package/app/database/entity/index.js +19 -19
- package/app/database/entity/index.ts +2 -2
- package/app/database/repository/Collection.repository.js +277 -180
- package/app/database/repository/Collection.repository.ts +291 -190
- package/app/database/repository/Logger.repository.js +24 -24
- package/app/database/repository/Logger.repository.ts +10 -10
- package/app/database/repository/Store.repository.js +324 -327
- package/app/database/repository/Store.repository.ts +349 -358
- package/app/database/repository/User.repository.js +11 -11
- package/app/database/repository/User.repository.ts +8 -8
- package/app/database/repository/index.js +19 -19
- package/app/database/repository/index.ts +2 -2
- package/app/enums/ChannelType.js +62 -61
- package/app/enums/ChannelType.ts +67 -65
- package/app/enums/StorageType.js +19 -18
- package/app/enums/{storageType.ts → StorageType.ts} +15 -14
- package/app/enums/index.js +18 -18
- package/app/enums/index.ts +2 -2
- package/app/helpers/command.js +27 -27
- package/app/helpers/command.ts +19 -19
- package/app/helpers/random.js +15 -15
- package/app/helpers/random.ts +12 -12
- package/app/main.js +97 -81
- package/app/main.ts +114 -94
- package/app/package-lock.json +2240 -2649
- package/app/package.json +23 -35
- package/app/types/AppAlert.js +2 -2
- package/app/types/AppAlert.ts +8 -8
- package/app/types/AuthUser.js +6 -6
- package/app/types/AuthUser.ts +5 -5
- package/app/types/Breadcrumb.js +2 -2
- package/app/types/Breadcrumb.ts +4 -4
- package/app/types/FontMetrics.ts +7 -0
- package/app/types/ImportOptions.js +6 -6
- package/app/types/ImportOptions.ts +3 -3
- package/app/types/QueryOptions.js +2 -2
- package/app/types/QueryOptions.ts +12 -12
- package/app/types/SystemConfig.js +2 -2
- package/app/types/SystemConfig.ts +127 -132
- package/app/types/SystemStats.js +2 -2
- package/app/types/SystemStats.ts +6 -7
- package/app/types/SystemTheme.js +2 -2
- package/app/types/SystemTheme.ts +5 -5
- package/app/types/index.js +24 -24
- package/app/types/index.ts +9 -8
- package/commitlint.config.mjs +3 -0
- package/e2e/main.spec.ts +57 -59
- package/e2e/playwright.config.ts +20 -19
- package/e2e/tsconfig.e2e.json +13 -13
- package/electron-builder.json +34 -71
- package/eslint.config.mjs +111 -0
- package/package.json +122 -123
- package/src/app/app.component.html +1 -2
- package/src/app/app.component.spec.ts +21 -0
- package/src/app/app.component.ts +40 -37
- package/src/app/core/core.module.ts +8 -11
- package/src/app/core/services/database/database.service.ts +410 -286
- package/src/app/core/services/electron/electron.service.spec.ts +12 -12
- package/src/app/core/services/electron/electron.service.ts +62 -62
- package/src/app/core/services/index.ts +5 -14
- package/src/app/core/services/message/message.service.ts +291 -289
- package/src/app/core/services/news/news.service.ts +52 -94
- package/src/app/core/services/presentation/presentation.service.ts +253 -341
- package/src/app/home/home.component.html +7 -0
- package/src/app/home/home.component.spec.ts +31 -0
- package/src/app/home/home.component.ts +15 -0
- package/src/app/layout/aside/aside.component.html +35 -9
- package/src/app/layout/aside/aside.component.ts +60 -26
- package/src/app/layout/footer/footer.component.html +5 -11
- package/src/app/layout/footer/footer.component.ts +13 -32
- package/src/app/layout/header/header.component.html +118 -35
- package/src/app/layout/header/header.component.ts +52 -229
- package/src/app/layout/layout.component.html +32 -14
- package/src/app/layout/layout.component.ts +19 -23
- package/src/app/layout/main/main.component.html +46 -0
- package/src/app/layout/main/main.component.ts +20 -0
- package/src/app/layout/navigation/library/library.component.ts +51 -0
- package/src/app/layout/navigation/navigation.component.html +126 -142
- package/src/app/layout/navigation/navigation.component.ts +372 -302
- package/src/app/layout/navigation/stats/stats.component.ts +56 -0
- package/src/app/settings/ai-keys/ai-keys.component.html +56 -0
- package/src/app/settings/ai-keys/ai-keys.component.ts +44 -0
- package/src/app/settings/danger-zone/danger-zone.component.html +22 -0
- package/src/app/settings/danger-zone/danger-zone.component.ts +37 -0
- package/src/app/settings/logs/logs.component.html +41 -0
- package/src/app/settings/logs/logs.component.ts +43 -0
- package/src/app/settings/news-api/news-api.component.html +40 -0
- package/src/app/settings/news-api/news-api.component.ts +57 -0
- package/src/app/settings/pages/general/general.component.html +7 -0
- package/src/app/settings/pages/general/general.component.ts +20 -0
- package/src/app/settings/pages/logs/logs.component.html +1 -0
- package/src/app/settings/pages/logs/logs.component.ts +10 -0
- package/src/app/settings/pages/system/system.component.html +1 -0
- package/src/app/settings/pages/system/system.component.ts +10 -0
- package/src/app/settings/quick-actions/quick-actions.component.html +49 -0
- package/src/app/settings/quick-actions/quick-actions.component.ts +40 -0
- package/src/app/settings/settings.component.html +47 -0
- package/src/app/settings/settings.component.ts +10 -0
- package/src/app/settings/system-info/system-info.component.html +21 -0
- package/src/app/settings/system-info/system-info.component.ts +42 -0
- package/src/app/settings/theme/theme.component.html +15 -0
- package/src/app/settings/theme/theme.component.ts +22 -0
- package/src/app/shared/components/collapsible-panel/collapsible-panel.component.ts +43 -0
- package/src/app/shared/components/context-menu/context-menu.component.ts +67 -0
- package/src/app/shared/components/datagrid/datagrid.component.html +96 -0
- package/src/app/shared/components/datagrid/datagrid.component.ts +49 -0
- package/src/app/shared/components/index.ts +11 -17
- package/src/app/shared/components/inspector/inspector.component.html +140 -0
- package/src/app/shared/components/inspector/inspector.component.ts +41 -0
- package/src/app/shared/components/page-not-found/page-not-found.component.html +1 -3
- package/src/app/shared/components/page-not-found/page-not-found.component.spec.ts +22 -0
- package/src/app/shared/components/page-not-found/page-not-found.component.ts +14 -12
- package/src/app/shared/components/panel/panel.component.html +5 -0
- package/src/app/shared/components/panel/panel.component.ts +8 -0
- package/src/app/shared/components/preview/preview.component.html +36 -0
- package/src/app/shared/components/preview/preview.component.ts +68 -0
- package/src/app/shared/components/prompt-dialog/prompt-dialog.component.html +36 -0
- package/src/app/shared/components/prompt-dialog/prompt-dialog.component.ts +39 -0
- package/src/app/shared/components/search/search.component.html +329 -54
- package/src/app/shared/components/search/search.component.ts +221 -77
- package/src/app/shared/components/spinner/spinner.component.ts +10 -26
- package/src/app/shared/components/toolbar/toolbar.component.html +207 -0
- package/src/app/shared/components/toolbar/toolbar.component.ts +99 -0
- package/src/app/shared/components/waterfall/waterfall.component.html +119 -0
- package/src/app/shared/components/waterfall/waterfall.component.ts +119 -0
- package/src/app/shared/directives/autofocus/autofocus.directive.ts +14 -0
- package/src/app/shared/directives/index.ts +2 -2
- package/src/app/shared/directives/webview/webview.directive.spec.ts +8 -0
- package/src/app/shared/directives/webview/webview.directive.ts +9 -8
- package/src/app/shared/shared.module.ts +13 -94
- package/src/assets/background.jpg +0 -0
- package/src/assets/fonts/md/MaterialIcons-Regular.woff2 +0 -0
- package/src/assets/fonts/md/MaterialSymbolsOutlined.woff2 +0 -0
- package/src/assets/i18n/en.json +12 -12
- package/src/assets/icons/electron.bmp +0 -0
- package/src/assets/icons/favicon.256x256.png +0 -0
- package/src/assets/icons/favicon.512x512.png +0 -0
- package/src/assets/icons/favicon.icns +0 -0
- package/src/assets/icons/favicon.ico +0 -0
- package/src/assets/icons/favicon.png +0 -0
- package/src/environments/environment.dev.ts +4 -4
- package/src/environments/environment.prod.ts +4 -4
- package/src/environments/environment.ts +4 -4
- package/src/favicon.ico +0 -0
- package/src/index.html +19 -13
- package/src/main.ts +65 -15
- package/src/styles/base/reset.css +41 -0
- package/src/styles/base/variables.css +356 -0
- package/src/styles/components/buttons.css +108 -0
- package/src/styles/components/cards.css +60 -0
- package/src/styles/components/forms.css +70 -0
- package/src/styles/components/navigation.css +24 -0
- package/src/styles/components/scrollbox.css +50 -0
- package/src/styles/components/spinner.css +7 -0
- package/src/styles/components/splash.css +47 -0
- package/src/styles/components/toolbar.css +92 -0
- package/src/styles/fonts/md.css +39 -0
- package/src/styles/themes/dashboard.scss +293 -294
- package/src/styles/themes/euphoria.scss +284 -283
- package/src/styles/themes/mellow.scss +281 -280
- package/src/styles/themes/midnight.scss +284 -283
- package/src/styles/themes/passion.scss +281 -280
- package/src/styles/themes/swiss.scss +284 -283
- package/src/styles.css +17 -0
- package/src/tsconfig.app.json +11 -20
- package/src/tsconfig.spec.json +10 -23
- package/src/typings.d.ts +9 -9
- package/src/vitest.d.ts +1 -0
- package/tsconfig.json +38 -44
- package/tsconfig.serve.json +27 -27
- package/.eslintrc.json +0 -54
- package/.github/ISSUE_TEMPLATE/bug_report.md +0 -29
- package/.github/ISSUE_TEMPLATE/feature_request.md +0 -20
- package/.github/stale.yml +0 -17
- package/CONTRIBUTING.md +0 -72
- package/LICENSE +0 -21
- package/angular.webpack.js +0 -35
- package/app/core/FontInstaller.js +0 -67
- package/app/core/FontInstaller.ts +0 -63
- package/docs/logo.png +0 -0
- package/docs/screenshots/readme.md +0 -7
- package/docs/screenshots/screen-grab1.png +0 -0
- package/docs/screenshots/screen-grab2.png +0 -0
- package/docs/screenshots/screen-grab3.png +0 -0
- package/src/app/app-routing.module.ts +0 -15
- package/src/app/app.component.scss +0 -0
- package/src/app/app.component.spec.ts.dist +0 -33
- package/src/app/app.module.ts +0 -63
- package/src/app/core/model/AuthUserModel.ts +0 -7
- package/src/app/core/model/CustomThemeModel.ts +0 -7
- package/src/app/core/model/DbConnectionModel.ts +0 -16
- package/src/app/core/model/ImportOptionsModel.ts +0 -5
- package/src/app/core/model/LatestNewsModel.ts +0 -7
- package/src/app/core/model/SearchFormModel.ts +0 -10
- package/src/app/core/model/index.ts +0 -6
- package/src/app/core/services/alert/alert.service.spec.ts +0 -22
- package/src/app/core/services/alert/alert.service.ts +0 -71
- package/src/app/core/services/auth/auth.service.ts +0 -30
- package/src/app/core/services/boot/boot.service.ts +0 -33
- package/src/app/core/services/breadcrumb/breadcrumb.service.spec.ts +0 -22
- package/src/app/core/services/breadcrumb/breadcrumb.service.ts +0 -30
- package/src/app/core/services/config/config.service.ts +0 -63
- package/src/app/core/services/font/font.service.ts +0 -79
- package/src/app/core/services/gravatar/gravatar.service.spec.ts +0 -16
- package/src/app/core/services/gravatar/gravatar.service.ts +0 -18
- package/src/app/core/services/modal/modal.service.spec.ts +0 -16
- package/src/app/core/services/modal/modal.service.ts +0 -34
- package/src/app/core/services/utils/utils.service.spec.ts +0 -12
- package/src/app/core/services/utils/utils.service.ts +0 -144
- package/src/app/layout/aside/aside.component.scss +0 -1
- package/src/app/layout/footer/footer.component.scss +0 -27
- package/src/app/layout/header/header.component.scss +0 -60
- package/src/app/layout/layout.component.scss +0 -119
- package/src/app/layout/layout.module.ts +0 -32
- package/src/app/layout/layout.service.ts +0 -13
- package/src/app/layout/navigation/navigation.component.scss +0 -0
- package/src/app/pages/main/grid/grid.component.html +0 -183
- package/src/app/pages/main/grid/grid.component.scss +0 -3
- package/src/app/pages/main/grid/grid.component.ts +0 -98
- package/src/app/pages/main/inspect/inspect.component.html +0 -7
- package/src/app/pages/main/inspect/inspect.component.scss +0 -3
- package/src/app/pages/main/inspect/inspect.component.ts +0 -66
- package/src/app/pages/main/main-routing.module.ts +0 -30
- package/src/app/pages/main/main.component.html +0 -6
- package/src/app/pages/main/main.component.scss +0 -191
- package/src/app/pages/main/main.component.ts +0 -35
- package/src/app/pages/main/main.module.ts +0 -28
- package/src/app/pages/main/preview/preview.component.html +0 -46
- package/src/app/pages/main/preview/preview.component.scss +0 -0
- package/src/app/pages/main/preview/preview.component.ts +0 -239
- package/src/app/pages/main/toolbar/toolbar.component.html +0 -78
- package/src/app/pages/main/toolbar/toolbar.component.scss +0 -3
- package/src/app/pages/main/toolbar/toolbar.component.ts +0 -132
- package/src/app/shared/components/alert/alert.component.html +0 -9
- package/src/app/shared/components/alert/alert.component.scss +0 -0
- package/src/app/shared/components/alert/alert.component.ts +0 -37
- package/src/app/shared/components/breadcrumbs/breadcrumbs.component.html +0 -14
- package/src/app/shared/components/breadcrumbs/breadcrumbs.component.scss +0 -0
- package/src/app/shared/components/breadcrumbs/breadcrumbs.component.ts +0 -36
- package/src/app/shared/components/button/button.component.html +0 -3
- package/src/app/shared/components/button/button.component.scss +0 -2
- package/src/app/shared/components/button/button.component.ts +0 -22
- package/src/app/shared/components/collection/create-collection/create-collection.component.html +0 -41
- package/src/app/shared/components/collection/create-collection/create-collection.component.scss +0 -0
- package/src/app/shared/components/collection/create-collection/create-collection.component.ts +0 -68
- package/src/app/shared/components/collection/form-collection/form-collection.component.html +0 -1
- package/src/app/shared/components/collection/form-collection/form-collection.component.scss +0 -0
- package/src/app/shared/components/collection/form-collection/form-collection.component.ts +0 -15
- package/src/app/shared/components/collection/update-collection/update-collection.component.html +0 -40
- package/src/app/shared/components/collection/update-collection/update-collection.component.scss +0 -0
- package/src/app/shared/components/collection/update-collection/update-collection.component.ts +0 -71
- package/src/app/shared/components/glyph-list/glyph-list.component.html +0 -40
- package/src/app/shared/components/glyph-list/glyph-list.component.scss +0 -1
- package/src/app/shared/components/glyph-list/glyph-list.component.ts +0 -165
- package/src/app/shared/components/glyph-view/glyph-view.component.html +0 -23
- package/src/app/shared/components/glyph-view/glyph-view.component.scss +0 -0
- package/src/app/shared/components/glyph-view/glyph-view.component.ts +0 -228
- package/src/app/shared/components/loading/loading.component.html +0 -1
- package/src/app/shared/components/loading/loading.component.scss +0 -0
- package/src/app/shared/components/loading/loading.component.ts +0 -26
- package/src/app/shared/components/modal/modal.component.html +0 -31
- package/src/app/shared/components/modal/modal.component.scss +0 -0
- package/src/app/shared/components/modal/modal.component.ts +0 -51
- package/src/app/shared/components/page-not-found/page-not-found.component.scss +0 -0
- package/src/app/shared/components/paginator/paginator.component.html +0 -21
- package/src/app/shared/components/paginator/paginator.component.scss +0 -3
- package/src/app/shared/components/paginator/paginator.component.ts +0 -133
- package/src/app/shared/components/pairing/pairing.component.html +0 -20
- package/src/app/shared/components/pairing/pairing.component.scss +0 -0
- package/src/app/shared/components/pairing/pairing.component.ts +0 -33
- package/src/app/shared/components/search/search.component.scss +0 -10
- package/src/app/shared/components/settings/database/database.component.html +0 -151
- package/src/app/shared/components/settings/database/database.component.scss +0 -0
- package/src/app/shared/components/settings/database/database.component.ts +0 -145
- package/src/app/shared/components/settings/general/general.component.html +0 -170
- package/src/app/shared/components/settings/general/general.component.scss +0 -0
- package/src/app/shared/components/settings/general/general.component.ts +0 -150
- package/src/app/shared/components/settings/logs/logs.component.html +0 -44
- package/src/app/shared/components/settings/logs/logs.component.scss +0 -0
- package/src/app/shared/components/settings/logs/logs.component.ts +0 -37
- package/src/app/shared/components/settings/news/news.component.html +0 -41
- package/src/app/shared/components/settings/news/news.component.scss +0 -0
- package/src/app/shared/components/settings/news/news.component.ts +0 -52
- package/src/app/shared/components/settings/settings.component.html +0 -46
- package/src/app/shared/components/settings/settings.component.scss +0 -0
- package/src/app/shared/components/settings/settings.component.ts +0 -23
- package/src/app/shared/components/settings/system/system.component.html +0 -30
- package/src/app/shared/components/settings/system/system.component.scss +0 -0
- package/src/app/shared/components/settings/system/system.component.ts +0 -43
- package/src/app/shared/components/settings/theme/theme.component.html +0 -22
- package/src/app/shared/components/settings/theme/theme.component.scss +0 -0
- package/src/app/shared/components/settings/theme/theme.component.ts +0 -50
- package/src/app/shared/components/spinner/spinner.component.html +0 -1
- package/src/app/shared/components/spinner/spinner.component.scss +0 -0
- package/src/app/shared/components/splash-screen/splash-screen.component.html +0 -5
- package/src/app/shared/components/splash-screen/splash-screen.component.scss +0 -31
- package/src/app/shared/components/splash-screen/splash-screen.component.ts +0 -17
- package/src/app/shared/components/store/store.component.html +0 -22
- package/src/app/shared/components/store/store.component.scss +0 -10
- package/src/app/shared/components/store/store.component.ts +0 -58
- package/src/app/shared/components/tables/tables.component.html +0 -47
- package/src/app/shared/components/tables/tables.component.scss +0 -10
- package/src/app/shared/components/tables/tables.component.ts +0 -81
- package/src/app/shared/components/typescale/typescale.component.html +0 -53
- package/src/app/shared/components/typescale/typescale.component.scss +0 -0
- package/src/app/shared/components/typescale/typescale.component.ts +0 -168
- package/src/app/shared/directives/gravatar/gravatar.directive.ts +0 -29
- package/src/app/shared/pipes/installable.pipe.ts +0 -10
- package/src/app/shared/pipes/prettybytes.pipe.ts +0 -10
- package/src/app/shared/pipes/safehtml.pipe.ts +0 -10
- package/src/assets/fonts/fa/fa-brands-400.eot +0 -0
- package/src/assets/fonts/fa/fa-brands-400.svg +0 -3535
- package/src/assets/fonts/fa/fa-brands-400.ttf +0 -0
- package/src/assets/fonts/fa/fa-brands-400.woff +0 -0
- package/src/assets/fonts/fa/fa-brands-400.woff2 +0 -0
- package/src/assets/fonts/fa/fa-regular-400.eot +0 -0
- package/src/assets/fonts/fa/fa-regular-400.svg +0 -803
- package/src/assets/fonts/fa/fa-regular-400.ttf +0 -0
- package/src/assets/fonts/fa/fa-regular-400.woff +0 -0
- package/src/assets/fonts/fa/fa-regular-400.woff2 +0 -0
- package/src/assets/fonts/fa/fa-solid-900.eot +0 -0
- package/src/assets/fonts/fa/fa-solid-900.svg +0 -4700
- package/src/assets/fonts/fa/fa-solid-900.ttf +0 -0
- package/src/assets/fonts/fa/fa-solid-900.woff +0 -0
- package/src/assets/fonts/fa/fa-solid-900.woff2 +0 -0
- package/src/assets/fonts/md/MaterialIcons-Regular.eot +0 -0
- package/src/assets/fonts/md/MaterialIcons-Regular.ijmap +0 -1
- package/src/assets/fonts/md/MaterialIcons-Regular.svg +0 -2373
- package/src/assets/fonts/md/MaterialIcons-Regular.ttf +0 -0
- package/src/assets/fonts/md/MaterialIcons-Regular.woff +0 -0
- package/src/assets/fonts/md/README.md +0 -9
- package/src/assets/fonts/md/codepoints +0 -932
- package/src/assets/fonts/md/material-icons.css +0 -36
- package/src/assets/fonts/octicon/octicon.eot +0 -0
- package/src/assets/fonts/octicon/octicon.svg +0 -378
- package/src/assets/fonts/octicon/octicon.ttf +0 -0
- package/src/assets/fonts/octicon/octicon.woff +0 -0
- package/src/assets/fonts/octicon/octicon.woff2 +0 -0
- package/src/assets/fonts/roboto/Roboto-Bold-webfont.eot +0 -0
- package/src/assets/fonts/roboto/Roboto-Bold-webfont.svg +0 -607
- package/src/assets/fonts/roboto/Roboto-Bold-webfont.ttf +0 -0
- package/src/assets/fonts/roboto/Roboto-Bold-webfont.woff +0 -0
- package/src/assets/fonts/roboto/Roboto-Medium-webfont.eot +0 -0
- package/src/assets/fonts/roboto/Roboto-Medium-webfont.svg +0 -607
- package/src/assets/fonts/roboto/Roboto-Medium-webfont.ttf +0 -0
- package/src/assets/fonts/roboto/Roboto-Medium-webfont.woff +0 -0
- package/src/assets/fonts/roboto/Roboto-Regular-webfont.eot +0 -0
- package/src/assets/fonts/roboto/Roboto-Regular-webfont.svg +0 -635
- package/src/assets/fonts/roboto/Roboto-Regular-webfont.ttf +0 -0
- package/src/assets/fonts/roboto/Roboto-Regular-webfont.woff +0 -0
- package/src/assets/icons/android-chrome-192x192.png +0 -0
- package/src/assets/icons/android-chrome-512x512.png +0 -0
- package/src/assets/icons/apple-touch-icon.png +0 -0
- package/src/assets/icons/favicon-16x16.png +0 -0
- package/src/assets/icons/favicon-32x32.png +0 -0
- package/src/assets/icons/site.webmanifest +0 -1
- package/src/assets/images/electron.bmp +0 -0
- package/src/bin/activator +0 -0
- package/src/bin/activator.exe +0 -0
- package/src/environments/environment.web.prod.ts +0 -4
- package/src/environments/environment.web.ts +0 -4
- package/src/karma.conf.js +0 -50
- package/src/polyfills-test.ts +0 -1
- package/src/polyfills.ts +0 -53
- package/src/styles/base/loading.scss +0 -87
- package/src/styles/base/reset.scss +0 -408
- package/src/styles/components/alert.scss +0 -54
- package/src/styles/components/buttons.scss +0 -93
- package/src/styles/components/forms.scss +0 -96
- package/src/styles/components/inputs.scss +0 -113
- package/src/styles/components/modal.scss +0 -52
- package/src/styles/components/paginator.scss +0 -108
- package/src/styles/components/scrollbox.scss +0 -71
- package/src/styles/components/spinner.scss +0 -11
- package/src/styles/components/tables.scss +0 -42
- package/src/styles/components/tabs.scss +0 -44
- package/src/styles/components/toolbar.scss +0 -130
- package/src/styles/fonts/md.scss +0 -54
- package/src/styles/fonts/octicons.scss +0 -1351
- package/src/styles/fonts/roboto.scss +0 -32
- package/src/styles/tools/_functions.scss +0 -5
- package/src/styles/tools/_mixins.scss +0 -159
- package/src/styles/tools/_placeholders.scss +0 -17
- package/src/styles/tools/_variables.scss +0 -67
- package/src/styles.scss +0 -37
- package/src/test.ts +0 -17
- package/tailwind.config.js +0 -67
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
import { Component, inject } from '@angular/core';
|
|
2
|
+
import { NewsService } from '../../../core/services';
|
|
3
|
+
import { CollapsiblePanelComponent } from '../../../shared/components/collapsible-panel/collapsible-panel.component';
|
|
4
|
+
|
|
5
|
+
@Component({
|
|
6
|
+
selector: 'app-news-stats',
|
|
7
|
+
standalone: true,
|
|
8
|
+
imports: [CollapsiblePanelComponent],
|
|
9
|
+
template: `
|
|
10
|
+
@if (news.hasArticles()) {
|
|
11
|
+
<app-collapsible-panel title="News">
|
|
12
|
+
<ul class="flex flex-col py-0.5">
|
|
13
|
+
<li class="flex items-center px-3.5 py-1 text-xs font-normal">
|
|
14
|
+
<span
|
|
15
|
+
class="material-symbols-outlined icon-sm mr-2"
|
|
16
|
+
style="font-variation-settings: 'opsz' 20, 'wght' 300; color: var(--text-muted)"
|
|
17
|
+
>article</span
|
|
18
|
+
>
|
|
19
|
+
<span class="flex-1">Total Articles</span>
|
|
20
|
+
<span class="text-[11px]" style="color: var(--text-muted)">{{ news.articleCount() }}</span>
|
|
21
|
+
</li>
|
|
22
|
+
<li class="flex items-center px-3.5 py-1 text-xs font-normal">
|
|
23
|
+
<span
|
|
24
|
+
class="material-symbols-outlined icon-sm mr-2"
|
|
25
|
+
style="font-variation-settings: 'opsz' 20, 'wght' 300; color: var(--text-muted)"
|
|
26
|
+
>source</span
|
|
27
|
+
>
|
|
28
|
+
<span class="flex-1">Sources</span>
|
|
29
|
+
<span class="text-[11px]" style="color: var(--text-muted)">{{ news.sourceCount() }}</span>
|
|
30
|
+
</li>
|
|
31
|
+
<li class="flex items-center px-3.5 py-1 text-xs font-normal">
|
|
32
|
+
<span
|
|
33
|
+
class="material-symbols-outlined icon-sm mr-2"
|
|
34
|
+
style="font-variation-settings: 'opsz' 20, 'wght' 300; color: var(--text-muted)"
|
|
35
|
+
>person</span
|
|
36
|
+
>
|
|
37
|
+
<span class="flex-1">Authors</span>
|
|
38
|
+
<span class="text-[11px]" style="color: var(--text-muted)">{{ news.authorCount() }}</span>
|
|
39
|
+
</li>
|
|
40
|
+
<li class="flex items-center px-3.5 py-1 text-xs font-normal">
|
|
41
|
+
<span
|
|
42
|
+
class="material-symbols-outlined icon-sm mr-2"
|
|
43
|
+
style="font-variation-settings: 'opsz' 20, 'wght' 300; color: var(--text-muted)"
|
|
44
|
+
>schedule</span
|
|
45
|
+
>
|
|
46
|
+
<span class="flex-1">Last Fetched</span>
|
|
47
|
+
<span class="text-[11px]" style="color: var(--text-muted)">{{ news.lastFetched() }}</span>
|
|
48
|
+
</li>
|
|
49
|
+
</ul>
|
|
50
|
+
</app-collapsible-panel>
|
|
51
|
+
}
|
|
52
|
+
`,
|
|
53
|
+
})
|
|
54
|
+
export class NewsStatsComponent {
|
|
55
|
+
readonly news = inject(NewsService);
|
|
56
|
+
}
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
<div class="card">
|
|
2
|
+
<div class="card-header">
|
|
3
|
+
<div class="card-title"><h3>AI API Keys</h3></div>
|
|
4
|
+
</div>
|
|
5
|
+
<div class="card-body space-y-4">
|
|
6
|
+
<div>
|
|
7
|
+
<label class="form-label" for="anthropicKey">Anthropic</label>
|
|
8
|
+
<input
|
|
9
|
+
type="password"
|
|
10
|
+
id="anthropicKey"
|
|
11
|
+
class="form-input w-full"
|
|
12
|
+
[ngModel]="anthropicKey()"
|
|
13
|
+
(ngModelChange)="anthropicKey.set($event)"
|
|
14
|
+
name="anthropicKey"
|
|
15
|
+
placeholder="sk-ant-..."
|
|
16
|
+
autocomplete="off"
|
|
17
|
+
/>
|
|
18
|
+
</div>
|
|
19
|
+
<div>
|
|
20
|
+
<label class="form-label" for="googleKey">Google Gemini</label>
|
|
21
|
+
<input
|
|
22
|
+
type="password"
|
|
23
|
+
id="googleKey"
|
|
24
|
+
class="form-input w-full"
|
|
25
|
+
[ngModel]="googleKey()"
|
|
26
|
+
(ngModelChange)="googleKey.set($event)"
|
|
27
|
+
name="googleKey"
|
|
28
|
+
placeholder="AIza..."
|
|
29
|
+
autocomplete="off"
|
|
30
|
+
/>
|
|
31
|
+
</div>
|
|
32
|
+
<div>
|
|
33
|
+
<label class="form-label" for="openaiKey">OpenAI</label>
|
|
34
|
+
<input
|
|
35
|
+
type="password"
|
|
36
|
+
id="openaiKey"
|
|
37
|
+
class="form-input w-full"
|
|
38
|
+
[ngModel]="openaiKey()"
|
|
39
|
+
(ngModelChange)="openaiKey.set($event)"
|
|
40
|
+
name="openaiKey"
|
|
41
|
+
placeholder="sk-..."
|
|
42
|
+
autocomplete="off"
|
|
43
|
+
/>
|
|
44
|
+
</div>
|
|
45
|
+
</div>
|
|
46
|
+
<div class="card-footer">
|
|
47
|
+
<div>
|
|
48
|
+
@if (saveStatus() === 'saved') {
|
|
49
|
+
<span class="text-xs" [style.color]="'var(--text-muted)'">Keys saved.</span>
|
|
50
|
+
}
|
|
51
|
+
</div>
|
|
52
|
+
<div class="card-tools">
|
|
53
|
+
<button class="btn btn-sm btn-theme" (click)="onSave()">Save</button>
|
|
54
|
+
</div>
|
|
55
|
+
</div>
|
|
56
|
+
</div>
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import { Component, inject, signal, OnInit } from '@angular/core';
|
|
2
|
+
import { FormsModule } from '@angular/forms';
|
|
3
|
+
import { MessageService } from '../../core/services';
|
|
4
|
+
import { StorageType } from '@main/enums';
|
|
5
|
+
|
|
6
|
+
interface AiKeys {
|
|
7
|
+
anthropic: string;
|
|
8
|
+
google: string;
|
|
9
|
+
openai: string;
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
@Component({
|
|
13
|
+
selector: 'app-settings-ai-keys',
|
|
14
|
+
standalone: true,
|
|
15
|
+
imports: [FormsModule],
|
|
16
|
+
templateUrl: './ai-keys.component.html',
|
|
17
|
+
})
|
|
18
|
+
export class SettingsAiKeysComponent implements OnInit {
|
|
19
|
+
private readonly message = inject(MessageService);
|
|
20
|
+
|
|
21
|
+
readonly anthropicKey = signal('');
|
|
22
|
+
readonly googleKey = signal('');
|
|
23
|
+
readonly openaiKey = signal('');
|
|
24
|
+
readonly saveStatus = signal<'idle' | 'saved'>('idle');
|
|
25
|
+
|
|
26
|
+
async ngOnInit() {
|
|
27
|
+
const keys = (await this.message.get(StorageType.AiKeys, null)) as AiKeys | null;
|
|
28
|
+
if (keys) {
|
|
29
|
+
this.anthropicKey.set(keys.anthropic ?? '');
|
|
30
|
+
this.googleKey.set(keys.google ?? '');
|
|
31
|
+
this.openaiKey.set(keys.openai ?? '');
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
async onSave() {
|
|
36
|
+
await this.message.set(StorageType.AiKeys, {
|
|
37
|
+
anthropic: this.anthropicKey(),
|
|
38
|
+
google: this.googleKey(),
|
|
39
|
+
openai: this.openaiKey(),
|
|
40
|
+
});
|
|
41
|
+
this.saveStatus.set('saved');
|
|
42
|
+
setTimeout(() => this.saveStatus.set('idle'), 2000);
|
|
43
|
+
}
|
|
44
|
+
}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
<div class="card" style="border-color: #dc2626">
|
|
2
|
+
<div class="card-header" style="border-color: #dc2626">
|
|
3
|
+
<div class="card-title"><h3 style="color: #dc2626">Danger Zone</h3></div>
|
|
4
|
+
</div>
|
|
5
|
+
<div class="card-body space-y-4">
|
|
6
|
+
<div class="flex items-center justify-between">
|
|
7
|
+
<div>
|
|
8
|
+
<p class="text-xs font-semibold" [style.color]="'var(--text-primary)'">Drop active database</p>
|
|
9
|
+
<p class="text-xs" [style.color]="'var(--text-muted)'">Drops the database and all its data.</p>
|
|
10
|
+
</div>
|
|
11
|
+
<button class="btn btn-sm btn-danger" (click)="onDropDatabase()">Drop</button>
|
|
12
|
+
</div>
|
|
13
|
+
|
|
14
|
+
<div class="flex items-center justify-between">
|
|
15
|
+
<div>
|
|
16
|
+
<p class="text-xs font-semibold" [style.color]="'var(--text-primary)'">Clear system storage</p>
|
|
17
|
+
<p class="text-xs" [style.color]="'var(--text-muted)'">Removes all settings from system store.</p>
|
|
18
|
+
</div>
|
|
19
|
+
<button class="btn btn-sm btn-danger" (click)="onClearStore()">Clear</button>
|
|
20
|
+
</div>
|
|
21
|
+
</div>
|
|
22
|
+
</div>
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import { Component, inject } from '@angular/core';
|
|
2
|
+
import { MessageService } from '../../core/services';
|
|
3
|
+
|
|
4
|
+
@Component({
|
|
5
|
+
selector: 'app-settings-danger-zone',
|
|
6
|
+
standalone: true,
|
|
7
|
+
templateUrl: './danger-zone.component.html',
|
|
8
|
+
})
|
|
9
|
+
export class SettingsDangerZoneComponent {
|
|
10
|
+
private readonly message = inject(MessageService);
|
|
11
|
+
|
|
12
|
+
async onDropDatabase() {
|
|
13
|
+
const response = await this.message.showMessageBox({
|
|
14
|
+
type: 'question',
|
|
15
|
+
buttons: ['Yes', 'No'],
|
|
16
|
+
title: 'Confirm',
|
|
17
|
+
message: 'Are you sure you want to drop the database?',
|
|
18
|
+
});
|
|
19
|
+
if (response?.response === 0) {
|
|
20
|
+
await this.message.dropDatabase();
|
|
21
|
+
this.message.reloadWindow();
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
async onClearStore() {
|
|
26
|
+
const response = await this.message.showMessageBox({
|
|
27
|
+
type: 'question',
|
|
28
|
+
buttons: ['Yes', 'No'],
|
|
29
|
+
title: 'Confirm',
|
|
30
|
+
message: 'Are you sure you want to delete the application store?',
|
|
31
|
+
});
|
|
32
|
+
if (response?.response === 0) {
|
|
33
|
+
await this.message.clearStore();
|
|
34
|
+
this.message.reloadWindow();
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
<div class="card">
|
|
2
|
+
<div class="card-header">
|
|
3
|
+
<div class="card-title"><h3>System Logs</h3></div>
|
|
4
|
+
<div class="card-tools">
|
|
5
|
+
<button class="btn btn-sm btn-default" (click)="onTruncateLogs()">Clear All</button>
|
|
6
|
+
</div>
|
|
7
|
+
</div>
|
|
8
|
+
<div class="card-body p-0">
|
|
9
|
+
@if (logs().length) {
|
|
10
|
+
<table class="w-full text-xs" [style.color]="'var(--text-primary)'">
|
|
11
|
+
<thead>
|
|
12
|
+
<tr [style.border-bottom]="'1px solid var(--border-subtle)'">
|
|
13
|
+
<th class="text-left px-4 py-2 font-semibold" [style.color]="'var(--text-secondary)'">Message</th>
|
|
14
|
+
<th class="text-center px-4 py-2 font-semibold w-24" [style.color]="'var(--text-secondary)'">Status</th>
|
|
15
|
+
<th class="text-center px-4 py-2 font-semibold w-40" [style.color]="'var(--text-secondary)'">Created</th>
|
|
16
|
+
<th class="text-center px-4 py-2 font-semibold w-16" [style.color]="'var(--text-secondary)'"></th>
|
|
17
|
+
</tr>
|
|
18
|
+
</thead>
|
|
19
|
+
<tbody>
|
|
20
|
+
@for (item of logs(); track item.id) {
|
|
21
|
+
<tr [style.border-bottom]="'1px solid var(--border-subtle)'">
|
|
22
|
+
<td class="px-4 py-2">
|
|
23
|
+
<span class="font-semibold">{{ filterMessageType(item.type) }}</span>
|
|
24
|
+
<span class="ml-2" [style.color]="'var(--text-muted)'">{{ item.message.substring(0, 80) }}</span>
|
|
25
|
+
</td>
|
|
26
|
+
<td class="text-center px-4 py-2">{{ filterMessageStatus(item.status) }}</td>
|
|
27
|
+
<td class="text-center px-4 py-2" [style.color]="'var(--text-muted)'">{{ item.created | date: 'medium' }}</td>
|
|
28
|
+
<td class="text-center px-4 py-2">
|
|
29
|
+
<button class="btn btn-xs btn-ghost" (click)="onDeleteLog(item.id)" title="Delete">
|
|
30
|
+
<span class="material-symbols-outlined icon-xs">delete</span>
|
|
31
|
+
</button>
|
|
32
|
+
</td>
|
|
33
|
+
</tr>
|
|
34
|
+
}
|
|
35
|
+
</tbody>
|
|
36
|
+
</table>
|
|
37
|
+
} @else {
|
|
38
|
+
<div class="px-4 py-6 text-center text-xs" [style.color]="'var(--text-muted)'">No log entries.</div>
|
|
39
|
+
}
|
|
40
|
+
</div>
|
|
41
|
+
</div>
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import { Component, inject, signal, OnInit } from '@angular/core';
|
|
2
|
+
import { DatePipe } from '@angular/common';
|
|
3
|
+
import { MessageService } from '../../core/services';
|
|
4
|
+
import type { Logger } from '@main/database/entity/Logger.schema';
|
|
5
|
+
|
|
6
|
+
@Component({
|
|
7
|
+
selector: 'app-settings-logs',
|
|
8
|
+
standalone: true,
|
|
9
|
+
imports: [DatePipe],
|
|
10
|
+
templateUrl: './logs.component.html',
|
|
11
|
+
})
|
|
12
|
+
export class SettingsLogsComponent implements OnInit {
|
|
13
|
+
private readonly message = inject(MessageService);
|
|
14
|
+
|
|
15
|
+
readonly logs = signal<Logger[]>([]);
|
|
16
|
+
|
|
17
|
+
async ngOnInit() {
|
|
18
|
+
await this.loadLogs();
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
async loadLogs() {
|
|
22
|
+
const result = await this.message.loggerFind({ order: { id: 'DESC' } });
|
|
23
|
+
this.logs.set(result);
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
filterMessageType(type: number): string {
|
|
27
|
+
return type === 1 ? 'System Action' : 'User Action';
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
filterMessageStatus(status: number): string {
|
|
31
|
+
return status === 1 ? 'Error' : 'Success';
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
async onDeleteLog(id: number) {
|
|
35
|
+
await this.message.loggerDelete(id);
|
|
36
|
+
await this.loadLogs();
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
async onTruncateLogs() {
|
|
40
|
+
await this.message.loggerTruncate();
|
|
41
|
+
await this.loadLogs();
|
|
42
|
+
}
|
|
43
|
+
}
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
<div class="card">
|
|
2
|
+
<div class="card-header">
|
|
3
|
+
<div class="card-title"><h3>News API</h3></div>
|
|
4
|
+
<div class="card-tools">
|
|
5
|
+
@if (newsApiKey()) {
|
|
6
|
+
<button class="btn btn-sm btn-default" (click)="onFetchNews()" [disabled]="fetchStatus() === 'fetching'">
|
|
7
|
+
{{ fetchStatus() === 'fetching' ? 'Fetching...' : 'Fetch Articles' }}
|
|
8
|
+
</button>
|
|
9
|
+
}
|
|
10
|
+
</div>
|
|
11
|
+
</div>
|
|
12
|
+
<div class="card-body">
|
|
13
|
+
<div>
|
|
14
|
+
<label class="form-label" for="apiKey">API Key</label>
|
|
15
|
+
<input
|
|
16
|
+
type="text"
|
|
17
|
+
id="apiKey"
|
|
18
|
+
class="form-input w-full"
|
|
19
|
+
[ngModel]="newsApiKey()"
|
|
20
|
+
(ngModelChange)="newsApiKey.set($event)"
|
|
21
|
+
name="apiKey"
|
|
22
|
+
placeholder="Enter your newsapi.org key"
|
|
23
|
+
/>
|
|
24
|
+
<p class="text-xs mt-1" [style.color]="'var(--text-muted)'">Visit newsapi.org to obtain your free key.</p>
|
|
25
|
+
</div>
|
|
26
|
+
</div>
|
|
27
|
+
<div class="card-footer">
|
|
28
|
+
<div>
|
|
29
|
+
@if (saveStatus() === 'saved') {
|
|
30
|
+
<span class="text-xs" [style.color]="'var(--text-muted)'">API key saved.</span>
|
|
31
|
+
}
|
|
32
|
+
@if (fetchMessage()) {
|
|
33
|
+
<span class="text-xs" [style.color]="fetchStatus() === 'error' ? '#dc2626' : 'var(--text-muted)'">{{ fetchMessage() }}</span>
|
|
34
|
+
}
|
|
35
|
+
</div>
|
|
36
|
+
<div class="card-tools">
|
|
37
|
+
<button class="btn btn-sm btn-theme" (click)="onSaveNewsApiKey()">Save</button>
|
|
38
|
+
</div>
|
|
39
|
+
</div>
|
|
40
|
+
</div>
|
|
@@ -0,0 +1,57 @@
|
|
|
1
|
+
import { Component, inject, signal, OnInit } from '@angular/core';
|
|
2
|
+
import { FormsModule } from '@angular/forms';
|
|
3
|
+
import { MessageService, NewsService } from '../../core/services';
|
|
4
|
+
import { StorageType } from '@main/enums';
|
|
5
|
+
import type { NewsType } from '@main/types';
|
|
6
|
+
|
|
7
|
+
@Component({
|
|
8
|
+
selector: 'app-settings-news-api',
|
|
9
|
+
standalone: true,
|
|
10
|
+
imports: [FormsModule],
|
|
11
|
+
templateUrl: './news-api.component.html',
|
|
12
|
+
})
|
|
13
|
+
export class SettingsNewsApiComponent implements OnInit {
|
|
14
|
+
private readonly message = inject(MessageService);
|
|
15
|
+
private readonly news = inject(NewsService);
|
|
16
|
+
|
|
17
|
+
readonly newsApiKey = signal('');
|
|
18
|
+
readonly saveStatus = signal<'idle' | 'saved'>('idle');
|
|
19
|
+
readonly fetchStatus = signal<'idle' | 'fetching' | 'done' | 'error'>('idle');
|
|
20
|
+
readonly fetchMessage = signal('');
|
|
21
|
+
|
|
22
|
+
async ngOnInit() {
|
|
23
|
+
const config = (await this.message.get(StorageType.News, null)) as NewsType | null;
|
|
24
|
+
if (config?.apiKey) {
|
|
25
|
+
this.newsApiKey.set(config.apiKey);
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
async onSaveNewsApiKey() {
|
|
30
|
+
const apiKey = this.newsApiKey();
|
|
31
|
+
if (!apiKey) return;
|
|
32
|
+
await this.message.set(StorageType.News, { apiKey });
|
|
33
|
+
this.saveStatus.set('saved');
|
|
34
|
+
await this.news.refresh();
|
|
35
|
+
setTimeout(() => this.saveStatus.set('idle'), 2000);
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
async onFetchNews() {
|
|
39
|
+
this.fetchStatus.set('fetching');
|
|
40
|
+
this.fetchMessage.set('');
|
|
41
|
+
try {
|
|
42
|
+
const endpoint = `https://newsapi.org/v2/top-headlines?country=us&apiKey=${this.newsApiKey()}`;
|
|
43
|
+
const result = await this.message.fetchLatestNews({ endpoint });
|
|
44
|
+
const count = result?.articles?.length ?? 0;
|
|
45
|
+
this.fetchMessage.set(`Fetched ${count} article${count !== 1 ? 's' : ''}.`);
|
|
46
|
+
this.fetchStatus.set('done');
|
|
47
|
+
await this.news.refresh();
|
|
48
|
+
} catch {
|
|
49
|
+
this.fetchMessage.set('Failed to fetch articles.');
|
|
50
|
+
this.fetchStatus.set('error');
|
|
51
|
+
}
|
|
52
|
+
setTimeout(() => {
|
|
53
|
+
this.fetchStatus.set('idle');
|
|
54
|
+
this.fetchMessage.set('');
|
|
55
|
+
}, 3000);
|
|
56
|
+
}
|
|
57
|
+
}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { Component } from '@angular/core';
|
|
2
|
+
import { SettingsThemeComponent } from '../../theme/theme.component';
|
|
3
|
+
import { SettingsAiKeysComponent } from '../../ai-keys/ai-keys.component';
|
|
4
|
+
import { SettingsNewsApiComponent } from '../../news-api/news-api.component';
|
|
5
|
+
import { SettingsQuickActionsComponent } from '../../quick-actions/quick-actions.component';
|
|
6
|
+
import { SettingsDangerZoneComponent } from '../../danger-zone/danger-zone.component';
|
|
7
|
+
|
|
8
|
+
@Component({
|
|
9
|
+
selector: 'app-settings-general-page',
|
|
10
|
+
standalone: true,
|
|
11
|
+
imports: [
|
|
12
|
+
SettingsThemeComponent,
|
|
13
|
+
SettingsAiKeysComponent,
|
|
14
|
+
SettingsNewsApiComponent,
|
|
15
|
+
SettingsQuickActionsComponent,
|
|
16
|
+
SettingsDangerZoneComponent,
|
|
17
|
+
],
|
|
18
|
+
templateUrl: './general.component.html',
|
|
19
|
+
})
|
|
20
|
+
export class SettingsGeneralPageComponent {}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
<app-settings-logs />
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import { Component } from '@angular/core';
|
|
2
|
+
import { SettingsLogsComponent } from '../../logs/logs.component';
|
|
3
|
+
|
|
4
|
+
@Component({
|
|
5
|
+
selector: 'app-settings-logs-page',
|
|
6
|
+
standalone: true,
|
|
7
|
+
imports: [SettingsLogsComponent],
|
|
8
|
+
templateUrl: './logs.component.html',
|
|
9
|
+
})
|
|
10
|
+
export class SettingsLogsPageComponent {}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
<app-settings-system-info />
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import { Component } from '@angular/core';
|
|
2
|
+
import { SettingsSystemInfoComponent } from '../../system-info/system-info.component';
|
|
3
|
+
|
|
4
|
+
@Component({
|
|
5
|
+
selector: 'app-settings-system-page',
|
|
6
|
+
standalone: true,
|
|
7
|
+
imports: [SettingsSystemInfoComponent],
|
|
8
|
+
templateUrl: './system.component.html',
|
|
9
|
+
})
|
|
10
|
+
export class SettingsSystemPageComponent {}
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
<div class="card">
|
|
2
|
+
<div class="card-header">
|
|
3
|
+
<div class="card-title"><h3>Quick Actions</h3></div>
|
|
4
|
+
<div class="card-tools">
|
|
5
|
+
@if (syncing()) {
|
|
6
|
+
<span class="text-xs font-medium" [style.color]="'var(--text-muted)'">Syncing...</span>
|
|
7
|
+
}
|
|
8
|
+
</div>
|
|
9
|
+
</div>
|
|
10
|
+
<div class="card-body space-y-4">
|
|
11
|
+
<div class="flex items-center justify-between">
|
|
12
|
+
<div>
|
|
13
|
+
<p class="text-xs font-semibold" [style.color]="'var(--text-primary)'">Reset favorited fonts</p>
|
|
14
|
+
<p class="text-xs" [style.color]="'var(--text-muted)'">Resets fonts that have been favorited.</p>
|
|
15
|
+
</div>
|
|
16
|
+
<button class="btn btn-sm btn-default" (click)="onResetFavorites()">Run</button>
|
|
17
|
+
</div>
|
|
18
|
+
|
|
19
|
+
<div class="flex items-center justify-between">
|
|
20
|
+
<div>
|
|
21
|
+
<p class="text-xs font-semibold" [style.color]="'var(--text-primary)'">Synchronize system fonts</p>
|
|
22
|
+
<p class="text-xs" [style.color]="'var(--text-muted)'">Imports fonts from system fonts folder.</p>
|
|
23
|
+
</div>
|
|
24
|
+
<button class="btn btn-sm btn-default" (click)="onSyncSystemFonts()" [disabled]="syncing()">
|
|
25
|
+
@if (syncing()) {
|
|
26
|
+
<app-spinner [animating]="true" />
|
|
27
|
+
} @else {
|
|
28
|
+
Run
|
|
29
|
+
}
|
|
30
|
+
</button>
|
|
31
|
+
</div>
|
|
32
|
+
|
|
33
|
+
<div class="flex items-center justify-between">
|
|
34
|
+
<div>
|
|
35
|
+
<p class="text-xs font-semibold" [style.color]="'var(--text-primary)'">Restart application</p>
|
|
36
|
+
<p class="text-xs" [style.color]="'var(--text-muted)'">Restarts the application.</p>
|
|
37
|
+
</div>
|
|
38
|
+
<button class="btn btn-sm btn-default" (click)="onRestartApp()">Run</button>
|
|
39
|
+
</div>
|
|
40
|
+
|
|
41
|
+
<div class="flex items-center justify-between">
|
|
42
|
+
<div>
|
|
43
|
+
<p class="text-xs font-semibold" [style.color]="'var(--text-primary)'">Exit application</p>
|
|
44
|
+
<p class="text-xs" [style.color]="'var(--text-muted)'">Exits the application.</p>
|
|
45
|
+
</div>
|
|
46
|
+
<button class="btn btn-sm btn-default" (click)="onExitApp()">Run</button>
|
|
47
|
+
</div>
|
|
48
|
+
</div>
|
|
49
|
+
</div>
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
import { Component, inject, signal } from '@angular/core';
|
|
2
|
+
import { SpinnerComponent } from '../../shared/components/spinner/spinner.component';
|
|
3
|
+
import { DatabaseService, MessageService } from '../../core/services';
|
|
4
|
+
|
|
5
|
+
@Component({
|
|
6
|
+
selector: 'app-settings-quick-actions',
|
|
7
|
+
standalone: true,
|
|
8
|
+
imports: [SpinnerComponent],
|
|
9
|
+
templateUrl: './quick-actions.component.html',
|
|
10
|
+
})
|
|
11
|
+
export class SettingsQuickActionsComponent {
|
|
12
|
+
private readonly database = inject(DatabaseService);
|
|
13
|
+
private readonly message = inject(MessageService);
|
|
14
|
+
|
|
15
|
+
readonly syncing = signal(false);
|
|
16
|
+
|
|
17
|
+
async onResetFavorites() {
|
|
18
|
+
await this.database.resetFavorites();
|
|
19
|
+
await this.database.fetchSystemStats();
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
async onSyncSystemFonts() {
|
|
23
|
+
this.syncing.set(true);
|
|
24
|
+
try {
|
|
25
|
+
await this.database.syncSystemFonts();
|
|
26
|
+
} finally {
|
|
27
|
+
this.syncing.set(false);
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
onRestartApp() {
|
|
32
|
+
this.message.beep();
|
|
33
|
+
this.message.reloadWindow();
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
onExitApp() {
|
|
37
|
+
this.message.beep();
|
|
38
|
+
this.message.quitApplication();
|
|
39
|
+
}
|
|
40
|
+
}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
<div class="grid h-screen grid-rows-[56px_1fr_32px]" [style.background-color]="'var(--surface-0)'">
|
|
2
|
+
<!-- Header -->
|
|
3
|
+
<div
|
|
4
|
+
class="flex items-center justify-between px-4"
|
|
5
|
+
[style.background-color]="'var(--surface-2)'"
|
|
6
|
+
[style.border-bottom]="'1px solid var(--border-subtle)'"
|
|
7
|
+
>
|
|
8
|
+
<h1 class="text-sm font-bold" [style.color]="'var(--text-primary)'">Settings</h1>
|
|
9
|
+
<a routerLink="/home" class="btn btn-sm btn-default">Back</a>
|
|
10
|
+
</div>
|
|
11
|
+
|
|
12
|
+
<!-- Body -->
|
|
13
|
+
<div class="grid overflow-hidden" style="grid-template-columns: 200px 1fr">
|
|
14
|
+
<!-- Sidebar -->
|
|
15
|
+
<nav
|
|
16
|
+
class="overflow-auto py-4 px-3"
|
|
17
|
+
[style.background-color]="'var(--surface-2)'"
|
|
18
|
+
[style.border-right]="'1px solid var(--border-subtle)'"
|
|
19
|
+
>
|
|
20
|
+
<ul class="space-y-1">
|
|
21
|
+
<li>
|
|
22
|
+
<a routerLink="general" routerLinkActive="active" class="settings-nav-link">General</a>
|
|
23
|
+
</li>
|
|
24
|
+
<li>
|
|
25
|
+
<a routerLink="logs" routerLinkActive="active" class="settings-nav-link">Logs</a>
|
|
26
|
+
</li>
|
|
27
|
+
<li>
|
|
28
|
+
<a routerLink="system" routerLinkActive="active" class="settings-nav-link">System</a>
|
|
29
|
+
</li>
|
|
30
|
+
</ul>
|
|
31
|
+
</nav>
|
|
32
|
+
|
|
33
|
+
<!-- Content -->
|
|
34
|
+
<div class="overflow-auto" [style.background-color]="'var(--surface-1)'">
|
|
35
|
+
<div class="p-6">
|
|
36
|
+
<router-outlet />
|
|
37
|
+
</div>
|
|
38
|
+
</div>
|
|
39
|
+
</div>
|
|
40
|
+
|
|
41
|
+
<!-- Footer -->
|
|
42
|
+
<div
|
|
43
|
+
class="flex items-center px-4"
|
|
44
|
+
[style.background-color]="'var(--surface-2)'"
|
|
45
|
+
[style.border-top]="'1px solid var(--border-subtle)'"
|
|
46
|
+
></div>
|
|
47
|
+
</div>
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import { Component } from '@angular/core';
|
|
2
|
+
import { RouterLink, RouterLinkActive, RouterOutlet } from '@angular/router';
|
|
3
|
+
|
|
4
|
+
@Component({
|
|
5
|
+
selector: 'app-settings',
|
|
6
|
+
templateUrl: './settings.component.html',
|
|
7
|
+
standalone: true,
|
|
8
|
+
imports: [RouterOutlet, RouterLink, RouterLinkActive],
|
|
9
|
+
})
|
|
10
|
+
export class SettingsComponent {}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
<div class="card">
|
|
2
|
+
<div class="card-header">
|
|
3
|
+
<div class="card-title"><h3>System Information</h3></div>
|
|
4
|
+
</div>
|
|
5
|
+
<div class="card-body p-0">
|
|
6
|
+
@if (systemSettings().length) {
|
|
7
|
+
<table class="w-full text-xs" [style.color]="'var(--text-primary)'">
|
|
8
|
+
<tbody>
|
|
9
|
+
@for (item of systemSettings(); track item.key) {
|
|
10
|
+
<tr [style.border-bottom]="'1px solid var(--border-subtle)'">
|
|
11
|
+
<td class="px-4 py-2 font-semibold w-1/3" [style.color]="'var(--text-secondary)'">{{ item.key }}</td>
|
|
12
|
+
<td class="px-4 py-2" [style.color]="'var(--text-muted)'">{{ item.value }}</td>
|
|
13
|
+
</tr>
|
|
14
|
+
}
|
|
15
|
+
</tbody>
|
|
16
|
+
</table>
|
|
17
|
+
} @else {
|
|
18
|
+
<div class="px-4 py-6 text-center text-xs" [style.color]="'var(--text-muted)'">Loading system information...</div>
|
|
19
|
+
}
|
|
20
|
+
</div>
|
|
21
|
+
</div>
|