Simple Managment webapp [LLM]

This commit is contained in:
2025-12-03 17:20:50 +01:00
parent b521f74951
commit e7f613b5dc
76 changed files with 20009 additions and 1 deletions

View File

@@ -0,0 +1,74 @@
<div class="page-content">
@if (loading()) {
<div class="loading-container">
<nz-spin nzSimple nzSize="large"></nz-spin>
</div>
} @else if (message()) {
<div class="detail-header">
<button nz-button (click)="goBack()">
<span nz-icon nzType="arrow-left" nzTheme="outline"></span>
Back to Messages
</button>
<button
nz-button
nzType="primary"
nzDanger
nz-popconfirm
nzPopconfirmTitle="Are you sure you want to delete this message?"
nzPopconfirmPlacement="bottomRight"
(nzOnConfirm)="deleteMessage()"
[nzLoading]="deleting()"
>
<span nz-icon nzType="delete"></span>
Delete
</button>
</div>
<nz-card [nzTitle]="message()!.title">
<nz-descriptions nzBordered [nzColumn]="2">
<nz-descriptions-item nzTitle="Message ID" [nzSpan]="2">
<span class="mono">{{ message()!.message_id }}</span>
</nz-descriptions-item>
<nz-descriptions-item nzTitle="Channel">
{{ message()!.channel_internal_name }}
</nz-descriptions-item>
<nz-descriptions-item nzTitle="Priority">
<nz-tag [nzColor]="getPriorityColor(message()!.priority)">
{{ getPriorityLabel(message()!.priority) }}
</nz-tag>
</nz-descriptions-item>
<nz-descriptions-item nzTitle="Sender Name">
{{ message()!.sender_name || '-' }}
</nz-descriptions-item>
<nz-descriptions-item nzTitle="Sender IP">
{{ message()!.sender_ip }}
</nz-descriptions-item>
<nz-descriptions-item nzTitle="Timestamp" [nzSpan]="2">
{{ message()!.timestamp }} ({{ message()!.timestamp | relativeTime }})
</nz-descriptions-item>
<nz-descriptions-item nzTitle="User Message ID" [nzSpan]="2">
<span class="mono">{{ message()!.usr_message_id || '-' }}</span>
</nz-descriptions-item>
<nz-descriptions-item nzTitle="Used Key ID" [nzSpan]="2">
<span class="mono">{{ message()!.used_key_id }}</span>
</nz-descriptions-item>
</nz-descriptions>
@if (message()!.content) {
<nz-divider nzText="Content"></nz-divider>
<div class="message-content">
<pre>{{ message()!.content }}</pre>
</div>
}
</nz-card>
} @else {
<nz-card>
<div class="not-found">
<p>Message not found</p>
<button nz-button nzType="primary" (click)="goBack()">
Back to Messages
</button>
</div>
</nz-card>
}
</div>

View File

@@ -0,0 +1,31 @@
.loading-container {
display: flex;
justify-content: center;
align-items: center;
min-height: 300px;
}
.message-content {
background: #f5f5f5;
padding: 16px;
border-radius: 4px;
overflow-x: auto;
pre {
margin: 0;
white-space: pre-wrap;
word-wrap: break-word;
font-family: 'SF Mono', 'Monaco', 'Inconsolata', 'Fira Mono', 'Droid Sans Mono', 'Source Code Pro', monospace;
font-size: 13px;
}
}
.not-found {
text-align: center;
padding: 48px;
p {
color: #999;
margin-bottom: 16px;
}
}

View File

@@ -0,0 +1,102 @@
import { Component, inject, signal, OnInit } from '@angular/core';
import { CommonModule } from '@angular/common';
import { ActivatedRoute, Router } from '@angular/router';
import { NzCardModule } from 'ng-zorro-antd/card';
import { NzButtonModule } from 'ng-zorro-antd/button';
import { NzIconModule } from 'ng-zorro-antd/icon';
import { NzDescriptionsModule } from 'ng-zorro-antd/descriptions';
import { NzTagModule } from 'ng-zorro-antd/tag';
import { NzSpinModule } from 'ng-zorro-antd/spin';
import { NzPopconfirmModule } from 'ng-zorro-antd/popconfirm';
import { NzDividerModule } from 'ng-zorro-antd/divider';
import { ApiService } from '../../../core/services/api.service';
import { NotificationService } from '../../../core/services/notification.service';
import { Message } from '../../../core/models';
import { RelativeTimePipe } from '../../../shared/pipes/relative-time.pipe';
@Component({
selector: 'app-message-detail',
standalone: true,
imports: [
CommonModule,
NzCardModule,
NzButtonModule,
NzIconModule,
NzDescriptionsModule,
NzTagModule,
NzSpinModule,
NzPopconfirmModule,
NzDividerModule,
RelativeTimePipe,
],
templateUrl: './message-detail.component.html',
styleUrl: './message-detail.component.scss'
})
export class MessageDetailComponent implements OnInit {
private route = inject(ActivatedRoute);
private router = inject(Router);
private apiService = inject(ApiService);
private notification = inject(NotificationService);
message = signal<Message | null>(null);
loading = signal(true);
deleting = signal(false);
ngOnInit(): void {
const messageId = this.route.snapshot.paramMap.get('id');
if (messageId) {
this.loadMessage(messageId);
}
}
loadMessage(messageId: string): void {
this.loading.set(true);
this.apiService.getMessage(messageId).subscribe({
next: (message) => {
this.message.set(message);
this.loading.set(false);
},
error: () => {
this.loading.set(false);
}
});
}
goBack(): void {
this.router.navigate(['/messages']);
}
deleteMessage(): void {
const message = this.message();
if (!message) return;
this.deleting.set(true);
this.apiService.deleteMessage(message.message_id).subscribe({
next: () => {
this.notification.success('Message deleted');
this.router.navigate(['/messages']);
},
error: () => {
this.deleting.set(false);
}
});
}
getPriorityLabel(priority: number): string {
switch (priority) {
case 0: return 'Low';
case 1: return 'Normal';
case 2: return 'High';
default: return 'Unknown';
}
}
getPriorityColor(priority: number): string {
switch (priority) {
case 0: return 'default';
case 1: return 'blue';
case 2: return 'red';
default: return 'default';
}
}
}

View File

@@ -0,0 +1,118 @@
<div class="page-content">
<div class="page-header">
<h2>Messages</h2>
<button nz-button nzType="default" (click)="loadMessages()">
<span nz-icon nzType="reload"></span>
Refresh
</button>
</div>
<nz-card class="filter-card">
<div class="filter-bar">
<nz-input-group nzSearch [nzAddOnAfter]="searchButton" style="width: 300px;">
<input
type="text"
nz-input
placeholder="Search messages..."
[(ngModel)]="searchText"
(keyup.enter)="applyFilters()"
/>
</nz-input-group>
<ng-template #searchButton>
<button nz-button nzType="primary" nzSearch (click)="applyFilters()">
<span nz-icon nzType="search"></span>
</button>
</ng-template>
<nz-select
[(ngModel)]="priorityFilter"
nzPlaceHolder="Priority"
nzAllowClear
style="width: 150px;"
(ngModelChange)="applyFilters()"
>
<nz-option [nzValue]="0" nzLabel="Low"></nz-option>
<nz-option [nzValue]="1" nzLabel="Normal"></nz-option>
<nz-option [nzValue]="2" nzLabel="High"></nz-option>
</nz-select>
<input
type="text"
nz-input
placeholder="Channel name"
[(ngModel)]="channelFilter"
style="width: 200px;"
(keyup.enter)="applyFilters()"
/>
@if (searchText || priorityFilter !== null || channelFilter) {
<button nz-button (click)="clearFilters()">
<span nz-icon nzType="close"></span>
Clear
</button>
}
</div>
</nz-card>
<nz-card>
<nz-table
#messageTable
[nzData]="messages()"
[nzLoading]="loading()"
[nzShowPagination]="false"
nzSize="middle"
>
<thead>
<tr>
<th nzWidth="40%">Title</th>
<th nzWidth="15%">Channel</th>
<th nzWidth="15%">Sender</th>
<th nzWidth="10%">Priority</th>
<th nzWidth="20%">Time</th>
</tr>
</thead>
<tbody>
@for (message of messages(); track message.message_id) {
<tr class="clickable-row" (click)="viewMessage(message)">
<td>
<div class="message-title">{{ message.title }}</div>
@if (message.content && !message.trimmed) {
<div class="message-preview">{{ message.content | slice:0:100 }}{{ message.content.length > 100 ? '...' : '' }}</div>
}
</td>
<td>
<span class="mono">{{ message.channel_internal_name }}</span>
</td>
<td>
{{ message.sender_name || '-' }}
</td>
<td>
<nz-tag [nzColor]="getPriorityColor(message.priority)">
{{ getPriorityLabel(message.priority) }}
</nz-tag>
</td>
<td>
<span nz-tooltip [nzTooltipTitle]="message.timestamp">
{{ message.timestamp | relativeTime }}
</span>
</td>
</tr>
} @empty {
<tr>
<td colspan="5">
<nz-empty nzNotFoundContent="No messages found"></nz-empty>
</td>
</tr>
}
</tbody>
</nz-table>
@if (nextPageToken()) {
<div class="load-more">
<button nz-button nzType="default" (click)="loadMore()" [nzLoading]="loading()">
Load More
</button>
</div>
}
</nz-card>
</div>

View File

@@ -0,0 +1,31 @@
.page-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 16px;
h2 {
margin: 0;
}
}
.filter-card {
margin-bottom: 16px;
}
.message-title {
font-weight: 500;
color: #333;
}
.message-preview {
font-size: 12px;
color: #999;
margin-top: 4px;
}
.load-more {
display: flex;
justify-content: center;
padding: 16px 0;
}

View File

@@ -0,0 +1,138 @@
import { Component, inject, signal, OnInit } from '@angular/core';
import { CommonModule } from '@angular/common';
import { Router } from '@angular/router';
import { FormsModule } from '@angular/forms';
import { NzTableModule } from 'ng-zorro-antd/table';
import { NzButtonModule } from 'ng-zorro-antd/button';
import { NzInputModule } from 'ng-zorro-antd/input';
import { NzSelectModule } from 'ng-zorro-antd/select';
import { NzTagModule } from 'ng-zorro-antd/tag';
import { NzIconModule } from 'ng-zorro-antd/icon';
import { NzEmptyModule } from 'ng-zorro-antd/empty';
import { NzSpinModule } from 'ng-zorro-antd/spin';
import { NzCardModule } from 'ng-zorro-antd/card';
import { NzToolTipModule } from 'ng-zorro-antd/tooltip';
import { ApiService } from '../../../core/services/api.service';
import { AuthService } from '../../../core/services/auth.service';
import { Message, MessageListParams } from '../../../core/models';
import { RelativeTimePipe } from '../../../shared/pipes/relative-time.pipe';
@Component({
selector: 'app-message-list',
standalone: true,
imports: [
CommonModule,
FormsModule,
NzTableModule,
NzButtonModule,
NzInputModule,
NzSelectModule,
NzTagModule,
NzIconModule,
NzEmptyModule,
NzSpinModule,
NzCardModule,
NzToolTipModule,
RelativeTimePipe,
],
templateUrl: './message-list.component.html',
styleUrl: './message-list.component.scss'
})
export class MessageListComponent implements OnInit {
private apiService = inject(ApiService);
private authService = inject(AuthService);
private router = inject(Router);
messages = signal<Message[]>([]);
loading = signal(false);
nextPageToken = signal<string | null>(null);
// Filters
searchText = '';
priorityFilter: number | null = null;
channelFilter = '';
ngOnInit(): void {
this.loadMessages();
}
loadMessages(append = false): void {
this.loading.set(true);
const params: MessageListParams = {
page_size: 50,
trimmed: true,
};
if (this.searchText) {
params.search = this.searchText;
}
if (this.priorityFilter !== null) {
params.priority = this.priorityFilter;
}
if (this.channelFilter) {
params.channel = this.channelFilter;
}
if (append && this.nextPageToken()) {
params.next_page_token = this.nextPageToken()!;
}
this.apiService.getMessages(params).subscribe({
next: (response) => {
if (append) {
this.messages.update(msgs => [...msgs, ...response.messages]);
} else {
this.messages.set(response.messages);
}
this.nextPageToken.set(
response.next_page_token && response.next_page_token !== '@end'
? response.next_page_token
: null
);
this.loading.set(false);
},
error: () => {
this.loading.set(false);
}
});
}
applyFilters(): void {
this.loadMessages();
}
clearFilters(): void {
this.searchText = '';
this.priorityFilter = null;
this.channelFilter = '';
this.loadMessages();
}
loadMore(): void {
if (this.nextPageToken()) {
this.loadMessages(true);
}
}
viewMessage(message: Message): void {
this.router.navigate(['/messages', message.message_id]);
}
getPriorityLabel(priority: number): string {
switch (priority) {
case 0: return 'Low';
case 1: return 'Normal';
case 2: return 'High';
default: return 'Unknown';
}
}
getPriorityColor(priority: number): string {
switch (priority) {
case 0: return 'default';
case 1: return 'blue';
case 2: return 'red';
default: return 'default';
}
}
}