瀏覽代碼

'update:简历pdf文件的读取和上传'

abstract001 1 年之前
父節點
當前提交
918c54a778
共有 24 個文件被更改,包括 302 次插入44 次删除
  1. 10 0
      .idea/inspectionProfiles/Project_Default.xml
  2. 2 2
      app-angular/src/app/app.component.ts
  3. 12 0
      app-angular/src/modules/dark-light/dark-light.module.ts
  4. 16 0
      app-angular/src/modules/dark-light/theme.service.spec.ts
  5. 25 0
      app-angular/src/modules/dark-light/theme.service.ts
  6. 6 3
      app-angular/src/modules/home/interview-history/interview-history.component.ts
  7. 5 0
      app-angular/src/modules/home/mock-interviews/mock-interviews.component.scss
  8. 6 2
      app-angular/src/modules/home/mock-interviews/mock-interviews.component.ts
  9. 2 0
      app-angular/src/modules/home/myresume/myresume.component.scss
  10. 24 1
      app-angular/src/modules/home/nav-menu/nav-menu.component.html
  11. 9 0
      app-angular/src/modules/home/nav-menu/nav-menu.component.scss
  12. 30 7
      app-angular/src/modules/home/nav-menu/nav-menu.component.ts
  13. 2 2
      app-angular/src/modules/home/recommended-templates/recommended-templates.component.html
  14. 12 0
      app-angular/src/modules/home/recommended-templates/recommended-templates.component.scss
  15. 7 0
      app-angular/src/modules/home/resume-build/resume-build.component.scss
  16. 1 2
      app-angular/src/modules/home/suit-companies-comp/suit-companies-comp.component.html
  17. 3 1
      app-angular/src/styles.scss
  18. 13 0
      app-angular/src/theme/variables.css
  19. 14 24
      app-node/interview-parse/PDFToText.js
  20. 二進制
      app-node/interview-parse/uploads/1.pdf
  21. 15 0
      pc-electron/my-electron-app/dist/index.html
  22. 54 0
      pc-electron/my-electron-app/main.js
  23. 15 0
      pc-electron/my-electron-app/package.json
  24. 19 0
      pc-electron/my-electron-app/preload.js

+ 10 - 0
.idea/inspectionProfiles/Project_Default.xml

@@ -0,0 +1,10 @@
+<component name="InspectionProjectProfileManager">
+  <profile version="1.0">
+    <option name="myName" value="Project Default" />
+    <inspection_tool class="DuplicatedCode" enabled="true" level="WEAK WARNING" enabled_by_default="true">
+      <Languages>
+        <language minSize="63" name="TypeScript" />
+      </Languages>
+    </inspection_tool>
+  </profile>
+</component>

+ 2 - 2
app-angular/src/app/app.component.ts

@@ -1,4 +1,5 @@
-import { Component } from '@angular/core';
+import {Component} from '@angular/core';
+
 
 @Component({
   selector: 'app-root',
@@ -8,5 +9,4 @@ import { Component } from '@angular/core';
 export class AppComponent {
 
 
-
 }

+ 12 - 0
app-angular/src/modules/dark-light/dark-light.module.ts

@@ -0,0 +1,12 @@
+import { NgModule } from '@angular/core';
+import { CommonModule } from '@angular/common';
+
+
+
+@NgModule({
+  declarations: [],
+  imports: [
+    CommonModule
+  ]
+})
+export class DarkLightModule { }

+ 16 - 0
app-angular/src/modules/dark-light/theme.service.spec.ts

@@ -0,0 +1,16 @@
+import { TestBed } from '@angular/core/testing';
+
+import { ThemeService } from './theme.service';
+
+describe('ThemeService', () => {
+  let service: ThemeService;
+
+  beforeEach(() => {
+    TestBed.configureTestingModule({});
+    service = TestBed.inject(ThemeService);
+  });
+
+  it('should be created', () => {
+    expect(service).toBeTruthy();
+  });
+});

+ 25 - 0
app-angular/src/modules/dark-light/theme.service.ts

@@ -0,0 +1,25 @@
+import {Injectable, Renderer2, RendererFactory2} from '@angular/core';
+
+@Injectable({
+  providedIn: 'root'
+})
+export class ThemeService {
+  private renderer: Renderer2;
+
+  constructor(private rendererFactory: RendererFactory2) {
+    this.renderer = this.rendererFactory.createRenderer(null, null);
+  }
+
+  toggleTheme(isDark: boolean): void {
+    const element = document.querySelector('.inner-content');
+    const suitCompany = document.querySelector('.container');
+    const suitBox = document.querySelectorAll('.ant-card');
+
+    element?.classList.toggle('dark-theme', !isDark);
+    suitCompany?.classList.toggle('dark-theme', !isDark);
+    suitBox.forEach(element => {
+      element?.classList.toggle('suit-box', !isDark);
+    });
+  }
+
+}

+ 6 - 3
app-angular/src/modules/home/interview-history/interview-history.component.ts

@@ -77,9 +77,12 @@ export class InterviewHistoryComponent {
   skip: number = 0;//跳过多少条加载
   async onBottomLoad() {
     this.createBasicMessage("历史数据获取中");
-    this.skip = this.history.length;
-    let list = await this.searchHistory();
-    this.history = this.history.concat(list)
+    setTimeout(async () => {
+      this.skip = this.history.length;
+      let list = await this.searchHistory();
+      this.history = this.history.concat(list)
+    }, 1000)
+
   }
 
   history!: any[];

+ 5 - 0
app-angular/src/modules/home/mock-interviews/mock-interviews.component.scss

@@ -118,3 +118,8 @@
 }
 
 //小版本联级
+@media screen and (max-width: 400px) {
+  .chat-container {
+    max-height: calc(100vh - 200px);
+  }
+}

+ 6 - 2
app-angular/src/modules/home/mock-interviews/mock-interviews.component.ts

@@ -296,6 +296,7 @@ export class MockInterviewsComponent implements OnInit {
   handleCancel(): void {
     if (this.errorMessage == '您有未完成的面试环节,请问您是否要继续') {
       this.isContinue = false;
+      this.selectedIndex = 0;
     }
     this.isVisible = false;
   }
@@ -334,6 +335,7 @@ export class MockInterviewsComponent implements OnInit {
 
   }
 
+  //第一次来到页面判断
   async initialize() {
     const currentUser: any = Parse.User.current();
 
@@ -353,15 +355,17 @@ export class MockInterviewsComponent implements OnInit {
           this.errorMessage = '您有未完成的面试环节,请问您是否要继续';
           await this.showModal();
           if (this.isContinue) {
-            console.log("继续面试")
+            console.log("继续面试");
             this.messagesListUser = results[0].get('userList');
             this.finalResult = results[0].get('question');
             this.resultAnswers = results[0].get('answer');
             this.messagesList = results[0].get('aiList');
             this.typeQuestion = results[0].get('number');
             this.isStart = results[0].get('interviewResult');
+          } else {
+            results[0].set('interviewResult', 0);
+            await Parse.Object.saveAll(results)
           }
-
         }
       }
 

+ 2 - 0
app-angular/src/modules/home/myresume/myresume.component.scss

@@ -2,8 +2,10 @@
 .container {
   max-width: 1500px;
   margin: 0 auto;
+  overflow-y: scroll;
 
   @media (max-width: 1000px) {
+    max-height: 100%;
     width: 100%;
   }
 

+ 24 - 1
app-angular/src/modules/home/nav-menu/nav-menu.component.html

@@ -83,9 +83,32 @@
                   [nzType]="isCollapsed ? 'menu-unfold' : 'menu-fold'"
             ></span>
         </span>
+        <nz-switch
+          [(ngModel)]="isDark"
+          (click)="switchDark()"
+          [nzCheckedChildren]="checkedTemplate"
+          [nzUnCheckedChildren]="unCheckedTemplate"
+          [nzControl]="true"
+        ></nz-switch>
+
+        <ng-template #checkedTemplate><span class="icon" nz-icon><svg t="1702252356908" class="icon"
+                                                                      viewBox="0 0 1024 1024" version="1.1"
+                                                                      xmlns="http://www.w3.org/2000/svg" p-id="4240"
+                                                                      width="16" height="16"><path
+          d="M520.021333 208.284444c-165.987556 0-301.041778 135.04-301.041777 301.027556 0 165.987556 135.04 301.027556 301.041777 301.027556 166.001778 0 301.056-135.025778 301.056-301.027556 0-165.987556-135.054222-301.027556-301.056-301.027556z m0 545.194667c-134.613333 0-244.152889-109.525333-244.152889-244.152889 0-134.627556 109.525333-244.138667 244.152889-244.138666 134.641778 0 244.167111 109.511111 244.167111 244.138666 0 134.641778-109.525333 244.152889-244.167111 244.152889z"
+          fill="" p-id="4241"></path><path
+          d="M493.027556 317.240889a164.949333 164.949333 0 0 0-141.212445 80.426667 14.250667 14.250667 0 0 0 24.476445 14.535111 136.32 136.32 0 0 1 116.736-66.517334 14.222222 14.222222 0 1 0 0-28.444444zM350.307556 433.265778a14.307556 14.307556 0 0 0-17.095112 10.595555c-2.773333 11.747556-4.323556 19.015111-4.323555 32.284445a14.222222 14.222222 0 1 0 28.444444 0c0-10.183111 1.095111-15.260444 3.569778-25.784889a14.222222 14.222222 0 0 0-10.595555-17.095111zM531.484444 846.222222a28.444444 28.444444 0 0 0-28.444444 28.444445v85.333333a28.444444 28.444444 0 0 0 56.888889 0v-85.333333a28.444444 28.444444 0 0 0-28.444445-28.444445zM531.484444 177.777778a28.444444 28.444444 0 0 0 28.444445-28.444445v-85.333333a28.444444 28.444444 0 0 0-56.888889 0v85.333333a28.444444 28.444444 0 0 0 28.444444 28.444445zM242.56 745.642667l-60.330667 60.330666a28.444444 28.444444 0 1 0 40.220445 40.220445l60.330666-60.330667a28.444444 28.444444 0 1 0-40.220444-40.220444zM775.552 281.315556c7.281778 0 14.549333-2.773333 20.110222-8.334223l60.330667-60.330666a28.444444 28.444444 0 1 0-40.220445-40.220445l-60.330666 60.330667a28.444444 28.444444 0 0 0 20.110222 48.554667zM242.56 272.981333c5.560889 5.560889 12.828444 8.334222 20.110222 8.334223a28.444444 28.444444 0 0 0 20.110222-48.554667l-60.330666-60.330667a28.444444 28.444444 0 1 0-40.220445 40.220445l60.330667 60.330666zM795.662222 745.642667a28.444444 28.444444 0 1 0-40.220444 40.220444l60.330666 60.359111c5.560889 5.560889 12.828444 8.334222 20.110223 8.334222s14.549333-2.773333 20.110222-8.334222c11.093333-11.107556 11.093333-29.112889 0-40.220444l-60.330667-60.359111zM149.333333 490.666667h-85.333333a28.444444 28.444444 0 0 0 0 56.888889h85.333333a28.444444 28.444444 0 0 0 0-56.888889zM960 490.666667h-85.333333a28.444444 28.444444 0 0 0 0 56.888889h85.333333a28.444444 28.444444 0 0 0 0-56.888889z"
+          fill="" p-id="4242"></path></svg></span></ng-template>
+        <ng-template #unCheckedTemplate><span class="icon" nz-icon><svg t="1702252487224" class="icon"
+                                                                        viewBox="0 0 1024 1024"
+                                                                        version="1.1" xmlns="http://www.w3.org/2000/svg"
+                                                                        p-id="5327"
+                                                                        width="16" height="16"><path
+          d="M529.611373 1023.38565c-146.112965 0-270.826063-51.707812-374.344078-155.225827C51.74928 764.641808 0.041469 639.826318 0.041469 493.815745c0-105.053891 29.693595-202.326012 88.978393-292.22593 59.38719-89.797526 137.000103-155.942569 232.83874-198.63991 6.041111-4.607627 12.184613-3.788493 18.225724 2.252618 7.576986 4.607627 9.931996 11.365479 6.860244 20.580733C322.677735 83.736961 310.493122 142.202626 310.493122 201.589815c0 135.464227 48.328885 251.474031 144.986656 348.131801 96.657771 96.657771 212.667574 144.986656 348.131801 144.986656 74.541162 0 139.252721-11.365479 194.032283-34.19883C1003.684974 655.799424 1009.726084 656.618558 1015.767195 662.659669c7.576986 4.607627 9.931996 11.365479 6.860244 20.580733C983.104241 786.758417 918.802249 869.286132 829.721465 930.925939 740.743072 992.565746 640.706375 1023.38565 529.611373 1023.38565z"
+          p-id="5328"></path></svg></span></ng-template>
       </div>
     </nz-header>
-    <nz-content>
+    <nz-content class="content">
       <router-outlet></router-outlet>
     </nz-content>
   </nz-layout>

+ 9 - 0
app-angular/src/modules/home/nav-menu/nav-menu.component.scss

@@ -72,6 +72,10 @@ nz-header {
   box-shadow: 0 1px 4px rgba(0, 21, 41, .08);
 }
 
+.switch-button {
+
+}
+
 nz-content {
   margin: 24px;
   overflow: hidden;
@@ -94,4 +98,9 @@ nz-avatar {
   height: 64px;
 }
 
+//图标
+.icon {
+  display: flex;
+  vertical-align: middle;
+}
 

+ 30 - 7
app-angular/src/modules/home/nav-menu/nav-menu.component.ts

@@ -1,5 +1,6 @@
-import {Component, ViewChild, ElementRef} from '@angular/core';
+import {Component, ViewChild, ElementRef,} from '@angular/core';
 import {HwobsService} from "../../file/hwobs.service";
+import {ThemeService} from "../../dark-light/theme.service";
 
 @Component({
   selector: 'app-nav-menu',
@@ -7,6 +8,34 @@ import {HwobsService} from "../../file/hwobs.service";
   styleUrls: ['./nav-menu.component.scss']
 })
 export class NavMenuComponent {
+//  华为云obs配置
+  constructor(private hwobs: HwobsService, private themeService: ThemeService) {
+  }
+
+  ngAfterContentChecked() {
+    this.toggleTheme()
+  }
+
+  ngOnDestroy() {
+    console.log(1)
+  }
+
+
+  //黑暗模式
+
+  //触发黑暗模式
+  toggleTheme(): void {
+    this.themeService.toggleTheme(this.isDark);
+  }
+
+  isDark: boolean = true;
+  loading = false;
+
+  switchDark() {
+    this.isDark = !this.isDark;
+    console.log(this.isDark)
+    this.toggleTheme()
+  }
 
 
   @ViewChild('fileInput') fileInput: any;
@@ -67,11 +96,5 @@ export class NavMenuComponent {
     }
   }
 
-//  华为云obs配置
-  constructor(private hwobs: HwobsService
-  ) {
-
-  }
-
 
 }

+ 2 - 2
app-angular/src/modules/home/recommended-templates/recommended-templates.component.html

@@ -1,7 +1,7 @@
 <div class="card-container">
   <nz-tabset nzType="card">
     <nz-tab [nzTitle]="'岗位模板'">
-      <div nz-row [nzGutter]="16">
+      <div nz-row [nzGutter]="16" class="tabs">
         <div nz-col class="gutter-row" [nzXs]="24" [nzSm]="4" *ngFor="let item of jobCategories; let i = index">
           <button nz-button [ngClass]="{ 'blue-button': selectedButton === i + 1 }" (click)="selectButton(i + 1)">
             <p>{{ item }}</p>
@@ -10,7 +10,7 @@
       </div>
     </nz-tab>
     <nz-tab [nzTitle]="'风格模板'">
-      <div nz-row [nzGutter]="16">
+      <div nz-row [nzGutter]="16" class="tabs">
         <div nz-col class="gutter-row" [nzXs]="24" [nzSm]="4" *ngFor="let item of jobStyles; let i = index">
           <button nz-button [ngClass]="{ 'blue-button': selectedButton === i + 1 }" (click)="selectButton(i + 1)">
             {{ item }}

+ 12 - 0
app-angular/src/modules/home/recommended-templates/recommended-templates.component.scss

@@ -107,9 +107,21 @@ button {
 }
 
 .card-container {
+  max-width: 1500px;
+  margin: 0 auto;
   ////height: 750px;
   //margin: auto;
   //width: 750px;
 }
 
+.tabs {
+  padding: 10px;
+  max-height: 200px;
+  overflow-y: auto;
+}
+
+.container {
+  overflow-y: auto;
+}
+
 

+ 7 - 0
app-angular/src/modules/home/resume-build/resume-build.component.scss

@@ -1,7 +1,14 @@
 .container {
   display: flex;
   flex-direction: column;
+  max-height: 1000px;
   height: 100vh;
+  overflow-y: auto;
+  @media (max-width: 400px) {
+    overflow-y: scroll;
+    height: 100%;
+
+  }
 
   header {
     flex: 0 0 auto;

+ 1 - 2
app-angular/src/modules/home/suit-companies-comp/suit-companies-comp.component.html

@@ -1,11 +1,10 @@
-<div class="container" style="background: #ECECEC;padding:30px;">
+<div class="container" style="padding:30px;">
   <div class="carousel-wrapper">
     <nz-carousel [nzAutoPlay]="true" [nzEffect]="'fade'" [nzDotPosition]="'bottomCenter'">
       <div nz-carousel-content *ngFor="let img of images">
         <img [src]="img" alt="Carousel Image">
       </div>
     </nz-carousel>
-
   </div>
 
   <div class="component-wrapper">

+ 3 - 1
app-angular/src/styles.scss

@@ -3,10 +3,12 @@
 
 // 使用相对路径引入 ng-zorro-antd 样式文件
 @import '../node_modules/ng-zorro-antd/style/index.min.css';
-@import "../node_modules/ng-zorro-antd/button/style/index.min.css"; /* 引入基本样式 */
+@import "../node_modules/ng-zorro-antd/button/style/index.min.css";
+/* 引入基本样式 */
 
 /* Importing Bootstrap SCSS file. */
 @import "./node_modules/bootstrap/scss/bootstrap";
 
 /* Importing Datepicker SCSS file. */
 @import "node_modules/ngx-bootstrap/datepicker/bs-datepicker";
+@import "theme/variables.css";

+ 13 - 0
app-angular/src/theme/variables.css

@@ -4,3 +4,16 @@
 
 /* To quickly generate your own theme, check out the color generator */
 /* http://ionicframework.com/docs/theming/color-generator */
+
+.dark-theme {
+  background-color: #333;
+  color: #fff;
+  /* 其他黑暗模式下的样式规则 */
+}
+
+.suit-box {
+  background-color: #333333;
+  color: whitesmoke;
+  border: #333333;
+}
+

+ 14 - 24
app-node/interview-parse/PDFToText.js

@@ -7,8 +7,7 @@ const app = express();
 app.use(express.json());
 app.use(express.urlencoded({extended: true}));
 app.use((req, res, next) => {
-  res.setHeader("Access-Control-Allow-Origin", "http://localhost:4200");
-  res.setHeader("Access-Control-Allow-Methods", "GET, POST, OPTIONS");
+  res.setHeader("Access-Control-Allow-Origin", "*");
   res.setHeader("Access-Control-Allow-Headers", "Content-Type, x-requested-with");
   next();
 });
@@ -22,37 +21,28 @@ const storage = multer.diskStorage({
   }
 });
 
+const options = {
+  normalizeWhitespace: true,
+  pdfjsDataRangeTransportFactory: function () {
+    return new pdfjsDataRangeTransport();
+  }
+};
 const upload = multer({storage: storage});
-
+let hasPrinted = false;
 app.post("/upload", upload.single("file"), (req, res) => {
   try {
     const filePath = req.file.path;
     let dataBuffer = fs.readFileSync(filePath);
-    pdf(dataBuffer).then(function (data) {
-      // number of pages
-      console.log(data.numpages);
-      console.log("-------------------------------")
-      // number of rendered pages
-      console.log(data.numrender);
-      // PDF info
-      console.log("-------------------------------")
-
-      console.log(data.info);
-      console.log("-------------------------------")
-
-      // PDF metadata
-      console.log(data.metadata);
-      console.log("-------------------------------")
-
-
-      // PDF text
-      console.log(data.text);
-      console.log("-------------------------------")
-
+    pdf(dataBuffer, options).then(function (data) {
+      if (!hasPrinted) {
+        console.log(data.text);
+        hasPrinted = true
+      }
     });
   } catch (error) {
     res.status(500).json({error: error.message});
   }
+  hasPrinted = false
 });
 
 app.listen(3000, () => {

二進制
app-node/interview-parse/uploads/1.pdf


+ 15 - 0
pc-electron/my-electron-app/dist/index.html

@@ -0,0 +1,15 @@
+<!DOCTYPE html>
+<html>
+<head>
+ <meta charset="UTF-8">
+ <!-- https://developer.mozilla.org/zh-CN/docs/Web/HTTP/CSP -->
+ <meta http-equiv="Content-Security-Policy" content="default-src 'self'; script-src 'self'">
+ <title>你好!</title>
+</head>
+<body>
+<h1>你好!</h1>
+我们正在使用 Node.js <span id="node-version"></span>,
+Chromium <span id="chrome-version"></span>,
+和 Electron <span id="electron-version"></span>.
+</body>
+</html>

+ 54 - 0
pc-electron/my-electron-app/main.js

@@ -0,0 +1,54 @@
+// main.js
+
+// Modules to control application life and create native browser window
+const {app, BrowserWindow} = require('electron')
+const path = require('node:path')
+
+const createWindow = () => {
+  // Create the browser window.
+  const mainWindow = new BrowserWindow({
+    width: 800,
+    height: 600,
+    webPreferences: {
+      preload: path.join(__dirname, 'preload.js')
+    }
+  })
+
+  //监听
+  function handleSetTitle(event, title) {
+    const webContents = event.sender
+    const win = BrowserWindow.fromWebContents(webContents)
+    win.setTitle(title)
+  }
+
+  // 加载 index.html
+  mainWindow.loadFile('./dist/index.html')
+
+  // 打开开发工具
+  // mainWindow.webContents.openDevTools()
+}
+
+// 这段程序将会在 Electron 结束初始化
+// 和创建浏览器窗口的时候调用
+// 部分 API 在 ready 事件触发后才能使用。
+app.whenReady().then(() => {
+  createWindow()
+  ipcMain.on('set-title', handleSetTitle)
+
+
+  app.on('activate', () => {
+    // 在 macOS 系统内, 如果没有已开启的应用窗口
+    // 点击托盘图标时通常会重新创建一个新窗口
+    if (BrowserWindow.getAllWindows().length === 0) createWindow()
+  })
+})
+
+// 除了 macOS 外,当所有窗口都被关闭的时候退出程序。 因此, 通常
+// 对应用程序和它们的菜单栏来说应该时刻保持激活状态, 
+// 直到用户使用 Cmd + Q 明确退出
+app.on('window-all-closed', () => {
+  if (process.platform !== 'darwin') app.quit()
+})
+
+// 在当前文件中你可以引入所有的主进程代码
+// 也可以拆分成几个文件,然后用 require 导入。

+ 15 - 0
pc-electron/my-electron-app/package.json

@@ -0,0 +1,15 @@
+{
+  "name": "my-electron-app",
+  "version": "1.0.0",
+  "description": "hello",
+  "main": "main.js",
+  "scripts": {
+    "test": "echo \"Error: no test specified\" && exit 1",
+    "start": "electron ."
+  },
+  "author": "lj",
+  "license": "MIT",
+  "devDependencies": {
+    "electron": "^28.0.0"
+  }
+}

+ 19 - 0
pc-electron/my-electron-app/preload.js

@@ -0,0 +1,19 @@
+// preload.js
+
+// 所有的 Node.js API接口 都可以在 preload 进程中被调用.
+// 它拥有与Chrome扩展一样的沙盒。
+const {contextBridge, ipcRenderer} = require('electron')
+window.addEventListener('DOMContentLoaded', () => {
+  const replaceText = (selector, text) => {
+    const element = document.getElementById(selector)
+    if (element) element.innerText = text
+  }
+
+  for (const dependency of ['chrome', 'node', 'electron']) {
+    replaceText(`${dependency}-version`, process.versions[dependency])
+  }
+})
+
+contextBridge.exposeInMainWorld('electronAPI', {
+  setTitle: (title) => ipcRenderer.send('set-title', title)
+})