Browse Source

update:front

csdn1233 3 months ago
parent
commit
0a10bcda58

+ 102 - 55
AIart-app/src/app/interest-search/interest-search.component.html

@@ -1,85 +1,132 @@
 <!-- src/app/interest-search/interest-search.component.html -->
-<ion-header [translucent]="true">
+<ion-header [translucent]="true" class="ion-no-border">
   <ion-toolbar>
     <ion-buttons slot="start">
-      <ion-back-button default-href="/tabs/tab1" style="color:black;"></ion-back-button>
+      <ion-back-button default-href="/tabs/tab1"></ion-back-button>
     </ion-buttons>
-    <ion-title style="font-family: 'Courier New', Courier, monospace;">
-      <span style="font-weight: bold;">调查问卷</span>
-    </ion-title>
+    <ion-title>兴趣调查问卷</ion-title>
   </ion-toolbar>
 </ion-header>
 
-<ion-content color="light">
-  <!-- 一、基本情况 -->
-  <ion-list [inset]="true">
-    <ion-item>
-      <ion-label>一、基本情况</ion-label>
-    </ion-item>
-    <ion-item>
-      <ion-input [(ngModel)]="name" label="姓名"></ion-input>
-    </ion-item>
-    <ion-item>
-      <span style="margin-right: 50px;">生日</span>
-      <ion-datetime-button datetime="datetime"></ion-datetime-button>
-      <ion-modal [keepContentsMounted]="true">
-        <ng-template>
-          <ion-datetime id="datetime" presentation="date" (ionChange)="onDateTimeChange($event)"></ion-datetime>
-        </ng-template>
-      </ion-modal>
-    </ion-item>
-    <!-- <ion-item>
-      <ion-label position="floating">生日</ion-label>
-      <ion-datetime [(ngModel)]="birthday"></ion-datetime>
-    </ion-item> -->
-  </ion-list>
+<ion-content>
+  <!-- 问卷说明 -->
+  <div class="survey-intro">
+    <h2>欢迎参与兴趣调查</h2>
+    <p>本问卷旨在帮助您更好地了解自己的兴趣倾向,请认真填写。</p>
+  </div>
+
+  <!-- 基本信息部分 -->
+  <div class="section-container">
+    <div class="section-header">
+      <ion-icon name="person-outline"></ion-icon>
+      <h3>基本信息</h3>
+    </div>
 
-  <!-- 二、问卷问题 -->
-  <ion-list [inset]="true">
-    <!-- 遍历 questionsWithOptions 逐个显示问题 -->
-    <div *ngFor="let question of questionsWithOptions">
-      <!-- 问题文本 -->
+    <ion-list>
       <ion-item>
-        <ion-label>{{ question.questionText }}</ion-label>
+        <ion-label position="stacked">姓名</ion-label>
+        <ion-input [(ngModel)]="name" placeholder="请输入您的姓名"></ion-input>
       </ion-item>
 
-      <!-- 选项列表 -->
-      <ion-radio-group [(ngModel)]="answers[question.QuestionId]" [allowEmptySelection]="true">
-        <!-- 如果选项还未加载完,显示加载中 -->
-        <ion-item *ngIf="!question.optionsData || question.optionsData.length === 0">
-          <ion-label>加载中...</ion-label>
-        </ion-item>
+      <ion-item>
+        <ion-label position="stacked">性别</ion-label>
+        <ion-select [(ngModel)]="gender" placeholder="请选择">
+          <ion-select-option value="male">男</ion-select-option>
+          <ion-select-option value="female">女</ion-select-option>
+          <ion-select-option value="other">其他</ion-select-option>
+        </ion-select>
+      </ion-item>
+
+      <ion-item>
+        <ion-label position="stacked">年龄</ion-label>
+        <ion-input [(ngModel)]="age" type="number" placeholder="请输入您的年龄"></ion-input>
+      </ion-item>
+
+      <ion-item>
+        <span style="margin-right: 50px;">生日</span>
+        <ion-datetime-button datetime="datetime"></ion-datetime-button>
+        <ion-modal [keepContentsMounted]="true">
+          <ng-template>
+            <ion-datetime id="datetime" presentation="date" (ionChange)="onDateTimeChange($event)"></ion-datetime>
+          </ng-template>
+        </ion-modal>
+      </ion-item>
+
+      <ion-item>
+        <ion-label position="stacked">职业</ion-label>
+        <ion-select [(ngModel)]="occupation" placeholder="请选择">
+          <ion-select-option value="student">学生</ion-select-option>
+          <ion-select-option value="employee">上班族</ion-select-option>
+          <ion-select-option value="freelancer">自由职业</ion-select-option>
+          <ion-select-option value="other">其他</ion-select-option>
+        </ion-select>
+      </ion-item>
+    </ion-list>
+  </div>
 
-        <!-- 加载完成后显示选项 -->
-        <ion-item *ngFor="let option of question.optionsData">
+  <!-- 问卷问题部分 -->
+  <div class="section-container">
+    <div class="section-header">
+      <ion-icon name="list-outline"></ion-icon>
+      <h3>兴趣调查</h3>
+    </div>
+
+    <div class="question-container" *ngFor="let question of questionsWithOptions; let i = index">
+      <div class="question-header">
+        <span class="question-number">Q{{i + 1}}</span>
+        <h4>{{question.questionText}}</h4>
+      </div>
+
+      <ion-radio-group [(ngModel)]="answers[question.QuestionId]" class="options-container">
+        <ion-item *ngFor="let option of question.optionsData" lines="none" class="option-item">
           <ion-radio slot="start" [value]="option.OptionId"></ion-radio>
-          <ion-label>{{ option.optionText }}</ion-label>
+          <ion-label>{{option.optionText}}</ion-label>
         </ion-item>
       </ion-radio-group>
     </div>
-  </ion-list>
+  </div>
 
-  <!-- 保存和提交按钮 -->
-  <div style="display: flex; justify-content: space-between; align-items: center; margin: 0 15px;">
-    <ion-button style="width: 45%;" (click)="save()">保存</ion-button>
-    <ion-button id="yes/no" style="width: 45%;" (click)="submit()">提交</ion-button>
-    <ion-alert trigger="yes/no" header="测试结果正在加载中,请等待几秒钟。" [buttons]="alertButtons"></ion-alert>
+  <!-- 按钮区域 -->
+  <div class="button-container">
+    <ion-button expand="block" color="medium" (click)="save()">
+      <ion-icon name="save-outline" slot="start"></ion-icon>
+      保存
+    </ion-button>
+    <ion-button expand="block" (click)="submit()" id="submit-button">
+      <ion-icon name="send-outline" slot="start"></ion-icon>
+      提交
+    </ion-button>
+    <ion-button expand="block" color="tertiary" (click)="refreshQuestionnaire()">
+      <ion-icon name="refresh-outline" slot="start"></ion-icon>
+      更新问卷
+    </ion-button>
   </div>
 
-  <ion-modal [isOpen]="modalIsOpen" (didDismiss)="closeModal()">
+  <!-- 结果弹窗 -->
+  <ion-modal [isOpen]="modalIsOpen" (didDismiss)="closeModal()" class="result-modal">
     <ng-template>
-      <ion-header>
+      <ion-header class="modal-header">
         <ion-toolbar>
           <ion-title>兴趣分析结果</ion-title>
           <ion-buttons slot="end">
-            <ion-button (click)="closeModal()">关闭</ion-button>
+            <ion-button (click)="closeModal()">
+              <ion-icon name="close-outline"></ion-icon>
+            </ion-button>
           </ion-buttons>
         </ion-toolbar>
       </ion-header>
-      <ion-content>
-        <div [innerHTML]="modalContent"></div>
+
+      <ion-content class="result-content">
+        @if(isAnalyzing) {
+        <div class="loading-container">
+          <ion-spinner name="crescent"></ion-spinner>
+          <h3>AI正在分析您的兴趣倾向</h3>
+          <p>请稍候片刻,我们正在为您生成专业的分析报告...</p>
+        </div>
+        } @else {
+        <div class="result-container" [innerHTML]="modalContent"></div>
+        }
       </ion-content>
     </ng-template>
   </ion-modal>
-
 </ion-content>

+ 569 - 0
AIart-app/src/app/interest-search/interest-search.component.scss

@@ -1,3 +1,315 @@
+:host {
+    --primary-color: var(--ion-color-primary);
+    --background-color: #f5f5f5;
+}
+
+ion-header {
+    ion-toolbar {
+        --background: transparent;
+
+        ion-title {
+            font-size: 18px;
+            font-weight: 600;
+        }
+    }
+}
+
+.survey-intro {
+    padding: 20px;
+    background: white;
+    margin: 16px;
+    border-radius: 12px;
+    box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
+
+    h2 {
+        margin: 0 0 12px;
+        font-size: 20px;
+        font-weight: 600;
+        color: #333;
+    }
+
+    p {
+        margin: 0;
+        color: #666;
+        font-size: 14px;
+        line-height: 1.5;
+    }
+}
+
+.section-container {
+    background: white;
+    margin: 16px;
+    border-radius: 12px;
+    overflow: hidden;
+    box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
+
+    .section-header {
+        display: flex;
+        align-items: center;
+        padding: 16px;
+        border-bottom: 1px solid #eee;
+
+        ion-icon {
+            font-size: 24px;
+            color: var(--primary-color);
+            margin-right: 12px;
+        }
+
+        h3 {
+            margin: 0;
+            font-size: 18px;
+            font-weight: 600;
+            color: #333;
+        }
+    }
+}
+
+ion-list {
+    background: transparent;
+    padding: 0;
+
+    ion-item {
+        --padding-start: 16px;
+        --padding-end: 16px;
+        --padding-top: 12px;
+        --padding-bottom: 12px;
+        --background: transparent;
+
+        ion-label {
+            color: #333;
+            font-weight: 500;
+            margin-bottom: 8px;
+        }
+
+        ion-input,
+        ion-select {
+            --padding-start: 0;
+            --placeholder-color: #999;
+            font-size: 15px;
+        }
+    }
+}
+
+.question-container {
+    padding: 16px;
+    border-bottom: 1px solid #eee;
+
+    &:last-child {
+        border-bottom: none;
+    }
+
+    .question-header {
+        display: flex;
+        align-items: flex-start;
+        margin-bottom: 16px;
+
+        .question-number {
+            background: var(--primary-color);
+            color: white;
+            padding: 4px 8px;
+            border-radius: 4px;
+            font-size: 14px;
+            margin-right: 12px;
+            min-width: 40px;
+            text-align: center;
+        }
+
+        h4 {
+            margin: 0;
+            font-size: 16px;
+            font-weight: 500;
+            color: #333;
+            flex: 1;
+        }
+    }
+
+    .options-container {
+        padding-left: 52px;
+
+        .option-item {
+            --padding-start: 0;
+            --padding-end: 0;
+            --min-height: 44px;
+            --background: transparent;
+
+            ion-radio {
+                margin-right: 12px;
+                --color-checked: var(--primary-color);
+            }
+
+            ion-label {
+                font-size: 15px;
+                color: #666;
+            }
+        }
+    }
+}
+
+.button-container {
+    padding: 16px;
+    display: grid;
+    grid-template-columns: repeat(3, 1fr);
+    gap: 12px;
+    margin-bottom: 32px;
+
+    ion-button {
+        margin: 0;
+        height: 44px;
+        --border-radius: 8px;
+
+        ion-icon {
+            margin-right: 8px;
+        }
+
+        &[color="tertiary"] {
+            --background: var(--ion-color-tertiary);
+            --color: white;
+
+            &:hover {
+                --background: var(--ion-color-tertiary-shade);
+            }
+        }
+    }
+}
+
+.result-modal {
+    --height: 90%;
+    --border-radius: 20px 20px 0 0;
+
+    .modal-header {
+        ion-toolbar {
+            --background: var(--ion-color-primary);
+            --color: white;
+
+            ion-title {
+                font-size: 20px;
+                font-weight: 600;
+            }
+
+            ion-button {
+                --color: white;
+            }
+        }
+    }
+
+    .result-content {
+        --background: #f5f5f5;
+    }
+
+    .result-container {
+        padding: 20px;
+
+        h3 {
+            display: flex;
+            align-items: center;
+            font-size: 18px;
+            font-weight: 600;
+            color: #333;
+            margin: 0 0 16px;
+
+            ion-icon {
+                margin-right: 8px;
+                font-size: 24px;
+                color: var(--ion-color-primary);
+            }
+        }
+
+        .tags-section {
+            background: white;
+            border-radius: 16px;
+            padding: 20px;
+            margin-bottom: 16px;
+            box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
+
+            .tags-container {
+                display: flex;
+                flex-wrap: wrap;
+                gap: 12px;
+
+                .interest-tag {
+                    display: flex;
+                    align-items: center;
+                    background: var(--ion-color-primary-light);
+                    color: var(--ion-color-primary);
+                    padding: 8px 16px;
+                    border-radius: 20px;
+                    font-size: 14px;
+
+                    ion-icon {
+                        margin-right: 6px;
+                        font-size: 16px;
+                    }
+                }
+            }
+        }
+
+        .analysis-section {
+            background: white;
+            border-radius: 16px;
+            padding: 20px;
+            margin-bottom: 16px;
+            box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
+
+            .analysis-content {
+                color: #666;
+                line-height: 1.6;
+                font-size: 15px;
+
+                p {
+                    margin: 8px 0;
+                }
+            }
+        }
+
+        .action-buttons {
+            display: grid;
+            grid-template-columns: 1fr 1fr;
+            gap: 12px;
+            margin-top: 24px;
+
+            ion-button {
+                margin: 0;
+                height: 44px;
+                --border-radius: 8px;
+
+                ion-icon {
+                    margin-right: 8px;
+                }
+            }
+        }
+    }
+}
+
+// 添加动画效果
+@keyframes slideUp {
+    from {
+        transform: translateY(100%);
+    }
+
+    to {
+        transform: translateY(0);
+    }
+}
+
+.result-modal {
+    .result-container {
+        animation: slideUp 0.3s ease-out;
+    }
+}
+
+.section-container {
+    animation: fadeInUp 0.5s ease-out;
+    animation-fill-mode: both;
+}
+
+.section-container:nth-child(2) {
+    animation-delay: 0.2s;
+}
+
+.section-container:nth-child(3) {
+    animation-delay: 0.4s;
+}
+
 ion-checkbox {
     margin-left: 15px;
     --size: 20px;
@@ -46,4 +358,261 @@ ion-radio.radio-checked::part(mark) {
 
 .fontsize {
     font-size: 20px;
+}
+
+// 日期选择器样式
+.custom-datetime {
+    padding: 8px 0;
+    --background: var(--ion-color-light);
+    border-radius: 8px;
+    margin-top: 4px;
+
+    &::part(wheel-item) {
+        color: var(--ion-color-dark);
+        font-size: 16px;
+    }
+
+    &::part(wheel-item active) {
+        color: var(--ion-color-primary);
+        font-size: 18px;
+        font-weight: 600;
+    }
+
+    &::part(wheel) {
+        background: var(--ion-background-color);
+        border-radius: 8px;
+    }
+}
+
+// 日期选择器头部样式
+.datetime-header {
+    padding: 16px;
+    font-size: 18px;
+    font-weight: 600;
+    color: var(--ion-color-dark);
+    text-align: center;
+    border-bottom: 1px solid var(--ion-color-light-shade);
+}
+
+// 日期选择器按钮样式
+.datetime-buttons {
+    display: flex;
+    justify-content: flex-end;
+    padding: 8px;
+    border-top: 1px solid var(--ion-color-light-shade);
+
+    ion-button {
+        margin: 0 4px;
+        --padding-start: 16px;
+        --padding-end: 16px;
+        font-weight: 500;
+    }
+}
+
+// 调整 ion-item 中日期选择器的样式
+ion-item {
+    &.item-has-value {
+        ion-datetime {
+            --placeholder-color: var(--ion-color-dark);
+        }
+    }
+
+    ion-datetime {
+        width: 100%;
+        min-height: 44px;
+        --padding-start: 0;
+        --padding-end: 0;
+    }
+}
+
+.analysis-result {
+    padding: 20px;
+    background: #f8f9fa;
+
+    .result-section {
+        background: white;
+        border-radius: 20px;
+        padding: 24px;
+        margin-bottom: 24px;
+        box-shadow: 0 4px 20px rgba(0, 0, 0, 0.06);
+        transition: transform 0.3s ease;
+
+        &:hover {
+            transform: translateY(-2px);
+        }
+
+        .section-title {
+            display: flex;
+            align-items: center;
+            margin-bottom: 24px;
+            padding-bottom: 16px;
+            border-bottom: 2px solid #f0f0f0;
+
+            ion-icon {
+                font-size: 28px;
+                color: var(--ion-color-primary);
+                margin-right: 16px;
+            }
+
+            h2 {
+                margin: 0;
+                font-size: 20px;
+                font-weight: 600;
+                color: #333;
+                letter-spacing: 0.5px;
+            }
+        }
+    }
+
+    .tags-section {
+        .tags-wrapper {
+            display: flex;
+            flex-wrap: wrap;
+            gap: 16px;
+
+            .tag-item {
+                display: flex;
+                align-items: center;
+                background: linear-gradient(135deg, var(--ion-color-primary-light), var(--ion-color-primary-tint));
+                color: var(--ion-color-primary-contrast);
+                padding: 12px 20px;
+                border-radius: 50px;
+                font-size: 16px;
+                font-weight: 500;
+                transition: all 0.3s ease;
+                box-shadow: 0 4px 12px rgba(var(--ion-color-primary-rgb), 0.2);
+
+                &:hover {
+                    transform: translateY(-3px) scale(1.02);
+                    box-shadow: 0 6px 16px rgba(var(--ion-color-primary-rgb), 0.3);
+                }
+
+                .tag-icon {
+                    display: flex;
+                    align-items: center;
+                    justify-content: center;
+                    width: 24px;
+                    height: 24px;
+                    margin-right: 8px;
+
+                    ion-icon {
+                        font-size: 18px;
+                    }
+                }
+
+                span {
+                    letter-spacing: 0.5px;
+                }
+            }
+        }
+    }
+
+    .report-section {
+        .content-wrapper {
+            color: #444;
+            line-height: 1.8;
+
+            .analysis-item {
+                display: flex;
+                align-items: flex-start;
+                margin-bottom: 20px;
+                padding: 16px;
+                background: #f8f9fa;
+                border-radius: 12px;
+                transition: all 0.3s ease;
+
+                &:hover {
+                    background: #f0f2f5;
+                    transform: translateX(8px);
+                }
+
+                .bullet-point {
+                    flex-shrink: 0;
+                    width: 8px;
+                    height: 8px;
+                    border-radius: 50%;
+                    background: var(--ion-color-primary);
+                    margin-top: 8px;
+                    margin-right: 16px;
+                }
+
+                p {
+                    margin: 0;
+                    font-size: 15px;
+                    color: #333;
+                    line-height: 1.8;
+                    letter-spacing: 0.3px;
+                }
+            }
+        }
+    }
+}
+
+// 添加加载状态的样式
+.loading-container {
+    display: flex;
+    flex-direction: column;
+    align-items: center;
+    justify-content: center;
+    min-height: 300px;
+    padding: 32px;
+    text-align: center;
+
+    ion-spinner {
+        width: 48px;
+        height: 48px;
+        --color: var(--ion-color-primary);
+        margin-bottom: 24px;
+    }
+
+    h3 {
+        font-size: 18px;
+        font-weight: 600;
+        color: #333;
+        margin: 0 0 12px;
+    }
+
+    p {
+        font-size: 14px;
+        color: #666;
+        margin: 0;
+        max-width: 280px;
+        line-height: 1.5;
+    }
+}
+
+// 添加淡入动画
+@keyframes fadeIn {
+    from {
+        opacity: 0;
+    }
+
+    to {
+        opacity: 1;
+    }
+}
+
+.loading-container {
+    animation: fadeIn 0.3s ease-out;
+}
+
+// 添加动画效果
+@keyframes fadeInUp {
+    from {
+        opacity: 0;
+        transform: translateY(20px);
+    }
+
+    to {
+        opacity: 1;
+        transform: translateY(0);
+    }
+}
+
+.result-section {
+    animation: fadeInUp 0.5s ease-out forwards;
+
+    &:nth-child(2) {
+        animation-delay: 0.2s;
+    }
 }

+ 311 - 78
AIart-app/src/app/interest-search/interest-search.component.ts

@@ -2,7 +2,11 @@ import { Component, OnInit, ViewChild } from '@angular/core';
 import {
   IonTextarea, IonCheckbox, IonList, IonButton, IonContent, IonHeader, IonInput, IonTitle,
   IonToolbar, IonItem, IonLabel, IonRadioGroup, IonRadio, IonDatetimeButton, IonDatetime,
-  IonModal, IonAlert, IonBackButton, IonButtons
+  IonModal, IonAlert, IonBackButton, IonButtons,
+  IonIcon,
+  IonSelectOption,
+  IonSelect,
+  IonSpinner
 } from '@ionic/angular/standalone';
 import { CloudQuery, CloudObject, Pointer } from '../../lib/ncloud'; // 确保路径正确
 import { CommonModule, DatePipe } from '@angular/common'; // 导入 CommonModule
@@ -26,7 +30,7 @@ interface Question {
   QuestionId: string;
   questionnaireId: string; // 修改为字符串
   questionText: string;
-  options: string[]; // 修改为字符串数
+  options: string[]; // 组
 }
 
 interface Option {
@@ -68,13 +72,16 @@ interface QuestionWithOptions extends Question {
   imports: [IonTextarea, IonCheckbox, IonList, IonButton, IonContent, IonHeader, IonInput,
     IonTitle, IonToolbar, IonItem, IonLabel, IonRadioGroup, IonRadio, IonDatetimeButton,
     IonDatetime, IonModal, CommonModule, FormsModule, IonDatetime, IonModal, IonAlert,
-    IonBackButton, IonButtons, MarkdownPreviewModule
+    IonBackButton, IonButtons, MarkdownPreviewModule, IonIcon, IonSelectOption, IonSelect,
+    IonSpinner,
   ]
 })
 export class InterestSearchComponent implements OnInit {
   // 固定字段
   name: string = '';
   birthday: string = '';
+  maxDate = new Date().toISOString().split('T')[0];
+  minDate = '1900-01-01';
 
   // 动态问卷数据
   questionnaire: Questionnaire | null = null;
@@ -91,23 +98,42 @@ export class InterestSearchComponent implements OnInit {
   modalIsOpen: boolean = false; // 使用 isOpen 控制 Modal 的显示状态
   modalContent: string = ''; // 保存弹窗的内容
 
+  gender: string = '';
+  age: number | null = null;
+  occupation: string = '';
+
+  @ViewChild(IonDatetime) datetime!: IonDatetime;
+
+  isAnalyzing: boolean = false; // 添加分析状态标志
+
+  // 修改问卷ID数组
+  questionnaires = ['q1', 'q2', 'q3']; // 确保这些ID在数据库中存在
+
   constructor() { }
 
   // 定义方法,用于获取 <ion-datetime> 组件选择的值
   onDateTimeChange(event: any) {
-    this.birthday = event.detail.value;
-    // // 使用DatePipe进行日期格式化,只保留年、月、日
-    // this.birthday = this.datePipe.transform(this.birthday, 'yyyy-MM-dd')!;
-    console.log('选择的日期为:', this.birthday);
-  }
+    if (event && event.detail && event.detail.value) {
+      this.birthday = event.detail.value.split('T')[0]; // 只保留日期部分
 
-  alertButtons = ['确定'];
+      // 计算年龄
+      const birthDate = new Date(this.birthday);
+      const today = new Date();
+      let age = today.getFullYear() - birthDate.getFullYear();
+      const monthDiff = today.getMonth() - birthDate.getMonth();
 
-  ngOnInit() {
-    this.loadQuestionnaireData('q1'); // 使用 QuestionnaireId 'q1'
-    //this.loadQuestionnaireData(this.getRandomQuestionnaire());
+      if (monthDiff < 0 || (monthDiff === 0 && today.getDate() < birthDate.getDate())) {
+        age--;
+      }
 
+      this.age = age;
+      console.log('Selected date:', this.birthday);
+      console.log('Calculated age:', this.age);
+    }
   }
+
+  alertButtons = ['确定'];
+
   getRandomQuestionnaire() {
     const questionnaires = ['q1', 'q2', 'q3']; // List of your questionnaires
     const randomIndex = Math.floor(Math.random() * questionnaires.length); // Generate a random index
@@ -117,14 +143,18 @@ export class InterestSearchComponent implements OnInit {
 
   async loadQuestionnaireData(questionnaireId: string) {
     try {
+      console.log('开始加载问卷:', questionnaireId);
+
+      // 清空现有数据
+      this.questionsWithOptions = [];
+      this.answers = {};
+
       const questionnaireQuery = new CloudQuery("Questionnaire");
       questionnaireQuery.equalTo("QuestionnaireId", questionnaireId);
       const questionnaireObj = await questionnaireQuery.first();
 
       if (questionnaireObj) {
         const questionnaireData = questionnaireObj.data as Questionnaire;
-
-        // 确保 objectId 存在且为字符串
         this.questionnaire = {
           ...questionnaireData,
           objectId: String(questionnaireObj.id)
@@ -132,15 +162,17 @@ export class InterestSearchComponent implements OnInit {
 
         console.log("加载到的问卷数据:", this.questionnaire);
 
-        // 确保 questions 被正确传入
-        if (this.questionnaire.questions) {
+        if (this.questionnaire.questions && this.questionnaire.questions.length > 0) {
           await this.loadQuestions(this.questionnaire.questions);
+        } else {
+          throw new Error('问卷中没有问题');
         }
       } else {
-        console.error(`未找到 QuestionnaireId 为 ${questionnaireId} 的问卷`);
+        throw new Error(`未找到 QuestionnaireId 为 ${questionnaireId} 的问卷`);
       }
     } catch (error) {
       console.error("加载问卷数据时出错:", error);
+      throw error; // 向上传递错误,让调用者处理
     }
   }
   async loadQuestions(questionIds: string[]) {
@@ -167,7 +199,7 @@ export class InterestSearchComponent implements OnInit {
             }
           });
 
-          // 可选:每加载一个题,立即触发渲染
+          // 可选:每加载一个题,立即触发渲染
           console.log("已加载问题:", question);
         }
       } catch (error) {
@@ -208,7 +240,11 @@ export class InterestSearchComponent implements OnInit {
         return;
       }
 
-      // 创建一个数组保存选中的 OptionId
+      // 显示加载状态
+      this.isAnalyzing = true;
+      this.modalIsOpen = true;
+
+      // 创建一个数组保存选的 OptionId
       const answersArray: string[] = [];
 
       // 遍历每个问题,获取用户选择的选项
@@ -237,21 +273,42 @@ export class InterestSearchComponent implements OnInit {
       console.log("问卷提交成功。");
 
 
-      // 构建用于 AI 模型分析提示词
+      // 构建用于 AI 模型分析提示词
       const aiPrompt = this.createAiPrompt(answersArray);
 
       // 调用 AI 模型分析,强制等待结果
       const aiResponse = await this.callAiModel(aiPrompt);
 
-      // 如果 AI 响应有效,执行以下逻辑
+      // 如果 AI 响应有效,执行以下逻辑
       if (aiResponse) {
         this.aiAnalysisResult = aiResponse; // 保存 AI 响应结果
-        this.showAnalysisResult(); // 显示结果给用户
         await this.saveAnalysisResult(aiResponse); // 保存到数据库
+
+        // 准备显示结果
+        this.modalContent = this.formatContent(
+          JSON.stringify({
+            interestTags: this.aiAnalysisResult.interestTags,
+            content: this.aiAnalysisResult.content
+          })
+        );
+
+        // 关闭加载状态
+        this.isAnalyzing = false;
       }
 
     } catch (error) {
       console.error("提交问卷时出错:", error);
+      this.isAnalyzing = false; // 确保出错时也关闭加载状态
+      this.modalIsOpen = false;
+
+      // 显示错误提示
+      const toast = document.createElement('ion-toast');
+      toast.message = '分析失败,请重试';
+      toast.duration = 2000;
+      toast.position = 'top';
+      toast.color = 'danger';
+      document.body.appendChild(toast);
+      await toast.present();
     }
   }
 
@@ -270,13 +327,13 @@ export class InterestSearchComponent implements OnInit {
     您是一名专业的兴趣分析师,请根据用户填写的问卷内容以及选项分析用户的兴趣并且生成以下格式的响应:
     
 {
-  "interestTags": ["标签1", "标签2", "标签3", "标签4"], // 生成用户最感兴趣的四个标签(如:书法、绘画、摄影
-  "content": "标签描述" // 针对每个标签生成简洁、生动的描述,帮助用户更清楚了解兴趣特点。描述可包括用户行为、倾向和相关建议。
+  "interestTags": ["标签1", "标签2", "标签3", "标签4"], // 生成用户最感兴趣的四个标签(如:书法、绘画、摄影)
+  "content": "标签描述" // 针对每个标签生成简洁、生动的描述,帮助用户更清楚了解兴趣特点。描述可包括用户行为、倾向和相关建议。
 }   
     请根据以下信息进行分析:
     问题:${questionTexts.join(',')}
     选项:${optionTexts.join(',')}
-    意:
+    意:
    - 仅选择用户**最感兴趣**的四个标签。
    - 生成的描述需要具体、生动,反映用户的兴趣深度或行为倾向。
    - 请忽略与用户兴趣无关的内容。
@@ -309,11 +366,11 @@ export class InterestSearchComponent implements OnInit {
                   this.isComplete = true;
                 }
 
-                // 如果消息完成且内容符合 JSON 格式,则解
+                // 如果消息完成且内容符合 JSON 格式,则解
                 if (this.isComplete) {
                   const cleanedContent = fullContent.trim();
 
-                  // 检查是否有效的 JSON 格式
+                  // 检查是否有效的 JSON 格式
                   if (cleanedContent.startsWith('{') && cleanedContent.endsWith('}')) {
                     try {
                       // 清理掉换行符和多余空格
@@ -333,7 +390,7 @@ export class InterestSearchComponent implements OnInit {
                         if (typeof content === 'string') {
                           contentStr = content; // 如果已经是字符串,直接使用
                         } else if (typeof content === 'object') {
-                          // 如果对象,转换为 JSON 字符串
+                          // 如果对象,转换为 JSON 字符串
                           contentStr = JSON.stringify(content, null, 2); // 美化 JSON 字符串格式
                           count = 0;
                           this.isComplete = false;
@@ -349,7 +406,7 @@ export class InterestSearchComponent implements OnInit {
                     } catch (err) {
                       console.log(fullContent);
                       console.error("解析 AI 响应失败:", err);
-                      reject(new Error("解 AI 响应失败"));
+                      reject(new Error("解 AI 响应失败"));
                     }
                   } else {
                     reject(new Error("返回的内容不是有效的 JSON 格式"));
@@ -396,7 +453,7 @@ export class InterestSearchComponent implements OnInit {
 
     const alert = await this.alertController.create({
       header: '兴趣分析结果',
-      message: `${content}`,  // 使用 pre 标签来保持格式
+      message: `${content}`,  // 用 pre 标签来保持格式
       buttons: ['确定']
     });
 
@@ -407,13 +464,13 @@ export class InterestSearchComponent implements OnInit {
   // 格式化 content 为可读的文本格式
   formatContent(content: string): string {
     try {
-      // 尝试 content 解析为 JSON 对象
+      // 尝试 content 解析为 JSON 对象
       const contentObj = JSON.parse(content);
 
       // 构建格式化后的文本
       let formattedContent = '';
 
-      // 遍历 JSON 象,生成类似 "标签: 描述" 格式
+      // 遍历 JSON 象,生成类似 "标签: 描述" 格式
       for (const [tag, description] of Object.entries(contentObj)) {
         formattedContent += `${tag}: \n${description}\n\n`;
       }
@@ -426,55 +483,74 @@ export class InterestSearchComponent implements OnInit {
     }
   }
 *//*
-                                                    // 格式化 AI 响应内容
-                                                    formatContent(content: string): string {
-                                                      try {
-                                                        const contentObj = JSON.parse(content);
-                                                        return Object.entries(contentObj)
-                                                          .map(
-                                                            ([key, value]) =>
-                                                              `<strong>${key}:</strong><br>${value}<br><br>`
-                                                          )
-                                                          .join('');
-                                                      } catch (error) {
-                                                        console.error('格式化 AI 响应时出错:', error);
-                                                        return '分析结果格式错误。';
-                                                      }
-                                                    }
-                                                  */
+                                                                                                      // 格式化 AI 响应内容
+                                                                                                      formatContent(content: string): string {
+                                                                                                        try {
+                                                                                                          const contentObj = JSON.parse(content);
+                                                                                                          return Object.entries(contentObj)
+                                                                                                            .map(
+                                                                                                              ([key, value]) =>
+                                                                                                                `<strong>${key}:</strong><br>${value}<br><br>`
+                                                                                                            )
+                                                                                                            .join('');
+                                                                                                        } catch (error) {
+                                                                                                          console.error('格式化 AI 响应时出错:', error);
+                                                                                                          return '分析结果格式错误。';
+                                                                                                        }
+                                                                                                      }
+                                                                                                    */
   // 格式化 AI 响应内容
   formatContent(content: string): string {
     try {
       const contentObj = JSON.parse(content);
 
-      console.log(contentObj)
-
-      // 提取 interestTags 数组并格式化为一行展示
-      const interestTags = contentObj.interestTags || [];
-      const interestTagsFormatted = Array.isArray(interestTags)
-        ? `${interestTags.join(',')}` // 标签用逗号分隔
-        : '';
-
-      // 提取 content 对象内容并格式化
-      const contentDetails = contentObj.content || {};
-
-      const contentFormatted = contentDetails
-        .replace(/\"/g, '') // 移除转义字符如 \"
-        .replace(/\n/g, '') // 移除换行符 \n
-        .replace(/,/g, '') // 移除,
-        .replace(/{/g, '') // 移除{
-        .replace(/}/g, '') // 移除}
-        .replace(/。/g, '。<br /><br />'); // 在每个句号 "。" 后插入换行 <br />
-      // 冒号前加粗,描述部分保持普通
-
-      // 拼接“兴趣描述”标题和换行
-      const fullContent = `<strong class="fontsize">兴趣描述</strong>:<br />${contentFormatted}`;
-
-      // 拼接最终输出
+      // 构建兴趣标签 HTML
+      const tagsHtml = contentObj.interestTags.map((tag: string) => `
+        <div class="tag-item">
+          <div class="tag-icon">
+            <ion-icon name="star"></ion-icon>
+          </div>
+          <span>${tag}</span>
+        </div>
+      `).join('');
+
+      // 处理分析内容,将内容按句号分段并添加样式
+      const contentText = contentObj.content;
+      const paragraphs = contentText
+        .split('。')
+        .filter((p: string) => p.trim())
+        .map((p: string) => `
+          <div class="analysis-item">
+            <div class="bullet-point"></div>
+            <p>${p}。</p>
+          </div>
+        `)
+        .join('');
+
+      // 返回完整的 HTML 结构
       return `
-      <strong class="fontsize">兴趣标签:</strong><br> ${interestTagsFormatted}<br><br>
-      ${fullContent}
-    `;
+        <div class="analysis-result">
+          <div class="result-section tags-section">
+            <div class="section-title">
+              <ion-icon name="ribbon"></ion-icon>
+              <h2>您的兴趣标签</h2>
+            </div>
+            <div class="tags-wrapper">
+              ${tagsHtml}
+            </div>
+          </div>
+
+          <div class="result-section report-section">
+            <div class="section-title">
+              <ion-icon name="document-text"></ion-icon>
+              <h2>个性化分析报告</h2>
+            </div>
+            <div class="content-wrapper">
+              ${paragraphs}
+            </div>
+          </div>
+        </div>
+      `;
     } catch (error) {
       console.error('格式化 AI 响应时出错:', error);
       return '分析结果格式错误。';
@@ -495,7 +571,9 @@ export class InterestSearchComponent implements OnInit {
 
   // 关闭 Modal
   closeModal() {
-    this.modalIsOpen = false; // 关闭 Modal
+    if (!this.isAnalyzing) { // 只有在不是分析状态时才允许关闭
+      this.modalIsOpen = false;
+    }
   }
 
   // 保存 AI 分析结果到数据库
@@ -504,16 +582,171 @@ export class InterestSearchComponent implements OnInit {
       const userInterestProfile = new CloudObject("UserInterestProfile");
 
       userInterestProfile.set({
-        userId: { __type: "Pointer", className: "_User", objectId: "user1" },  // 假这是当前用户ID
+        userId: { __type: "Pointer", className: "_User", objectId: "user1" },  // 假这是当前用户ID
         QuestionnaireId: this.questionnaire?.QuestionnaireId,
         interestTags: aiResponse.interestTags,
         content: aiResponse.content
       });
 
       await userInterestProfile.save();
-      console.log("分析结果保存");
+      console.log("分析结果保存");
     } catch (error) {
       console.error('保存分析结果时出错:', error);
     }
   }
+
+  async cancelDate() {
+    await this.datetime.cancel(true);
+  }
+
+  async confirmDate() {
+    await this.datetime.confirm(true);
+  }
+
+  async saveProgress() {
+    try {
+      // 创建一个新的对象来保存问卷进度
+      const surveyProgress = new CloudObject("SurveyProgress");
+
+      // 保存当前的问卷状态
+      surveyProgress.set({
+        userId: { __type: "Pointer", className: "_User", objectId: "user1" }, // 替换为实际的用户ID
+        questionnaireId: this.questionnaire?.QuestionnaireId,
+        answers: this.answers,
+        personalInfo: {
+          name: this.name,
+          gender: this.gender,
+          age: this.age,
+          birthday: this.birthday,
+          occupation: this.occupation
+        },
+        lastUpdated: new Date(),
+        isCompleted: false
+      });
+
+      await surveyProgress.save();
+
+      // 显示保存成功提示
+      const toast = document.createElement('ion-toast');
+      toast.message = '进度保存成功';
+      toast.duration = 2000;
+      toast.position = 'top';
+      toast.color = 'success';
+      document.body.appendChild(toast);
+      await toast.present();
+
+    } catch (error) {
+      console.error('保存进度失败:', error);
+
+      // 显示错误提示
+      const toast = document.createElement('ion-toast');
+      toast.message = '保存失败,请重试';
+      toast.duration = 2000;
+      toast.position = 'top';
+      toast.color = 'danger';
+      document.body.appendChild(toast);
+      await toast.present();
+    }
+  }
+
+  async ngOnInit() {
+    // 加载问卷数据
+    await this.loadQuestionnaireData('q1');
+
+    // 尝试加载保存的进度
+    await this.loadSavedProgress();
+  }
+
+  async loadSavedProgress() {
+    try {
+      const progressQuery = new CloudQuery("SurveyProgress");
+      progressQuery.equalTo("userId", { __type: "Pointer", className: "_User", objectId: "user1" });
+      // progressQuery.order("-lastUpdated"); // 使用 order 方法,负号表示降序
+
+      const savedProgress = await progressQuery.first();
+
+      if (savedProgress) {
+        const progressData = savedProgress.data;
+
+        // 恢复个人信息
+        this.name = progressData['personalInfo']['name'];
+        this.gender = progressData['personalInfo']['gender'];
+        this.age = progressData['personalInfo']['age'];
+        this.birthday = progressData['personalInfo']['birthday'];
+        this.occupation = progressData['personalInfo']['occupation'];
+
+        // 恢复答案
+        this.answers = progressData['answers'];
+
+        console.log('已加载保存的进度');
+      }
+    } catch (error) {
+      console.error('加载保存的进度失败:', error);
+    }
+  }
+
+  // 修改刷新问卷的方法
+  async refreshQuestionnaire() {
+    try {
+      // 显示加载提示
+      const loadingToast = document.createElement('ion-toast');
+      loadingToast.message = '正在更新问卷...';
+      loadingToast.duration = 1000;
+      loadingToast.position = 'top';
+      document.body.appendChild(loadingToast);
+      await loadingToast.present();
+
+      await this.getNewQuestionnaire();
+    } catch (error) {
+      console.error('更新问卷失败:', error);
+      const toast = document.createElement('ion-toast');
+      toast.message = '更新问卷失败,请重试';
+      toast.duration = 2000;
+      toast.position = 'top';
+      toast.color = 'danger';
+      document.body.appendChild(toast);
+      await toast.present();
+    }
+  }
+
+  // 修改获取新问卷的方法
+  private async getNewQuestionnaire() {
+    try {
+      // 获取当前问卷ID
+      const currentId = this.questionnaire?.QuestionnaireId || 'q1';
+
+      // 从问卷列表中随机选择一个不同的问卷
+      let availableQuestionnaires = this.questionnaires.filter(id => id !== currentId);
+      if (availableQuestionnaires.length === 0) {
+        availableQuestionnaires = this.questionnaires;
+      }
+
+      const randomIndex = Math.floor(Math.random() * availableQuestionnaires.length);
+      const newQuestionnaireId = availableQuestionnaires[randomIndex];
+
+      console.log('当前问卷ID:', currentId);
+      console.log('新问卷ID:', newQuestionnaireId);
+
+      // 重置状态
+      this.answers = {};
+      this.questionsWithOptions = [];
+      this.questionnaire = null;
+
+      // 加载新问卷
+      await this.loadQuestionnaireData(newQuestionnaireId);
+
+      // 显示成功提示
+      const toast = document.createElement('ion-toast');
+      toast.message = '问卷已更新';
+      toast.duration = 2000;
+      toast.position = 'top';
+      toast.color = 'success';
+      document.body.appendChild(toast);
+      await toast.present();
+
+    } catch (error) {
+      console.error('获取新问卷失败:', error);
+      throw error; // 向上传递错误,让调用者处理
+    }
+  }
 }

+ 4 - 1
AIart-app/src/app/tabs/tabs.page.ts

@@ -28,6 +28,8 @@ import {
   checkmarkCircle,
   handRight,
   medal,
+  saveOutline,
+  sendOutline,
 } from 'ionicons/icons';
 
 
@@ -56,7 +58,8 @@ export class TabsPage {
       homeOutline, peopleOutline, trashOutline, chatbubbleOutline, informationCircleOutline,
       pricetagOutline, closeCircle, brushOutline, musicalNotesOutline, bodyOutline,
       cameraOutline, codeOutline, restaurantOutline, fitnessOutline, languageOutline,
-      helpCircleOutline, leafOutline, flame, checkmarkCircle, handRight, medal,
+      helpCircleOutline, leafOutline, flame, checkmarkCircle, handRight, medal, saveOutline,
+      sendOutline,
 
     });
   }