Compare commits

...

2 commits

Author SHA1 Message Date
bf3c950da0
🧭 Add navigation sidebar and 📏 enforce minimum window dimensions
- Add sidebar component to enhance site navigation and improve user experience
- Implement window size constraints (min 1366x768) to ensure proper display across devices
2025-04-08 18:42:38 +02:00
aa8d8bb4f3
🐛 Fix dialog rendering issue to ensure full-page coverage
 Add optional page transitions for improved navigation
🎨 Enhance dashboard UI/UX for better user experience
2025-04-04 17:54:56 +02:00
16 changed files with 652 additions and 105 deletions

View file

Before

Width:  |  Height:  |  Size: 80 KiB

After

Width:  |  Height:  |  Size: 80 KiB

BIN
src/assets/logo.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 737 KiB

View file

@ -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>
`, `,
)} )}

View file

@ -2,6 +2,7 @@ import { LitElement, css, html } from 'lit';
import { customElement, property } from 'lit/decorators.js'; import { customElement, property } from 'lit/decorators.js';
import '@components/EveLink'; import '@components/EveLink';
import { ifDefined } from 'lit/directives/if-defined.js';
import { when } from 'lit/directives/when.js'; import { when } from 'lit/directives/when.js';
@customElement('arx-app-icon') @customElement('arx-app-icon')
@ -18,43 +19,56 @@ export class AppIcon extends LitElement {
@property() @property()
name = 'App'; name = 'App';
@property({ type: Boolean })
small = false;
@property({ type: Boolean })
selected = false;
private iconElement?: HTMLElement; private iconElement?: HTMLElement;
static override styles = [ static override styles = [
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 +80,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 +88,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 {
@ -92,17 +107,26 @@ export class AppIcon extends LitElement {
} }
} }
:host([small]) .icon {
width: 64px;
height: 64px;
border-radius: 12px;
}
.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 +138,53 @@ 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;
}
:host([selected]) .app-name {
color: var(--color-primary-content);
}
:host([small]) .app-name {
font-size: 14px;
margin-top: 2px;
} }
.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);
}
:host([small]) .app-icon:hover .icon {
transform: scale(1.05) translateZ(5px) rotateX(var(--rotate-x, 0deg)) rotateY(var(--rotate-y, 0deg));
}
.app-icon:hover .app-name {
opacity: 1;
transform: translateY(2px) scale(1.02);
}
.app-icon:hover .icon-svg {
transform: scale(1.08) translateZ(15px);
}
:host([small]) .app-icon:hover .icon-svg {
transform: scale(1.08) translateZ(8px);
} }
.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 +193,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 +201,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 +216,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,12 +250,13 @@ 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 () =>
icon="${this.icon}" html`<iconify-icon
class="icon-svg" icon="${ifDefined(this.icon)}"
width="64" class="icon-svg"
height="64" width="${this.small ? '48' : '64'}"
></iconify-icon>`, height="${this.small ? '48' : '64'}"
></iconify-icon>`,
)} )}
</div> </div>
<span class="app-name">${this.name}</span> <span class="app-name">${this.name}</span>

View file

@ -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"

View file

@ -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};"
> >

View file

@ -35,10 +35,6 @@ export class Header extends LitElement {
header { header {
background: var(--color-primary); background: var(--color-primary);
backdrop-filter: blur(10px); backdrop-filter: blur(10px);
border: var(--border) solid var(--color-primary-content);
box-shadow: calc(var(--depth) * 4px) calc(var(--depth) * 4px)
calc(var(--depth) * 8px)
oklch(from var(--color-base-content) l c h / 0.2);
height: var(--font-2xl, 3rem); height: var(--font-2xl, 3rem);
font-size: var(--font-md, 1rem); font-size: var(--font-md, 1rem);
transition: all 0.3s ease; transition: all 0.3s ease;

View file

@ -1,8 +1,8 @@
import defaultAvatar from '@assets/default-avatar.png';
import type { NDKUserProfile } from '@nostr-dev-kit/ndk'; import type { NDKUserProfile } from '@nostr-dev-kit/ndk';
import { LitElement, css, html } from 'lit-element'; import { LitElement, css, html } from 'lit-element';
import { customElement, property } from 'lit/decorators.js'; import { customElement, property } from 'lit/decorators.js';
type AvatarSize = 'short' | 'medium' | 'large' | 'huge'; type AvatarSize = 'short' | 'medium' | 'large' | 'huge';
import defaultAvatar from '@/default-avatar.png';
@customElement('arx-nostr-avatar') @customElement('arx-nostr-avatar')
export class ArxNostrAvatar extends LitElement { export class ArxNostrAvatar extends LitElement {

267
src/components/Sidebar.ts Normal file
View file

@ -0,0 +1,267 @@
import defaultAvatar from '@assets/default-avatar.png';
import logo from '@assets/logo.png';
import type { NDKUserProfile } from '@nostr-dev-kit/ndk';
import { LitElement, css, html } from 'lit';
import { customElement, property } from 'lit/decorators.js';
import { map } from 'lit/directives/map.js';
@customElement('arx-sidebar')
export default class Sidebar extends LitElement {
@property({ type: String })
currentPath = '';
@property({ type: String })
userNpub = '';
@property({ type: Object })
userProfile: NDKUserProfile | undefined = undefined;
@property({ type: Array })
apps = [
{
id: 1,
href: 'beacon',
name: 'Beacon',
color: '#FF8C00',
icon: 'fa-solid:sun',
},
{
id: 2,
href: 'arbor',
name: 'Arbor',
color: '#FF4040',
icon: 'fa-solid:tree',
},
{
id: 3,
href: 'wallet',
name: 'Wallet',
color: '#1E90FF',
icon: 'fa-solid:spa',
},
{
id: 4,
href: 'settings',
name: 'Settings',
color: '#7B68EE',
icon: 'fa-solid:tools',
},
];
static override styles = css`
:host {
overflow-x: hidden;
background: var(--color-base-200);
padding: 0 0 1.5rem 0;
display: flex;
flex-direction: column;
align-items: center;
overflow-y: overlay;
scrollbar-width: none;
position: relative;
height: 100%;
}
::-webkit-scrollbar {
width: 4px;
height: 4px;
background: transparent;
}
::-webkit-scrollbar-track {
background: transparent;
margin: 3px;
}
::-webkit-scrollbar-thumb {
background: rgba(100, 100, 100, 0.4);
border-radius: 10px;
transition: all 0.3s ease;
}
::-webkit-scrollbar-thumb:hover {
background: rgba(100, 100, 100, 0.7);
}
:host:hover {
scrollbar-color: rgba(100, 100, 100, 0.4) transparent;
}
:host::after {
content: '';
position: absolute;
right: 0;
top: 0;
width: 10px;
height: 100%;
opacity: 0;
pointer-events: none;
background: linear-gradient(to right, transparent, rgba(0, 0, 0, 0.03));
transition: opacity 0.3s ease;
}
:host:hover::after {
opacity: 1;
}
.app-icon-small {
margin-bottom: 1.5rem;
width: 80px;
}
.logo {
height: 64px;
margin-bottom: 1.5rem;
cursor: pointer;
transition: transform 0.2s ease, filter 0.3s ease;
border-radius: var(--radius-field);
padding: 8px;
background: none;
filter:
drop-shadow(0 0 2px rgba(0, 0, 0, 1))
drop-shadow(0 0 1px rgba(255, 255, 255, 1));
}
.logo-container {
width: 100%;
background: var(--color-accent);
display: flex;
justify-content: center;
padding: 12px 0;
margin-bottom: 1.5rem;
position: relative;
}
.logo-container::after {
content: '';
position: absolute;
bottom: -5px;
left: 0;
right: 0;
height: 6px;
background: rgba(0, 0, 0, 0.08);
border-radius: 50%;
filter: blur(3px);
}
.logo-container .logo {
margin-bottom: 0;
transform: translateY(0);
}
.logo:hover {
transform: translateY(-2px) scale(1.02);
filter:
drop-shadow(0 0 3px rgba(0, 0, 0, 0.8))
drop-shadow(0 0 2px rgba(255, 255, 255, 0.6));
}
.logo:active {
transform: translateY(1px) scale(0.98);
filter:
drop-shadow(0 0 1px rgba(0, 0, 0, 0.6));
transition-duration: 0.1s;
}
.app-icon-small.active {
transform: scale(1.15);
position: relative;
z-index: 10;
}
.app-icon-small.active::before {
content: '';
position: absolute;
top: -8px;
left: -8px;
right: -8px;
bottom: -8px;
background: var(--color-accent);
z-index: -1;
}
.sidebar-item {
position: relative;
margin-bottom: 1.5rem;
transition: transform 0.3s cubic-bezier(0.34, 1.56, 0.64, 1);
}
.sidebar-item:hover {
transform: translateY(-2px);
}
.profile-avatar {
width: 64px;
height: 64px;
border-radius: 50%;
object-fit: cover;
transition: transform 0.3s cubic-bezier(0.34, 1.56, 0.64, 1);
border: 2px solid var(--color-base-300);
}
.profile-avatar.active {
border-color: var(--color-primary);
transform: scale(1.15);
box-shadow: 0 3px 10px rgba(0, 0, 0, 0.15);
}
.profile-item {
margin-bottom: 1.5rem;
transition: transform 0.3s cubic-bezier(0.34, 1.56, 0.64, 1);
display: flex;
justify-content: center;
}
.profile-item:hover .profile-avatar {
transform: translateY(-2px) scale(1.05);
}
`;
private handleLogoClick() {
this.dispatchEvent(new CustomEvent('navigate', { detail: 'home' }));
}
override render() {
const activePath = this.currentPath.split('/')[0];
return html`
<div class="sidebar-item logo-container">
<img
src="${logo}"
alt="Eve"
class="logo"
@click=${this.handleLogoClick}
/>
</div>
${map(
this.apps,
(app) => html`
<div class="sidebar-item">
<arx-app-icon
class="app-icon-small ${activePath === app.href.split('/')[0] ? 'active' : ''}"
?selected=${activePath === app.href.split('/')[0]}
.icon=${app.icon}
.color=${app.color}
.href=${`#${app.href}`}
.name=${app.name}
?small=${true}
></arx-app-icon>
</div>
`,
)}
<div style="flex-grow: 1;"></div>
<div class="profile-item">
<a href="#profile/${this.userNpub}" class="profile-link">
<img
class="profile-avatar ${activePath === 'profile' ? 'active' : ''}"
src=${this.userProfile?.picture || defaultAvatar}
alt="Profile"
@error=${(e: Event) => {
(e.target as HTMLImageElement).src = defaultAvatar;
}}
/>
</a>
</div>
`;
}
}

Binary file not shown.

Before

Width:  |  Height:  |  Size: 80 KiB

View file

@ -38,9 +38,11 @@ ipcMain.handle('relay:status', () => {
function createWindow(): void { function createWindow(): void {
const mainWindow = new BrowserWindow({ const mainWindow = new BrowserWindow({
width: 1024, width: 1366,
height: 768, height: 768,
show: false, show: false,
minWidth: 1366,
minHeight: 768,
autoHideMenuBar: true, autoHideMenuBar: true,
webPreferences: { webPreferences: {
preload: path.join(__dirname, '../preload/preload.mjs'), preload: path.join(__dirname, '../preload/preload.mjs'),

View file

@ -1,13 +1,12 @@
import './style.css';
import '@components/ErrorView';
import '@components/NostrAvatar';
import '@components/LoadingView';
import '@components/NostrProfile';
import '@components/Breadcrumbs'; import '@components/Breadcrumbs';
import '@components/ErrorView';
import '@components/Header'; import '@components/Header';
import '@routes/router';
import '@components/LoadingView'; import '@components/LoadingView';
import '@components/NostrAvatar';
import '@components/NostrProfile';
import '@routes/router';
import type EveRouter from '@routes/router'; import type EveRouter from '@routes/router';
import './style.css';
function checkRelayUp() { function checkRelayUp() {
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {

View file

@ -193,7 +193,7 @@ export class Home extends LitElement {
<arx-card class="home-container"> <arx-card class="home-container">
<arx-card class="welcome-section"> <arx-card class="welcome-section">
<arx-nostr-avatar <arx-nostr-avatar
.profile=${this.profile} .profile=${this.profile as NDKUserProfile}
size="huge" size="huge"
></arx-nostr-avatar> ></arx-nostr-avatar>
<div class="welcome-text"> <div class="welcome-text">

View file

@ -1,5 +1,5 @@
import defaultAvatar from '@/default-avatar.png';
import { getSigner, getUserProfile, ndk } from '@/ndk'; import { getSigner, getUserProfile, ndk } from '@/ndk';
import defaultAvatar from '@assets/default-avatar.png';
import type { ArxInputChangeEvent } from '@components/General/Input'; import type { ArxInputChangeEvent } from '@components/General/Input';
import { NDKEvent, type NDKUserProfile } from '@nostr-dev-kit/ndk'; import { NDKEvent, type NDKUserProfile } from '@nostr-dev-kit/ndk';
import { LitElement, css, html } from 'lit'; import { LitElement, css, html } from 'lit';
@ -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(

View file

@ -1,4 +1,5 @@
import '@components/InitialSetup'; import '@components/InitialSetup';
import '@components/Sidebar';
import '@routes/404Page'; import '@routes/404Page';
import '@routes/Arbor/Home'; import '@routes/Arbor/Home';
import '@routes/Arbor/NewPost'; import '@routes/Arbor/NewPost';
@ -10,10 +11,14 @@ import '@routes/Profile';
import '@routes/Settings'; import '@routes/Settings';
import '@routes/Wallet'; import '@routes/Wallet';
import { getNpub, getUserProfile } from '@/ndk';
import type { NDKUserProfile } from '@nostr-dev-kit/ndk';
import { spread } from '@open-wc/lit-helpers'; 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 +29,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,17 +94,34 @@ 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)[] = [];
@state()
private userProfile: NDKUserProfile | undefined = undefined;
@state()
private userNpub = '';
static override styles = css` static override styles = css`
:host { :host {
height: 100vh; position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
display: grid; display: grid;
grid-template-rows: auto 1fr; grid-template-rows: auto 1fr;
grid-template-columns: 100px 1fr;
overflow: hidden; overflow: hidden;
} }
@ -124,9 +145,34 @@ export default class EveRouter extends LitElement {
::-webkit-scrollbar-thumb:hover { ::-webkit-scrollbar-thumb:hover {
background: var(--color-neutral); background: var(--color-neutral);
} }
.window { .window {
overflow: auto; overflow: auto;
grid-column: 2;
grid-row: 2;
position: relative;
}
.window::after {
content: '';
position: absolute;
right: 0;
top: 0;
width: 10px;
height: 100%;
opacity: 0;
pointer-events: none;
background: linear-gradient(to right, transparent, rgba(0, 0, 0, 0.03));
transition: opacity 0.3s ease;
}
.window:hover::after {
opacity: 1;
}
arx-sidebar {
grid-column: 1;
grid-row: 1 / span 2;
} }
.window-content { .window-content {
@ -135,22 +181,58 @@ 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 {
overflow: hidden; overflow: hidden;
} }
arx-header {
grid-column: 2;
grid-row: 1;
}
`; `;
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')!);
} }
override connectedCallback(): void { override connectedCallback(): void {
super.connectedCallback(); super.connectedCallback();
this.setupEventListeners(); this.setupEventListeners();
if (this.ccnSetup) {
this.loadUserProfile();
}
} }
override disconnectedCallback(): void { override disconnectedCallback(): void {
@ -171,11 +253,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 +334,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 +345,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--;
@ -283,7 +384,7 @@ export default class EveRouter extends LitElement {
renderSetup() { renderSetup() {
return html` return html`
<div class="window"> <div class="window" style="grid-column: 1 / span 2;">
<div class="window-content"> <div class="window-content">
<arx-initial-setup <arx-initial-setup
@finish=${() => this.finishSetup()} @finish=${() => this.finishSetup()}
@ -293,9 +394,25 @@ export default class EveRouter extends LitElement {
`; `;
} }
async loadUserProfile() {
try {
this.userNpub = await getNpub();
this.userProfile = await getUserProfile();
} catch (error) {
console.error('Failed to load user profile:', error);
}
}
override render() { override render() {
if (!this.ccnSetup) return this.renderSetup(); if (!this.ccnSetup) return this.renderSetup();
return html` return html`
<arx-sidebar
.currentPath=${this.currentPath}
.userNpub=${this.userNpub}
.userProfile=${this.userProfile}
@navigate=${(e: CustomEvent) => this.navigate(e.detail)}
></arx-sidebar>
<arx-header <arx-header
?canGoBack=${this.currentIndex > 0} ?canGoBack=${this.currentIndex > 0}
?canGoForward=${this.currentIndex < this.history.length - 1} ?canGoForward=${this.currentIndex < this.history.length - 1}
@ -306,10 +423,14 @@ 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.currentRoute.params, this.isTransitioning,
html` () => html`<arx-loading-view></arx-loading-view>`,
() =>
keyed(
this.currentRoute.params,
html`
<${this.currentRoute.component} <${this.currentRoute.component}
${spread(this.currentRoute.params)} ${spread(this.currentRoute.params)}
path=${this.currentPath} path=${this.currentPath}
@ -318,6 +439,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>

View file

@ -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);