diff --git a/ui/angular.json b/ui/angular.json index 02a3fde..d9afffa 100644 --- a/ui/angular.json +++ b/ui/angular.json @@ -26,19 +26,13 @@ "src/favicon.ico", "src/assets", "src/manifest.json", - "src/robots.txt", - { - "glob": "**/*", - "input": "src/app/modules/eqmac-components/assets", - "output": "assets/" - } - + "src/robots.txt" ], "styles": [ "src/styles.scss" ], "scripts": [], - "aot": false, + "aot": true, "vendorChunk": true, "extractLicenses": false, "buildOptimizer": false, @@ -58,12 +52,11 @@ "optimization": true, "outputHashing": "all", "sourceMap": false, - "extractCss": true, "namedChunks": false, "aot": true, "extractLicenses": true, "vendorChunk": false, - "buildOptimizer": true, + "buildOptimizer": false, "budgets": [ { "type": "initial", diff --git a/ui/package.json b/ui/package.json index 2a9cd0e..a03cc85 100644 --- a/ui/package.json +++ b/ui/package.json @@ -3,8 +3,8 @@ "version": "3.0.0", "scripts": { "lint": "npx eslint .", - "start": "ng serve --port 8080 --host 0.0.0.0 --disable-host-check", - "build": "rm -rf dist/ && ng build --configuration production && node -e \"console.log(require('./package.json').version)\" > dist/version.txt && cd dist/ && zip -r -D ui.zip * -x '*.DS_Store' && cp ui.zip ../../native/app/Embedded" + "start": "../node_modules/.bin/ng serve --port 8080 --host 0.0.0.0 --disable-host-check", + "build": "rm -rf dist/ && ../node_modules/.bin/ng build --configuration production && node -e \"console.log(require('./package.json').version)\" > dist/version.txt && cd dist/ && zip -r -D ui.zip * -x '*.DS_Store' && cp ui.zip ../../native/app/Embedded" }, "private": true, "dependencies": { diff --git a/ui/src/app/app.component.html b/ui/src/app/app.component.html index 6c82356..5a43f91 100644 --- a/ui/src/app/app.component.html +++ b/ui/src/app/app.component.html @@ -23,9 +23,9 @@ - + - + diff --git a/ui/src/app/app.component.scss b/ui/src/app/app.component.scss index 2faf513..75114fd 100644 --- a/ui/src/app/app.component.scss +++ b/ui/src/app/app.component.scss @@ -1,25 +1,35 @@ @import "./styles/colors"; ::ng-deep *:not(input) { - user-select: none; - -moz-user-select: none; - -khtml-user-select: none; - -webkit-user-select: none; - -o-user-select: none; - cursor: default; + -webkit-touch-callout: none !important; /* iOS Safari */ + -webkit-user-select: none !important; /* Safari */ + -khtml-user-select: none !important; /* Konqueror HTML */ + -moz-user-select: none !important; /* Old versions of Firefox */ + -ms-user-select: none !important; /* Internet Explorer/Edge */ + user-select: none !important; /* Non-prefixed version, currently + supported by Chrome, Edge, Opera and Firefox */ &::-webkit-scrollbar { display: none; } -} +} -// ::ng-deep * : +::ng-deep .pointer { + cursor: pointer !important; + + & * { + cursor: pointer !important; + } +} $noise: url(''); ::ng-deep html, body { - width: 100vw; - height: 100vh; + width: 100vw !important; + height: 100vh !important; + max-width: 100vw !important; + max-height: 100vh !important; + margin: 0; overflow: hidden; background-image: $noise; @@ -31,6 +41,9 @@ $noise: url(' text-rendering: optimizeLegibility; text-shadow: rgba(0, 0, 0, .01) 0 0 1px; -webkit-font-smoothing: subpixel-antialiased; + &::-webkit-scrollbar { + display: none; + } transform: perspective(1px) translateZ(0); backface-visibility: hidden; } @@ -84,6 +97,11 @@ $noise: url(' .dropdown-section { width: 100%; + background-color: $gradient-end; + box-shadow: 0px 8px 77px 2px rgba(0,0,0,0.56); + padding: 10px; + $border: 1px solid black; + border-bottom: $border; max-width: 400px; box-sizing: border-box; position: absolute; diff --git a/ui/src/app/app.component.ts b/ui/src/app/app.component.ts index 80446c0..850659b 100644 --- a/ui/src/app/app.component.ts +++ b/ui/src/app/app.component.ts @@ -12,6 +12,7 @@ import { TransitionService } from './services/transitions.service' import { AnalyticsService } from './services/analytics.service' import { ApplicationService } from './services/app.service' import { SettingsService, IconMode } from './sections/settings/settings.service' +import { ToastService } from './services/toast.service' import { OptionsDialogComponent } from './components/options-dialog/options-dialog.component' import { Option, Options } from './components/options/options.component' import { HeaderComponent } from './sections/header/header.component' @@ -59,7 +60,8 @@ export class AppComponent implements OnInit, AfterContentInit { public transitions: TransitionService, public analytics: AnalyticsService, public app: ApplicationService, - public settings: SettingsService + public settings: SettingsService, + public toast: ToastService ) { this.app.ref = this } @@ -88,10 +90,38 @@ export class AppComponent implements OnInit, AfterContentInit { return minHeight } + get minWidth () { + return 400 + } + + get maxHeight () { + const divider = 3 + + const { + volumeFeatureEnabled, balanceFeatureEnabled, + equalizersFeatureEnabled, + outputFeatureEnabled + } = this.ui.settings + let maxHeight = this.header.height + divider + + ((volumeFeatureEnabled || balanceFeatureEnabled) ? (this.volumeBoosterBalance.height + divider) : 0) + + (equalizersFeatureEnabled ? (this.equalizers.maxHeight + divider) : 0) + + (outputFeatureEnabled ? this.outputs.height : 0) + + const dropdownSection = document.getElementById('dropdown-section') + if (dropdownSection) { + const dropdownHeight = dropdownSection.offsetHeight + this.header.height + divider + if (dropdownHeight > maxHeight) { + maxHeight = dropdownHeight + } + } + + return maxHeight + } + async ngOnInit () { await this.sync() - this.startHeightSync() await this.fixUIMode() + this.startDimensionsSync() await this.setupPrivacy() } @@ -201,6 +231,7 @@ This data would help us improve and grow the product.` async ngAfterContentInit () { await this.utils.delay(this.animationDuration) this.loaded = true + await this.utils.delay(1000) this.ui.loaded() } @@ -210,23 +241,39 @@ This data would help us improve and grow the product.` ]) } - async startHeightSync () { - this.syncHeight() + async startDimensionsSync () { + this.previousMinHeight = this.minHeight + this.previousMaxHeight = this.maxHeight setInterval(() => { - this.syncHeight() + this.syncMinHeight() + this.syncMaxHeight() }, 1000) } - private previousMinHeight - async syncHeight () { + private previousMinHeight: number + async syncMinHeight () { const diff = this.minHeight - this.previousMinHeight this.previousMinHeight = this.minHeight - await this.ui.setMinHeight({ minHeight: this.minHeight }) + if (diff !== 0) { + this.ui.onMinHeightChanged.emit() + await this.ui.setMinHeight({ minHeight: this.minHeight }) + } + if (diff < 0) { this.ui.changeHeight({ diff }) } } + private previousMaxHeight + async syncMaxHeight () { + const diff = this.maxHeight - this.previousMaxHeight + this.previousMaxHeight = this.maxHeight + await this.ui.setMaxHeight({ maxHeight: this.maxHeight }) + if (diff > 0) { + // this.ui.changeHeight({ diff }) + } + } + async getTransitionSettings () { const settings = await this.transitions.getSettings() this.animationDuration = settings.duration @@ -239,6 +286,12 @@ This data would help us improve and grow the product.` } } + openDropdownSection (section: string) { + for (const key in this.showDropdownSections) { + this.showDropdownSections[key] = key === section + } + } + async fixUIMode () { const [ mode, iconMode ] = await Promise.all([ this.ui.getMode(), @@ -250,7 +303,7 @@ This data would help us improve and grow the product.` } } - closeDropdownSection (section: string, event?: any) { + closeDropdownSection (section: string, event?: MouseEvent) { // if (event && event.target && ['backdrop', 'mat-dialog'].some(e => event.target.className.includes(e))) return if (this.dialog.openDialogs.length > 0) return if (section in this.showDropdownSections) { diff --git a/ui/src/app/services/app.service.ts b/ui/src/app/services/app.service.ts index 179271d..d964981 100644 --- a/ui/src/app/services/app.service.ts +++ b/ui/src/app/services/app.service.ts @@ -12,6 +12,25 @@ export interface Info { isOpenSource: boolean driverVersion?: string } + +export const SystemSounds = [ + 'Basso', + 'Blow', + 'Bottle', + 'From', + 'Funk', + 'Glass', + 'Hero', + 'Morse', + 'Ping', + 'Pop', + 'Purr', + 'Sosumi', + 'Submarine', + 'Tink' +] as const +export type SystemSound = typeof SystemSounds[number] + @Injectable({ providedIn: 'root' }) @@ -85,4 +104,17 @@ export class ApplicationService extends DataService { this.enabled = enabled return this.request({ method: 'POST', endpoint: '/enabled', data: { enabled } }) } + + async getBundleIcon (bundleId: string): Promise { + const resp = await this.request({ method: 'GET', endpoint: '/bundle-icon', data: { bundleId } }) + return resp?.base64 + } + + playAlertSound () { + return this.request({ method: 'GET', endpoint: '/alert-sound' }) + } + + playSystemSound (name: SystemSound) { + return this.request({ method: 'POST', endpoint: '/system-sound', data: { name } }) + } } diff --git a/ui/src/app/services/constants.service.ts b/ui/src/app/services/constants.service.ts index 4996028..b94975a 100644 --- a/ui/src/app/services/constants.service.ts +++ b/ui/src/app/services/constants.service.ts @@ -5,8 +5,8 @@ import { Injectable } from '@angular/core' }) export class ConstantsService { readonly DOMAIN = 'eqmac.app' - readonly FAQ_URL = new URL(`https://${this.DOMAIN}/faq`) + readonly FAQ_URL = new URL(`https://${this.DOMAIN}#faq`) + readonly FEATURES_URL = new URL(`https://${this.DOMAIN}#features`) + readonly ACCOUNT_URL = new URL(`https://${this.DOMAIN}/account`) readonly BUG_REPORT_URL = new URL(`https://${this.DOMAIN}/report-bug`) - readonly LOCAL_API_URL = 'https://127.0.0.1' - readonly REMOTE_API_URL = `https://api.${this.DOMAIN}` } diff --git a/ui/src/app/services/context.service.ts b/ui/src/app/services/context.service.ts index 4f524a7..7a9a6a1 100644 --- a/ui/src/app/services/context.service.ts +++ b/ui/src/app/services/context.service.ts @@ -1,6 +1,6 @@ import { Injectable } from '@angular/core' -export type Context = 'EQ_TYPE_EXPERT' +export type Context = never @Injectable({ providedIn: 'root' diff --git a/ui/src/app/services/toast.service.ts b/ui/src/app/services/toast.service.ts index 1308da6..c4cad7a 100644 --- a/ui/src/app/services/toast.service.ts +++ b/ui/src/app/services/toast.service.ts @@ -29,7 +29,6 @@ export class ToastService { const toast = this.snackBar.open(message, action, { horizontalPosition: 'center', verticalPosition: 'top', - duration, panelClass: [ bgClass, textClass ] }) setTimeout(() => { diff --git a/ui/src/app/services/types.service.ts b/ui/src/app/services/types.service.ts new file mode 100644 index 0000000..e0eb1f5 --- /dev/null +++ b/ui/src/app/services/types.service.ts @@ -0,0 +1,6 @@ + +export class Types { + static unreachable (type: never) { + console.error(`Should not have reached type: ${type}`) + } +} diff --git a/ui/src/app/services/ui.service.ts b/ui/src/app/services/ui.service.ts index 45f63e0..065fef6 100644 --- a/ui/src/app/services/ui.service.ts +++ b/ui/src/app/services/ui.service.ts @@ -15,7 +15,6 @@ export interface UISettings { equalizersFeatureEnabled?: boolean outputFeatureEnabled?: boolean - showReverbs?: boolean showEqualizers?: boolean reverbsShownBefore?: boolean @@ -116,6 +115,12 @@ export class UIService extends DataService { return this.request({ method: 'POST', endpoint: '/width', data: { width } }) } + async changeWidth ({ diff }: { diff: number }) { + const currentWidth = await this.getWidth() + const width = currentWidth + diff + await this.setWidth(width) + } + async getHeight (): Promise { const { height } = await this.request({ method: 'GET', endpoint: '/height' }) return height @@ -203,6 +208,7 @@ export class UIService extends DataService { ]) } + onMinHeightChanged = new EventEmitter() async getMinHeight (): Promise { const { minHeight } = await this.request({ method: 'GET', endpoint: '/min-height' }) return minHeight @@ -212,6 +218,33 @@ export class UIService extends DataService { return this.request({ method: 'POST', endpoint: '/min-height', data: { minHeight } }) } + async getMinWidth (): Promise { + const { minWidth } = await this.request({ method: 'GET', endpoint: '/min-width' }) + return minWidth + } + + async setMinWidth ({ minWidth }: { minWidth: number }) { + return this.request({ method: 'POST', endpoint: '/min-width', data: { minWidth } }) + } + + async getMaxHeight (): Promise { + const { maxHeight } = await this.request({ method: 'GET', endpoint: '/max-height' }) + return maxHeight + } + + async setMaxHeight ({ maxHeight }: { maxHeight?: number }) { + return this.request({ method: 'POST', endpoint: '/max-height', data: { maxHeight } }) + } + + async getMaxWidth (): Promise { + const { maxWidth } = await this.request({ method: 'GET', endpoint: '/max-width' }) + return maxWidth + } + + async setMaxWidth ({ maxWidth }: { maxWidth?: number }) { + return this.request({ method: 'POST', endpoint: '/max-width', data: { maxWidth } }) + } + onShownChanged (cb: UIShownChangedEventCallback) { this.on('/shown', cb) } diff --git a/ui/src/app/services/utilities.service.ts b/ui/src/app/services/utilities.service.ts index 54d9616..db2cc86 100644 --- a/ui/src/app/services/utilities.service.ts +++ b/ui/src/app/services/utilities.service.ts @@ -12,6 +12,15 @@ export class UtilitiesService { return (value - inMin) * (outMax - outMin) / (inMax - inMin) + outMin } + clampValue ({ value, min, max }: { value: number, min: number, max: number }) { + if (value < min) { + value = min + } else if (value > max) { + value = max + } + return value + } + getTimestampFromDurationAndProgress (duration, progress = 1) { const currentSecond = Math.floor(duration * progress) let minutes = Math.floor(currentSecond / 60).toString() @@ -55,6 +64,16 @@ export class UtilitiesService { }) } + getCoordinatesInsideElementFromEvent (event: MouseEvent, element?: HTMLElement) { + const el = element || event.target as HTMLElement + const rect = el.getBoundingClientRect() + const scale = rect.width / el.clientWidth + return { + x: (event.clientX - rect.left) / scale, + y: (event.clientY - rect.top) / scale + } + } + static async injectScript ({ src, id }: { src: string, id?: string }) { return new Promise((resolve, reject) => { const script = document.createElement('script') @@ -74,4 +93,17 @@ export class UtilitiesService { head.appendChild(script) }) } + + quickHash (str: string) { return UtilitiesService.quickHash(str) } + static quickHash (str: string) { + let hash = 0 + let chr: number + if (str.length === 0) return hash + for (let i = 0; i < str.length; i++) { + chr = str.charCodeAt(i) + hash = ((hash << 5) - hash) + chr + hash |= 0 // Convert to 32bit integer + } + return hash + } } diff --git a/ui/src/polyfills.ts b/ui/src/polyfills.ts index 86b3024..d7e311b 100644 --- a/ui/src/polyfills.ts +++ b/ui/src/polyfills.ts @@ -61,14 +61,13 @@ import 'web-animations-js' // Run `npm install --save web-animations-js`. // (window as any).__Zone_disable_requestAnimationFrame = true; // disable patch requestAnimationFrame // (window as any).__Zone_disable_on_property = true; // disable patch onProperty such as onclick -// (window as any).__zone_symbol__BLACK_LISTED_EVENTS = ['scroll', 'mousemove']; // disable patch specified eventNames /* * in IE/Edge developer tools, the addEventListener will also be wrapped by zone.js * with the following flag, it will bypass `zone.js` patch for IE/Edge */ // (window as any).__Zone_enable_cross_context_check = true; - +import './zone-flags' /*************************************************************************************************** * Zone JS is required by default for Angular itself. */ diff --git a/ui/src/styles.scss b/ui/src/styles.scss index 91ee7ee..3f4c0ce 100644 --- a/ui/src/styles.scss +++ b/ui/src/styles.scss @@ -2,14 +2,40 @@ @import '~@angular/cdk/overlay-prebuilt.css'; @import "~@angular/material/prebuilt-themes/pink-bluegrey.css"; -.mat-dialog-container { - padding: 12px !important; -} - .w-100 { width: 100%; } .h-100 { height: 100%; +} + +.mat-dialog-container { + padding: 12px !important; +} + +.mat-menu-panel { + min-height: 0px !important; +} + +.underline { + text-decoration: underline; +} + +.clickable { + cursor: pointer !important; + + * { + cursor: pointer !important; + } +} + +.not-available-overlay { + position: absolute; + top: 0; + left: 0; + width: 100%; + height: 100%; + background-color: rgba(0, 0, 0, 0.5); + backdrop-filter: blur(10px); } \ No newline at end of file diff --git a/ui/src/tsconfig.app.json b/ui/src/tsconfig.app.json index 1853b0e..1e452e9 100644 --- a/ui/src/tsconfig.app.json +++ b/ui/src/tsconfig.app.json @@ -7,7 +7,7 @@ }, "angularCompilerOptions": { "enableIvy": true, - "strictTemplates": true + "strictTemplates": true, }, "exclude": [], "paths": { "@angular/*": [ "./node_modules/@angular/*" ] } diff --git a/ui/src/zone-flags.ts b/ui/src/zone-flags.ts new file mode 100644 index 0000000..86f499b --- /dev/null +++ b/ui/src/zone-flags.ts @@ -0,0 +1,2 @@ +;(window as any).__Zone_disable_requestAnimationFrame = true +;(window as any).__zone_symbol__BLACK_LISTED_EVENTS = [ 'mousewheel', 'wheel', 'mousemove', 'scroll' ] // disable patch specified eventNames