当前位置: 首页 > news >正文

xhEditor pdf导入识别图片和图表

山西老码农的680元"Office全家桶"改造计划

各位老铁好啊!我是山西那个天天跟Word文档"干仗"的前端码农,最近接了个企业官网的外包活儿,客户突然要加个"Office全家桶"功能…预算还只有680块!这不得好好跟大家唠唠我是怎么用最少的钱办最大的事儿~

需求拆解:客户要的到底是啥?

客户其实就是想要个"傻瓜式"操作:

  • 大妈也能一键粘贴Word不丢样式(领导们年纪大了,搞不来复杂操作)
  • 能导入各种Office文档(连PPT动画都要保留你敢信?)
  • 公式要高清显示(从手机到小程序都不能糊)
  • 最好能直接复制微信公众号文章(客户说他们经常这么干)

技术选型:680块怎么花最值?

前端方案:

  • 继续用xhEditor?老古董了,改不动改不动!
  • 换UEditor?百度都不维护了,放弃!
  • TinyMCE开源版+插件大法(最终选择!)
    • 买了个学生优惠插件包(花了99)
    • 自己魔改了下公式处理(头发掉了不少)

后端方案:

  • 手撸了个Node.js中间层(反正客户服务器能跑)
  • 图片处理用sharp库(比ImageMagick轻量)
  • 公式转换用mathjax-node(免费的真香)

代码实战:能跑就行!

前端部分(Vue3版)

// 在setup函数里搞事情import{onMounted}from'vue'importtinymcefrom'tinymce/tinymce'import'tinymce/icons/default'import'tinymce/themes/silver'import'tinymce/plugins/paste'import'tinymce/plugins/image'exportdefault{setup(){onMounted(()=>{tinymce.init({selector:'#editor',plugins:'paste image wordcount',toolbar:'pasteword importoffice',height:800,// 自定义按钮!!!setup:(editor)=>{editor.ui.registry.addButton('pasteword',{icon:'paste',tooltip:'粘贴Word内容',onAction:()=>{navigator.clipboard.readText().then(text=>{// 这里调用咱的后端API处理Word内容fetch('/api/process-word',{method:'POST',body:JSON.stringify({content:text})}).then(res=>res.text()).then(html=>{editor.insertContent(html)})})}})// 文件导入按钮editor.ui.registry.addButton('importoffice',{text:'导入Office',onAction:()=>{// 偷偷创建一个文件inputconstinput=document.createElement('input')input.type='file'input.accept='.docx,.xlsx,.pptx,.pdf'input.onchange=(e)=>{constfile=e.target.files[0]constformData=newFormData()formData.append('file',file)fetch('/api/import-office',{method:'POST',body:formData}).then(res=>res.text()).then(html=>{editor.insertContent(html)})}input.click()}})}})})}}

后端部分(Node.js版)

constexpress=require('express')constmulter=require('multer')constmammoth=require('mammoth')constsharp=require('sharp')constfs=require('fs')constapp=express()constupload=multer({dest:'uploads/'})// 处理Word粘贴app.post('/api/process-word',(req,res)=>{const{content}=req.body// 这里其实应该用mammoth处理,但演示就简单点了consthtml=content.replace(/]+src="data:image[^"]+"[^>]*>/g,match=>{// 提取base64图片constbase64Data=match.match(/src="([^"]+)"/)[1]constbuffer=Buffer.from(base64Data.split(',')[1],'base64')// 用sharp处理图片并保存constfilename=`images/${Date.now()}.jpg`sharp(buffer).resize(800)// 限制宽度.toFile(`public/${filename}`)return``})res.send(html)})// 处理Office文件导入app.post('/api/import-office',upload.single('file'),async(req,res)=>{const{path,originalname}=req.filetry{lethtmlif(originalname.endsWith('.docx')){constresult=awaitmammoth.convertToHtml({path})html=result.value}else{// 其他格式简单返回文件名(实际应该用对应的库处理)html=`已上传文件:${originalname}`}// 处理图片上传(同上)html=html.replace(/]+src="[^"]+"[^>]*>/g,match=>{// 图片处理逻辑...})res.send(html)}finally{fs.unlinkSync(path)// 删除临时文件}})app.listen(3000)

踩坑大全:都是泪啊!

  1. 公式显示问题:最终用了这个方案:

    // 前端公式渲染document.querySelectorAll('.math').forEach(el=>{MathJax.typesetPromise([el]).catch(err=>{console.error('公式渲染失败:',err)})})
  2. PPT动画丢失:老老实实跟客户说做不到,加钱也做不到!(客户最后妥协了)

  3. 微信公众号特殊样式:加了这段黑魔法:

    // 处理微信公众号的特殊标签html=html.replace(/]*><\/mpchecktext>/g,'')

680元预算分配

  1. TinyMCE基础插件包(学生优惠):99元
  2. 阿里云OSS存储包(最便宜那档):60元/年
  3. 腾讯云函数计算(跑公式转换):0元(免费额度够用)
  4. 剩下521元…当然是买咖啡熬夜改bug啊!

老码农的私房话

看到你们群里那个推广提成,说实话我有点心动…但我更心疼那些被忽悠的新人。这年头,什么"轻松年入40万"都是鬼扯!

我接这个680元的项目,前后改了8版需求,最后实际收入也就够交一个月房租。但是!这个案例成了我接下一个2万块项目的敲门砖——客户看到我能在这么低的预算下实现复杂功能,当场就签了合同。

所以啊,与其天天群里发广告,不如静下心来:

  • 把GitHub仓库搞漂亮点
  • 在掘金写两篇技术文章
  • 把做过的项目好好整理成案例

(小声说:那个QQ群我还是加了,领了3块钱红包就跑路了…)


最后的最后:完整项目我开源在了GitHub(其实并没有,客户不让),但核心代码都在上面了,自己拿去魔改吧!

P.S.那个公式转换的坑我填了3天3夜…你们要是遇到类似问题,记住一定要用mathjax-node,别像我一样头铁非要自己写!

将插件目录复制到项目中

引入插件文件

定义插件图标

初始化插件

在工具栏中添加插件按钮

效果

编辑器

导入Word文档,支持doc,docx

导入Excel文档,支持xls,xlsx

粘贴Word

一键粘贴Word内容,自动上传Word中的图片,保留文字样式。

Word转图片

一键导入Word文件,并将Word文件转换成图片上传到服务器中。

导入PDF

一键导入PDF文件,并将PDF转换成图片上传到服务器中。

导入PPT

一键导入PPT文件,并将PPT转换成图片上传到服务器中。

上传网络图片

一键自动上传网络图片,自动下载远程服务器图片,自动上传远程服务器图片

下载示例

点击下载完整示例

http://www.jsqmd.com/news/192733/

相关文章:

  • 塔吉克族鹰舞表演:演员数字人展翅飞翔
  • GitHub镜像网站推荐 + HeyGem系统部署:加速开源项目落地
  • PHP 8.7即将上线:现在不看这份基准报告就晚了!
  • 普米语韩规节祭祀:祭司数字人带领族群缅怀祖先
  • 如何用Swoole+Consul实现PHP高性能服务注册?(实战代码曝光)
  • 西门子 S7 PLC 通信 WPF 应用分析笔记
  • 塞浦路斯语婚礼习俗讲解:长辈数字人传授传统仪式细节
  • 格鲁吉亚语高加索登山指南:向导数字人介绍徒步路线
  • 为什么顶级公司都在测试PHP 8.7?(内部基准数据首次曝光)
  • 服务注册突然失效?PHP微服务容灾机制紧急应对指南
  • PHP WebSocket 实时消息推送全解析(从入门到高并发架构设计)
  • AI口型同步技术新突破!HeyGem数字人系统WebUI版全面解析
  • 巴斯克语工业安全培训:工厂主管数字人强调操作规范
  • 西班牙语旅游口语:度假村接待员教你常用表达
  • HeyGem数字人系统常见问题解答:处理慢、分辨率支持、并发任务解析
  • ‌测试左移避坑:开发写单元测试 ≠ 测试介入
  • HeyGem数字人系统适合哪些场景?教育、客服、短视频应用全解析
  • C#组件化扩展实践全记录,手把手教你构建可插拔业务模块
  • 波兰语法律咨询服务:律师数字人解答常见民事问题
  • 拉祜语咖啡种植技术:农户数字人交流种植心得
  • 【Python小技巧】-PyCharm配置专业开发规范模板完全指南
  • 【PHP 8.7性能优化终极指南】:基于最新基准测试的调优策略
  • PHP微服务上线必看:服务注册配置的8个致命误区
  • C#企业级扩展技术深度解析(从反射到MEF再到依赖注入)
  • Consul vs ZooKeeper vs Eureka:PHP微服务注册该用谁?
  • 【性能调优关键时刻】如何在macOS上高效调试C# HTTP拦截器?
  • 前端新手必看:用事件委托轻松搞定动态元素交互(附实战技巧)
  • 威尔士语学校教育改革:校长数字人宣布新政措施
  • 读共生:4_0时代的人机关系07工作者
  • PHP 8.7性能飞跃真相:5大基准测试结果震惊开发者社区