3
1

2 Ревизии 1bbc278b1f ... 88a64b3ba6

Автор SHA1 Съобщение Дата
  未来全栈 88a64b3ba6 Merge branch 'master' of http://git.fmode.cn:3000/csdn1233/s202226701049 преди 10 месеца
  未来全栈 8b8955cee0 feat: ncloud.ts преди 10 месеца

+ 32 - 48
AIart-app/src/app/interest-search/interest-search.component.html

@@ -1,3 +1,4 @@
+<!-- src/app/interest-search/interest-search.component.html -->
 <ion-header [translucent]="true">
   <ion-toolbar>
     <ion-buttons slot="start">
@@ -10,66 +11,49 @@
 </ion-header>
 
 <ion-content color="light">
+  <!-- 一、基本情况 -->
   <ion-list [inset]="true">
     <ion-item>
       <ion-label>一、基本情况</ion-label>
     </ion-item>
     <ion-item>
-      <ion-input label="姓名"></ion-input>
+      <ion-label position="floating">姓名</ion-label>
+      <ion-input [(ngModel)]="name"></ion-input>
     </ion-item>
     <ion-item>
-      <span style="margin-right: 30px;">生日</span>
-      <ion-datetime-button datetime="datetime"></ion-datetime-button>
-      <ion-modal [keepContentsMounted]="true">
-        <ng-template>
-          <ion-datetime id="datetime"></ion-datetime>
-        </ng-template>
-      </ion-modal>
+      <ion-label position="floating">生日</ion-label>
+      <ion-datetime displayFormat="YYYY-MM-DD" [(ngModel)]="birthday"></ion-datetime>
     </ion-item>
   </ion-list>
 
+  <!-- 二、问卷问题 -->
   <ion-list [inset]="true">
-    <ion-item>
-      <ion-label>二、工作状况</ion-label>
-    </ion-item>
-    <p style="margin-left: 15px;">工作几年?</p>
-    <ion-radio-group [allowEmptySelection]="true" value="turtles">
-      <ion-radio value="1年-3年" labelPlacement="end" aria-label="Custom checkbox">1年-3年</ion-radio><br />
-      <ion-radio value="3年-5年" labelPlacement="end" aria-label="Custom checkbox">3年-5年</ion-radio><br />
-      <ion-radio value="5年以上" labelPlacement="end" aria-label="Custom checkbox">5年以上</ion-radio><br />
-    </ion-radio-group>
-    <p style="margin-left: 15px;">从事什么职业?</p>
-    <ion-radio-group [allowEmptySelection]="true" value="turtles">
-      <ion-radio value="前端开发" labelPlacement="end" aria-label="Custom checkbox">前端开发</ion-radio><br />
-      <ion-radio value="后端开发" labelPlacement="end" aria-label="Custom checkbox">后端开发</ion-radio><br />
-      <ion-radio value="测试" labelPlacement="end" aria-label="Custom checkbox">测试</ion-radio><br />
-      <ion-radio value="产品经理" labelPlacement="end" aria-label="Custom checkbox">产品经理</ion-radio><br />
-      <ion-radio value="其他" labelPlacement="end" aria-label="Custom checkbox">其他</ion-radio><br />
-      <br />
-    </ion-radio-group>
-  </ion-list>
+    <!-- 遍历 questionsWithOptions 逐个显示问题 -->
+    <div *ngFor="let question of questionsWithOptions">
+      <!-- 问题文本 -->
+      <ion-item>
+        <ion-label>{{ question.questionText }}</ion-label>
+      </ion-item>
 
-  <ion-list [inset]="true">
-    <ion-item>
-      <ion-label>三、其他情况</ion-label>
-    </ion-item>
-    <p style="margin-left: 15px;">是否喜欢吃水果?</p>
-    <ion-radio-group [allowEmptySelection]="true" value="turtles">
-      <ion-radio value="是" labelPlacement="end" aria-label="Custom checkbox">是</ion-radio><br />
-      <ion-radio value="否" labelPlacement="end" aria-label="Custom checkbox">否</ion-radio><br />
-    </ion-radio-group>
-    <p style="margin-left: 15px;">喜欢什么运动?</p>
-    <ion-radio-group [allowEmptySelection]="true" value="turtles">
-      <ion-radio value="篮球" labelPlacement="end" aria-label="Custom checkbox">篮球</ion-radio><br />
-      <ion-radio value="足球" labelPlacement="end" aria-label="Custom checkbox">足球</ion-radio><br />
-      <ion-radio value="羽毛球" labelPlacement="end" aria-label="Custom checkbox">羽毛球</ion-radio><br />
-      <ion-radio value="其他" labelPlacement="end" aria-label="Custom checkbox">其他</ion-radio><br />
-      <br />
-    </ion-radio-group>
+      <!-- 选项列表 -->
+      <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 *ngFor="let option of question.optionsData">
+          <ion-radio slot="start" [value]="option.OptionId"></ion-radio>
+          <ion-label>{{ option.optionText }}</ion-label>
+        </ion-item>
+      </ion-radio-group>
+    </div>
   </ion-list>
-  <div style="display: flex;justify-content: space-between;align-items: center;margin: 0 15px 0 15px;">
-    <ion-button style="width: 45%;">保存</ion-button>
-    <ion-button id="yes/no" style="width: 45%;">提交</ion-button>
-    <ion-alert trigger="yes/no" header="是否确定提交" [buttons]="alertButtons"></ion-alert>
+
+  <!-- 保存和提交按钮 -->
+  <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 style="width: 45%;" (click)="submit()">提交</ion-button>
   </div>
 </ion-content>

+ 356 - 8
AIart-app/src/app/interest-search/interest-search.component.ts

@@ -1,22 +1,370 @@
 import { Component, OnInit } from '@angular/core';
-import { IonTextarea, IonCheckbox, IonList, IonButton, IonContent, IonHeader, IonInput, IonTitle, IonToolbar, IonItem, IonLabel, IonRadioGroup, IonRadio, IonDatetimeButton, IonDatetime, IonModal, IonAlert, IonBackButton, IonButtons } from '@ionic/angular/standalone';
+import {
+  IonTextarea,
+  IonCheckbox,
+  IonList,
+  IonButton,
+  IonContent,
+  IonHeader,
+  IonInput,
+  IonTitle,
+  IonToolbar,
+  IonItem,
+  IonLabel,
+  IonRadioGroup,
+  IonRadio,
+  IonDatetimeButton,
+  IonDatetime,
+  IonModal
+} from '@ionic/angular/standalone';
+import { CloudQuery, CloudObject, Pointer } from '../../lib/ncloud'; // 确保路径正确
+import { CommonModule } from '@angular/common'; // 导入 CommonModule
+import { FormsModule } from '@angular/forms';   // 导入 FormsModule
+
+// 定义接口以确保类型安全
+interface Questionnaire {
+  objectId: string;
+  createdAt: string;
+  QuestionnaireId: string;
+  title: string;
+  status: string;
+  questions: string[]; // 修改为字符串数组
+}
+
+interface Question {
+  objectId: string;
+  createdAt: string;
+  QuestionId: string;
+  questionnaireId: string; // 修改为字符串
+  questionText: string;
+  options: string[]; // 修改为字符串数组
+}
+
+interface Option {
+  objectId: string;
+  createdAt: string;
+  OptionId: string;
+  questionId: string; // 修改为字符串
+  optionText: string;
+  isSelected: boolean;
+}
+
+interface QuestionnaireResult {
+  objectId: string;
+  createdAt: string;
+  QuestionnaireResultId: string;
+  userId: Pointer;
+  questionnaireId: Pointer;
+  answers: Pointer[];
+}
+
+interface QuestionWithOptions extends Question {
+  optionsData: Option[];
+}
 
 @Component({
   selector: 'app-interest-search',
   templateUrl: './interest-search.component.html',
   styleUrls: ['./interest-search.component.scss'],
   standalone: true,
-  imports: [IonHeader, IonToolbar, IonTitle, IonContent, IonButton, IonTextarea, IonInput,
-    IonList, IonItem, IonLabel, IonCheckbox, IonRadioGroup, IonRadio, IonDatetimeButton,
-    IonDatetime, IonModal, IonAlert, IonBackButton, IonButtons,
-  ],
+  imports: [
+    IonTextarea,
+    IonCheckbox,
+    IonList,
+    IonButton,
+    IonContent,
+    IonHeader,
+    IonInput,
+    IonTitle,
+    IonToolbar,
+    IonItem,
+    IonLabel,
+    IonRadioGroup,
+    IonRadio,
+    IonDatetimeButton,
+    IonDatetime,
+    IonModal,
+    CommonModule,
+    FormsModule
+  ]
 })
 export class InterestSearchComponent implements OnInit {
+  // 固定字段
+  name: string = '';
+  birthday: string = '';
+
+  // 动态问卷数据
+  questionnaire: Questionnaire | null = null;
+  questionsWithOptions: QuestionWithOptions[] = [];
+  answers: { [questionId: string]: string } = {}; // 存储用户答案
+
+  constructor() { }
 
-  constructor() {
+  ngOnInit() {
+    this.loadQuestionnaireData('test_q1'); // 使用 QuestionnaireId 'test_q1'
   }
 
-  alertButtons = ['确定'];
-  ngOnInit() { }
 
+  async loadQuestionnaireData(questionnaireId: string) {
+    try {
+      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)
+        };
+
+        console.log("加载到的问卷数据:", this.questionnaire);
+
+        // 确保 questions 被正确传入
+        if (this.questionnaire.questions) {
+          await this.loadQuestions(this.questionnaire.questions);
+        }
+      } else {
+        console.error(`未找到 QuestionnaireId 为 ${questionnaireId} 的问卷。`);
+      }
+    } catch (error) {
+      console.error("加载问卷数据时出错:", error);
+    }
+  }
+  async loadQuestions(questionIds: string[]) {
+    this.questionsWithOptions = []; // 初始化问题列表
+
+    for (const questionId of questionIds) {
+      try {
+        const questionQuery = new CloudQuery("Question");
+        questionQuery.equalTo("QuestionId", questionId);
+        const questionObj = await questionQuery.first();
+
+        if (questionObj) {
+          const question = questionObj.data as Question;
+
+          // 异步加载选项并立即显示问题
+          this.questionsWithOptions.push({ ...question, optionsData: [] });
+
+          this.loadOptions(question.options).then((options) => {
+            const index = this.questionsWithOptions.findIndex(
+              (q) => q.QuestionId === question.QuestionId
+            );
+            if (index !== -1) {
+              this.questionsWithOptions[index].optionsData = options;
+            }
+          });
+
+          // 可选:每加载一个问题,立即触发渲染
+          console.log("已加载问题:", question);
+        }
+      } catch (error) {
+        console.error(`加载问题 ID ${questionId} 时出错:`, error);
+      }
+    }
+  }
+  async loadOptions(optionIds: string[]): Promise<Option[]> {
+    try {
+      if (!optionIds || optionIds.length === 0) return [];
+      const optionQuery = new CloudQuery("Option");
+      optionQuery.containedIn("OptionId", optionIds); // 批量查询
+      const optionObjs = await optionQuery.find();
+      return optionObjs.map((optionObj: any) => optionObj.data as Option);
+    } catch (error) {
+      console.error("加载选项时出错:", error);
+      return [];
+    }
+  }
+
+  /*
+  async loadQuestions(questionIds: string[]) {
+    try {
+      const questionQuery = new CloudQuery("Question");
+      questionQuery.containedIn("QuestionId", questionIds); // 一次查询多个 QuestionId
+      const questionObjs = await questionQuery.find();
+
+      const questionsWithOptions: QuestionWithOptions[] = await Promise.all(
+        questionObjs.map(async (questionObj: any) => {
+          const question = questionObj.data as Question;
+          const options = await this.loadOptions(question.options);
+          return { ...question, optionsData: options };
+        })
+      );
+
+      this.questionsWithOptions = questionsWithOptions;
+    } catch (error) {
+      console.error("批量加载问题时出错:", error);
+    }
+  }
+  async loadOptions(optionIds: string[]): Promise<Option[]> {
+    try {
+      const optionQuery = new CloudQuery("Option");
+      optionQuery.containedIn("OptionId", optionIds); // 批量查询所有 OptionId
+      const optionObjs = await optionQuery.find();
+
+      return optionObjs.map((optionObj: any) => optionObj.data as Option);
+    } catch (error) {
+      console.error("批量加载选项时出错:", error);
+      return [];
+    }
+  }
+*/
+  /*
+    // 加载问卷中的问题及选项
+    async loadQuestions(questionIds: string[]) {
+      try {
+        const questions: QuestionWithOptions[] = [];
+  
+        for (const questionId of questionIds) {
+          const questionQuery = new CloudQuery("Question");
+          questionQuery.equalTo("QuestionId", questionId);
+          const questionObj = await questionQuery.first();
+  
+          if (!questionObj) {
+            console.error(`未找到问题ID为 ${questionId} 的问题。`);
+            continue;
+          }
+  
+          const question = questionObj.data as Question;
+          const options = await this.loadOptions(question.options); // 加载选项
+  
+          questions.push({
+            ...question,
+            optionsData: options
+          });
+        }
+  
+        this.questionsWithOptions = questions;
+      } catch (error) {
+        console.error("加载问题时出错:", error);
+      }
+    }
+  
+    // 加载选项
+    async loadOptions(optionIds: string[]): Promise<Option[]> {
+      try {
+        const options: Option[] = [];
+  
+        for (const optionId of optionIds) {
+          const optionQuery = new CloudQuery("Option");
+          optionQuery.equalTo("OptionId", optionId);
+          const optionObj = await optionQuery.first();
+  
+          if (!optionObj) {
+            console.error(`未找到选项ID为 ${optionId} 的选项。`);
+            continue;
+          }
+  
+          const option = optionObj.data as Option;
+          options.push(option);
+        }
+  
+        return options;
+      } catch (error) {
+        console.error("加载选项时出错:", error);
+        return [];
+      }
+    }
+  */
+
+
+  // 保存功能(可选)
+  async save() {
+    try {
+      // 实现保存逻辑,例如保存到本地存储或发送到后台
+      console.log("保存的答案:", this.answers);
+      console.log("姓名:", this.name);
+      console.log("生日:", this.birthday);
+    } catch (error) {
+      console.error("保存答案时出错:", error);
+    }
+  }
+
+  // 提交功能
+  async submit() {
+    try {
+      if (!this.questionnaire) {
+        console.error("未加载问卷数据。");
+        return;
+      }
+
+      // 创建一个数组保存选中的 OptionId
+      const answersArray: string[] = [];
+
+      // 遍历每个问题,获取用户选择的选项
+      for (const question of this.questionsWithOptions) {
+        const selectedOptionId = this.answers[question.QuestionId];
+        if (selectedOptionId) {
+          // 将选中的 OptionId 存入 answersArray
+          answersArray.push(selectedOptionId);
+        }
+      }
+
+      // 创建一个新的 QuestionnaireResult 对象
+      const questionnaireResult = new CloudObject("QuestionnaireResult");
+
+      // 设置 QuestionnaireResult 的属性
+      questionnaireResult.set({
+        QuestionnaireResultId: `qr_${new Date().getTime()}`, // 生成唯一的 QuestionnaireResultId
+        userId: { __type: "Pointer", className: "_User", objectId: "user1" }, // 替换为实际的用户ID
+        // 使用 Pointer 类型的引用方式来设置 questionnaireId
+        questionnaireId: { __type: "Pointer", className: "Questionnaire", objectId: this.questionnaire.objectId },
+        answers: answersArray // 将选中的 OptionId 数组存入 answers 字段
+      });
+
+      // 保存 QuestionnaireResult 对象
+      await questionnaireResult.save();
+      console.log("问卷提交成功。");
+
+      // 可选:清空表单
+      this.name = '';
+      this.birthday = '';
+      this.answers = {};
+    } catch (error) {
+      console.error("提交问卷时出错:", error);
+    }
+  }
+
+
+  /*async submit() {
+    try {
+      if (!this.questionnaire) {
+        console.error("未加载问卷数据。");
+        return;
+      }
+
+      const answersPointers: Pointer[] = [];
+      for (const question of this.questionsWithOptions) {
+        const selectedOptionId = this.answers[question.QuestionId];
+        if (selectedOptionId) {
+          const optionPointer: Pointer = {
+            __type: "Pointer",
+            className: "Option",
+            objectId: selectedOptionId // 确保使用 objectId
+          };
+          answersPointers.push(optionPointer);
+        }
+      }
+
+      const questionnaireResult = new CloudObject("QuestionnaireResult");
+      questionnaireResult.set({
+        QuestionnaireResultId: `qr_${new Date().getTime()}`, // 生成唯一的 QuestionnaireResultId
+        userId: { __type: "Pointer", className: "_User", objectId: "user1" }, // 替换为实际的用户ID
+        questionnaireId: { __type: "Pointer", className: "Questionnaire", QuestionnaireId: this.questionnaire.QuestionnaireId },
+        answers: answersPointers
+      });
+
+      await questionnaireResult.save();
+      console.log("问卷提交成功。");
+
+      // 可选:清空表单
+      this.name = '';
+      this.birthday = '';
+      this.answers = {};
+    } catch (error) {
+      console.error("提交问卷时出错:", error);
+    }
+  }*/
 }

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

@@ -12,6 +12,7 @@ import {
   addCircleOutline, addOutline, optionsOutline
 } from 'ionicons/icons';
 
+
 @Component({
   selector: 'app-tabs',
   templateUrl: 'tabs.page.html',

+ 244 - 205
AIart-app/src/lib/ncloud.ts

@@ -1,23 +1,58 @@
+/*export interface Pointer {
+    __type: string;
+    className: string;
+    objectId?: string; // 这里保持 objectId 作为可选字段
+    //QuestionnaireId?: string; // 自定义主键
+    //QuestionId?: string; // 自定义主键
+    //OptionId?: string; // 自定义主键
+    //_UserId?: string; // 自定义主键
+}
 
-// CloudObject.ts
 export class CloudObject {
+    id: string | null = null; // 数据库生成的 objectId
     className: string;
-    id: string | null = null;
-    createdAt: any;
-    updatedAt: any;
+    createdAt: string | null = null;
+    updatedAt: string | null = null;
     data: Record<string, any> = {};
 
+    // 新增的自定义主键
+    QuestionnaireId?: string;
+    QuestionId?: string;
+    OptionId?: string;
+    _UserId?: string;
+
     constructor(className: string) {
         this.className = className;
     }
 
-    toPointer() {
-        return { "__type": "Pointer", "className": this.className, "objectId": this.id };
+    toPointer(): Pointer {
+        // 使用自定义主键生成指针
+        let pointer: Pointer = { "__type": "Pointer", "className": this.className };
+
+        // 根据类名设置相应的 ID
+        switch (this.className) {
+            case "Questionnaire":
+                pointer.objectId = this.QuestionnaireId || "";
+                break;
+            case "Question":
+                pointer.objectId = this.QuestionId || "";
+                break;
+            case "Option":
+                pointer.objectId = this.OptionId || "";
+                break;
+            case "_User":
+                pointer.objectId = this._UserId || "";
+                break;
+            default:
+                pointer.objectId = this.id || ""; // 保持 objectId 作为后备
+        }
+
+        return pointer;
     }
 
     set(json: Record<string, any>) {
         Object.keys(json).forEach(key => {
-            if (["objectId", "id", "createdAt", "updatedAt"].indexOf(key) > -1) {
+            if (["objectId", "id", "createdAt", "updatedAt", "ACL"].includes(key)) {
                 return;
             }
             this.data[key] = json[key];
@@ -28,7 +63,7 @@ export class CloudObject {
         return this.data[key] || null;
     }
 
-    async save() {
+    async save(): Promise<this> {
         let method = "POST";
         let url = `http://dev.fmode.cn:1337/parse/classes/${this.className}`;
 
@@ -50,29 +85,27 @@ export class CloudObject {
             credentials: "omit"
         });
 
-        const result = await response?.json();
+        const result = await response.json();
         if (result?.error) {
-            console.error(result?.error);
+            console.error(result.error);
         }
         if (result?.objectId) {
-            this.id = result?.objectId;
+            this.id = result.objectId;
         }
         return this;
     }
 
-    async destroy() {
-        if (!this.id) return;
+    async destroy(): Promise<boolean> {
+        if (!this.id) return false;
         const response = await fetch(`http://dev.fmode.cn:1337/parse/classes/${this.className}/${this.id}`, {
             headers: {
                 "x-parse-application-id": "dev"
             },
-            body: null,
             method: "DELETE",
             mode: "cors",
             credentials: "omit"
         });
-
-        const result = await response?.json();
+        const result = await response.json();
         if (result) {
             this.id = null;
         }
@@ -80,300 +113,306 @@ export class CloudObject {
     }
 }
 
-// CloudQuery.ts
 export class CloudQuery {
     className: string;
-    queryParams: Record<string, any> = {};
+    whereOptions: Record<string, any> = {};
 
     constructor(className: string) {
         this.className = className;
     }
 
-    include(...fileds: string[]) {
-        this.queryParams["include"] = fileds;
-    }
     greaterThan(key: string, value: any) {
-        if (!this.queryParams["where"][key]) this.queryParams["where"][key] = {};
-        this.queryParams["where"][key]["$gt"] = value;
+        if (!this.whereOptions[key]) this.whereOptions[key] = {};
+        this.whereOptions[key]["$gt"] = value;
+        return this;
     }
 
     greaterThanAndEqualTo(key: string, value: any) {
-        if (!this.queryParams["where"][key]) this.queryParams["where"][key] = {};
-        this.queryParams["where"][key]["$gte"] = value;
+        if (!this.whereOptions[key]) this.whereOptions[key] = {};
+        this.whereOptions[key]["$gte"] = value;
+        return this;
     }
 
     lessThan(key: string, value: any) {
-        if (!this.queryParams["where"][key]) this.queryParams["where"][key] = {};
-        this.queryParams["where"][key]["$lt"] = value;
+        if (!this.whereOptions[key]) this.whereOptions[key] = {};
+        this.whereOptions[key]["$lt"] = value;
+        return this;
     }
 
     lessThanAndEqualTo(key: string, value: any) {
-        if (!this.queryParams["where"][key]) this.queryParams["where"][key] = {};
-        this.queryParams["where"][key]["$lte"] = value;
+        if (!this.whereOptions[key]) this.whereOptions[key] = {};
+        this.whereOptions[key]["$lte"] = value;
+        return this;
     }
 
     equalTo(key: string, value: any) {
-        this.queryParams["where"][key] = value;
+        this.whereOptions[key] = value;
+        return this;
     }
 
-    async get(id: string) {
-        const url = `http://dev.fmode.cn:1337/parse/classes/${this.className}/${id}?`;
-
+    async get(id: string): Promise<Record<string, any>> {
+        const url = `http://dev.fmode.cn:1337/parse/classes/${this.className}/${id}`;
         const response = await fetch(url, {
             headers: {
-                "if-none-match": "W/\"1f0-ghxH2EwTk6Blz0g89ivf2adBDKY\"",
                 "x-parse-application-id": "dev"
             },
-            body: null,
             method: "GET",
             mode: "cors",
             credentials: "omit"
         });
-
-        const json = await response?.json();
+        const json = await response.json();
         return json || {};
     }
 
-    async find() {
+    async find(): Promise<CloudObject[]> {
         let url = `http://dev.fmode.cn:1337/parse/classes/${this.className}?`;
 
-        let queryStr = ``
-        Object.keys(this.queryParams).forEach(key => {
-            let paramStr = JSON.stringify(this.queryParams[key]);
-            if (key == "include") {
-                paramStr = this.queryParams[key]?.join(",")
-            }
-            if (queryStr) {
-                url += `${key}=${paramStr}`;
-            } else {
-                url += `&${key}=${paramStr}`;
-            }
-        })
-        // if (Object.keys(this.queryParams["where"]).length) {
-
-        // }
+        if (Object.keys(this.whereOptions).length) {
+            const whereStr = JSON.stringify(this.whereOptions);
+            url += `where=${encodeURIComponent(whereStr)}`;
+        }
 
         const response = await fetch(url, {
             headers: {
-                "if-none-match": "W/\"1f0-ghxH2EwTk6Blz0g89ivf2adBDKY\"",
                 "x-parse-application-id": "dev"
             },
-            body: null,
             method: "GET",
             mode: "cors",
             credentials: "omit"
         });
-
-        const json = await response?.json();
-        let list = json?.results || []
-        let objList = list.map((item: any) => this.dataToObj(item))
-        return objList || [];
+        const json = await response.json();
+        return json?.results.map((item: any) => {
+            const cloudObject = new CloudObject(this.className);
+            cloudObject.set(item);
+            cloudObject.id = item.objectId;
+            cloudObject.createdAt = item.createdAt;
+            cloudObject.updatedAt = item.updatedAt;
+            return cloudObject;
+        }) || [];
     }
 
-
-    async first() {
+    async first(): Promise<CloudObject | null> {
         let url = `http://dev.fmode.cn:1337/parse/classes/${this.className}?`;
 
-        if (Object.keys(this.queryParams["where"]).length) {
-            const whereStr = JSON.stringify(this.queryParams["where"]);
-            url += `where=${whereStr}`;
+        if (Object.keys(this.whereOptions).length) {
+            const whereStr = JSON.stringify(this.whereOptions);
+            url += `where=${encodeURIComponent(whereStr)}&limit=1`;
+        } else {
+            url += `limit=1`;
         }
 
         const response = await fetch(url, {
             headers: {
-                "if-none-match": "W/\"1f0-ghxH2EwTk6Blz0g89ivf2adBDKY\"",
                 "x-parse-application-id": "dev"
             },
-            body: null,
             method: "GET",
             mode: "cors",
             credentials: "omit"
         });
-
-        const json = await response?.json();
+        const json = await response.json();
         const exists = json?.results?.[0] || null;
         if (exists) {
-            let existsObject = this.dataToObj(exists)
-            return existsObject;
+            const cloudObject = new CloudObject(this.className);
+            cloudObject.set(exists);
+            cloudObject.id = exists.objectId;
+            cloudObject.createdAt = exists.createdAt;
+            cloudObject.updatedAt = exists.updatedAt;
+            return cloudObject;
         }
-        return null
+        return null;
     }
+}*/
+// CloudObject.ts
+export interface Pointer {
+    __type: string;
+    className: string;
+    objectId?: string; // 这里保持 objectId 作为可选字段
+}
+export class CloudObject {
+    id: string | null = null;
+    className: string;
+    data: Record<string, any> = {};
+    createdAt: string | null = null;
+    updatedAt: string | null = null;
 
-    dataToObj(exists: any): CloudObject {
-        let existsObject = new CloudObject(this.className);
-        existsObject.set(exists);
-        existsObject.id = exists.objectId;
-        existsObject.createdAt = exists.createdAt;
-        existsObject.updatedAt = exists.updatedAt;
-        return existsObject;
+    constructor(className: string) {
+        this.className = className;
     }
-}
 
-// CloudUser.ts
-export class CloudUser extends CloudObject {
-    constructor() {
-        super("_User"); // 假设用户类在Parse中是"_User"
-        // 读取用户缓存信息
-        let userCacheStr = localStorage.getItem("NCloud/dev/User")
-        if (userCacheStr) {
-            let userData = JSON.parse(userCacheStr)
-            // 设置用户信息
-            this.id = userData?.objectId;
-            this.sessionToken = userData?.sessionToken;
-            this.data = userData; // 保存用户数据
-        }
+    toPointer() {
+        return { "__type": "Pointer", "className": this.className, "objectId": this.id };
     }
 
-    sessionToken: string | null = ""
-    /** 获取当前用户信息 */
-    async current() {
-        if (!this.sessionToken) {
-            console.error("用户未登录");
-            return null;
-        }
-        return this;
-        // const response = await fetch(`http://dev.fmode.cn:1337/parse/users/me`, {
-        //     headers: {
-        //         "x-parse-application-id": "dev",
-        //         "x-parse-session-token": this.sessionToken // 使用sessionToken进行身份验证
-        //     },
-        //     method: "GET"
-        // });
-
-        // const result = await response?.json();
-        // if (result?.error) {
-        //     console.error(result?.error);
-        //     return null;
-        // }
-        // return result;
+    set(json: Record<string, any>) {
+        Object.keys(json).forEach(key => {
+            if (["objectId", "id", "createdAt", "updatedAt", "ACL"].includes(key)) {
+                return;
+            }
+            this.data[key] = json[key];
+        });
+    }
+
+    get(key: string) {
+        return this.data[key] || null;
     }
 
-    /** 登录 */
-    async login(username: string, password: string): Promise<CloudUser | null> {
-        const response = await fetch(`http://dev.fmode.cn:1337/parse/login`, {
+    async save(): Promise<CloudObject> {
+        let method = "POST";
+        let url = `http://dev.fmode.cn:1337/parse/classes/${this.className}`;
+
+        // 更新
+        if (this.id) {
+            url += `/${this.id}`;
+            method = "PUT";
+        }
+
+        const body = JSON.stringify(this.data);
+        const response = await fetch(url, {
             headers: {
-                "x-parse-application-id": "dev",
-                "Content-Type": "application/json"
+                "Content-Type": "application/json;charset=UTF-8",
+                "x-parse-application-id": "dev"
             },
-            body: JSON.stringify({ username, password }),
-            method: "POST"
+            body,
+            method,
+            mode: "cors",
+            credentials: "omit"
         });
 
-        const result = await response?.json();
+        const result = await response.json();
         if (result?.error) {
-            console.error(result?.error);
-            return null;
+            console.error(result.error);
+        }
+        if (result?.objectId) {
+            this.id = result.objectId;
         }
-
-        // 设置用户信息
-        this.id = result?.objectId;
-        this.sessionToken = result?.sessionToken;
-        this.data = result; // 保存用户数据
-        // 缓存用户信息
-        console.log(result)
-        localStorage.setItem("NCloud/dev/User", JSON.stringify(result))
         return this;
     }
 
-    /** 登出 */
-    async logout() {
-        if (!this.sessionToken) {
-            console.error("用户未登录");
-            return;
-        }
+    async destroy(): Promise<boolean> {
+        if (!this.id) return false;
 
-        const response = await fetch(`http://dev.fmode.cn:1337/parse/logout`, {
+        const response = await fetch(`http://dev.fmode.cn:1337/parse/classes/${this.className}/${this.id}`, {
             headers: {
-                "x-parse-application-id": "dev",
-                "x-parse-session-token": this.sessionToken
+                "x-parse-application-id": "dev"
             },
-            method: "POST"
+            method: "DELETE",
+            mode: "cors",
+            credentials: "omit"
         });
 
         const result = await response?.json();
-        if (result?.error) {
-            console.error(result?.error);
-            return false;
+        if (result) {
+            this.id = null;
         }
-
-        // 清除用户信息
-        localStorage.removeItem("NCloud/dev/User")
-        this.id = null;
-        this.sessionToken = null;
-        this.data = {};
         return true;
     }
+}
+
+// CloudQuery.ts
+export class CloudQuery {
+    className: string;
+    whereOptions: Record<string, any> = {};
+
+    constructor(className: string) {
+        this.className = className;
+    }
 
-    /** 注册 */
-    async signUp(username: string, password: string, additionalData: Record<string, any> = {}) {
-        const userData = {
-            username,
-            password,
-            ...additionalData // 合并额外的用户数据
-        };
+    greaterThan(key: string, value: any) {
+        this.whereOptions[key] = { "$gt": value };
+    }
 
-        const response = await fetch(`http://dev.fmode.cn:1337/parse/users`, {
+    greaterThanAndEqualTo(key: string, value: any) {
+        this.whereOptions[key] = { "$gte": value };
+    }
+
+    lessThan(key: string, value: any) {
+        this.whereOptions[key] = { "$lt": value };
+    }
+
+    lessThanAndEqualTo(key: string, value: any) {
+        this.whereOptions[key] = { "$lte": value };
+    }
+
+    equalTo(key: string, value: any) {
+        this.whereOptions[key] = value;
+    }
+
+    containedIn(key: string, valueArray: any[]) {
+        this.whereOptions[key] = { "$in": valueArray };
+    }
+
+    async get(id: string): Promise<Record<string, any>> {
+        const url = `http://dev.fmode.cn:1337/parse/classes/${this.className}/${id}?`;
+
+        const response = await fetch(url, {
             headers: {
-                "x-parse-application-id": "dev",
-                "Content-Type": "application/json"
+                "if-none-match": "W/\"1f0-ghxH2EwTk6Blz0g89ivf2adBDKY\"",
+                "x-parse-application-id": "dev"
             },
-            body: JSON.stringify(userData),
-            method: "POST"
+            method: "GET",
+            mode: "cors",
+            credentials: "omit"
         });
 
-        const result = await response?.json();
-        if (result?.error) {
-            console.error(result?.error);
-            return null;
+        const json = await response.json();
+        return json || {};
+    }
+
+    async find(): Promise<CloudObject[]> {
+        let url = `http://dev.fmode.cn:1337/parse/classes/${this.className}?`;
+
+        if (Object.keys(this.whereOptions).length) {
+            const whereStr = JSON.stringify(this.whereOptions);
+            url += `where=${encodeURIComponent(whereStr)}`;
         }
 
-        // 设置用户信息
-        // 缓存用户信息
-        console.log(result)
-        localStorage.setItem("NCloud/dev/User", JSON.stringify(result))
-        this.id = result?.objectId;
-        this.sessionToken = result?.sessionToken;
-        this.data = result; // 保存用户数据
-        return this;
+        const response = await fetch(url, {
+            headers: {
+                "if-none-match": "W/\"1f0-ghxH2EwTk6Blz0g89ivf2BDKY\"",
+                "x-parse-application-id": "dev"
+            },
+            method: "GET",
+            mode: "cors",
+            credentials: "omit"
+        });
+
+        const json = await response.json();
+        return json?.results.map((item: any) => {
+            const cloudObject = new CloudObject(this.className);
+            cloudObject.set(item);
+            cloudObject.id = item.objectId;
+            return cloudObject;
+        }) || [];
     }
 
-    override async save() {
-        let method = "POST";
-        let url = `http://dev.fmode.cn:1337/parse/users`;
+    async first(): Promise<CloudObject | null> {
+        let url = `http://dev.fmode.cn:1337/parse/classes/${this.className}?`;
 
-        // 更新用户信息
-        if (this.id) {
-            url += `/${this.id}`;
-            method = "PUT";
+        if (Object.keys(this.whereOptions).length) {
+            const whereStr = JSON.stringify(this.whereOptions);
+            url += `where=${encodeURIComponent(whereStr)}`;
         }
 
-        let data: any = JSON.parse(JSON.stringify(this.data))
-        delete data.createdAt
-        delete data.updatedAt
-        delete data.ACL
-        delete data.objectId
-        const body = JSON.stringify(data);
-        let headersOptions: any = {
-            "content-type": "application/json;charset=UTF-8",
-            "x-parse-application-id": "dev",
-            "x-parse-session-token": this.sessionToken, // 添加sessionToken以进行身份验证
-        }
         const response = await fetch(url, {
-            headers: headersOptions,
-            body: body,
-            method: method,
+            headers: {
+                "if-none-match": "W/\"1f0-ghxH2EwTk6Blz0g89ivf2adBDKY\"",
+                "x-parse-application-id": "dev"
+            },
+            method: "GET",
             mode: "cors",
             credentials: "omit"
         });
 
-        const result = await response?.json();
-        if (result?.error) {
-            console.error(result?.error);
-        }
-        if (result?.objectId) {
-            this.id = result?.objectId;
+        const json = await response.json();
+        const exists = json?.results?.[0] || null;
+        if (exists) {
+            const existsObject = new CloudObject(this.className);
+            existsObject.set(exists);
+            existsObject.id = exists.objectId;
+            existsObject.createdAt = exists.createdAt;
+            existsObject.updatedAt = exists.updatedAt;
+            return existsObject;
         }
-        localStorage.setItem("NCloud/dev/User", JSON.stringify(this.data))
-        return this;
+        return null;
     }
 }

+ 5 - 0
Alart-server/lib/ncloud.js

@@ -90,6 +90,11 @@ class CloudQuery {
     equalTo(key, value) {
         this.whereOptions[key] = value
     }
+    // 支持 $in 查询(这个就是你需要的 containedIn 方法)
+    containedIn(key, valueArray) {
+        if (!this.whereOptions[key]) this.whereOptions[key] = {}
+        this.whereOptions[key]["$in"] = valueArray
+    }
 
     async get(id) {
         let url = "http://dev.fmode.cn:1337/parse/classes/" + this.className + "/" + id + "?"

+ 241 - 122
Alart-server/migration/data.js

@@ -1,59 +1,4 @@
-module.exports.QuestionnaireResultList = [
-    {
-        "objectId": "result1",
-        "createdAt": "2024-12-10T11:20:24Z",
-        "QuestionnaireResultId": "qr1", // 新增的非数据库生成的主键
-        "userId": {
-            "__type": "Pointer",
-            "className": "User",
-            "objectId": "user1"
-        },
-        "questionnaireId": {
-            "__type": "Pointer",
-            "className": "Questionnaire",
-            "objectId": "questionnaire1"
-        },
-        "answers": [
-            {
-                "__type": "Pointer",
-                "className": "Option",
-                "objectId": "option1"
-            },
-            {
-                "__type": "Pointer",
-                "className": "Option",
-                "objectId": "option2"
-            }
-        ]
-    },
-    {
-        "objectId": "result2",
-        "createdAt": "2024-12-10T11:20:24Z",
-        "QuestionnaireResultId": "qr2", // 新增的非数据库生成的主键
-        "userId": {
-            "__type": "Pointer",
-            "className": "User",
-            "objectId": "user2"
-        },
-        "questionnaireId": {
-            "__type": "Pointer",
-            "className": "Questionnaire",
-            "objectId": "questionnaire1"
-        },
-        "answers": [
-            {
-                "__type": "Pointer",
-                "className": "Option",
-                "objectId": "option3"
-            },
-            {
-                "__type": "Pointer",
-                "className": "Option",
-                "objectId": "option4"
-            }
-        ]
-    }
-];
+
 
 module.exports.QuestionnaireList = [
     {
@@ -63,18 +8,24 @@ module.exports.QuestionnaireList = [
         "title": "兴趣调查问卷",
         "status": "已发布",
         "questions": [
-            {
-                "__type": "Pointer",
-                "className": "Question",
-                "objectId": "question1"
-            },
-            {
-                "__type": "Pointer",
-                "className": "Question",
-                "objectId": "question2"
-            }
+            "question1", // 修改为 String 类型
+            "question2"  // 修改为 String 类型
+        ]
+    },
+
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "QuestionnaireId": "test_q1", // 新增的非数据库生成的主键
+        "title": "基于基本情况的兴趣调查问卷",
+        "status": "已发布",
+        "questions": [
+            "test_q1_q1", // 修改为 String 类型
+            "test_q1_q2", // 修改为 String 类型
+            "test_q1_q3", // 修改为 String 类型
+            "test_q1_q4"  // 修改为 String 类型
         ]
     }
+
 ];
 
 module.exports.QuestionList = [
@@ -82,46 +33,68 @@ module.exports.QuestionList = [
         "objectId": "question1",
         "createdAt": "2024-12-10T11:20:24Z",
         "QuestionId": "q1", // 新增的非数据库生成的主键
-        "questionnaireId": {
-            "__type": "Pointer",
-            "className": "Questionnaire",
-            "objectId": "questionnaire1"
-        },
+        "questionnaireId": "questionnaire1", // 修改为 String 类型
         "questionText": "你最喜欢的活动是什么?",
         "options": [
-            {
-                "__type": "Pointer",
-                "className": "Option",
-                "objectId": "option1"
-            },
-            {
-                "__type": "Pointer",
-                "className": "Option",
-                "objectId": "option2"
-            }
+            "option1", // 修改为 String 类型
+            "option2"  // 修改为 String 类型
         ]
     },
     {
         "objectId": "question2",
         "createdAt": "2024-12-10T11:20:24Z",
         "QuestionId": "q2", // 新增的非数据库生成的主键
-        "questionnaireId": {
-            "__type": "Pointer",
-            "className": "Questionnaire",
-            "objectId": "questionnaire1"
-        },
+        "questionnaireId": "questionnaire1", // 修改为 String 类型
         "questionText": "你喜欢的休闲方式是什么?",
         "options": [
-            {
-                "__type": "Pointer",
-                "className": "Option",
-                "objectId": "option3"
-            },
-            {
-                "__type": "Pointer",
-                "className": "Option",
-                "objectId": "option4"
-            }
+            "option3", // 修改为 String 类型
+            "option4"  // 修改为 String 类型
+        ]
+    },
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "QuestionId": "test_q1_q1", // 新增的非数据库生成的主键
+        "questionnaireId": "test_q1", // 修改为 String 类型
+        "questionText": "你工作几年了?",
+        "options": [
+            "test_q1_q1_o1", // 修改为 String 类型
+            "test_q1_q1_o2", // 修改为 String 类型
+            "test_q1_q1_o3"  // 修改为 String 类型
+        ]
+    },
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "QuestionId": "test_q1_q2", // 新增的非数据库生成的主键
+        "questionnaireId": "test_q1", // 修改为 String 类型
+        "questionText": "从事什么类型的职业?",
+        "options": [
+            "test_q1_q2_o1", // 修改为 String 类型
+            "test_q1_q2_o2", // 修改为 String 类型
+            "test_q1_q2_o3", // 修改为 String 类型
+            "test_q1_q2_o4", // 修改为 String 类型
+            "test_q1_q2_o5"  // 修改为 String 类型
+        ]
+    },
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "QuestionId": "test_q1_q3", // 新增的非数据库生成的主键
+        "questionnaireId": "test_q1", // 修改为 String 类型
+        "questionText": "是否喜欢吃水果?",
+        "options": [
+            "test_q1_q3_o1", // 修改为 String 类型
+            "test_q1_q3_o2"  // 修改为 String 类型
+        ]
+    },
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "QuestionId": "test_q1_q4", // 新增的非数据库生成的主键
+        "questionnaireId": "test_q1", // 修改为 String 类型
+        "questionText": "喜欢什么运动?",
+        "options": [
+            "test_q1_q4_o1", // 修改为 String 类型
+            "test_q1_q4_o2", // 修改为 String 类型
+            "test_q1_q4_o3", // 修改为 String 类型
+            "test_q1_q4_o4"  // 修改为 String 类型
         ]
     }
 ];
@@ -131,11 +104,7 @@ module.exports.OptionList = [
         "objectId": "option1",
         "createdAt": "2024-12-10T11:20:24Z",
         "OptionId": "o1", // 新增的非数据库生成的主键
-        "questionId": {
-            "__type": "Pointer",
-            "className": "Question",
-            "objectId": "question1"
-        },
+        "QuestionId": "question1", // 修改为 String 类型
         "optionText": "户外活动",
         "isSelected": false
     },
@@ -143,11 +112,7 @@ module.exports.OptionList = [
         "objectId": "option2",
         "createdAt": "2024-12-10T11:20:24Z",
         "OptionId": "o2", // 新增的非数据库生成的主键
-        "questionId": {
-            "__type": "Pointer",
-            "className": "Question",
-            "objectId": "question1"
-        },
+        "QuestionId": "question1", // 修改为 String 类型
         "optionText": "阅读",
         "isSelected": false
     },
@@ -155,11 +120,7 @@ module.exports.OptionList = [
         "objectId": "option3",
         "createdAt": "2024-12-10T11:20:24Z",
         "OptionId": "o3", // 新增的非数据库生成的主键
-        "questionId": {
-            "__type": "Pointer",
-            "className": "Question",
-            "objectId": "question2"
-        },
+        "QuestionId": "question2", // 修改为 String 类型
         "optionText": "看电影",
         "isSelected": false
     },
@@ -167,13 +128,171 @@ module.exports.OptionList = [
         "objectId": "option4",
         "createdAt": "2024-12-10T11:20:24Z",
         "OptionId": "o4", // 新增的非数据库生成的主键
-        "questionId": {
-            "__type": "Pointer",
-            "className": "Question",
-            "objectId": "question2"
-        },
+        "QuestionId": "question2", // 修改为 String 类型
         "optionText": "旅游",
         "isSelected": false
+    },
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "OptionId": "test_q1_q1_o1", // 新增的非数据库生成的主键
+        "QuestionId": "test_q1_q1", // 修改为 String 类型
+        "optionText": "0-3年",
+        "isSelected": false
+    },
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "OptionId": "test_q1_q1_o2", // 新增的非数据库生成的主键
+        "QuestionId": "test_q1_q1", // 修改为 String 类型
+        "optionText": "3-5年",
+        "isSelected": false
+    },
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "OptionId": "test_q1_q1_o3", // 新增的非数据库生成的主键
+        "QuestionId": "test_q1_q1", // 修改为 String 类型
+        "optionText": "5年以上",
+        "isSelected": false
+    },
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "OptionId": "test_q1_q2_o1", // 新增的非数据库生成的主键
+        "QuestionId": "test_q1_q2", // 修改为 String 类型
+        "optionText": "前端开发",
+        "isSelected": false
+    },
+
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "OptionId": "test_q1_q2_o2", // 新增的非数据库生成的主键
+        "QuestionId": "test_q1_q2", // 修改为 String 类型
+        "optionText": "后端开发",
+        "isSelected": false
+    },
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "OptionId": "test_q1_q2_o3", // 新增的非数据库生成的主键
+        "QuestionId": "test_q1_q2", // 修改为 String 类型
+        "optionText": "产品经理",
+        "isSelected": false
+    },
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "OptionId": "test_q1_q2_o4", // 新增的非数据库生成的主键
+        "QuestionId": "test_q1_q2", // 修改为 String 类型
+        "optionText": "UI设计师",
+        "isSelected": false
+    },
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "OptionId": "test_q1_q2_o5", // 新增的非数据库生成的主键
+        "QuestionId": "test_q1_q2", // 修改为 String 类型
+        "optionText": "其他",
+        "isSelected": false
+    },
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "OptionId": "test_q1_q3_o1", // 新增的非数据库生成的主键
+        "QuestionId": "test_q1_q3",
+        "optionText": "是",
+        "isSelected": false
+    },
+
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "OptionId": "test_q1_q3_o2", // 新增的非数据库生成的主键
+        "QuestionId": "test_q1_q3",
+        "optionText": "否",
+        "isSelected": false
+    },
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "OptionId": "test_q1_q4_o1", // 新增的非数据库生成的主键
+        "QuestionId": "test_q1_q4",
+        "optionText": "篮球",
+        "isSelected": false
+    },
+
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "OptionId": "test_q1_q4_o2", // 新增的非数据库生成的主键
+        "QuestionId": "test_q1_q4",
+        "optionText": "足球",
+        "isSelected": false
+    },
+
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "OptionId": "test_q1_q4_o3", // 新增的非数据库生成的主键
+        "QuestionId": "test_q1_q4",
+        "optionText": "羽毛球",
+        "isSelected": false
+    },
+
+    {
+        "createdAt": "2024-12-10T11:20:24Z",
+        "OptionId": "test_q1_q4_o4", // 新增的非数据库生成的主键
+        "QuestionId": "test_q1_q4",
+        "optionText": "其他",
+        "isSelected": false
+    }
+
+
+];
+
+module.exports.QuestionnaireResultList = [
+    {
+        "objectId": "result1",
+        "createdAt": "2024-12-10T11:20:24Z",
+        "QuestionnaireResultId": "qr1", // 新增的非数据库生成的主键
+        "userId": {
+            "__type": "Pointer",
+            "className": "_User",  // 修改为 _User,符合标准类型
+            "objectId": "user1"  // 修改为 objectId,符合 Pointer 类型格式
+        },
+        "questionnaireId": {
+            "__type": "Pointer",
+            "className": "Questionnaire",
+            "objectId": "questionnaire1"  // 修改为 objectId,符合 Pointer 类型格式
+        },
+        "answers": [
+            {
+                "__type": "Pointer",
+                "className": "Option",
+                "objectId": "option1"  // 修改为 objectId,符合 Pointer 类型格式
+            },
+            {
+                "__type": "Pointer",
+                "className": "Option",
+                "objectId": "option2"  // 修改为 objectId,符合 Pointer 类型格式
+            }
+        ]
+    },
+    {
+        "objectId": "result2",
+        "createdAt": "2024-12-10T11:20:24Z",
+        "QuestionnaireResultId": "qr2", // 新增的非数据库生成的主键
+        "userId": {
+            "__type": "Pointer",
+            "className": "_User",  // 修改为 _User,符合标准类型
+            "objectId": "user2"  // 修改为 objectId,符合 Pointer 类型格式
+        },
+        "questionnaireId": {
+            "__type": "Pointer",
+            "className": "Questionnaire",
+            "objectId": "questionnaire1"  // 修改为 objectId,符合 Pointer 类型格式
+        },
+        "answers": [
+            {
+                "__type": "Pointer",
+                "className": "Option",
+                "objectId": "option3"  // 修改为 objectId,符合 Pointer 类型格式
+            },
+            {
+                "__type": "Pointer",
+                "className": "Option",
+                "objectId": "option4"  // 修改为 objectId,符合 Pointer 类型格式
+            }
+        ]
     }
 ];
 
@@ -183,15 +302,15 @@ module.exports.UserInterestProfileList = [
         "createdAt": "2024-12-10T11:20:24Z",
         "userId": {
             "__type": "Pointer",
-            "className": "User",
-            "objectId": "user1"
+            "className": "_User",  // 修改为 _User,符合标准类型
+            "objectId": "user1"  // 修改为 objectId,符合 Pointer 类型格式
         },
         "interestTags": ["户外活动", "阅读"],
         "results": [
             {
                 "__type": "Pointer",
                 "className": "QuestionnaireResult",
-                "objectId": "result1"
+                "objectId": "result1"  // 修改为 objectId,符合 Pointer 类型格式
             }
         ],
         "content": "用户喜欢户外活动和阅读,喜欢在自然中放松和探索新书籍。"
@@ -201,15 +320,15 @@ module.exports.UserInterestProfileList = [
         "createdAt": "2024-12-10T11:20:24Z",
         "userId": {
             "__type": "Pointer",
-            "className": "User",
-            "objectId": "user2"
+            "className": "_User",  // 修改为 _User,符合标准类型
+            "objectId": "user2"  // 修改为 objectId,符合 Pointer 类型格式
         },
         "interestTags": ["看电影", "旅游"],
         "results": [
             {
                 "__type": "Pointer",
                 "className": "QuestionnaireResult",
-                "objectId": "result2"
+                "objectId": "result2"  // 修改为 objectId,符合 Pointer 类型格式
             }
         ],
         "content": "用户热爱看电影和旅游,喜欢通过电影探索不同的文化和风景。"

+ 202 - 10
Alart-server/migration/import-data.js

@@ -1,4 +1,4 @@
-const { CloudQuery, CloudObject } = require("../lib/ncloud");
+/*const { CloudQuery, CloudObject } = require("../lib/ncloud");
 const { QuestionnaireResultList, QuestionnaireList, QuestionList, OptionList, UserInterestProfileList } = require("./data");
 
 importData();
@@ -51,9 +51,12 @@ async function importData() {
 }
 
 async function importObject(className, data) {
-    // 查重 srcId 数据源列表中的 objectId 并非数据库生成的唯一 ID,因此需要有一个 srcId 字段进行记录,并查重
+    // 获取类名对应的 ID 字段名称(如 OptionId, QuestionnaireId 等)
+    let idField = `${className}Id`;
+    let srcId = data[idField];  // 使用 classnameId 作为 srcId 的值
+
+    // 查重:查找是否已经存在相同 srcId 的数据
     let query = new CloudQuery(className);
-    let srcId = data.objectId;
     query.equalTo("srcId", srcId);
     let importObj = await query.first();
     console.log(importObj);
@@ -61,32 +64,44 @@ async function importObject(className, data) {
     // 导入前批量处理 Pointer 类型数据,进行重定向
     Object.keys(data)?.forEach(key => {
         let field = data[key];
-        let srcId = field?.objectId;
-        if (srcId) { // 是 Pointer 类型
+        let fieldSrcId = field?.objectId || field?.OptionId || field?.QuestionId || field?.UserInterestProfileId;
+
+        if (fieldSrcId) { // 是 Pointer 类型
             if (key === "userId" || key === "questionnaireId" || key === "questionId") {
-                data[key] = DataMap?.[key.replace("Id", "")]?.[srcId]?.toPointer();
+                // 确保插入的是一个 Pointer 对象,而不是直接的 id
+                data[key] = {
+                    "__type": "Pointer",
+                    "className": key.replace("Id", ""),
+                    "objectId": DataMap?.[key.replace("Id", "")]?.[fieldSrcId]?.id || fieldSrcId
+                };
             } else if (Array.isArray(field)) {
                 data[key] = field.map(item => {
-                    return DataMap?.[item.className]?.[item.objectId]?.toPointer();
+                    return {
+                        "__type": "Pointer",
+                        "className": item.className,
+                        "objectId": DataMap?.[item.className]?.[item.objectId]?.id || item.objectId
+                    };
                 });
             }
         }
     });
+
     // 若未添加,则创建新对象并保存
     if (!importObj?.id) {
         importObj = new CloudObject(className);
     }
 
     // 保存或更新数据
-    data.srcId = srcId;
+    data.srcId = srcId; // 将 srcId 设置为与 classnameId 相同的值
     importObj.set(data);
     importObj = await importObj.save();
 
+    // 更新 DataMap,确保 srcId 和 classnameId 保持一致
     DataMap[className][srcId] = importObj;
 }
 
 
-/// 展示当前问卷内的问题及其选项
+// 展示当前问卷内的问题及其选项
 async function displayQuestionnaireDetails(questionnaireId) {
     // 查询问卷
     let questionnaireQuery = new CloudQuery("Questionnaire");
@@ -151,4 +166,181 @@ async function displayQuestionnaireDetails(questionnaireId) {
 }
 
 // 示例调用
-displayQuestionnaireDetails("jbg6EYXk3G");
+displayQuestionnaireDetails("jbg6EYXk3G");
+*/
+
+const { CloudQuery, CloudObject } = require("../lib/ncloud");
+const { QuestionnaireResultList, QuestionnaireList, QuestionList, OptionList, UserInterestProfileList } = require("./data");
+
+//importData();
+
+let DataMap = {
+    QuestionnaireResult: {},
+    Questionnaire: {},
+    Question: {},
+    Option: {},
+    UserInterestProfile: {}
+};
+
+async function importData() {
+    // 确保 UserInterestProfileList 已经加载
+    if (!UserInterestProfileList || UserInterestProfileList.length === 0) {
+        console.log("UserInterestProfileList 数据为空或未定义!");
+        return;
+    }
+
+    // 导入问卷结果数据
+    let questionnaireResultList = QuestionnaireResultList;
+    for (let index = 0; index < questionnaireResultList.length; index++) {
+        let result = questionnaireResultList[index];
+        result = await importObject("QuestionnaireResult", result);
+    }
+
+    // 导入问卷数据
+    let questionnaireList = QuestionnaireList;
+    for (let index = 0; index < questionnaireList.length; index++) {
+        let questionnaire = questionnaireList[index];
+        questionnaire = await importObject("Questionnaire", questionnaire);
+    }
+
+    // 导入问题数据
+    let questionList = QuestionList;
+    for (let index = 0; index < questionList.length; index++) {
+        let question = questionList[index];
+        question = await importObject("Question", question);
+    }
+
+    // 导入选项数据
+    let optionList = OptionList;
+    for (let index = 0; index < optionList.length; index++) {
+        let option = optionList[index];
+        option = await importObject("Option", option);
+    }
+
+    // 导入用户兴趣画像数据
+    let userInterestProfileList = UserInterestProfileList;
+    for (let index = 0; index < userInterestProfileList.length; index++) {
+        let profile = userInterestProfileList[index];
+        profile = await importObject("UserInterestProfile", profile);
+    }
+
+    // console.log(DataMap);
+}
+
+async function importObject(className, data) {
+    // 获取类名对应的 ID 字段名称(如 OptionId, QuestionnaireId 等)
+    let idField = `${className}Id`;
+    let srcId = data[idField];  // 使用 classnameId 作为 srcId 的值
+
+    // 查重:查找是否已经存在相同 srcId 的数据
+    let query = new CloudQuery(className);
+    query.equalTo("srcId", srcId);
+    let importObj = await query.first();
+    console.log(importObj);
+
+    // 导入前批量处理 Array 类型数据,进行重定向
+    Object.keys(data)?.forEach(key => {
+        let field = data[key];
+        let fieldSrcId = field?.objectId || field?.OptionId || field?.QuestionId || field?.UserInterestProfileId;
+
+        if (fieldSrcId) { // 是 Pointer 类型的字段
+            if (key === "userIds" || key === "questionnaireIds" || key === "questionIds") {
+                // 确保插入的是一个数组而不是 Pointer 对象
+                data[key] = Array.isArray(field) ? field.map(item => {
+                    return {
+                        className: item.className,
+                        srcId: DataMap?.[item.className]?.[item.srcId]?.id || item.srcId
+                    };
+                }) : [{
+                    className: key.replace("Ids", ""),
+                    srcId: DataMap?.[key.replace("Ids", "")]?.[fieldSrcId]?.id || fieldSrcId
+                }];
+            } else if (Array.isArray(field)) {
+                data[key] = field.map(item => {
+                    return {
+                        className: item.className,
+                        srcId: DataMap?.[item.className]?.[item.srcId]?.id || item.srcId
+                    };
+                });
+            }
+        }
+    });
+
+    // 若未添加,则创建新对象并保存
+    if (!importObj?.id) {
+        importObj = new CloudObject(className);
+    }
+
+    // 保存或更新数据
+    data.srcId = srcId; // 将 srcId 设置为与 classnameId 相同的值
+    importObj.set(data);
+    importObj = await importObj.save();
+
+    // 更新 DataMap,确保 srcId 和 classnameId 保持一致
+    DataMap[className][srcId] = importObj;
+}
+
+// 展示当前问卷内的问题及其选项
+async function displayQuestionnaireDetails(questionnaireId) {
+    // 查询问卷
+    let questionnaireQuery = new CloudQuery("Questionnaire");
+    questionnaireQuery.equalTo("QuestionnaireId", questionnaireId); // 使用 QuestionnaireId 查询
+    let questionnaire = await questionnaireQuery.first();
+
+    // 检查问卷是否存在
+    if (!questionnaire) {
+        console.log("问卷未找到,无法展示问题和选项。");
+        return;
+    }
+
+    console.log(`问卷标题: ${questionnaire.get('title')}`);
+    console.log(`问卷状态: ${questionnaire.get('status')}`);
+    console.log("问题列表:");
+
+    // 获取问题 IDs(数组字段)
+    let questionIds = questionnaire.get('questions');
+
+    // 查询所有相关问题
+    let questionQuery = new CloudQuery("Question");
+    questionQuery.containedIn("QuestionId", questionIds); // 查询 QuestionId 在问题 ID 列表中的问题
+    let questions = await questionQuery.find();
+
+    // 检查是否有问题
+    if (questions.length === 0) {
+        console.log("当前问卷没有问题。");
+        return;
+    }
+
+    // 遍历问题并查询选项
+    for (let question of questions) {
+        // 检查问题是否存在
+        if (!question) {
+            console.log("问题未找到,无法展示选项。");
+            continue; // 跳过当前循环,继续下一个问题
+        }
+
+        console.log(`  问题: ${question.questionText}`);
+
+        // 获取问题中的选项 IDs(数组字段)
+        let optionIds = question.options;
+
+        // 查询所有相关选项
+        let optionQuery = new CloudQuery("Option");
+        optionQuery.containedIn("OptionId", optionIds); // 查询 OptionId 在选项 ID 列表中的选项
+        let options = await optionQuery.find();
+
+        // 输出选项
+        console.log("  选项:");
+        if (options.length === 0) {
+            console.log("    - 此问题没有选项。");
+        } else {
+            for (let option of options) {
+                console.log(`    - ${option.optionText} (已选: ${option.isSelected ? "是" : "否"})`);
+            }
+        }
+        console.log(""); // 添加空行以便于阅读
+    }
+}
+
+// 示例调用
+displayQuestionnaireDetails("test_q1");