3 次代碼提交 617ed9b56d ... 3ff4a8fae7

作者 SHA1 備註 提交日期
  csdn1233 3ff4a8fae7 Merge branch 'master' of http://git.fmode.cn:3000/csdn1233/s202226701049 3 月之前
  csdn1233 e85f181a34 update:Front-end changes 3 月之前
  csdn1233 056b58915e fix:small 3 月之前
共有 40 個文件被更改,包括 2689 次插入391 次删除
  1. 56 0
      AIart-app/src/app/art-detail/art-detail.component.html
  2. 217 0
      AIart-app/src/app/art-detail/art-detail.component.scss
  3. 22 0
      AIart-app/src/app/art-detail/art-detail.component.spec.ts
  4. 63 0
      AIart-app/src/app/art-detail/art-detail.component.ts
  5. 0 50
      AIart-app/src/app/art-detail/art-detail.page.html
  6. 0 90
      AIart-app/src/app/art-detail/art-detail.page.scss
  7. 0 89
      AIart-app/src/app/art-detail/art-detail.page.ts
  8. 76 0
      AIart-app/src/app/attendance/attendance.component.html
  9. 224 0
      AIart-app/src/app/attendance/attendance.component.scss
  10. 22 0
      AIart-app/src/app/attendance/attendance.component.spec.ts
  11. 64 0
      AIart-app/src/app/attendance/attendance.component.ts
  12. 27 0
      AIart-app/src/app/category-modal/category-modal.component.html
  13. 46 0
      AIart-app/src/app/category-modal/category-modal.component.scss
  14. 24 0
      AIart-app/src/app/category-modal/category-modal.component.ts
  15. 3 0
      AIart-app/src/app/drawing/drawing.component.html
  16. 132 0
      AIart-app/src/app/drawing/drawing.component.scss
  17. 22 0
      AIart-app/src/app/drawing/drawing.component.spec.ts
  18. 306 0
      AIart-app/src/app/drawing/drawing.component.ts
  19. 20 5
      AIart-app/src/app/edit-tag/edit-tag.component.html
  20. 86 0
      AIart-app/src/app/edit-tag/edit-tag.component.scss
  21. 21 12
      AIart-app/src/app/edit-tag/edit-tag.component.ts
  22. 93 33
      AIart-app/src/app/interest-picture/interest-picture.component.html
  23. 285 0
      AIart-app/src/app/interest-picture/interest-picture.component.scss
  24. 141 43
      AIart-app/src/app/interest-picture/interest-picture.component.ts
  25. 60 0
      AIart-app/src/app/mailbox/mailbox.component.html
  26. 171 0
      AIart-app/src/app/mailbox/mailbox.component.scss
  27. 22 0
      AIart-app/src/app/mailbox/mailbox.component.spec.ts
  28. 18 0
      AIart-app/src/app/mailbox/mailbox.component.ts
  29. 22 22
      AIart-app/src/app/tab1/tab1.page.html
  30. 56 0
      AIart-app/src/app/tab1/tab1.page.ts
  31. 8 7
      AIart-app/src/app/tab3/tab3.page.html
  32. 33 7
      AIart-app/src/app/tab3/tab3.page.ts
  33. 0 1
      AIart-app/src/app/tab4/tab4.page.html
  34. 23 9
      AIart-app/src/app/tabs/tabs.page.ts
  35. 16 1
      AIart-app/src/app/tabs/tabs.routes.ts
  36. 25 12
      AIart-app/src/app/user-login/user-login.component.html
  37. 35 6
      AIart-app/src/app/user-login/user-login.component.ts
  38. 43 3
      AIart-app/src/app/view-all/view-all.component.html
  39. 157 0
      AIart-app/src/app/view-all/view-all.component.scss
  40. 50 1
      AIart-app/src/app/view-all/view-all.component.ts

+ 56 - 0
AIart-app/src/app/art-detail/art-detail.component.html

@@ -0,0 +1,56 @@
+<ion-header class="ion-no-border">
+  <ion-toolbar>
+    <ion-buttons slot="start">
+      <ion-back-button defaultHref="/tabs/tab3"></ion-back-button>
+    </ion-buttons>
+    <ion-title>作品详情</ion-title>
+    <ion-buttons slot="end">
+      <ion-button (click)="shareArtwork()">
+        <ion-icon name="share-social-outline"></ion-icon>
+      </ion-button>
+    </ion-buttons>
+  </ion-toolbar>
+</ion-header>
+
+<ion-content>
+  <div class="art-detail-container" *ngIf="artwork">
+    <div class="image-container">
+      <img [src]="artwork.image" [alt]="artwork.title">
+    </div>
+
+    <div class="info-container">
+      <h2>{{artwork.title}}</h2>
+      <p class="description">{{artwork.description}}</p>
+
+      <div class="user-info">
+        <img [src]="artwork.avatarUrl" [alt]="artwork.userName" class="avatar">
+        <div class="user-details">
+          <span class="username">{{artwork.userName}}</span>
+          <span class="date">{{artwork.date}}</span>
+        </div>
+      </div>
+
+      <div class="interaction-bar">
+        <div class="likes" (click)="toggleLike()">
+          <ion-icon [name]="artwork.isLiked ? 'heart' : 'heart-outline'"
+            [color]="artwork.isLiked ? 'danger' : 'medium'">
+          </ion-icon>
+          <span>{{artwork.likes}}</span>
+        </div>
+        <div class="share">
+          <ion-icon name="share-social-outline"></ion-icon>
+          <span>分享</span>
+        </div>
+      </div>
+    </div>
+
+    <!-- 添加评论区域 -->
+    <div class="comments-section">
+      <h3>评论</h3>
+      <div class="comment-input">
+        <ion-input placeholder="写下你的评论..."></ion-input>
+        <ion-button expand="block">发送</ion-button>
+      </div>
+    </div>
+  </div>
+</ion-content>

+ 217 - 0
AIart-app/src/app/art-detail/art-detail.component.scss

@@ -0,0 +1,217 @@
+.art-detail {
+    .main-image {
+        width: 100%;
+        height: 300px;
+        object-fit: cover;
+    }
+
+    .content-section {
+        padding: 20px;
+
+        h1 {
+            font-size: 24px;
+            font-weight: 600;
+            margin: 0 0 12px;
+            color: #333;
+        }
+
+        .description {
+            font-size: 16px;
+            line-height: 1.6;
+            color: #666;
+            margin-bottom: 20px;
+        }
+
+        .interaction-bar {
+            display: flex;
+            align-items: center;
+            gap: 16px;
+            padding: 8px 0;
+            border-bottom: 1px solid #eee;
+        }
+    }
+
+    .comments-section {
+        padding: 20px;
+
+        h2 {
+            font-size: 18px;
+            font-weight: 600;
+            margin: 0 0 16px;
+            color: #333;
+        }
+
+        .comment-input {
+            margin-bottom: 24px;
+
+            ion-button {
+                margin-top: 12px;
+            }
+        }
+
+        .comments-list {
+            .comment-item {
+                margin-bottom: 20px;
+                padding-bottom: 20px;
+                border-bottom: 1px solid #eee;
+
+                .comment-header {
+                    display: flex;
+                    align-items: center;
+                    gap: 8px;
+                    margin-bottom: 8px;
+
+                    img {
+                        width: 32px;
+                        height: 32px;
+                        border-radius: 50%;
+                    }
+
+                    .username {
+                        font-weight: 500;
+                        color: #333;
+                    }
+
+                    .time {
+                        font-size: 12px;
+                        color: #999;
+                    }
+                }
+
+                .comment-content {
+                    font-size: 14px;
+                    line-height: 1.5;
+                    color: #666;
+                    margin: 0;
+                }
+            }
+        }
+    }
+}
+
+.art-detail-container {
+    .image-container {
+        width: 100%;
+        height: 300px;
+        overflow: hidden;
+        position: relative;
+
+        img {
+            width: 100%;
+            height: 100%;
+            object-fit: cover;
+            transition: transform 0.3s ease;
+
+            &:hover {
+                transform: scale(1.05);
+            }
+        }
+    }
+
+    .info-container {
+        padding: 20px;
+        background: white;
+        border-radius: 20px;
+        margin-top: -20px;
+        position: relative;
+        z-index: 1;
+        box-shadow: 0 -4px 10px rgba(0, 0, 0, 0.1);
+
+        h2 {
+            font-size: 24px;
+            font-weight: 600;
+            color: #333;
+            margin-bottom: 12px;
+        }
+
+        .description {
+            font-size: 16px;
+            color: #666;
+            line-height: 1.5;
+            margin-bottom: 20px;
+        }
+
+        .user-info {
+            display: flex;
+            align-items: center;
+            margin-bottom: 20px;
+
+            .avatar {
+                width: 48px;
+                height: 48px;
+                border-radius: 50%;
+                margin-right: 12px;
+                border: 2px solid white;
+                box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
+            }
+
+            .user-details {
+                display: flex;
+                flex-direction: column;
+
+                .username {
+                    font-size: 16px;
+                    font-weight: 500;
+                    color: #333;
+                }
+
+                .date {
+                    font-size: 14px;
+                    color: #999;
+                }
+            }
+        }
+
+        .interaction-bar {
+            display: flex;
+            gap: 20px;
+            padding: 15px 0;
+            border-top: 1px solid #eee;
+
+            .likes,
+            .share {
+                display: flex;
+                align-items: center;
+                gap: 8px;
+                color: #666;
+                cursor: pointer;
+                transition: all 0.3s ease;
+
+                &:hover {
+                    color: var(--ion-color-primary);
+                }
+
+                ion-icon {
+                    font-size: 24px;
+                }
+            }
+        }
+    }
+
+    .comments-section {
+        padding: 20px;
+        background: white;
+        margin-top: 12px;
+
+        h3 {
+            font-size: 18px;
+            font-weight: 600;
+            color: #333;
+            margin-bottom: 16px;
+        }
+
+        .comment-input {
+            ion-input {
+                --background: #f5f5f5;
+                --padding-start: 16px;
+                --padding-end: 16px;
+                --border-radius: 8px;
+                margin-bottom: 12px;
+            }
+
+            ion-button {
+                --border-radius: 8px;
+            }
+        }
+    }
+}

+ 22 - 0
AIart-app/src/app/art-detail/art-detail.component.spec.ts

@@ -0,0 +1,22 @@
+import { ComponentFixture, TestBed, waitForAsync } from '@angular/core/testing';
+
+import { ArtDetailComponent } from './art-detail.component';
+
+describe('ArtDetailComponent', () => {
+  let component: ArtDetailComponent;
+  let fixture: ComponentFixture<ArtDetailComponent>;
+
+  beforeEach(waitForAsync(() => {
+    TestBed.configureTestingModule({
+      imports: [ArtDetailComponent],
+    }).compileComponents();
+
+    fixture = TestBed.createComponent(ArtDetailComponent);
+    component = fixture.componentInstance;
+    fixture.detectChanges();
+  }));
+
+  it('should create', () => {
+    expect(component).toBeTruthy();
+  });
+});

+ 63 - 0
AIart-app/src/app/art-detail/art-detail.component.ts

@@ -0,0 +1,63 @@
+import { Component, OnInit } from '@angular/core';
+import { ActivatedRoute, Router } from '@angular/router';
+import { CommonModule } from '@angular/common';
+import { IonicModule } from '@ionic/angular';
+
+@Component({
+  selector: 'app-art-detail',
+  templateUrl: './art-detail.component.html',
+  styleUrls: ['./art-detail.component.scss'],
+  standalone: true,
+  imports: [IonicModule, CommonModule]
+})
+export class ArtDetailComponent implements OnInit {
+  artId: string = '';
+  artwork: any = null;
+
+  constructor(
+    private route: ActivatedRoute,
+    private router: Router
+  ) {
+    // 获取路由传递的数据
+    const navigation = this.router.getCurrentNavigation();
+    if (navigation?.extras.state) {
+      this.artwork = navigation.extras.state['artwork'];
+    }
+  }
+
+  ngOnInit() {
+    this.artId = this.route.snapshot.paramMap.get('id') || '';
+    if (!this.artwork) {
+      this.loadArtwork();
+    }
+  }
+
+  loadArtwork() {
+    // 如果没有通过路由获取到数据,则加载默认数据
+    this.artwork = {
+      id: this.artId,
+      image: '../../assets/img/xingkong.png',
+      title: '梵高-星空高清画作',
+      description: '致敬梵高 #艺术品 #星空',
+      avatarUrl: '../../assets/img/book1.png',
+      userName: '星空艺术家',
+      date: '12-01',
+      likes: 1687,
+      category: 'art'
+    };
+  }
+
+  // 添加点赞功能
+  toggleLike() {
+    if (this.artwork) {
+      this.artwork.isLiked = !this.artwork.isLiked;
+      this.artwork.likes += this.artwork.isLiked ? 1 : -1;
+    }
+  }
+
+  // 添加分享功能
+  shareArtwork() {
+    // 实现分享逻辑
+    console.log('Sharing artwork:', this.artwork.title);
+  }
+}

+ 0 - 50
AIart-app/src/app/art-detail/art-detail.page.html

@@ -1,50 +0,0 @@
-<ion-header [translucent]="true" class="ion-no-border">
-    <ion-toolbar>
-        <ion-buttons slot="start">
-            <ion-back-button defaultHref="/tabs/tab3"></ion-back-button>
-        </ion-buttons>
-        <ion-title>作品详情</ion-title>
-    </ion-toolbar>
-</ion-header>
-
-<ion-content [fullscreen]="true">
-    <div class="art-detail">
-        <img [src]="artDetail.image" [alt]="artDetail.title" class="main-image">
-
-        <div class="content-section">
-            <h1>{{artDetail.title}}</h1>
-            <p class="description">{{artDetail.description}}</p>
-
-            <div class="interaction-bar">
-                <ion-button fill="clear" (click)="toggleLike()" [color]="isLiked ? 'danger' : 'medium'">
-                    <ion-icon slot="start" [name]="isLiked ? 'heart' : 'heart-outline'"></ion-icon>
-                    {{likeCount}}
-                </ion-button>
-            </div>
-        </div>
-
-        <div class="comments-section">
-            <h2>评论</h2>
-
-            <div class="comment-input">
-                <ion-item>
-                    <ion-textarea [(ngModel)]="newComment" placeholder="写下你的评论..." rows="3"></ion-textarea>
-                </ion-item>
-                <ion-button expand="block" (click)="addComment()">发表评论</ion-button>
-            </div>
-
-            <div class="comments-list">
-                @for(comment of comments; track comment.id) {
-                <div class="comment-item">
-                    <div class="comment-header">
-                        <img [src]="comment.userAvatar || 'assets/default-avatar.png'" alt="avatar">
-                        <span class="username">{{comment.username}}</span>
-                        <span class="time">{{comment.createdAt | date:'MM-dd HH:mm'}}</span>
-                    </div>
-                    <p class="comment-content">{{comment.content}}</p>
-                </div>
-                }
-            </div>
-        </div>
-    </div>
-</ion-content>

+ 0 - 90
AIart-app/src/app/art-detail/art-detail.page.scss

@@ -1,90 +0,0 @@
-.art-detail {
-    .main-image {
-        width: 100%;
-        height: 300px;
-        object-fit: cover;
-    }
-
-    .content-section {
-        padding: 20px;
-
-        h1 {
-            font-size: 24px;
-            font-weight: 600;
-            margin: 0 0 12px;
-            color: #333;
-        }
-
-        .description {
-            font-size: 16px;
-            line-height: 1.6;
-            color: #666;
-            margin-bottom: 20px;
-        }
-
-        .interaction-bar {
-            display: flex;
-            align-items: center;
-            gap: 16px;
-            padding: 8px 0;
-            border-bottom: 1px solid #eee;
-        }
-    }
-
-    .comments-section {
-        padding: 20px;
-
-        h2 {
-            font-size: 18px;
-            font-weight: 600;
-            margin: 0 0 16px;
-            color: #333;
-        }
-
-        .comment-input {
-            margin-bottom: 24px;
-
-            ion-button {
-                margin-top: 12px;
-            }
-        }
-
-        .comments-list {
-            .comment-item {
-                margin-bottom: 20px;
-                padding-bottom: 20px;
-                border-bottom: 1px solid #eee;
-
-                .comment-header {
-                    display: flex;
-                    align-items: center;
-                    gap: 8px;
-                    margin-bottom: 8px;
-
-                    img {
-                        width: 32px;
-                        height: 32px;
-                        border-radius: 50%;
-                    }
-
-                    .username {
-                        font-weight: 500;
-                        color: #333;
-                    }
-
-                    .time {
-                        font-size: 12px;
-                        color: #999;
-                    }
-                }
-
-                .comment-content {
-                    font-size: 14px;
-                    line-height: 1.5;
-                    color: #666;
-                    margin: 0;
-                }
-            }
-        }
-    }
-}

+ 0 - 89
AIart-app/src/app/art-detail/art-detail.page.ts

@@ -1,89 +0,0 @@
-import { Component, OnInit, CUSTOM_ELEMENTS_SCHEMA } from '@angular/core';
-import { ActivatedRoute } from '@angular/router';
-import { IonHeader, IonToolbar, IonTitle, IonContent, IonButton, IonIcon, IonBackButton, IonItem, IonLabel, IonInput, IonTextarea, IonButtons } from '@ionic/angular/standalone';
-import { CommonModule } from '@angular/common';
-import { FormsModule } from '@angular/forms';
-import { CloudObject, CloudQuery } from 'src/lib/ncloud';
-
-@Component({
-    selector: 'app-art-detail',
-    templateUrl: './art-detail.page.html',
-    styleUrls: ['./art-detail.page.scss'],
-    standalone: true,
-    schemas: [CUSTOM_ELEMENTS_SCHEMA],
-    imports: [
-        CommonModule,
-        FormsModule,
-        IonHeader,
-        IonToolbar,
-        IonTitle,
-        IonContent,
-        IonButton,
-        IonIcon,
-        IonBackButton,
-        IonItem,
-        IonLabel,
-        IonInput,
-        IonTextarea,
-        IonButtons
-    ]
-})
-export class ArtDetailPage implements OnInit {
-    artDetail: any = {};
-    comments: any[] = [];
-    newComment: string = '';
-    isLiked: boolean = false;
-    likeCount: number = 0;
-
-    constructor(private route: ActivatedRoute) { }
-
-    ngOnInit() {
-        const id = this.route.snapshot.paramMap.get('id');
-        if (id) {
-            this.loadArtDetail(id);
-            this.loadComments(id);
-        }
-    }
-
-    async loadArtDetail(id: string) {
-        const query = new CloudQuery('ArtWork');
-        const result = await query.get(id);
-        this.artDetail = result;
-        this.likeCount = result['get']('likes') || 0;
-        this.checkLikeStatus();
-    }
-
-    async loadComments(artId: string) {
-        const query = new CloudQuery('Comments');
-        query.equalTo('artId', artId);
-        const results = await query.find();
-        this.comments = results || [];
-    }
-
-    async toggleLike() {
-        if (!this.isLiked) {
-            this.likeCount++;
-            this.isLiked = true;
-            this.artDetail.set('likes', this.likeCount);
-            await this.artDetail.save();
-        }
-    }
-
-    async addComment() {
-        if (!this.newComment.trim()) return;
-
-        const comment = new CloudObject('Comments');
-        // comment.set('artId', this.artDetail.id);
-        // comment.set('content', this.newComment);
-        // comment.set('createdAt', new Date());
-        await comment.save();
-
-        await this.loadComments(this.artDetail.id);
-        this.newComment = '';
-    }
-
-    private async checkLikeStatus() {
-        // 这里可以添加检查用户是否已点赞的逻辑
-        // 可以使用 localStorage 或者数据库来存储用户的点赞状态
-    }
-} 

+ 76 - 0
AIart-app/src/app/attendance/attendance.component.html

@@ -0,0 +1,76 @@
+<ion-header class="ion-no-border">
+  <ion-toolbar>
+    <ion-buttons slot="start">
+      <ion-back-button default-href="/tabs/tab1" text=""></ion-back-button>
+    </ion-buttons>
+    <ion-title>每日签到</ion-title>
+  </ion-toolbar>
+</ion-header>
+
+<ion-content>
+  <!-- 签到卡片 -->
+  <div class="sign-in-card">
+    <div class="date-info">
+      <h2>{{ formatDate(currentDate) }}</h2>
+      <div class="streak-info">
+        <ion-icon name="flame"></ion-icon>
+        <span>已连续签到 {{ currentStreak }} 天</span>
+      </div>
+    </div>
+
+    <div class="sign-in-button" [class.signed]="todaySigned" (click)="signIn()">
+      <ion-icon [name]="todaySigned ? 'checkmark-circle' : 'hand-right'"></ion-icon>
+      <span>{{ todaySigned ? '已签到' : '立即签到' }}</span>
+    </div>
+  </div>
+
+  <!-- 签到统计 -->
+  <div class="stats-card">
+    <div class="stat-item">
+      <div class="stat-value">{{ currentStreak }}</div>
+      <div class="stat-label">连续签到</div>
+    </div>
+    <div class="stat-divider"></div>
+    <div class="stat-item">
+      <div class="stat-value">{{ totalDays }}</div>
+      <div class="stat-label">总签到天数</div>
+    </div>
+  </div>
+
+  <!-- 奖励列表 -->
+  <div class="rewards-section">
+    <h3>签到奖励</h3>
+    <div class="rewards-grid">
+      @for(reward of rewards; track reward.days) {
+      <div class="reward-item" [class.claimed]="reward.claimed"
+        [class.available]="!reward.claimed && currentStreak >= reward.days" (click)="claimReward(reward)">
+        <div class="day-count">{{ reward.days }}天</div>
+        <div class="coin-amount">
+          <ion-icon name="medal"></ion-icon>
+          {{ reward.coins }}
+        </div>
+        <div class="claim-status">
+          @if(reward.claimed) {
+          <ion-icon name="checkmark-circle"></ion-icon>
+          } @else if(currentStreak >= reward.days) {
+          <span>可领取</span>
+          } @else {
+          <span>未达成</span>
+          }
+        </div>
+      </div>
+      }
+    </div>
+  </div>
+
+  <!-- 签到规则 -->
+  <div class="rules-section">
+    <h3>签到规则</h3>
+    <div class="rules-content">
+      <p>1. 每日签到可获得5个积分</p>
+      <p>2. 连续签到可获得额外奖励</p>
+      <p>3. 断签将重置连续签到天数</p>
+      <p>4. 积分可用于兑换精美礼品</p>
+    </div>
+  </div>
+</ion-content>

+ 224 - 0
AIart-app/src/app/attendance/attendance.component.scss

@@ -0,0 +1,224 @@
+:host {
+    --primary-color: var(--ion-color-primary);
+    --background-color: #f5f5f5;
+}
+
+ion-header {
+    ion-toolbar {
+        --background: transparent;
+
+        ion-title {
+            font-size: 18px;
+            font-weight: 600;
+        }
+    }
+}
+
+.sign-in-card {
+    background: white;
+    margin: 16px;
+    padding: 20px;
+    border-radius: 16px;
+    box-shadow: 0 2px 12px rgba(0, 0, 0, 0.1);
+
+    .date-info {
+        text-align: center;
+        margin-bottom: 20px;
+
+        h2 {
+            margin: 0;
+            font-size: 20px;
+            color: #333;
+        }
+
+        .streak-info {
+            display: flex;
+            align-items: center;
+            justify-content: center;
+            gap: 8px;
+            margin-top: 8px;
+            color: var(--primary-color);
+
+            ion-icon {
+                font-size: 20px;
+            }
+        }
+    }
+
+    .sign-in-button {
+        background: var(--primary-color);
+        color: white;
+        padding: 16px;
+        border-radius: 8px;
+        display: flex;
+        align-items: center;
+        justify-content: center;
+        gap: 8px;
+        cursor: pointer;
+        transition: all 0.3s ease;
+
+        &.signed {
+            background: #8bc34a;
+        }
+
+        ion-icon {
+            font-size: 24px;
+        }
+
+        span {
+            font-size: 16px;
+            font-weight: 500;
+        }
+    }
+}
+
+.stats-card {
+    background: white;
+    margin: 16px;
+    padding: 20px;
+    border-radius: 16px;
+    display: flex;
+    justify-content: space-around;
+    align-items: center;
+    box-shadow: 0 2px 12px rgba(0, 0, 0, 0.1);
+
+    .stat-item {
+        text-align: center;
+
+        .stat-value {
+            font-size: 24px;
+            font-weight: 600;
+            color: var(--primary-color);
+        }
+
+        .stat-label {
+            font-size: 14px;
+            color: #666;
+            margin-top: 4px;
+        }
+    }
+
+    .stat-divider {
+        width: 1px;
+        height: 40px;
+        background: #eee;
+    }
+}
+
+.rewards-section {
+    background: white;
+    margin: 16px;
+    padding: 20px;
+    border-radius: 16px;
+    box-shadow: 0 2px 12px rgba(0, 0, 0, 0.1);
+
+    h3 {
+        margin: 0 0 16px;
+        font-size: 18px;
+        color: #333;
+    }
+
+    .rewards-grid {
+        display: grid;
+        grid-template-columns: repeat(auto-fill, minmax(140px, 1fr));
+        gap: 12px;
+
+        .reward-item {
+            background: #f5f5f5;
+            padding: 16px;
+            border-radius: 8px;
+            text-align: center;
+            transition: all 0.3s ease;
+
+            &.claimed {
+                background: #e8f5e9;
+
+                .claim-status {
+                    color: #4caf50;
+                }
+            }
+
+            &.available {
+                background: #e3f2fd;
+                cursor: pointer;
+
+                &:hover {
+                    transform: translateY(-2px);
+                }
+            }
+
+            .day-count {
+                font-size: 16px;
+                font-weight: 500;
+                color: #333;
+            }
+
+            .coin-amount {
+                display: flex;
+                align-items: center;
+                justify-content: center;
+                gap: 4px;
+                margin: 8px 0;
+                color: var(--primary-color);
+                font-size: 20px;
+                font-weight: 600;
+
+                ion-icon {
+                    font-size: 24px;
+                }
+            }
+
+            .claim-status {
+                font-size: 14px;
+                color: #666;
+
+                ion-icon {
+                    font-size: 18px;
+                }
+            }
+        }
+    }
+}
+
+.rules-section {
+    background: white;
+    margin: 16px;
+    padding: 20px;
+    border-radius: 16px;
+    box-shadow: 0 2px 12px rgba(0, 0, 0, 0.1);
+
+    h3 {
+        margin: 0 0 16px;
+        font-size: 18px;
+        color: #333;
+    }
+
+    .rules-content {
+        p {
+            margin: 8px 0;
+            font-size: 14px;
+            color: #666;
+            line-height: 1.5;
+        }
+    }
+}
+
+// 添加动画效果
+@keyframes fadeInUp {
+    from {
+        opacity: 0;
+        transform: translateY(20px);
+    }
+
+    to {
+        opacity: 1;
+        transform: translateY(0);
+    }
+}
+
+.sign-in-card,
+.stats-card,
+.rewards-section,
+.rules-section {
+    animation: fadeInUp 0.5s ease-out;
+}

+ 22 - 0
AIart-app/src/app/attendance/attendance.component.spec.ts

@@ -0,0 +1,22 @@
+import { ComponentFixture, TestBed, waitForAsync } from '@angular/core/testing';
+
+import { AttendanceComponent } from './attendance.component';
+
+describe('AttendanceComponent', () => {
+  let component: AttendanceComponent;
+  let fixture: ComponentFixture<AttendanceComponent>;
+
+  beforeEach(waitForAsync(() => {
+    TestBed.configureTestingModule({
+      imports: [AttendanceComponent],
+    }).compileComponents();
+
+    fixture = TestBed.createComponent(AttendanceComponent);
+    component = fixture.componentInstance;
+    fixture.detectChanges();
+  }));
+
+  it('should create', () => {
+    expect(component).toBeTruthy();
+  });
+});

+ 64 - 0
AIart-app/src/app/attendance/attendance.component.ts

@@ -0,0 +1,64 @@
+import { Component, OnInit } from '@angular/core';
+import { IonicModule } from '@ionic/angular';
+import { CommonModule } from '@angular/common';
+
+@Component({
+  selector: 'app-attendance',
+  templateUrl: './attendance.component.html',
+  styleUrls: ['./attendance.component.scss'],
+  standalone: true,
+  imports: [IonicModule, CommonModule],
+})
+export class AttendanceComponent implements OnInit {
+  currentDate = new Date();
+  currentStreak = 3; // 当前连续签到天数
+  totalDays = 15;    // 总签到天数
+  todaySigned = false; // 今日是否已签到
+  rewards = [
+    { days: 1, coins: 5, claimed: true },
+    { days: 3, coins: 15, claimed: true },
+    { days: 7, coins: 40, claimed: false },
+    { days: 15, coins: 100, claimed: false },
+    { days: 30, coins: 250, claimed: false }
+  ];
+
+  constructor() { }
+
+  ngOnInit() {
+    // 这里可以从服务器获取用户签到数据
+  }
+
+  signIn() {
+    if (!this.todaySigned) {
+      this.todaySigned = true;
+      this.currentStreak++;
+      this.totalDays++;
+      // 这里添加签到成功的提示
+      this.showToast('签到成功!获得5个积分');
+    }
+  }
+
+  async showToast(message: string) {
+    const toast = document.createElement('ion-toast');
+    toast.message = message;
+    toast.position = 'top';
+    toast.color = 'success';
+    document.body.appendChild(toast);
+    await toast.present();
+  }
+
+  claimReward(reward: any) {
+    if (!reward.claimed && this.currentStreak >= reward.days) {
+      reward.claimed = true;
+      this.showToast(`领取成功!获得${reward.coins}个积分`);
+    }
+  }
+
+  formatDate(date: Date): string {
+    return date.toLocaleDateString('zh-CN', {
+      month: 'long',
+      day: 'numeric',
+      weekday: 'long'
+    });
+  }
+}

+ 27 - 0
AIart-app/src/app/category-modal/category-modal.component.html

@@ -0,0 +1,27 @@
+<ion-header>
+    <ion-toolbar>
+        <ion-title>全部分类</ion-title>
+        <ion-buttons slot="end">
+            <ion-button (click)="dismiss()">
+                <ion-icon name="close-outline"></ion-icon>
+            </ion-button>
+        </ion-buttons>
+    </ion-toolbar>
+</ion-header>
+
+<ion-content>
+    <div class="category-container">
+        @for(category of categories; track category.title) {
+        <div class="category-section">
+            <h3 class="category-title">{{category.title}}</h3>
+            <div class="category-items">
+                @for(item of category.items; track item) {
+                <div class="category-item" (click)="selectCategory(item)">
+                    {{item}}
+                </div>
+                }
+            </div>
+        </div>
+        }
+    </div>
+</ion-content>

+ 46 - 0
AIart-app/src/app/category-modal/category-modal.component.scss

@@ -0,0 +1,46 @@
+.category-container {
+    padding: 16px;
+
+    .category-section {
+        margin-bottom: 24px;
+
+        .category-title {
+            font-size: 18px;
+            font-weight: 600;
+            color: #333;
+            margin-bottom: 16px;
+            padding-left: 8px;
+            border-left: 4px solid var(--ion-color-primary);
+        }
+
+        .category-items {
+            display: grid;
+            grid-template-columns: repeat(4, 1fr);
+            gap: 12px;
+
+            .category-item {
+                padding: 12px;
+                text-align: center;
+                background: #f5f5f5;
+                border-radius: 8px;
+                font-size: 14px;
+                color: #666;
+                cursor: pointer;
+                transition: all 0.3s ease;
+
+                &:hover {
+                    background: var(--ion-color-primary);
+                    color: white;
+                    transform: translateY(-2px);
+                }
+            }
+        }
+    }
+}
+
+// 适配小屏幕
+@media (max-width: 480px) {
+    .category-items {
+        grid-template-columns: repeat(3, 1fr) !important;
+    }
+}

+ 24 - 0
AIart-app/src/app/category-modal/category-modal.component.ts

@@ -0,0 +1,24 @@
+import { Component, Input } from '@angular/core';
+import { IonicModule, ModalController } from '@ionic/angular';
+import { CommonModule } from '@angular/common';
+
+@Component({
+    selector: 'app-category-modal',
+    templateUrl: './category-modal.component.html',
+    styleUrls: ['./category-modal.component.scss'],
+    standalone: true,
+    imports: [IonicModule, CommonModule]
+})
+export class CategoryModalComponent {
+    @Input() categories: any[] = [];
+
+    constructor(private modalCtrl: ModalController) { }
+
+    dismiss() {
+        this.modalCtrl.dismiss();
+    }
+
+    selectCategory(category: string) {
+        this.modalCtrl.dismiss(category);
+    }
+} 

+ 3 - 0
AIart-app/src/app/drawing/drawing.component.html

@@ -0,0 +1,3 @@
+<p>
+  drawing works!
+</p>

+ 132 - 0
AIart-app/src/app/drawing/drawing.component.scss

@@ -0,0 +1,132 @@
+:host {
+    --primary-color: var(--ion-color-primary);
+    --background-color: #f5f5f5;
+}
+
+ion-header {
+    ion-toolbar {
+        --background: transparent;
+
+        ion-title {
+            font-size: 18px;
+            font-weight: 600;
+        }
+    }
+}
+
+.drawing-container {
+    position: relative;
+    width: 100%;
+    height: calc(100% - 180px);
+    background: white;
+    overflow: hidden;
+
+    canvas {
+        touch-action: none;
+        background: white;
+    }
+}
+
+.tools-panel {
+    position: fixed;
+    bottom: 0;
+    left: 0;
+    right: 0;
+    background: white;
+    padding: 16px;
+    box-shadow: 0 -2px 10px rgba(0, 0, 0, 0.1);
+    z-index: 1000;
+
+    .tools-section {
+        display: flex;
+        justify-content: space-around;
+        margin-bottom: 16px;
+
+        ion-button {
+            --padding-start: 8px;
+            --padding-end: 8px;
+            height: 60px;
+            flex-direction: column;
+
+            &.selected {
+                --color: var(--primary-color);
+                position: relative;
+
+                &::after {
+                    content: '';
+                    position: absolute;
+                    bottom: 0;
+                    left: 25%;
+                    width: 50%;
+                    height: 2px;
+                    background: var(--primary-color);
+                    border-radius: 1px;
+                }
+            }
+
+            ion-icon {
+                font-size: 24px;
+                margin-bottom: 4px;
+            }
+
+            span {
+                font-size: 12px;
+            }
+        }
+    }
+
+    .color-section {
+        display: flex;
+        justify-content: center;
+        gap: 12px;
+        margin-bottom: 16px;
+        padding: 0 16px;
+
+        .color-item {
+            width: 30px;
+            height: 30px;
+            border-radius: 50%;
+            border: 2px solid #eee;
+            cursor: pointer;
+            transition: transform 0.2s ease;
+
+            &.selected {
+                transform: scale(1.2);
+                border-color: var(--primary-color);
+            }
+
+            &:hover {
+                transform: scale(1.1);
+            }
+        }
+    }
+
+    .width-section {
+        padding: 0 16px;
+
+        ion-range {
+            --bar-height: 4px;
+            --bar-background: #eee;
+            --bar-background-active: var(--primary-color);
+            --height: 40px;
+            --knob-size: 20px;
+            --knob-background: white;
+            --knob-box-shadow: 0 2px 4px rgba(0, 0, 0, 0.2);
+        }
+    }
+}
+
+// 添加动画效果
+@keyframes slideUp {
+    from {
+        transform: translateY(100%);
+    }
+
+    to {
+        transform: translateY(0);
+    }
+}
+
+.tools-panel {
+    animation: slideUp 0.3s ease-out;
+}

+ 22 - 0
AIart-app/src/app/drawing/drawing.component.spec.ts

@@ -0,0 +1,22 @@
+import { ComponentFixture, TestBed, waitForAsync } from '@angular/core/testing';
+
+import { DrawingComponent } from './drawing.component';
+
+describe('DrawingComponent', () => {
+  let component: DrawingComponent;
+  let fixture: ComponentFixture<DrawingComponent>;
+
+  beforeEach(waitForAsync(() => {
+    TestBed.configureTestingModule({
+      imports: [DrawingComponent],
+    }).compileComponents();
+
+    fixture = TestBed.createComponent(DrawingComponent);
+    component = fixture.componentInstance;
+    fixture.detectChanges();
+  }));
+
+  it('should create', () => {
+    expect(component).toBeTruthy();
+  });
+});

+ 306 - 0
AIart-app/src/app/drawing/drawing.component.ts

@@ -0,0 +1,306 @@
+import { Component, OnInit, ViewChild, ElementRef, AfterViewInit } from '@angular/core';
+import { IonicModule } from '@ionic/angular';
+import { CommonModule } from '@angular/common';
+import { FormsModule } from '@angular/forms';
+
+@Component({
+  selector: 'app-drawing',
+  templateUrl: './drawing.component.html',
+  styleUrls: ['./drawing.component.scss'],
+  standalone: true,
+  imports: [IonicModule, CommonModule, FormsModule],
+})
+export class DrawingComponent implements OnInit, AfterViewInit {
+  @ViewChild('canvas') canvasRef!: ElementRef<HTMLCanvasElement>;
+  private ctx!: CanvasRenderingContext2D;
+  private isDrawing = false;
+  private lastX = 0;
+  private lastY = 0;
+
+  currentColor = '#000000';
+  currentWidth = 5;
+
+  customColor = '#000000';
+  showGrid = false;
+  opacity = 1;
+  smoothing = true;
+  showLayers = false;
+  currentLayer = 'layer1';
+
+  layers = [
+    { id: 'layer1', name: '图层 1', visible: true },
+    { id: 'layer2', name: '图层 2', visible: true }
+  ];
+
+  history: ImageData[] = [];
+  historyIndex = -1;
+
+  tools = [
+    { id: 'brush', icon: 'brush', name: '画笔' },
+    { id: 'eraser', icon: 'trash', name: '橡皮擦' },
+    { id: 'line', icon: 'remove', name: '直线' },
+    { id: 'rect', icon: 'square-outline', name: '矩形' },
+    { id: 'circle', icon: 'ellipse-outline', name: '圆形' },
+    { id: 'clear', icon: 'refresh', name: '清空' }
+  ];
+
+  colors = [
+    '#000000', '#ffffff', '#ff0000', '#00ff00', '#0000ff',
+    '#ffff00', '#00ffff', '#ff00ff', '#808080', '#800000'
+  ];
+
+  selectedTool = 'brush';
+
+  constructor() { }
+
+  ngOnInit() { }
+
+  ngAfterViewInit() {
+    const canvas = this.canvasRef.nativeElement;
+    this.ctx = canvas.getContext('2d')!;
+    this.resizeCanvas();
+    this.initializeCanvas();
+
+    window.addEventListener('resize', () => {
+      this.resizeCanvas();
+    });
+  }
+
+  private resizeCanvas() {
+    const canvas = this.canvasRef.nativeElement;
+    canvas.width = window.innerWidth;
+    canvas.height = window.innerHeight - 200; // 减去顶部工具栏的高度
+  }
+
+  private initializeCanvas() {
+    this.ctx.fillStyle = '#ffffff';
+    this.ctx.fillRect(0, 0, this.canvasRef.nativeElement.width, this.canvasRef.nativeElement.height);
+  }
+
+  startDrawing(event: TouchEvent | MouseEvent) {
+    this.isDrawing = true;
+    const pos = this.getPosition(event);
+    this.lastX = pos.x;
+    this.lastY = pos.y;
+  }
+
+  draw(event: TouchEvent | MouseEvent) {
+    if (!this.isDrawing) return;
+    event.preventDefault();
+
+    const pos = this.getPosition(event);
+
+    switch (this.selectedTool) {
+      case 'brush':
+      case 'eraser':
+        this.drawFreehand(pos);
+        break;
+      case 'line':
+        this.drawLine(pos);
+        break;
+      case 'rect':
+        this.drawRect(pos);
+        break;
+      case 'circle':
+        this.drawCircle(pos);
+        break;
+    }
+  }
+
+  private drawFreehand(pos: { x: number, y: number }) {
+    this.ctx.beginPath();
+    this.ctx.moveTo(this.lastX, this.lastY);
+    this.ctx.lineTo(pos.x, pos.y);
+    this.ctx.strokeStyle = this.selectedTool === 'eraser' ? '#ffffff' : this.currentColor;
+    this.ctx.lineWidth = this.currentWidth;
+    this.ctx.lineCap = 'round';
+    this.ctx.stroke();
+
+    this.lastX = pos.x;
+    this.lastY = pos.y;
+  }
+
+  private drawLine(pos: { x: number, y: number }) {
+    const tempCanvas = document.createElement('canvas');
+    tempCanvas.width = this.canvasRef.nativeElement.width;
+    tempCanvas.height = this.canvasRef.nativeElement.height;
+    const tempCtx = tempCanvas.getContext('2d')!;
+
+    tempCtx.drawImage(this.canvasRef.nativeElement, 0, 0);
+    tempCtx.beginPath();
+    tempCtx.moveTo(this.lastX, this.lastY);
+    tempCtx.lineTo(pos.x, pos.y);
+    tempCtx.strokeStyle = this.currentColor;
+    tempCtx.lineWidth = this.currentWidth;
+    tempCtx.stroke();
+
+    this.ctx.clearRect(0, 0, this.canvasRef.nativeElement.width, this.canvasRef.nativeElement.height);
+    this.ctx.drawImage(tempCanvas, 0, 0);
+  }
+
+  private drawRect(pos: { x: number, y: number }) {
+    const tempCanvas = document.createElement('canvas');
+    tempCanvas.width = this.canvasRef.nativeElement.width;
+    tempCanvas.height = this.canvasRef.nativeElement.height;
+    const tempCtx = tempCanvas.getContext('2d')!;
+
+    tempCtx.drawImage(this.canvasRef.nativeElement, 0, 0);
+    tempCtx.beginPath();
+    tempCtx.rect(this.lastX, this.lastY, pos.x - this.lastX, pos.y - this.lastY);
+    tempCtx.strokeStyle = this.currentColor;
+    tempCtx.lineWidth = this.currentWidth;
+    tempCtx.stroke();
+
+    this.ctx.clearRect(0, 0, this.canvasRef.nativeElement.width, this.canvasRef.nativeElement.height);
+    this.ctx.drawImage(tempCanvas, 0, 0);
+  }
+
+  private drawCircle(pos: { x: number, y: number }) {
+    const tempCanvas = document.createElement('canvas');
+    tempCanvas.width = this.canvasRef.nativeElement.width;
+    tempCanvas.height = this.canvasRef.nativeElement.height;
+    const tempCtx = tempCanvas.getContext('2d')!;
+
+    const radius = Math.sqrt(
+      Math.pow(pos.x - this.lastX, 2) + Math.pow(pos.y - this.lastY, 2)
+    );
+
+    tempCtx.drawImage(this.canvasRef.nativeElement, 0, 0);
+    tempCtx.beginPath();
+    tempCtx.arc(this.lastX, this.lastY, radius, 0, Math.PI * 2);
+    tempCtx.strokeStyle = this.currentColor;
+    tempCtx.lineWidth = this.currentWidth;
+    tempCtx.stroke();
+
+    this.ctx.clearRect(0, 0, this.canvasRef.nativeElement.width, this.canvasRef.nativeElement.height);
+    this.ctx.drawImage(tempCanvas, 0, 0);
+  }
+
+  stopDrawing() {
+    if (this.isDrawing) {
+      this.isDrawing = false;
+      this.saveToHistory();
+    }
+  }
+
+  private getPosition(event: TouchEvent | MouseEvent) {
+    const canvas = this.canvasRef.nativeElement;
+    const rect = canvas.getBoundingClientRect();
+
+    let x, y;
+    if (event instanceof TouchEvent) {
+      x = event.touches[0].clientX - rect.left;
+      y = event.touches[0].clientY - rect.top;
+    } else {
+      x = event.clientX - rect.left;
+      y = event.clientY - rect.top;
+    }
+
+    return { x, y };
+  }
+
+  selectTool(toolId: string) {
+    this.selectedTool = toolId;
+    if (toolId === 'clear') {
+      this.clearCanvas();
+    }
+  }
+
+  selectColor(color: string) {
+    this.currentColor = color;
+    this.selectedTool = 'brush';
+  }
+
+  clearCanvas() {
+    this.ctx.fillStyle = '#ffffff';
+    this.ctx.fillRect(0, 0, this.canvasRef.nativeElement.width, this.canvasRef.nativeElement.height);
+  }
+
+  async saveDrawing() {
+    const canvas = this.canvasRef.nativeElement;
+    try {
+      const dataUrl = canvas.toDataURL('image/png');
+      const response = await fetch(dataUrl);
+      const blob = await response.blob();
+      const url = window.URL.createObjectURL(blob);
+      const a = document.createElement('a');
+      a.href = url;
+      a.download = `绘画_${new Date().getTime()}.png`;
+      document.body.appendChild(a);
+      a.click();
+      window.URL.revokeObjectURL(url);
+      document.body.removeChild(a);
+      this.showToast('保存成功');
+    } catch (error) {
+      this.showToast('保存失败,请重试');
+    }
+  }
+
+  private async showToast(message: string) {
+    const toast = document.createElement('ion-toast');
+    toast.message = message;
+    toast.duration = 2000;
+    toast.position = 'top';
+    document.body.appendChild(toast);
+    await toast.present();
+  }
+
+  undo() {
+    if (this.historyIndex > 0) {
+      this.historyIndex--;
+      const imageData = this.history[this.historyIndex];
+      this.ctx.putImageData(imageData, 0, 0);
+    }
+  }
+
+  redo() {
+    if (this.historyIndex < this.history.length - 1) {
+      this.historyIndex++;
+      const imageData = this.history[this.historyIndex];
+      this.ctx.putImageData(imageData, 0, 0);
+    }
+  }
+
+  saveToHistory() {
+    this.historyIndex++;
+    this.history = this.history.slice(0, this.historyIndex);
+    this.history.push(this.ctx.getImageData(0, 0, this.canvasRef.nativeElement.width, this.canvasRef.nativeElement.height));
+  }
+
+  toggleGrid() {
+    this.showGrid = !this.showGrid;
+  }
+
+  toggleOpacity() {
+    this.opacity = this.opacity === 1 ? 0.5 : 1;
+    this.ctx.globalAlpha = this.opacity;
+  }
+
+  toggleSmooth() {
+    this.smoothing = !this.smoothing;
+    this.ctx.imageSmoothingEnabled = this.smoothing;
+  }
+
+  toggleLayers() {
+    this.showLayers = !this.showLayers;
+  }
+
+  addLayer() {
+    const newId = `layer${this.layers.length + 1}`;
+    this.layers.push({
+      id: newId,
+      name: `图层 ${this.layers.length + 1}`,
+      visible: true
+    });
+    this.currentLayer = newId;
+  }
+
+  deleteLayer(layerId: string) {
+    if (this.layers.length > 1) {
+      this.layers = this.layers.filter(layer => layer.id !== layerId);
+      if (this.currentLayer === layerId) {
+        this.currentLayer = this.layers[0].id;
+      }
+    }
+  }
+}

+ 20 - 5
AIart-app/src/app/edit-tag/edit-tag.component.html

@@ -1,5 +1,20 @@
-<ion-input [value]="textResult" (ionInput)="userInput($event)" type="text" placeholder="请输入内容"></ion-input>
-<p>当前输入:{{textResult}}</p> <ion-button (click)="Add_tags()">添加标签</ion-button>
-@for (tag of user_tag;track tag) {
-<ion-chip>{{tag}}<span (click)="Delete_tag(tag)">&nbsp;x</span></ion-chip>
-}
+<div class="tag-input-container">
+    <ion-input [value]="textResult" (ionInput)="userInput($event)" type="text" placeholder="输入标签内容..."
+        class="tag-input">
+        <ion-icon name="pricetag-outline" slot="start"></ion-icon>
+    </ion-input>
+
+    <ion-button (click)="Add_tags()" class="add-button" [disabled]="!textResult">
+        <ion-icon name="add-outline" slot="start"></ion-icon>
+        添加标签
+    </ion-button>
+</div>
+
+<!-- <div class="tags-container">
+    @for (tag of user_tag; track tag) {
+    <ion-chip class="custom-chip">
+        <ion-label>{{tag}}</ion-label>
+        <ion-icon name="close-circle" (click)="Delete_tag(tag)"></ion-icon>
+    </ion-chip>
+    }
+</div> -->

+ 86 - 0
AIart-app/src/app/edit-tag/edit-tag.component.scss

@@ -0,0 +1,86 @@
+.tag-input-container {
+    display: flex;
+    gap: 12px;
+    margin-bottom: 16px;
+
+    .tag-input {
+        flex: 1;
+        --padding-start: 12px;
+        --padding-end: 12px;
+        --background: var(--ion-color-light);
+        --border-radius: 8px;
+        --placeholder-color: #999;
+        --placeholder-opacity: 0.8;
+
+        &::part(native) {
+            padding: 8px 12px;
+        }
+
+        ion-icon {
+            color: var(--ion-color-medium);
+            margin-right: 8px;
+        }
+    }
+
+    .add-button {
+        --padding-start: 16px;
+        --padding-end: 16px;
+        --border-radius: 8px;
+
+        ion-icon {
+            margin-right: 4px;
+        }
+    }
+}
+
+.tags-container {
+    display: flex;
+    flex-wrap: wrap;
+    gap: 8px;
+    min-height: 40px;
+
+    .custom-chip {
+        --background: var(--ion-color-light);
+        --color: var(--ion-color-dark);
+        padding: 8px 12px;
+        border-radius: 16px;
+        transition: all 0.2s ease;
+
+        ion-label {
+            margin-right: 4px;
+            font-size: 14px;
+        }
+
+        ion-icon {
+            font-size: 16px;
+            color: var(--ion-color-medium);
+            cursor: pointer;
+            transition: color 0.2s ease;
+
+            &:hover {
+                color: var(--ion-color-danger);
+            }
+        }
+
+        &:hover {
+            --background: var(--ion-color-light-shade);
+        }
+    }
+}
+
+// 添加动画效果
+@keyframes fadeIn {
+    from {
+        opacity: 0;
+        transform: translateY(10px);
+    }
+
+    to {
+        opacity: 1;
+        transform: translateY(0);
+    }
+}
+
+ion-chip {
+    animation: fadeIn 0.3s ease-out;
+}

+ 21 - 12
AIart-app/src/app/edit-tag/edit-tag.component.ts

@@ -1,30 +1,39 @@
-import { Component, input, OnInit } from '@angular/core';
-import { IonInput, IonButton, IonChip } from '@ionic/angular/standalone';
+import { Component, OnInit, Output, EventEmitter } from '@angular/core';
+import { IonInput, IonButton, IonChip, IonIcon, IonLabel } from '@ionic/angular/standalone';
+
 @Component({
   selector: 'edit-tag',
   templateUrl: './edit-tag.component.html',
   styleUrls: ['./edit-tag.component.scss'],
   standalone: true,
-  imports: [IonInput, IonButton, IonChip],
+  imports: [IonInput, IonButton, IonChip, IonIcon, IonLabel],
 })
 export class EditTagComponent implements OnInit {
+  @Output() onTagAdd = new EventEmitter<string>();
+
+  user_tag: string[] = [];
+  textResult: string = "";
 
-  user_tag: Array<string> = []
-  textResult: string = ""
   userInput(ev: any) {
-    console.log(ev)
-    this.textResult = ev.detail.value
+    this.textResult = ev.detail.value;
   }
+
   Add_tags() {
-    this.user_tag.push(this.textResult)
-    this.textResult = ""
+    if (this.textResult.trim()) {
+      this.user_tag.push(this.textResult.trim());
+      this.onTagAdd.emit(this.textResult.trim());
+      this.textResult = "";
+    }
   }
+
   Delete_tag(tag: string) {
-    let idx = this.user_tag.findIndex(item => item == this.textResult)
-    this.user_tag.splice(idx, 1)
+    const index = this.user_tag.indexOf(tag);
+    if (index > -1) {
+      this.user_tag.splice(index, 1);
+    }
   }
+
   constructor() { }
 
   ngOnInit() { }
-
 }

+ 93 - 33
AIart-app/src/app/interest-picture/interest-picture.component.html

@@ -1,39 +1,99 @@
-<ion-header [translucent]="true">
+<ion-header class="ion-no-border">
   <ion-toolbar>
-    <ion-title>
-      意境呈现{{imagineWork?.progress || 0}}
-    </ion-title>
+    <ion-buttons slot="start">
+      <ion-back-button defaultHref="/tabs/tab1" text=""></ion-back-button>
+    </ion-buttons>
+    <ion-title>意境呈现</ion-title>
+    @if(imagineWork?.progress !== undefined) {
+    <ion-progress-bar [value]="(imagineWork?.progress || 0) / 100"></ion-progress-bar>
+    }
   </ion-toolbar>
 </ion-header>
 
-<ion-content [fullscreen]="true">
-  <div style="background-color: #fcfafafa;margin: 15px;border-radius: 10px;padding-top: 10px;">
-    <!-- 生成提示词 -->
-    <span style="font-weight: bold;font-size: 20px;margin-left: 10px;">logo设想</span>
-    <ion-textarea [value]="userPrompt" (ionInput)="promptInput($event)" placeholder="logo创作" autoGrow="true"
-      style="margin-left: 10px;text-decoration: underline;opacity: 0.5;"></ion-textarea>
-  </div>
-  <div style="display: flex;align-items: center;justify-content: center;margin-bottom: 10px;">
-    <ion-button (click)="createImage()" expand="block" style="width: 90%;">logo创作</ion-button>
-  </div>
-  <!-- 意境画面提示词 -->
-  <div>
-    {{PictureDescResult}}
-  </div>
-  <!-- 生成结果 -->
-  @if(images.length) {
-  @for(imageUrl of images;track imageUrl){
-  <img [src]="imageUrl" alt="" srcset="">
-  }
-  }
-  <!-- 生成状态 -->
-  @if(!images.length){
-  @if(imagineWork){
-  <h1>生成中</h1>
-  }
-  @if(!imagineWork){
-  <h1>未开始</h1>
-  }
-  }
+<ion-content>
+  <div class="creation-container">
+    <!-- 创作输入区域 -->
+    <div class="input-section">
+      <h2>创作灵感</h2>
+      <p class="subtitle">请描述您想要创作的画面意境</p>
+
+      <ion-textarea [value]="userPrompt" (ionInput)="promptInput($event)"
+        placeholder="例如:一个充满科技感和艺术气息的场景,展现AI与艺术的完美融合..." autoGrow="true" class="prompt-input" rows="4"></ion-textarea>
+    </div>
 
+    <!-- 创作按钮 -->
+    <div class="action-section">
+      <ion-button (click)="createImage()" expand="block" class="create-button">
+        <ion-icon name="color-wand-outline" slot="start"></ion-icon>
+        开始创作
+      </ion-button>
+    </div>
+
+    <!-- AI解读结果 -->
+    @if(PictureDescResult) {
+    <div class="interpretation-section">
+      <h3>AI解读</h3>
+      <p class="interpretation-text">{{PictureDescResult}}</p>
+    </div>
+    }
+
+    <!-- 生成结果展示 -->
+    <div class="results-section">
+      @if(images.length) {
+      <div class="image-grid">
+        @for(imageUrl of images; track imageUrl) {
+        <div class="image-card">
+          <div class="image-wrapper">
+            <img [src]="imageUrl" alt="AI创作图片">
+            <div class="image-overlay">
+              <div class="overlay-content">
+                <ion-button fill="clear" class="action-button" (click)="downloadImage(imageUrl)">
+                  <ion-icon name="download-outline"></ion-icon>
+                  <span>下载</span>
+                </ion-button>
+                <ion-button fill="clear" class="action-button" (click)="shareImage(imageUrl)">
+                  <ion-icon name="share-social-outline"></ion-icon>
+                  <span>分享</span>
+                </ion-button>
+              </div>
+            </div>
+          </div>
+          <div class="image-info">
+            <div class="info-text">
+              <span class="creation-time">创作于 {{getCurrentTime()}}</span>
+              <span class="resolution">1024 x 1024</span>
+            </div>
+            <div class="action-buttons">
+              <ion-button fill="clear" size="small" (click)="likeImage(imageUrl)">
+                <ion-icon [name]="isImageLiked(imageUrl) ? 'heart' : 'heart-outline'"
+                  [color]="isImageLiked(imageUrl) ? 'danger' : 'medium'">
+                </ion-icon>
+              </ion-button>
+              <ion-button fill="clear" size="small" (click)="saveToCollection(imageUrl)">
+                <ion-icon name="bookmark-outline"></ion-icon>
+              </ion-button>
+            </div>
+          </div>
+        </div>
+        }
+      </div>
+      } @else {
+      <div class="status-display">
+        @if(imagineWork) {
+        <div class="generating">
+          <ion-spinner name="crescent"></ion-spinner>
+          <p>AI正在创作中...</p>
+          <p class="generating-tip">这可能需要一点时间,请耐心等待</p>
+        </div>
+        } @else {
+        <div class="empty-state">
+          <ion-icon name="image-outline"></ion-icon>
+          <p>等待您的创作灵感</p>
+          <p class="empty-tip">描述您的想法,AI将为您呈现独特的艺术作品</p>
+        </div>
+        }
+      </div>
+      }
+    </div>
+  </div>
 </ion-content>

+ 285 - 0
AIart-app/src/app/interest-picture/interest-picture.component.scss

@@ -0,0 +1,285 @@
+:host {
+    --primary-color: var(--ion-color-primary);
+    --background-color: #f5f5f5;
+}
+
+ion-header {
+    ion-toolbar {
+        --background: transparent;
+
+        ion-title {
+            font-size: 18px;
+            font-weight: 600;
+        }
+
+        ion-progress-bar {
+            --progress-background: var(--primary-color);
+        }
+    }
+}
+
+.creation-container {
+    padding: 16px;
+
+    // 输入区域样式
+    .input-section {
+        background: white;
+        border-radius: 16px;
+        padding: 20px;
+        margin-bottom: 20px;
+        box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
+
+        h2 {
+            margin: 0 0 8px;
+            font-size: 20px;
+            font-weight: 600;
+            color: #333;
+        }
+
+        .subtitle {
+            color: #666;
+            font-size: 14px;
+            margin-bottom: 16px;
+        }
+
+        .prompt-input {
+            --background: #f5f5f5;
+            --padding-start: 16px;
+            --padding-end: 16px;
+            --padding-top: 12px;
+            --padding-bottom: 12px;
+            border-radius: 8px;
+            margin-top: 8px;
+
+            &::part(native) {
+                color: #333;
+                font-size: 15px;
+            }
+        }
+    }
+
+    // 按钮样式
+    .action-section {
+        margin-bottom: 24px;
+
+        .create-button {
+            --border-radius: 8px;
+            --box-shadow: none;
+            margin: 0;
+            height: 48px;
+
+            ion-icon {
+                margin-right: 8px;
+            }
+        }
+    }
+
+    // AI解读区域
+    .interpretation-section {
+        background: white;
+        border-radius: 16px;
+        padding: 20px;
+        margin-bottom: 20px;
+        box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
+
+        h3 {
+            margin: 0 0 12px;
+            font-size: 16px;
+            font-weight: 600;
+            color: #333;
+        }
+
+        .interpretation-text {
+            color: #666;
+            font-size: 14px;
+            line-height: 1.6;
+            margin: 0;
+        }
+    }
+
+    // 结果展示区域
+    .results-section {
+        .image-grid {
+            display: grid;
+            grid-template-columns: repeat(auto-fill, minmax(300px, 1fr));
+            gap: 24px;
+            padding: 16px;
+
+            .image-card {
+                background: white;
+                border-radius: 16px;
+                overflow: hidden;
+                box-shadow: 0 4px 12px rgba(0, 0, 0, 0.08);
+                transition: transform 0.3s ease;
+
+                &:hover {
+                    transform: translateY(-4px);
+                    box-shadow: 0 8px 24px rgba(0, 0, 0, 0.12);
+
+                    .image-overlay {
+                        opacity: 1;
+                    }
+                }
+
+                .image-wrapper {
+                    position: relative;
+                    width: 100%;
+                    padding-bottom: 100%; // 1:1 aspect ratio
+                    overflow: hidden;
+
+                    img {
+                        position: absolute;
+                        top: 0;
+                        left: 0;
+                        width: 100%;
+                        height: 100%;
+                        object-fit: cover;
+                        transition: transform 0.3s ease;
+                    }
+
+                    .image-overlay {
+                        position: absolute;
+                        top: 0;
+                        left: 0;
+                        right: 0;
+                        bottom: 0;
+                        background: rgba(0, 0, 0, 0.5);
+                        display: flex;
+                        align-items: center;
+                        justify-content: center;
+                        opacity: 0;
+                        transition: opacity 0.3s ease;
+
+                        .overlay-content {
+                            display: flex;
+                            gap: 16px;
+
+                            .action-button {
+                                --color: white;
+                                font-size: 14px;
+
+                                ion-icon {
+                                    font-size: 24px;
+                                    margin-bottom: 4px;
+                                }
+
+                                span {
+                                    display: block;
+                                    text-align: center;
+                                }
+                            }
+                        }
+                    }
+                }
+
+                .image-info {
+                    padding: 16px;
+                    display: flex;
+                    justify-content: space-between;
+                    align-items: center;
+
+                    .info-text {
+                        display: flex;
+                        flex-direction: column;
+                        gap: 4px;
+
+                        .creation-time {
+                            font-size: 14px;
+                            color: #666;
+                        }
+
+                        .resolution {
+                            font-size: 12px;
+                            color: #999;
+                        }
+                    }
+
+                    .action-buttons {
+                        display: flex;
+                        gap: 8px;
+
+                        ion-button {
+                            --padding-start: 8px;
+                            --padding-end: 8px;
+
+                            ion-icon {
+                                font-size: 20px;
+                            }
+                        }
+                    }
+                }
+            }
+        }
+
+        .status-display {
+            display: flex;
+            justify-content: center;
+            align-items: center;
+            min-height: 200px;
+
+            .generating {
+                text-align: center;
+
+                ion-spinner {
+                    width: 48px;
+                    height: 48px;
+                    --color: var(--primary-color);
+                }
+
+                p {
+                    margin-top: 16px;
+                    color: #666;
+                }
+
+                .generating-tip {
+                    color: #999;
+                    font-size: 14px;
+                    margin-top: 8px;
+                }
+            }
+
+            .empty-state {
+                text-align: center;
+                color: #999;
+
+                ion-icon {
+                    font-size: 48px;
+                    margin-bottom: 12px;
+                }
+
+                p {
+                    margin: 0;
+                    font-size: 14px;
+                }
+
+                .empty-tip {
+                    color: #999;
+                    font-size: 14px;
+                    margin-top: 8px;
+                    max-width: 240px;
+                    margin-left: auto;
+                    margin-right: auto;
+                }
+            }
+        }
+    }
+}
+
+// 添加动画效果
+@keyframes fadeIn {
+    from {
+        opacity: 0;
+        transform: translateY(20px);
+    }
+
+    to {
+        opacity: 1;
+        transform: translateY(0);
+    }
+}
+
+.input-section,
+.interpretation-section,
+.image-card {
+    animation: fadeIn 0.5s ease-out;
+}

+ 141 - 43
AIart-app/src/app/interest-picture/interest-picture.component.ts

@@ -1,6 +1,6 @@
 import { Component, OnInit } from '@angular/core';
-import { IonHeader, IonToolbar, IonTitle, IonContent } from '@ionic/angular/standalone';
-import { IonTextarea, IonButton, IonInput } from "@ionic/angular/standalone";
+import { IonHeader, IonToolbar, IonTitle, IonContent, IonProgressBar, IonButtons, IonBackButton, IonSpinner, IonIcon, IonButton } from '@ionic/angular/standalone';
+import { IonTextarea } from "@ionic/angular/standalone";
 import { DalleOptions, FmodeChatCompletion, ImagineWork } from 'fmode-ng';
 
 @Component({
@@ -10,61 +10,159 @@ import { DalleOptions, FmodeChatCompletion, ImagineWork } from 'fmode-ng';
   standalone: true,
   imports: [
     IonHeader, IonToolbar, IonTitle, IonContent,
-    IonButton,
-    IonInput,
-    IonTextarea
+    IonProgressBar, IonButtons, IonBackButton,
+    IonButton, IonTextarea, IonSpinner, IonIcon
   ],
 })
 export class InterestPictureComponent implements OnInit {
-  ngOnInit(): void {
+  userPrompt: string = "请描述您想要创作的画面意境";
+  PictureDescResult: string = '';
+  imagineWork: ImagineWork | undefined;
+  images: Array<string> = [];
+  likedImages: Set<string> = new Set();
 
+  constructor() {
+    this.initializeWork();
   }
-  userPrompt: string = "请输入项目的简要内容"
+
+  ngOnInit(): void { }
+
+  private initializeWork() {
+    // 初始化或重置工作状态
+    this.imagineWork = undefined;
+    this.images = [];
+    this.PictureDescResult = '';
+  }
+
   promptInput(ev: any) {
     this.userPrompt = ev.detail.value;
   }
-  imagineWork: ImagineWork | undefined
-  images: Array<string> = []
-  constructor(
-  ) {
-    // 示例任务,自己生成图片后请存储新的ID
-    this.imagineWork = new ImagineWork("lpJGiFwWeA");
-    this.imagineWork.fetchTask().then(work => {
-      this.images = this.imagineWork?.images || [];
-    })
-  }
 
-  PictureDescResult = ``
   async createImage() {
+    if (!this.userPrompt.trim()) {
+      this.showToast('请输入创作描述');
+      return;
+    }
+
+    this.initializeWork();
     this.imagineWork = new ImagineWork();
-    let PromptTemplate = `
-    您是一名专业的logo设计师,请您根据项目的内容,将其描述的内容等用最简短的语言表达,
-    项目内容如下:
-    ${this.userPrompt}
-    `
-    let completion = new FmodeChatCompletion([
-      { role: "system", content: "" },
-      { role: "user", content: PromptTemplate }
-    ])
-    completion.sendCompletion().subscribe((message: any) => {
-      // 打印消息体
-      console.log(message.content)
-      // 赋值消息内容给组件内属性
-      this.PictureDescResult = message.content
-      if (message?.complete) {
-        let PicturePrompt = `${this.PictureDescResult}\n根据描述的内容生成一张与项目名称“AI艺速”相关的logo图片,画面包含元素:画笔,ai字母,风。风格简约大气,画面以彩色为主色调且不带任何文字(画面越简单越好)`
-        let options: DalleOptions = { prompt: PicturePrompt }
-        this.imagineWork?.draw(options).subscribe(work => {
-          console.log("imagineWork", work?.toJSON())
-          console.log("images", work?.get("images"))
-          if (work?.get("images")?.length) {
-            this.images = work?.get("images");
-          }
-        })
+
+    const promptTemplate = `
+      您是一名专业的艺术创作顾问,请根据用户的描述,用简练的专业语言重新诠释其创作意图:
+      ${this.userPrompt}
+    `;
+
+    const completion = new FmodeChatCompletion([
+      { role: "system", content: "您是一位专业的艺术顾问,擅长理解和优化创作意图。" },
+      { role: "user", content: promptTemplate }
+    ]);
+
+    completion.sendCompletion().subscribe({
+      next: (message: any) => {
+        this.PictureDescResult = message.content;
+
+        if (message?.complete) {
+          this.generateImage();
+        }
+      },
+      error: (error) => {
+        this.showToast('创作解读失败,请重试');
+        this.initializeWork();
+      }
+    });
+  }
+
+  private generateImage() {
+    const picturePrompt = `${this.PictureDescResult}\n请根据以上描述创作一幅意境图画。要求:画面构图完整,色彩和谐,风格现代。`;
+
+    const options: DalleOptions = {
+      prompt: picturePrompt,
+      size: '1024x1024',
+    };
+
+    this.imagineWork?.draw(options).subscribe({
+      next: (work) => {
+        if (work?.get("images")?.length) {
+          this.images = work.get("images");
+        }
+      },
+      error: (error) => {
+        this.showToast('图片生成失败,请重试');
+        this.initializeWork();
       }
-    })
+    });
   }
 
+  private async showToast(message: string) {
+    const toast = document.createElement('ion-toast');
+    toast.message = message;
+    toast.duration = 2000;
+    toast.position = 'top';
+    document.body.appendChild(toast);
+    await toast.present();
+  }
+
+  getCurrentTime(): string {
+    return new Date().toLocaleString('zh-CN', {
+      month: 'numeric',
+      day: 'numeric',
+      hour: 'numeric',
+      minute: 'numeric'
+    });
+  }
+
+  async downloadImage(imageUrl: string) {
+    try {
+      const response = await fetch(imageUrl);
+      const blob = await response.blob();
+      const url = window.URL.createObjectURL(blob);
+      const a = document.createElement('a');
+      a.href = url;
+      a.download = `AI创作_${new Date().getTime()}.png`;
+      document.body.appendChild(a);
+      a.click();
+      window.URL.revokeObjectURL(url);
+      document.body.removeChild(a);
+      this.showToast('下载成功');
+    } catch (error) {
+      this.showToast('下载失败,请重试');
+    }
+  }
+
+  async shareImage(imageUrl: string) {
+    if (navigator.share) {
+      try {
+        await navigator.share({
+          title: 'AI艺术创作',
+          text: '查看我用AI创作的艺术作品',
+          url: imageUrl
+        });
+      } catch (error) {
+        this.showToast('分享失败,请重试');
+      }
+    } else {
+      // 复制图片链接
+      await navigator.clipboard.writeText(imageUrl);
+      this.showToast('图片链接已复制到剪贴板');
+    }
+  }
+
+  likeImage(imageUrl: string) {
+    if (this.likedImages.has(imageUrl)) {
+      this.likedImages.delete(imageUrl);
+    } else {
+      this.likedImages.add(imageUrl);
+    }
+  }
+
+  isImageLiked(imageUrl: string): boolean {
+    return this.likedImages.has(imageUrl);
+  }
+
+  async saveToCollection(imageUrl: string) {
+    // 这里可以添加保存到收藏的逻辑
+    this.showToast('已添加到收藏');
+  }
 }
 
 // # "AI艺速"项目策划书

+ 60 - 0
AIart-app/src/app/mailbox/mailbox.component.html

@@ -0,0 +1,60 @@
+<ion-header class="ion-no-border">
+  <ion-toolbar>
+    <ion-buttons slot="start">
+      <ion-back-button default-href="/tabs/tab1" text=""></ion-back-button>
+    </ion-buttons>
+    <ion-title>消息中心</ion-title>
+    <ion-buttons slot="end">
+      <ion-button>
+        <ion-icon name="settings-outline"></ion-icon>
+      </ion-button>
+    </ion-buttons>
+  </ion-toolbar>
+</ion-header>
+
+<ion-content>
+  <!-- 消息类型选择器 -->
+  <div class="message-types">
+    <div class="type-item active">
+      <ion-icon name="notifications-outline"></ion-icon>
+      <span>通知</span>
+    </div>
+    <div class="type-item">
+      <ion-icon name="chatbubble-outline"></ion-icon>
+      <span>私信</span>
+    </div>
+    <div class="type-item">
+      <ion-icon name="heart-outline"></ion-icon>
+      <span>点赞</span>
+    </div>
+  </div>
+
+  <!-- 空状态展示 -->
+  <div class="empty-state">
+    <div class="empty-icon">
+      <ion-icon name="mail-outline"></ion-icon>
+    </div>
+    <h2>暂无消息</h2>
+    <p>这里会显示你的通知、私信和点赞消息</p>
+    <ion-button fill="clear" class="refresh-btn">
+      <ion-icon name="refresh-outline" slot="start"></ion-icon>
+      刷新
+    </ion-button>
+  </div>
+
+  <!-- 消息列表(当有消息时显示) -->
+  <div class="message-list" style="display: none;">
+    <ion-list>
+      <ion-item>
+        <ion-avatar slot="start">
+          <img src="assets/img/avatar1.png" alt="用户头像">
+        </ion-avatar>
+        <ion-label>
+          <h2>系统通知</h2>
+          <p>欢迎使用AI艺速!开始探索你的兴趣吧</p>
+          <span class="time">刚刚</span>
+        </ion-label>
+      </ion-item>
+    </ion-list>
+  </div>
+</ion-content>

+ 171 - 0
AIart-app/src/app/mailbox/mailbox.component.scss

@@ -0,0 +1,171 @@
+// 头部样式
+ion-header {
+    ion-toolbar {
+        --background: transparent;
+
+        ion-title {
+            font-size: 18px;
+            font-weight: 600;
+        }
+
+        ion-back-button {
+            --color: var(--ion-color-dark);
+        }
+
+        ion-icon {
+            font-size: 24px;
+            color: var(--ion-color-dark);
+        }
+    }
+}
+
+// 消息类型选择器
+.message-types {
+    display: flex;
+    justify-content: space-around;
+    padding: 16px;
+    background: white;
+    margin-bottom: 8px;
+    box-shadow: 0 1px 2px rgba(0, 0, 0, 0.1);
+
+    .type-item {
+        display: flex;
+        flex-direction: column;
+        align-items: center;
+        gap: 8px;
+        padding: 8px 16px;
+        border-radius: 8px;
+        transition: all 0.3s ease;
+
+        &.active {
+            background: var(--ion-color-primary-light);
+            color: var(--ion-color-primary);
+
+            ion-icon {
+                color: var(--ion-color-primary);
+            }
+        }
+
+        ion-icon {
+            font-size: 24px;
+            color: #666;
+        }
+
+        span {
+            font-size: 14px;
+            color: #333;
+        }
+    }
+}
+
+// 空状态样式
+.empty-state {
+    display: flex;
+    flex-direction: column;
+    align-items: center;
+    justify-content: center;
+    height: calc(100vh - 200px);
+    padding: 32px;
+    text-align: center;
+
+    .empty-icon {
+        width: 80px;
+        height: 80px;
+        border-radius: 50%;
+        background: var(--ion-color-light);
+        display: flex;
+        align-items: center;
+        justify-content: center;
+        margin-bottom: 24px;
+
+        ion-icon {
+            font-size: 40px;
+            color: var(--ion-color-medium);
+        }
+    }
+
+    h2 {
+        font-size: 20px;
+        font-weight: 600;
+        color: #333;
+        margin-bottom: 8px;
+    }
+
+    p {
+        font-size: 14px;
+        color: #666;
+        margin-bottom: 24px;
+    }
+
+    .refresh-btn {
+        --color: var(--ion-color-primary);
+        font-size: 14px;
+
+        ion-icon {
+            margin-right: 4px;
+        }
+    }
+}
+
+// 消息列表样式
+.message-list {
+    ion-item {
+        --padding-start: 16px;
+        --padding-end: 16px;
+        --padding-top: 12px;
+        --padding-bottom: 12px;
+
+        ion-avatar {
+            width: 48px;
+            height: 48px;
+        }
+
+        ion-label {
+            h2 {
+                font-size: 16px;
+                font-weight: 500;
+                color: #333;
+                margin-bottom: 4px;
+            }
+
+            p {
+                font-size: 14px;
+                color: #666;
+                margin: 0;
+            }
+
+            .time {
+                font-size: 12px;
+                color: #999;
+                position: absolute;
+                top: 12px;
+                right: 16px;
+            }
+        }
+    }
+}
+
+// 添加一些动画效果
+@keyframes fadeIn {
+    from {
+        opacity: 0;
+        transform: translateY(20px);
+    }
+
+    to {
+        opacity: 1;
+        transform: translateY(0);
+    }
+}
+
+.empty-state {
+    animation: fadeIn 0.5s ease-out;
+}
+
+.type-item {
+    cursor: pointer;
+
+    &:hover {
+        background: var(--ion-color-light);
+    }
+}

+ 22 - 0
AIart-app/src/app/mailbox/mailbox.component.spec.ts

@@ -0,0 +1,22 @@
+import { ComponentFixture, TestBed, waitForAsync } from '@angular/core/testing';
+
+import { MailboxComponent } from './mailbox.component';
+
+describe('MailboxComponent', () => {
+  let component: MailboxComponent;
+  let fixture: ComponentFixture<MailboxComponent>;
+
+  beforeEach(waitForAsync(() => {
+    TestBed.configureTestingModule({
+      imports: [MailboxComponent],
+    }).compileComponents();
+
+    fixture = TestBed.createComponent(MailboxComponent);
+    component = fixture.componentInstance;
+    fixture.detectChanges();
+  }));
+
+  it('should create', () => {
+    expect(component).toBeTruthy();
+  });
+});

+ 18 - 0
AIart-app/src/app/mailbox/mailbox.component.ts

@@ -0,0 +1,18 @@
+import { Component, OnInit } from '@angular/core';
+import { IonicModule } from '@ionic/angular';
+import { CommonModule } from '@angular/common';
+
+@Component({
+  selector: 'app-mailbox',
+  templateUrl: './mailbox.component.html',
+  styleUrls: ['./mailbox.component.scss'],
+  standalone: true,
+  imports: [IonicModule, CommonModule],
+})
+export class MailboxComponent implements OnInit {
+
+  constructor() { }
+
+  ngOnInit() { }
+
+}

+ 22 - 22
AIart-app/src/app/tab1/tab1.page.html

@@ -16,10 +16,10 @@
   <div class="search-section">
     <ion-searchbar mode="ios" placeholder="搜索你感兴趣的内容" class="custom-searchbar"></ion-searchbar>
     <div class="action-buttons">
-      <ion-button fill="clear" class="icon-button">
+      <ion-button fill="clear" class="icon-button" (click)="goTomailbox()">
         <ion-icon name="mail-outline"></ion-icon>
       </ion-button>
-      <ion-button fill="clear" class="sign-in-button">
+      <ion-button fill="clear" class="sign-in-button" (click)="goToAttendance()">
         <ion-icon name="calendar-outline"></ion-icon>
         <span>签到</span>
       </ion-button>
@@ -72,46 +72,46 @@
   <!-- 功能导航区 -->
   <div class="features-grid">
     <div class="features-row">
-      <div class="feature-item">
-        <ion-icon name="storefront-outline"></ion-icon>
-        <span>商业</span>
+      <div class="feature-item" (click)="goTodrawing()">
+        <ion-icon name="brush-outline"></ion-icon>
+        <span>绘画</span>
       </div>
       <div class="feature-item">
-        <ion-icon name="business-outline"></ion-icon>
-        <span>职场</span>
+        <ion-icon name="musical-notes-outline"></ion-icon>
+        <span>音乐</span>
       </div>
       <div class="feature-item">
-        <ion-icon name="book-outline"></ion-icon>
-        <span>历史</span>
+        <ion-icon name="body-outline"></ion-icon>
+        <span>舞蹈</span>
       </div>
       <div class="feature-item">
-        <ion-icon name="heart-outline"></ion-icon>
-        <span>心理学</span>
+        <ion-icon name="camera-outline"></ion-icon>
+        <span>摄影</span>
       </div>
       <div class="feature-item">
-        <ion-icon name="football-outline"></ion-icon>
-        <span>运动</span>
+        <ion-icon name="book-outline"></ion-icon>
+        <span>文学</span>
       </div>
     </div>
 
     <div class="features-row">
       <div class="feature-item">
-        <ion-icon name="rocket-outline"></ion-icon>
-        <span>训练营</span>
+        <ion-icon name="code-outline"></ion-icon>
+        <span>编程</span>
       </div>
       <div class="feature-item">
-        <ion-icon name="documents-outline"></ion-icon>
-        <span>清单广场</span>
+        <ion-icon name="restaurant-outline"></ion-icon>
+        <span>烹饪</span>
       </div>
       <div class="feature-item">
-        <ion-icon name="sparkles-outline"></ion-icon>
-        <span>兴趣视频</span>
+        <ion-icon name="fitness-outline"></ion-icon>
+        <span>健身</span>
       </div>
       <div class="feature-item">
-        <ion-icon name="medal-outline"></ion-icon>
-        <span>会员专享</span>
+        <ion-icon name="language-outline"></ion-icon>
+        <span>外语</span>
       </div>
-      <div class="feature-item">
+      <div class="feature-item" (click)="openCategoryModal()">
         <ion-icon name="list-outline"></ion-icon>
         <span>全部分类</span>
       </div>

+ 56 - 0
AIart-app/src/app/tab1/tab1.page.ts

@@ -10,6 +10,7 @@ import { ExploreContainerComponent } from '../explore-container/explore-containe
 import { Router } from '@angular/router';
 import { CloudUser } from 'src/lib/ncloud';
 import { openUserLoginModal } from 'src/lib/user/model-user-login/model-user-login.component';
+import { CategoryModalComponent } from 'src/app/category-modal/category-modal.component';
 
 interface Slide {
   image: string;
@@ -45,6 +46,37 @@ export class Tab1Page implements OnInit, OnDestroy {
   private touchDeltaX = 0;
   private slideWidth = 0;
 
+  // 添加分类数据
+  categories = [
+    {
+      title: '艺术创作',
+      items: ['绘画', '音乐', '舞蹈', '摄影', '手工艺', '书法', '设计']
+    },
+    {
+      title: '运动健身',
+      items: ['瑜伽', '健身', '游泳', '跑步', '球类运动', '户外运动', '武术']
+    },
+    {
+      title: '文化学习',
+      items: ['文学', '历史', '哲学', '外语', '写作', '戏剧', '诗歌']
+    },
+    {
+      title: '科技创新',
+      items: ['编程', '人工智能', '机器人', '电子制作', '3D打印', '虚拟现实']
+    },
+    {
+      title: '生活技能',
+      items: ['烹饪', '园艺', '美妆', '摄影', '旅行', '茶艺', '插花']
+    },
+    {
+      title: '职业发展',
+      items: ['商业', '金融', '管理', '营销', '创业', '职场技能']
+    }
+  ];
+
+  // 添加控制分类弹窗显示的变量
+  isCategoryModalOpen = false;
+
   constructor(
     private router: Router,
     private modalCtrl: ModalController
@@ -167,5 +199,29 @@ export class Tab1Page implements OnInit, OnDestroy {
   goToInterestSearch() {
     this.checkLoginAndNavigate('/tabs/interest-search');
   }
+  goTomailbox() {
+    this.router.navigate(['/tabs/mailbox'])
+  }
+  goToAttendance() {
+    this.router.navigate(['/tabs/attendance'])
+  }
+  goTodrawing() {
+    this.router.navigate(['/tabs/drawing'])
+  }
+
+  // 添加打开分类弹窗的方法
+  async openCategoryModal() {
+    const modal = await this.modalCtrl.create({
+      component: CategoryModalComponent,
+      componentProps: {
+        categories: this.categories
+      },
+      breakpoints: [0, 0.9],
+      initialBreakpoint: 0.9,
+      cssClass: 'category-modal'
+    });
+
+    await modal.present();
+  }
 
 }

+ 8 - 7
AIart-app/src/app/tab3/tab3.page.html

@@ -12,7 +12,7 @@
   </ion-toolbar>
 </ion-header>
 
-<ion-content [fullscreen]="true">
+<ion-content [fullscreen]="true" [scrollY]="true">
   <!-- 分类导航 -->
   <div class="category-nav">
     <ion-toolbar>
@@ -44,18 +44,19 @@
     <div class="content-card" (click)="goToDetail(artwork.id)">
       <img [src]="artwork.image" [alt]="artwork.title" class="card-image">
       <div class="card-content">
-        <div class="title">{{artwork.title}} {{artwork.description}}</div>
+        <div class="title">{{artwork.title}}</div>
+        <div class="description">{{artwork.description}}</div>
         <div class="user-info">
           <div class="user">
-            <img src="../../assets/img/book1.png" alt="头像" class="avatar">
+            <img [src]="artwork.avatarUrl" [alt]="artwork.userName" class="avatar">
             <div class="info">
-              <span class="name">霸王别姬</span>
-              <span class="date">11-27</span>
+              <span class="name" style="font-size: 12px;">{{artwork.userName}}</span>
+              <span class="date">{{artwork.date}}</span>
             </div>
           </div>
           <div class="likes">
             <ion-icon name="heart-outline"></ion-icon>
-            <span>1687</span>
+            <span>{{artwork.likes}}</span>
           </div>
         </div>
       </div>
@@ -103,5 +104,5 @@
 
 <!-- 主内容区域添加 id -->
 <div class="ion-page" id="main-content">
-  <!-- 其他内容持不变 -->
+  <!-- 其他内容持不变 -->
 </div>

+ 33 - 7
AIart-app/src/app/tab3/tab3.page.ts

@@ -1,6 +1,5 @@
 import { Component } from '@angular/core';
 import { IonHeader, IonToolbar, IonTitle, IonContent, IonSearchbar, IonIcon, IonTabButton, IonTabs, IonTabBar, IonLabel, IonMenu, IonList, IonItem, IonMenuButton } from '@ionic/angular/standalone';
-import { ExploreContainerComponent } from '../explore-container/explore-container.component';
 import { Router } from '@angular/router';
 
 @Component({
@@ -8,7 +7,8 @@ import { Router } from '@angular/router';
   templateUrl: 'tab3.page.html',
   styleUrls: ['tab3.page.scss'],
   standalone: true,
-  imports: [IonHeader, IonToolbar, IonTitle, IonContent, ExploreContainerComponent,
+  imports: [
+    IonHeader, IonToolbar, IonTitle, IonContent,
     IonSearchbar, IonIcon, IonTabButton, IonTabs, IonTabBar, IonLabel,
     IonMenu, IonList, IonItem, IonMenuButton
   ],
@@ -19,25 +19,45 @@ export class Tab3Page {
       id: '1',
       image: '../../assets/img/xingkong.png',
       title: '梵高-星空高清画作',
-      description: '致敬梵高 #艺术品 #星空'
+      description: '致敬梵高 #艺术品 #星空',
+      avatarUrl: '../../assets/img/book1.png',
+      userName: '星空艺术家',
+      date: '12-01',
+      likes: 1687,
+      category: 'art'
     },
     {
       id: '2',
       image: '../../assets/img/xiangrikui.png',
       title: '梵高《向日葵》',
-      description: '经典中的经典,笔触很牛!#花'
+      description: '经典中的经典,笔触很牛!#花',
+      avatarUrl: '../../assets/img/book2.png',
+      userName: '艺术鉴赏家',
+      date: '11-30',
+      likes: 2341,
+      category: 'art'
     },
     {
       id: '3',
       image: '../../assets/img/fangao.png',
       title: '梵高自画像',
-      description: '梵高生平自画像之其中🔟副👩‍🎨 #艺术欣赏 #美'
+      description: '梵高生平自画像之其中🔟副👩‍🎨 #艺术欣赏 #美',
+      avatarUrl: '../../assets/img/book3.png',
+      userName: '艺术探索者',
+      date: '11-29',
+      likes: 1892,
+      category: 'art'
     },
     {
       id: '4',
       image: '../../assets/img/cunzhuang.png',
       title: '阿尔小镇',
-      description: '走吧,梵高,去你最爱的阿尔看看,阿尔小镇'
+      description: '走吧,梵高,去你最爱的阿尔看看,阿尔小镇',
+      avatarUrl: '../../assets/img/book4.png',
+      userName: '风景画师',
+      date: '11-28',
+      likes: 2156,
+      category: 'art'
     }
   ];
 
@@ -61,6 +81,12 @@ export class Tab3Page {
 
   goToDetail(artId: string) {
     console.log('Navigating to artwork:', artId);
-    this.router.navigate(['/tabs/art-detail', artId]);
+    this.router.navigate(['/tabs/art-detail', artId], {
+      state: { artwork: this.artworks.find(art => art.id === artId) }
+    });
+  }
+
+  goToCategory(category: string) {
+    this.router.navigate(['/tabs/category', category]);
   }
 }

+ 0 - 1
AIart-app/src/app/tab4/tab4.page.html

@@ -2,7 +2,6 @@
   <ion-toolbar>
     <div class="header-container">
       <ion-title class="page-title">AI学习助手</ion-title>
-      <ion-icon name="help-circle-outline" class="help-icon"></ion-icon>
     </div>
   </ion-toolbar>
 </ion-header>

+ 23 - 9
AIart-app/src/app/tabs/tabs.page.ts

@@ -10,14 +10,24 @@ import {
   folderOutline, readerOutline, cartOutline, thumbsUpOutline, ticketOutline, documentTextOutline,
   timeOutline, cloudDownloadOutline, bagCheckOutline, cardOutline, searchOutline, menuOutline,
   addCircleOutline, addOutline, optionsOutline, bookmarkOutline, notificationsOutline,
-  chevronBackOutline, chatbubblesOutline, refreshOutline, createOutline,
-  logOutOutline,
-  closeCircleOutline,
-  bulbOutline,
-  homeOutline,
-  peopleOutline,
-  trashOutline,
-  chatbubbleOutline
+  chevronBackOutline, chatbubblesOutline, refreshOutline, createOutline, logOutOutline,
+  closeCircleOutline, bulbOutline, homeOutline, peopleOutline, trashOutline, chatbubbleOutline,
+  informationCircleOutline, pricetagOutline, closeCircle,
+  brushOutline,
+  bodyOutline,
+  musicalNotesOutline,
+  cameraOutline,
+  codeOutline,
+  restaurantOutline,
+  fitnessOutline,
+  languageOutline,
+  helpCircleOutline,
+  leafOutline,
+  ellipsisHorizontalOutline,
+  flame,
+  checkmarkCircle,
+  handRight,
+  medal,
 } from 'ionicons/icons';
 
 
@@ -43,7 +53,11 @@ export class TabsPage {
       cardOutline, searchOutline, menuOutline, addCircleOutline, addOutline, optionsOutline,
       bookmarkOutline, notificationsOutline, chevronBackOutline, chatbubblesOutline,
       refreshOutline, createOutline, logOutOutline, closeCircleOutline, bulbOutline,
-      homeOutline, peopleOutline, trashOutline, chatbubbleOutline
+      homeOutline, peopleOutline, trashOutline, chatbubbleOutline, informationCircleOutline,
+      pricetagOutline, closeCircle, brushOutline, musicalNotesOutline, bodyOutline,
+      cameraOutline, codeOutline, restaurantOutline, fitnessOutline, languageOutline,
+      helpCircleOutline, leafOutline, flame, checkmarkCircle, handRight, medal,
+
     });
   }
 }

+ 16 - 1
AIart-app/src/app/tabs/tabs.routes.ts

@@ -56,9 +56,24 @@ export const routes: Routes = [
         loadComponent: () =>
           import('../user-login/user-login.component').then((m) => m.UserLoginComponent),
       },
+      {
+        path: 'mailbox',
+        loadComponent: () =>
+          import('../mailbox/mailbox.component').then((m) => m.MailboxComponent),
+      },
+      {
+        path: 'attendance',
+        loadComponent: () =>
+          import('../attendance/attendance.component').then((m) => m.AttendanceComponent),
+      },
+      {
+        path: 'drawing',
+        loadComponent: () =>
+          import('../drawing/drawing.component').then((m) => m.DrawingComponent),
+      },
       {
         path: 'art-detail/:id',
-        loadComponent: () => import('../art-detail/art-detail.page').then(m => m.ArtDetailPage)
+        loadComponent: () => import('../art-detail/art-detail.component').then((m) => m.ArtDetailComponent)
       },
     ],
   },

+ 25 - 12
AIart-app/src/app/user-login/user-login.component.html

@@ -67,21 +67,34 @@
 
   @if(currentUser?.id) {
   <ion-card class="memo-card">
-    <h2 class="memo-title">兴趣跟踪表</h2>
-    <p class="memo-description">写下您感兴趣的学习方向或者心动的科普知识,便于您下次查找(点击标签可删除)</p>
+    <div class="memo-header" style="display: flex; align-items: center;">
+      <ion-icon name="bookmark-outline" style="margin-bottom: 12px;"></ion-icon>
+      <h2 class="memo-title">兴趣跟踪表</h2>
+    </div>
+
+    <p class="memo-description">
+      请写下您感兴趣的学习方向或者心动的科普知识,便于您下次查找
+    </p>
 
-    <edit-tag (onTagChange)="setTagsValue($event)"></edit-tag>
+    <div class="tag-editor">
+      <edit-tag (onTagAdd)="onTagAdd($event)"></edit-tag>
+    </div>
 
     @if(editTags.length > 0) {
-    <h2 class="memo-title">收藏夹</h2>
-    <ul class="tag-list">
-      @for(tag of editTags; track tag) {
-      <li class="tag-item">
-        <span>{{ tag }}</span>
-        <ion-icon name="close-circle-outline"></ion-icon>
-      </li>
-      }
-    </ul>
+    <div class="favorites-section">
+      <div class="section-header" style="display: flex; align-items: center;">
+        <h2 class="section-title">收藏夹</h2>
+      </div>
+
+      <div class="tag-list">
+        @for(tag of editTags; track tag) {
+        <div class="tag-item">
+          <span class="tag-text">{{ tag }}</span>
+          <ion-icon name="close-circle" class="delete-icon" (click)="deleteTag(tag)"></ion-icon>
+        </div>
+        }
+      </div>
+    </div>
     }
   </ion-card>
   }

+ 35 - 6
AIart-app/src/app/user-login/user-login.component.ts

@@ -73,20 +73,49 @@ export class UserLoginComponent implements OnInit {
     openUserEditModal(this.modalCtrl)
   }
 
-  editTags: Array<String> = []
-  async setTagsValue(ev: any) {
+  editTags: string[] = [];
+
+  onTagAdd(newTag: string) {
+    // 检查标签是否已存在
+    if (!this.editTags.includes(newTag)) {
+      this.editTags.push(newTag);
+      this.saveTagsToUser(); // 保存到用户数据
+    } else {
+      // 可以添加一个提示说标签已存在
+      this.showToast('标签已存在');
+    }
+  }
+
+  deleteTag(tagToDelete: string) {
+    const index = this.editTags.indexOf(tagToDelete);
+    if (index > -1) {
+      this.editTags.splice(index, 1);
+      this.saveTagsToUser(); // 保存到用户数据
+    }
+  }
+
+  async saveTagsToUser() {
     let currentUser = new CloudUser();
-    let userPrompt = ``
     if (!currentUser?.id) {
       console.log("用户未登录,请登录后重试");
       let user = await openUserLoginModal(this.modalCtrl);
       if (!user?.id) {
-        return
+        return;
       }
       currentUser = user;
     }
-    //console.log("setTagsValue",ev);
-    this.editTags = ev;
+    // 这里可以添加保存标签到用户数据的逻辑
+    // 例如: currentUser.set('tags', this.editTags);
+    // await currentUser.save();
+  }
+
+  private async showToast(message: string) {
+    const toast = document.createElement('ion-toast');
+    toast.message = message;
+    toast.duration = 2000;
+    toast.position = 'top';
+    document.body.appendChild(toast);
+    await toast.present();
   }
 
 }

+ 43 - 3
AIart-app/src/app/view-all/view-all.component.html

@@ -1,3 +1,43 @@
-<p>
-  view-all works!
-</p>
+<ion-header class="ion-no-border">
+  <ion-toolbar>
+    <ion-buttons slot="start">
+      <ion-back-button defaultHref="/tabs/tab1" text=""></ion-back-button>
+    </ion-buttons>
+    <ion-title>最近在学</ion-title>
+  </ion-toolbar>
+</ion-header>
+
+<ion-content>
+  <div class="books-container">
+    @for(book of recentBooks; track book.id) {
+    <div class="book-card" (click)="goToBookDetail(book.id)">
+      <div class="book-image">
+        <img [src]="book.image" [alt]="book.title">
+        <div class="progress-overlay">
+          <div class="progress-bar">
+            <div class="progress-fill" [style.width]="formatProgress(book.progress)"></div>
+          </div>
+          <span class="progress-text">{{formatProgress(book.progress)}}</span>
+        </div>
+      </div>
+
+      <div class="book-info">
+        <h2>{{book.title}}</h2>
+        <p class="author">作者:{{book.author}}</p>
+        <p class="description">{{book.description}}</p>
+
+        <div class="book-meta">
+          <span class="last-read">
+            <ion-icon name="time-outline"></ion-icon>
+            最近阅读:{{book.lastRead}}
+          </span>
+          <ion-button fill="clear" size="small">
+            <ion-icon name="play-circle-outline" slot="start"></ion-icon>
+            继续学习
+          </ion-button>
+        </div>
+      </div>
+    </div>
+    }
+  </div>
+</ion-content>

+ 157 - 0
AIart-app/src/app/view-all/view-all.component.scss

@@ -0,0 +1,157 @@
+:host {
+    --primary-color: var(--ion-color-primary);
+    --background-color: #f5f5f5;
+}
+
+ion-header {
+    ion-toolbar {
+        --background: transparent;
+
+        ion-title {
+            font-size: 18px;
+            font-weight: 600;
+        }
+    }
+}
+
+.books-container {
+    padding: 16px;
+
+    .book-card {
+        background: white;
+        border-radius: 12px;
+        overflow: hidden;
+        margin-bottom: 16px;
+        box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
+        display: flex;
+        transition: transform 0.3s ease;
+
+        &:active {
+            transform: scale(0.98);
+        }
+
+        .book-image {
+            position: relative;
+            width: 120px;
+            height: 160px;
+
+            img {
+                width: 100%;
+                height: 100%;
+                object-fit: cover;
+            }
+
+            .progress-overlay {
+                position: absolute;
+                bottom: 0;
+                left: 0;
+                right: 0;
+                background: rgba(0, 0, 0, 0.7);
+                padding: 8px;
+
+                .progress-bar {
+                    height: 4px;
+                    background: rgba(255, 255, 255, 0.3);
+                    border-radius: 2px;
+                    overflow: hidden;
+                    margin-bottom: 4px;
+
+                    .progress-fill {
+                        height: 100%;
+                        background: var(--primary-color);
+                        border-radius: 2px;
+                    }
+                }
+
+                .progress-text {
+                    color: white;
+                    font-size: 12px;
+                }
+            }
+        }
+
+        .book-info {
+            flex: 1;
+            padding: 16px;
+            display: flex;
+            flex-direction: column;
+
+            h2 {
+                margin: 0 0 8px;
+                font-size: 18px;
+                font-weight: 600;
+                color: #333;
+            }
+
+            .author {
+                font-size: 14px;
+                color: #666;
+                margin: 0 0 8px;
+            }
+
+            .description {
+                font-size: 14px;
+                color: #666;
+                line-height: 1.4;
+                margin: 0;
+                display: -webkit-box;
+                -webkit-line-clamp: 2;
+                -webkit-box-orient: vertical;
+                overflow: hidden;
+                flex: 1;
+            }
+
+            .book-meta {
+                display: flex;
+                justify-content: space-between;
+                align-items: center;
+                margin-top: 12px;
+
+                .last-read {
+                    display: flex;
+                    align-items: center;
+                    gap: 4px;
+                    font-size: 12px;
+                    color: #999;
+
+                    ion-icon {
+                        font-size: 16px;
+                    }
+                }
+
+                ion-button {
+                    --color: var(--primary-color);
+                    font-size: 14px;
+
+                    ion-icon {
+                        margin-right: 4px;
+                    }
+                }
+            }
+        }
+    }
+}
+
+// 添加动画效果
+@keyframes fadeInUp {
+    from {
+        opacity: 0;
+        transform: translateY(20px);
+    }
+
+    to {
+        opacity: 1;
+        transform: translateY(0);
+    }
+}
+
+.book-card {
+    animation: fadeInUp 0.5s ease-out;
+    animation-fill-mode: both;
+
+    @for $i from 1 through 10 {
+        &:nth-child(#{$i}) {
+            animation-delay: #{$i * 0.1}s;
+        }
+    }
+}

+ 50 - 1
AIart-app/src/app/view-all/view-all.component.ts

@@ -1,15 +1,64 @@
 import { Component, OnInit } from '@angular/core';
+import { IonicModule } from '@ionic/angular';
+import { CommonModule } from '@angular/common';
+import { Router } from '@angular/router';
 
 @Component({
   selector: 'app-view-all',
   templateUrl: './view-all.component.html',
   styleUrls: ['./view-all.component.scss'],
   standalone: true,
+  imports: [IonicModule, CommonModule],
 })
 export class ViewAllComponent implements OnInit {
+  recentBooks = [
+    {
+      id: '1',
+      title: '城南旧事',
+      author: '林海音',
+      image: '../../assets/img/book1.png',
+      progress: 35,
+      description: '《城南旧事》是林海音创作的一部自传体小说集,以一个小女孩的视角讲述了在北京城南的生活往事。',
+      lastRead: '2024-12-15'
+    },
+    {
+      id: '2',
+      title: '朝花夕拾',
+      author: '鲁迅',
+      image: '../../assets/img/book2.png',
+      progress: 68,
+      description: '《朝花夕拾》是鲁迅创作的一部回忆性散文集,记录了作者的童年往事和对社会的思考。',
+      lastRead: '2024-12-14'
+    },
+    {
+      id: '3',
+      title: '狂人日记',
+      author: '鲁迅',
+      image: '../../assets/img/book3.png',
+      progress: 42,
+      description: '《狂人日记》是中国现代文学史上第一篇白话小说,以独特的视角揭示了封建社会的吃人本质。',
+      lastRead: '2024-12-13'
+    },
+    {
+      id: '4',
+      title: '彷徨',
+      author: '鲁迅',
+      image: '../../assets/img/book4.png',
+      progress: 25,
+      description: '《彷徨》是鲁迅创作的第二部小说集,收录了多篇反映社会现实的优秀作品。',
+      lastRead: '2024-12-12'
+    }
+  ];
 
-  constructor() { }
+  constructor(private router: Router) { }
 
   ngOnInit() { }
 
+  goToBookDetail(bookId: string) {
+    this.router.navigate(['/tabs/book-detail', bookId]);
+  }
+
+  formatProgress(progress: number): string {
+    return `${progress}%`;
+  }
 }