🐛 Fix dialog rendering issue to ensure full-page coverage
✨ Add optional page transitions for improved navigation 🎨 Enhance dashboard UI/UX for better user experience
This commit is contained in:
parent
ff01fc8503
commit
aa8d8bb4f3
7 changed files with 278 additions and 88 deletions
|
@ -14,26 +14,52 @@ export class AppGrid extends LitElement {
|
||||||
name: string;
|
name: string;
|
||||||
}[] = [];
|
}[] = [];
|
||||||
|
|
||||||
|
override connectedCallback() {
|
||||||
|
super.connectedCallback();
|
||||||
|
this.style.setProperty('--icons-count', this.apps.length.toString());
|
||||||
|
}
|
||||||
|
|
||||||
static override styles = [
|
static override styles = [
|
||||||
css`
|
css`
|
||||||
:host {
|
:host {
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: 80px;
|
grid-template-columns: 80px;
|
||||||
justify-content: space-around;
|
justify-content: space-around;
|
||||||
gap: 20px;
|
gap: 28px;
|
||||||
padding: 30px;
|
padding: 42px 30px;
|
||||||
margin-top: 10px;
|
margin-top: 10px;
|
||||||
|
--stagger-delay: calc(var(--animation-duration) * 2 / var(--icons-count));
|
||||||
|
|
||||||
@media (min-width: 500px) {
|
@media (min-width: 500px) {
|
||||||
grid-template-columns: repeat(2, 80px);
|
grid-template-columns: repeat(2, 80px);
|
||||||
|
grid-auto-rows: minmax(120px, auto);
|
||||||
}
|
}
|
||||||
|
|
||||||
@media (min-width: 768px) {
|
@media (min-width: 768px) {
|
||||||
grid-template-columns: repeat(3, 80px);
|
grid-template-columns: repeat(3, 80px);
|
||||||
|
gap: 32px 42px;
|
||||||
}
|
}
|
||||||
|
|
||||||
@media (min-width: 1024px) {
|
@media (min-width: 1024px) {
|
||||||
grid-template-columns: repeat(4, 80px);
|
grid-template-columns: repeat(4, 80px);
|
||||||
|
grid-auto-flow: dense;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
arx-app-icon {
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateY(20px) scale(0.8);
|
||||||
|
animation: float-in var(--transition) forwards;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes float-in {
|
||||||
|
0% {
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateY(20px) scale(0.8);
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
opacity: 1;
|
||||||
|
transform: translateY(0) scale(1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
|
@ -43,12 +69,13 @@ export class AppGrid extends LitElement {
|
||||||
return html`
|
return html`
|
||||||
${map(
|
${map(
|
||||||
this.apps,
|
this.apps,
|
||||||
(app) => html`
|
(app, index) => html`
|
||||||
<arx-app-icon
|
<arx-app-icon
|
||||||
.icon=${app.icon}
|
.icon=${app.icon}
|
||||||
.color=${app.color}
|
.color=${app.color}
|
||||||
.href=${app.href}
|
.href=${app.href}
|
||||||
.name=${app.name}
|
.name=${app.name}
|
||||||
|
style="animation-delay: calc(var(--stagger-delay) * ${index});"
|
||||||
></arx-app-icon>
|
></arx-app-icon>
|
||||||
`,
|
`,
|
||||||
)}
|
)}
|
||||||
|
|
|
@ -24,37 +24,44 @@ export class AppIcon extends LitElement {
|
||||||
css`
|
css`
|
||||||
:host {
|
:host {
|
||||||
display: flex;
|
display: flex;
|
||||||
--animation-speed: 0.25s;
|
--shadow-opacity: 0.2;
|
||||||
|
--hover-lift: -6px;
|
||||||
|
--tap-scale: 0.92;
|
||||||
}
|
}
|
||||||
|
|
||||||
.app-icon {
|
.app-icon {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 8px;
|
gap: 10px;
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
padding: 6px;
|
padding: 8px;
|
||||||
border-radius: 18px;
|
border-radius: 18px;
|
||||||
transition: transform var(--animation-speed)
|
transition: transform var(--transition);
|
||||||
cubic-bezier(0.175, 0.885, 0.32, 1.275);
|
position: relative;
|
||||||
|
will-change: transform;
|
||||||
|
perspective: 800px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.app-icon:hover {
|
.app-icon:hover {
|
||||||
transform: translateY(-2px);
|
transform: translateY(var(--hover-lift));
|
||||||
}
|
}
|
||||||
|
|
||||||
.icon {
|
.icon {
|
||||||
width: 96px;
|
width: 96px;
|
||||||
height: 96px;
|
height: 96px;
|
||||||
border-radius: 24px;
|
border-radius: 24px;
|
||||||
box-shadow: 0 3px 8px rgba(0, 0, 0, 0.12);
|
box-shadow: 0 3px 8px rgba(0, 0, 0, var(--shadow-opacity));
|
||||||
display: flex;
|
display: flex;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.2);
|
border: 1px solid rgba(255, 255, 255, 0.2);
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
position: relative;
|
position: relative;
|
||||||
transition: all var(--animation-speed) ease-out;
|
transition:
|
||||||
|
transform var(--overshoot-transition),
|
||||||
|
box-shadow var(--overshoot-transition);
|
||||||
|
transform-style: preserve-3d;
|
||||||
&::before {
|
&::before {
|
||||||
content: "";
|
content: "";
|
||||||
position: absolute;
|
position: absolute;
|
||||||
|
@ -66,7 +73,7 @@ export class AppIcon extends LitElement {
|
||||||
);
|
);
|
||||||
pointer-events: none;
|
pointer-events: none;
|
||||||
opacity: 1;
|
opacity: 1;
|
||||||
transition: opacity var(--animation-speed) ease-out;
|
transition: opacity var(--transition);
|
||||||
}
|
}
|
||||||
|
|
||||||
&::after {
|
&::after {
|
||||||
|
@ -74,13 +81,14 @@ export class AppIcon extends LitElement {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
inset: 0;
|
inset: 0;
|
||||||
background: linear-gradient(
|
background: linear-gradient(
|
||||||
var(--gradient-angle),
|
var(--gradient-angle, 145deg),
|
||||||
rgba(255, 255, 255, 0.25) 20%,
|
rgba(255, 255, 255, 0.35) 10%,
|
||||||
rgba(255, 255, 255, 0) 60%
|
rgba(255, 255, 255, 0) 70%
|
||||||
);
|
);
|
||||||
pointer-events: none;
|
pointer-events: none;
|
||||||
opacity: 0;
|
opacity: 0;
|
||||||
transition: opacity var(--animation-speed) ease-out;
|
transition: opacity var(--undershoot-transition);
|
||||||
|
z-index: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
&:hover::before {
|
&:hover::before {
|
||||||
|
@ -95,14 +103,17 @@ export class AppIcon extends LitElement {
|
||||||
.icon-svg {
|
.icon-svg {
|
||||||
color: white;
|
color: white;
|
||||||
filter: drop-shadow(
|
filter: drop-shadow(
|
||||||
calc(cos(var(--gradient-angle)) * 2px)
|
calc(cos(var(--gradient-angle, 145deg)) * 3px)
|
||||||
calc(sin(var(--gradient-angle)) * 2px) rgba(0, 0, 0, 0.2)
|
calc(sin(var(--gradient-angle, 145deg)) * 3px)
|
||||||
|
rgba(0, 0, 0, 0.25)
|
||||||
);
|
);
|
||||||
display: flex;
|
display: flex;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
|
transition: transform var(--overshoot-transition);
|
||||||
|
z-index: 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
.app-name {
|
.app-name {
|
||||||
|
@ -114,16 +125,36 @@ export class AppIcon extends LitElement {
|
||||||
color: var(--color-base-content);
|
color: var(--color-base-content);
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
transition: all var(--animation-speed) ease-out;
|
transition: all var(--transition);
|
||||||
|
transform: translateZ(0);
|
||||||
|
opacity: 0.85;
|
||||||
}
|
}
|
||||||
|
|
||||||
.app-icon:hover .icon {
|
.app-icon:hover .icon {
|
||||||
transform: scale(1.08);
|
transform: scale(1.05) translateZ(10px) rotateX(var(--rotate-x, 0deg)) rotateY(var(--rotate-y, 0deg));
|
||||||
box-shadow: 0 6px 12px rgba(0, 0, 0, 0.25);
|
box-shadow:
|
||||||
|
0 10px 20px -5px rgba(0, 0, 0, calc(var(--shadow-opacity) * 1.5)),
|
||||||
|
0 0 0 1px rgba(255, 255, 255, 0.15);
|
||||||
|
}
|
||||||
|
|
||||||
|
.app-icon:hover .app-name {
|
||||||
|
opacity: 1;
|
||||||
|
transform: translateY(2px) scale(1.02);
|
||||||
|
}
|
||||||
|
|
||||||
|
.app-icon:hover .icon-svg {
|
||||||
|
transform: scale(1.08) translateZ(15px);
|
||||||
}
|
}
|
||||||
|
|
||||||
.app-icon:active .icon {
|
.app-icon:active .icon {
|
||||||
transform: scale(0.96);
|
transform: scale(var(--tap-scale)) translateZ(0);
|
||||||
|
box-shadow: 0 2px 5px rgba(0, 0, 0, calc(var(--shadow-opacity) * 0.8));
|
||||||
|
transition: all var(--transition);
|
||||||
|
}
|
||||||
|
|
||||||
|
.app-icon:active .app-name {
|
||||||
|
transform: translateY(0) scale(0.98);
|
||||||
|
transition: all var(--transition);
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
];
|
];
|
||||||
|
@ -132,6 +163,7 @@ export class AppIcon extends LitElement {
|
||||||
this.iconElement = this.shadowRoot?.querySelector('.icon') as HTMLElement;
|
this.iconElement = this.shadowRoot?.querySelector('.icon') as HTMLElement;
|
||||||
if (this.iconElement) {
|
if (this.iconElement) {
|
||||||
this.iconElement.addEventListener('mousemove', this.handleMouseMove);
|
this.iconElement.addEventListener('mousemove', this.handleMouseMove);
|
||||||
|
this.iconElement.addEventListener('mouseleave', this.handleMouseLeave);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -139,6 +171,7 @@ export class AppIcon extends LitElement {
|
||||||
super.disconnectedCallback();
|
super.disconnectedCallback();
|
||||||
if (this.iconElement) {
|
if (this.iconElement) {
|
||||||
this.iconElement.removeEventListener('mousemove', this.handleMouseMove);
|
this.iconElement.removeEventListener('mousemove', this.handleMouseMove);
|
||||||
|
this.iconElement.removeEventListener('mouseleave', this.handleMouseLeave);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -153,7 +186,32 @@ export class AppIcon extends LitElement {
|
||||||
const dy = e.clientY - centerY;
|
const dy = e.clientY - centerY;
|
||||||
const angle = Math.atan2(dy, dx) * (180 / Math.PI);
|
const angle = Math.atan2(dy, dx) * (180 / Math.PI);
|
||||||
|
|
||||||
|
const maxTilt = 20;
|
||||||
|
const percentX = (e.clientX - rect.left) / rect.width - 0.5;
|
||||||
|
const percentY = (e.clientY - rect.top) / rect.height - 0.5;
|
||||||
|
|
||||||
|
const tiltX = -percentY * maxTilt * 1.2;
|
||||||
|
const tiltY = percentX * maxTilt;
|
||||||
|
|
||||||
this.iconElement.style.setProperty('--gradient-angle', `${angle}deg`);
|
this.iconElement.style.setProperty('--gradient-angle', `${angle}deg`);
|
||||||
|
this.iconElement.style.setProperty('--rotate-x', `${tiltX}deg`);
|
||||||
|
this.iconElement.style.setProperty('--rotate-y', `${tiltY}deg`);
|
||||||
|
};
|
||||||
|
|
||||||
|
handleMouseLeave = () => {
|
||||||
|
if (!this.iconElement) return;
|
||||||
|
|
||||||
|
this.iconElement.style.transition = 'transform var(--transition)';
|
||||||
|
this.iconElement.style.setProperty('--rotate-x', '0deg');
|
||||||
|
this.iconElement.style.setProperty('--rotate-y', '0deg');
|
||||||
|
|
||||||
|
this.iconElement.addEventListener(
|
||||||
|
'transitionend',
|
||||||
|
() => {
|
||||||
|
this.iconElement!.style.transition = '';
|
||||||
|
},
|
||||||
|
{ once: true },
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
override render() {
|
override render() {
|
||||||
|
@ -162,7 +220,8 @@ export class AppIcon extends LitElement {
|
||||||
<div class="icon" style="background: ${this.color};">
|
<div class="icon" style="background: ${this.color};">
|
||||||
${when(
|
${when(
|
||||||
this.icon,
|
this.icon,
|
||||||
() => html`<iconify-icon
|
() =>
|
||||||
|
html`<iconify-icon
|
||||||
icon="${this.icon}"
|
icon="${this.icon}"
|
||||||
class="icon-svg"
|
class="icon-svg"
|
||||||
width="64"
|
width="64"
|
||||||
|
|
|
@ -1,11 +1,13 @@
|
||||||
import { LitElement, css, html } from 'lit';
|
import { LitElement, css, html } from 'lit';
|
||||||
import { customElement, property, state } from 'lit/decorators.js';
|
import { customElement, property, state } from 'lit/decorators.js';
|
||||||
|
import { type Ref, createRef, ref } from 'lit/directives/ref.js';
|
||||||
import { when } from 'lit/directives/when.js';
|
import { when } from 'lit/directives/when.js';
|
||||||
|
|
||||||
import '@components/General/Button';
|
import '@components/General/Button';
|
||||||
import { StyledInput } from '@components/General/Input';
|
import { StyledInput } from '@components/General/Input';
|
||||||
import { StyledTextarea } from '@components/General/Textarea';
|
import { StyledTextarea } from '@components/General/Textarea';
|
||||||
import { StyledToggle } from '@components/General/Toggle';
|
import { StyledToggle } from '@components/General/Toggle';
|
||||||
|
import type { EveDialog } from '../General/Dialog';
|
||||||
|
|
||||||
interface InputEvent extends Event {
|
interface InputEvent extends Event {
|
||||||
detail: {
|
detail: {
|
||||||
|
@ -18,6 +20,8 @@ export class CalendarEventDialog extends LitElement {
|
||||||
@property({ type: Boolean })
|
@property({ type: Boolean })
|
||||||
open = false;
|
open = false;
|
||||||
|
|
||||||
|
dialogRef: Ref<EveDialog> = createRef();
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
private newEvent = {
|
private newEvent = {
|
||||||
title: '',
|
title: '',
|
||||||
|
@ -49,7 +53,8 @@ export class CalendarEventDialog extends LitElement {
|
||||||
`;
|
`;
|
||||||
|
|
||||||
private handleClose() {
|
private handleClose() {
|
||||||
this.dispatchEvent(new CustomEvent('close'));
|
this.dialogRef.value?.close(false);
|
||||||
|
this.dispatchEvent(new CustomEvent('close', { bubbles: false }));
|
||||||
}
|
}
|
||||||
|
|
||||||
private handleInputChange(e: Event) {
|
private handleInputChange(e: Event) {
|
||||||
|
@ -77,7 +82,7 @@ export class CalendarEventDialog extends LitElement {
|
||||||
if (!this.open) return html``;
|
if (!this.open) return html``;
|
||||||
|
|
||||||
return html`
|
return html`
|
||||||
<arx-dialog width="500px" title="Add Event" .open=${this.open}>
|
<arx-dialog ${ref(this.dialogRef)} width="500px" title="Add Event" .open=${this.open} @close=${this.handleClose}>
|
||||||
<arx-fieldset legend="Title">
|
<arx-fieldset legend="Title">
|
||||||
<arx-input
|
<arx-input
|
||||||
type="text"
|
type="text"
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import { LitElement, css, html } from 'lit';
|
import { LitElement, css, html } from 'lit';
|
||||||
import { customElement, property, state } from 'lit/decorators.js';
|
import { customElement, property } from 'lit/decorators.js';
|
||||||
import { classMap } from 'lit/directives/class-map.js';
|
import { classMap } from 'lit/directives/class-map.js';
|
||||||
|
|
||||||
@customElement('arx-dialog')
|
@customElement('arx-dialog')
|
||||||
|
@ -11,42 +11,59 @@ export class EveDialog extends LitElement {
|
||||||
@property({ type: String }) width = '420px';
|
@property({ type: String }) width = '420px';
|
||||||
@property({ type: String }) maxWidth = '90%';
|
@property({ type: String }) maxWidth = '90%';
|
||||||
|
|
||||||
@state() private _previousFocus: HTMLElement | null = null;
|
private rootWindow: HTMLDivElement | null = null;
|
||||||
|
private previousScrollTop = 0;
|
||||||
|
|
||||||
static override styles = css`
|
static override styles = css`
|
||||||
:host {
|
::-webkit-scrollbar {
|
||||||
display: block;
|
width: 12px;
|
||||||
position: fixed;
|
height: 12px;
|
||||||
top: 0;
|
|
||||||
left: 0;
|
|
||||||
width: 100%;
|
|
||||||
height: 100%;
|
|
||||||
z-index: 999999;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.overlay {
|
::-webkit-scrollbar-track {
|
||||||
position: fixed;
|
background: var(--color-base-200);
|
||||||
top: 0;
|
border-radius: var(--radius-field);
|
||||||
left: 0;
|
}
|
||||||
width: 100%;
|
|
||||||
height: 100%;
|
::-webkit-scrollbar-thumb {
|
||||||
|
background: var(--color-base-300);
|
||||||
|
border-radius: var(--radius-field);
|
||||||
|
border: 2px solid var(--color-base-200);
|
||||||
|
transition: var(--transition);
|
||||||
|
}
|
||||||
|
|
||||||
|
::-webkit-scrollbar-thumb:hover {
|
||||||
|
background: var(--color-neutral);
|
||||||
|
}
|
||||||
|
|
||||||
|
:host {
|
||||||
|
display: none;
|
||||||
|
position: fixed !important;
|
||||||
|
transform: translate3d(-50%, -50%, 1px);
|
||||||
|
will-change: transform;
|
||||||
|
top: 50%;
|
||||||
|
left: 50%;
|
||||||
|
width: 100vw;
|
||||||
|
height: 100vh;
|
||||||
|
z-index: 2147483647 !important;
|
||||||
|
contain: layout;
|
||||||
background-color: oklch(from var(--color-base-content) l c h / 0.6);
|
background-color: oklch(from var(--color-base-content) l c h / 0.6);
|
||||||
display: flex;
|
overflow: hidden;
|
||||||
align-items: center;
|
-webkit-app-region: no-drag;
|
||||||
justify-content: center;
|
|
||||||
z-index: 9999;
|
|
||||||
opacity: 0;
|
|
||||||
pointer-events: none;
|
|
||||||
transition: opacity 0.2s ease;
|
|
||||||
backdrop-filter: blur(4px);
|
backdrop-filter: blur(4px);
|
||||||
}
|
}
|
||||||
|
|
||||||
.overlay.active {
|
:host([open]) {
|
||||||
opacity: 1;
|
display: block !important;
|
||||||
pointer-events: all;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.dialog-container {
|
.dialog-container {
|
||||||
|
position: absolute;
|
||||||
|
top: 50%;
|
||||||
|
left: 50%;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
|
max-height: 80vh;
|
||||||
|
overflow: auto;
|
||||||
background-color: var(--color-base-100);
|
background-color: var(--color-base-100);
|
||||||
border-radius: var(--radius-box);
|
border-radius: var(--radius-box);
|
||||||
border: var(--border) solid var(--color-base-300);
|
border: var(--border) solid var(--color-base-300);
|
||||||
|
@ -58,15 +75,10 @@ export class EveDialog extends LitElement {
|
||||||
width: var(--dialog-width, 420px);
|
width: var(--dialog-width, 420px);
|
||||||
max-width: var(--dialog-max-width, 90%);
|
max-width: var(--dialog-max-width, 90%);
|
||||||
padding: 28px;
|
padding: 28px;
|
||||||
transform: scale(0.95) translateY(10px);
|
|
||||||
transition: transform 0.25s cubic-bezier(0.1, 1, 0.2, 1);
|
transition: transform 0.25s cubic-bezier(0.1, 1, 0.2, 1);
|
||||||
color: var(--color-base-content);
|
color: var(--color-base-content);
|
||||||
}
|
}
|
||||||
|
|
||||||
.overlay.active .dialog-container {
|
|
||||||
transform: scale(1) translateY(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
.dialog-header {
|
.dialog-header {
|
||||||
font-size: 18px;
|
font-size: 18px;
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
|
@ -76,6 +88,7 @@ export class EveDialog extends LitElement {
|
||||||
}
|
}
|
||||||
|
|
||||||
.dialog-content {
|
.dialog-content {
|
||||||
|
overflow: auto;
|
||||||
margin-bottom: 24px;
|
margin-bottom: 24px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -103,6 +116,9 @@ export class EveDialog extends LitElement {
|
||||||
override connectedCallback() {
|
override connectedCallback() {
|
||||||
super.connectedCallback();
|
super.connectedCallback();
|
||||||
document.addEventListener('keydown', this._handleKeyDown);
|
document.addEventListener('keydown', this._handleKeyDown);
|
||||||
|
this.rootWindow = document.body
|
||||||
|
.querySelector('arx-eve-router')!
|
||||||
|
.shadowRoot!.querySelector('.window') as HTMLDivElement;
|
||||||
}
|
}
|
||||||
|
|
||||||
override disconnectedCallback() {
|
override disconnectedCallback() {
|
||||||
|
@ -111,15 +127,9 @@ export class EveDialog extends LitElement {
|
||||||
}
|
}
|
||||||
|
|
||||||
override updated(changedProps: Map<string, unknown>) {
|
override updated(changedProps: Map<string, unknown>) {
|
||||||
if (changedProps.has('open') && this.open) {
|
if (changedProps.has('open')) {
|
||||||
this._previousFocus = document.activeElement as HTMLElement;
|
if (this.open) this.show();
|
||||||
// Focus the dialog container after rendering
|
else this.close();
|
||||||
setTimeout(() => {
|
|
||||||
const container = this.shadowRoot?.querySelector('.dialog-container') as HTMLElement;
|
|
||||||
if (container) container.focus();
|
|
||||||
}, 50);
|
|
||||||
} else if (changedProps.has('open') && !this.open && this._previousFocus) {
|
|
||||||
this._previousFocus.focus();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -129,24 +139,36 @@ export class EveDialog extends LitElement {
|
||||||
}
|
}
|
||||||
|
|
||||||
private _handleOverlayClick(e: MouseEvent) {
|
private _handleOverlayClick(e: MouseEvent) {
|
||||||
if (this.closeOnOverlayClick && e.target === e.currentTarget) {
|
if (this.closeOnOverlayClick && e.target === e.currentTarget) this.close();
|
||||||
this.close();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
show() {
|
show() {
|
||||||
this.open = true;
|
this.open = true;
|
||||||
|
this.style.display = 'block';
|
||||||
|
this.previousScrollTop = this.rootWindow?.scrollTop ?? 0;
|
||||||
|
this.style.height = `${this.rootWindow!.clientHeight * 2}px`; // this is a hack to prevent the overlay background from showing through
|
||||||
|
this.rootWindow?.style.setProperty('overflow', 'hidden', 'important');
|
||||||
|
this.rootWindow?.scrollTo({
|
||||||
|
top: 0,
|
||||||
|
behavior: 'smooth',
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
close() {
|
close(triggerEvent = true) {
|
||||||
this.open = false;
|
this.open = false;
|
||||||
this.dispatchEvent(new CustomEvent('close'));
|
this.style.display = 'none';
|
||||||
|
this.rootWindow?.style.setProperty('overflow', 'auto', 'important');
|
||||||
|
this.rootWindow?.scrollTo({
|
||||||
|
top: this.previousScrollTop,
|
||||||
|
behavior: 'smooth',
|
||||||
|
});
|
||||||
|
if (triggerEvent) this.dispatchEvent(new CustomEvent('close'));
|
||||||
}
|
}
|
||||||
|
|
||||||
override render() {
|
override render() {
|
||||||
return html`
|
return html`
|
||||||
<div
|
<div
|
||||||
class="${classMap({ overlay: true, active: this.open })}"
|
class="${classMap({ active: this.open })}"
|
||||||
@click=${this._handleOverlayClick}
|
@click=${this._handleOverlayClick}
|
||||||
style="--dialog-width: ${this.width}; --dialog-max-width: ${this.maxWidth};"
|
style="--dialog-width: ${this.width}; --dialog-max-width: ${this.maxWidth};"
|
||||||
>
|
>
|
||||||
|
|
|
@ -62,6 +62,7 @@ export class EveSettings extends LitElement {
|
||||||
@state() private profile: NDKUserProfile | undefined;
|
@state() private profile: NDKUserProfile | undefined;
|
||||||
@state() private error: string | undefined;
|
@state() private error: string | undefined;
|
||||||
@state() private darkMode = false;
|
@state() private darkMode = false;
|
||||||
|
@state() private pageTransitions = true;
|
||||||
@state() private relayStatus: { running: boolean; pid: number | null; logs: string[] } = {
|
@state() private relayStatus: { running: boolean; pid: number | null; logs: string[] } = {
|
||||||
running: false,
|
running: false,
|
||||||
pid: null,
|
pid: null,
|
||||||
|
@ -77,6 +78,7 @@ export class EveSettings extends LitElement {
|
||||||
try {
|
try {
|
||||||
this.profile = await getUserProfile();
|
this.profile = await getUserProfile();
|
||||||
this.darkMode = localStorage.getItem('darkMode') === 'true';
|
this.darkMode = localStorage.getItem('darkMode') === 'true';
|
||||||
|
this.pageTransitions = localStorage.getItem('pageTransitions') !== 'false';
|
||||||
this.updateRelayStatus();
|
this.updateRelayStatus();
|
||||||
this.loading = false;
|
this.loading = false;
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
|
@ -117,6 +119,12 @@ export class EveSettings extends LitElement {
|
||||||
document.body.classList.toggle('dark', this.darkMode);
|
document.body.classList.toggle('dark', this.darkMode);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private togglePageTransitions() {
|
||||||
|
this.pageTransitions = !this.pageTransitions;
|
||||||
|
localStorage.setItem('pageTransitions', this.pageTransitions.toString());
|
||||||
|
location.reload();
|
||||||
|
}
|
||||||
|
|
||||||
private reset() {
|
private reset() {
|
||||||
if (!confirm('Are you sure you want to reset the app?')) return;
|
if (!confirm('Are you sure you want to reset the app?')) return;
|
||||||
localStorage.clear();
|
localStorage.clear();
|
||||||
|
@ -133,12 +141,17 @@ export class EveSettings extends LitElement {
|
||||||
return html`
|
return html`
|
||||||
<arx-breadcrumbs .items=${breadcrumbItems}></arx-breadcrumbs>
|
<arx-breadcrumbs .items=${breadcrumbItems}></arx-breadcrumbs>
|
||||||
<arx-card>
|
<arx-card>
|
||||||
<arx-fieldset legend="Dark Mode">
|
<arx-fieldset legend="Visual">
|
||||||
<arx-toggle
|
<arx-toggle
|
||||||
label="Dark Mode"
|
label="Dark Mode"
|
||||||
.checked=${this.darkMode}
|
.checked=${this.darkMode}
|
||||||
@change=${() => this.toggleDarkMode()}
|
@change=${() => this.toggleDarkMode()}
|
||||||
></arx-toggle>
|
></arx-toggle>
|
||||||
|
<arx-toggle
|
||||||
|
label="Page Transitions"
|
||||||
|
.checked=${this.pageTransitions}
|
||||||
|
@change=${() => this.togglePageTransitions()}
|
||||||
|
></arx-toggle>
|
||||||
</arx-fieldset>
|
</arx-fieldset>
|
||||||
<arx-fieldset legend="Profile">
|
<arx-fieldset legend="Profile">
|
||||||
${when(
|
${when(
|
||||||
|
|
|
@ -14,6 +14,8 @@ import { spread } from '@open-wc/lit-helpers';
|
||||||
import { LitElement, css } from 'lit';
|
import { LitElement, css } from 'lit';
|
||||||
import { customElement, property, state } from 'lit/decorators.js';
|
import { customElement, property, state } from 'lit/decorators.js';
|
||||||
import { keyed } from 'lit/directives/keyed.js';
|
import { keyed } from 'lit/directives/keyed.js';
|
||||||
|
import { type Ref, createRef, ref } from 'lit/directives/ref.js';
|
||||||
|
import { when } from 'lit/directives/when.js';
|
||||||
import { type StaticValue, html, literal } from 'lit/static-html.js';
|
import { type StaticValue, html, literal } from 'lit/static-html.js';
|
||||||
|
|
||||||
export interface RouteParams {
|
export interface RouteParams {
|
||||||
|
@ -24,7 +26,6 @@ interface Route {
|
||||||
pattern: string;
|
pattern: string;
|
||||||
params: RouteParams;
|
params: RouteParams;
|
||||||
component: StaticValue;
|
component: StaticValue;
|
||||||
// component: typeof LitElement | ((params: RouteParams) => typeof LitElement);
|
|
||||||
title?: string;
|
title?: string;
|
||||||
meta?: Record<string, string>;
|
meta?: Record<string, string>;
|
||||||
}
|
}
|
||||||
|
@ -90,9 +91,15 @@ export default class EveRouter extends LitElement {
|
||||||
@state()
|
@state()
|
||||||
private currentIndex = -1;
|
private currentIndex = -1;
|
||||||
|
|
||||||
|
@state()
|
||||||
|
private isTransitioning = false;
|
||||||
|
|
||||||
@property()
|
@property()
|
||||||
public ccnSetup = false;
|
public ccnSetup = false;
|
||||||
|
|
||||||
|
windowContentRef: Ref<HTMLDivElement> = createRef();
|
||||||
|
pageTransitions = true;
|
||||||
|
|
||||||
private beforeEachGuards: ((to: Route, from: Route | null) => boolean)[] = [];
|
private beforeEachGuards: ((to: Route, from: Route | null) => boolean)[] = [];
|
||||||
private afterEachHooks: ((to: Route, from: Route | null) => void)[] = [];
|
private afterEachHooks: ((to: Route, from: Route | null) => void)[] = [];
|
||||||
|
|
||||||
|
@ -135,6 +142,33 @@ export default class EveRouter extends LitElement {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
margin: 0 auto;
|
margin: 0 auto;
|
||||||
padding: 1rem;
|
padding: 1rem;
|
||||||
|
opacity: 1;
|
||||||
|
transform-origin: left center;
|
||||||
|
transform: perspective(1200px) translateX(0);
|
||||||
|
transition: var(--transition);
|
||||||
|
backface-visibility: hidden;
|
||||||
|
filter: blur(0px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.window-content::after {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
right: 0;
|
||||||
|
width: 4px;
|
||||||
|
height: 100%;
|
||||||
|
opacity: 0;
|
||||||
|
transition: var(--transition);
|
||||||
|
}
|
||||||
|
|
||||||
|
.window-content.transitioning {
|
||||||
|
overflow: hidden;
|
||||||
|
transform: perspective(1200px) translateX(50vw);
|
||||||
|
filter: blur(50px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.window-content.transitioning::after {
|
||||||
|
opacity: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
.hide-overflow {
|
.hide-overflow {
|
||||||
|
@ -145,6 +179,7 @@ export default class EveRouter extends LitElement {
|
||||||
constructor() {
|
constructor() {
|
||||||
super();
|
super();
|
||||||
this.initializeRouter();
|
this.initializeRouter();
|
||||||
|
this.pageTransitions = localStorage.getItem('pageTransitions') !== 'false';
|
||||||
if (this.ccnSetup) window.relay.start(localStorage.getItem('encryption_key')!);
|
if (this.ccnSetup) window.relay.start(localStorage.getItem('encryption_key')!);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -171,11 +206,11 @@ export default class EveRouter extends LitElement {
|
||||||
window.addEventListener('popstate', this.handlePopState.bind(this));
|
window.addEventListener('popstate', this.handlePopState.bind(this));
|
||||||
}
|
}
|
||||||
|
|
||||||
private handleHashChange(): void {
|
private async handleHashChange(): Promise<void> {
|
||||||
const newPath = this.currentPath;
|
const newPath = this.currentPath;
|
||||||
if (newPath !== this.history[this.currentIndex]) {
|
if (newPath !== this.history[this.currentIndex]) {
|
||||||
|
await this.requestUpdateWithTransition();
|
||||||
this.updateHistory(newPath);
|
this.updateHistory(newPath);
|
||||||
this.requestUpdate();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -252,7 +287,7 @@ export default class EveRouter extends LitElement {
|
||||||
const canProceed = this.beforeEachGuards.every((guard) => guard(to, from));
|
const canProceed = this.beforeEachGuards.every((guard) => guard(to, from));
|
||||||
|
|
||||||
if (canProceed) {
|
if (canProceed) {
|
||||||
this.requestUpdate();
|
await this.requestUpdateWithTransition();
|
||||||
for (const hook of this.afterEachHooks) {
|
for (const hook of this.afterEachHooks) {
|
||||||
hook(to, from);
|
hook(to, from);
|
||||||
}
|
}
|
||||||
|
@ -263,6 +298,25 @@ export default class EveRouter extends LitElement {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async requestUpdateWithTransition(): Promise<void> {
|
||||||
|
if (!this.windowContentRef.value) return this.requestUpdate();
|
||||||
|
if (!this.pageTransitions) return this.requestUpdate();
|
||||||
|
this.isTransitioning = true;
|
||||||
|
this.requestUpdate();
|
||||||
|
|
||||||
|
await new Promise((resolve) => {
|
||||||
|
this.windowContentRef.value!.addEventListener(
|
||||||
|
'transitionend',
|
||||||
|
() => {
|
||||||
|
this.isTransitioning = false;
|
||||||
|
resolve(true);
|
||||||
|
},
|
||||||
|
{ once: true },
|
||||||
|
);
|
||||||
|
});
|
||||||
|
this.requestUpdate();
|
||||||
|
}
|
||||||
|
|
||||||
goBack(): void {
|
goBack(): void {
|
||||||
if (this.currentIndex > 0) {
|
if (this.currentIndex > 0) {
|
||||||
this.currentIndex--;
|
this.currentIndex--;
|
||||||
|
@ -306,8 +360,12 @@ export default class EveRouter extends LitElement {
|
||||||
title="Eve"
|
title="Eve"
|
||||||
></arx-header>
|
></arx-header>
|
||||||
<div class="window ${this.currentRoute.pattern === 'home' ? 'hide-overflow' : ''}">
|
<div class="window ${this.currentRoute.pattern === 'home' ? 'hide-overflow' : ''}">
|
||||||
<div class="window-content">
|
<div ${ref(this.windowContentRef)} class="window-content ${this.isTransitioning ? 'transitioning' : ''}">
|
||||||
${keyed(
|
${when(
|
||||||
|
this.isTransitioning,
|
||||||
|
() => html`<arx-loading-view></arx-loading-view>`,
|
||||||
|
() =>
|
||||||
|
keyed(
|
||||||
this.currentRoute.params,
|
this.currentRoute.params,
|
||||||
html`
|
html`
|
||||||
<${this.currentRoute.component}
|
<${this.currentRoute.component}
|
||||||
|
@ -318,6 +376,7 @@ export default class EveRouter extends LitElement {
|
||||||
@go-forward=${this.goForward}
|
@go-forward=${this.goForward}
|
||||||
></${this.currentRoute.component}>
|
></${this.currentRoute.component}>
|
||||||
`,
|
`,
|
||||||
|
),
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -6,8 +6,13 @@
|
||||||
--space-sm: clamp(1rem, 1.5vw, 1.5rem);
|
--space-sm: clamp(1rem, 1.5vw, 1.5rem);
|
||||||
--space-md: clamp(2rem, 3vw, 3rem);
|
--space-md: clamp(2rem, 3vw, 3rem);
|
||||||
|
|
||||||
--animation-curve: cubic-bezier(0.68, -0.55, 0.265, 1.55);
|
--undershoot-curve: cubic-bezier(0.38, 0, 0.618, 0.88);
|
||||||
--transition: 0.3s var(--animation-curve);
|
--animation-curve: cubic-bezier(0.18, 0, 0.618, 1);
|
||||||
|
--overshoot-curve: cubic-bezier(0.38, 0, 0.618, 1.12);
|
||||||
|
--animation-duration: 275ms;
|
||||||
|
--transition: var(--animation-duration) var(--animation-curve);
|
||||||
|
--overshoot-transition: calc(var(--animation-duration) * 1.2) var(--overshoot-curve);
|
||||||
|
--undershoot-transition: calc(var(--animation-duration) * 0.8) var(--undershoot-curve);
|
||||||
|
|
||||||
--color-base-100: oklch(98% 0.016 73.684);
|
--color-base-100: oklch(98% 0.016 73.684);
|
||||||
--color-base-200: oklch(95% 0.038 75.164);
|
--color-base-200: oklch(95% 0.038 75.164);
|
||||||
|
|
Loading…
Add table
Reference in a new issue