一次基于electron的图片上传插件的开发过程

背景:

项目前端有个图片压缩包上传功能,用户上传的时候会选择单反拍摄的巨幅图片,由于前端打不开压缩包,也没法读取压缩率并重新编码,只能原样发送,这给后台存储造成相当大的压力.同时由于压缩包内容涉及隐私,所以需要制作专门的加密压缩工具,使压缩包无法用通用的压缩工具打开.由于同组的C++同事都在忙别的项目,这个工具就由我来开发了.

预期功能点:
  • 选择文件夹,使用node遍历出所有图片文件,压缩编码成640*480分辨率以下的图片
  • 将图片打包生成压缩包,不勾选加密生成.zip,勾选加密生成biu包(biu是我起的加密压缩包名字)
  • 具备加密biu包和不加密zip包互相转换的功能
  • 识别无效的照片,包括非jpg、jpeg格式的图片
选型:
  • 遍历文件夹和生成压缩文件属于IO操作,选用node
  • 要兼容多版本windows系统,使用electron-vue
  • 要对图像进行压缩,选用resize-optimize-images,能知道这个包还是非常巧合,网上的博客,适合node图片压缩的包有两个: gm和images, images在win10中有很大的兼容性问题,而gm在electron中编译一直失败. 所以我在github上以images和resize为关键字,搜索结果中排名前三十的包全部试用了一遍,大部分在electron都存在问题,最后逼得我准备在electron中调用OpenCV库的时候,忽然发现了这个包非常小巧,而且正好能够满足要求
  • 获取图片分辨率,选用get-pixels
  • 打包图片生成压缩包,使用jszip
  • 加密压缩包,手写加密函数

下面是最终的成品: 耗时两天完成


工具界面
详细流程:
  • electron进程间通信:
    electron主进程捕获渲染进程发出的打开文件夹事件, 在electron渲染进程中设置监听,然后通过ipcRenderer.send发送给主进程,主进程通过ipcMain.on监听到消息后,调用 dialog.showOpenDialog就可以实现打开文件夹/文件操作了
    实现关键代码:
渲染进程
<template>
     <button class="btn"
                @click="showModalHandler">选择要压缩的文件夹</button>
</template>

<script>
export default {
  methods: {
    showModalHandler() {
      ipcRenderer.send("open-directory-dialog", {
        openMethod: "openDirectory",  // 参数openDirectory为打开文件夹,参数openFile为打开文件
      });
    },
</script>
主进程
ipcMain.on('open-directory-dialog', function (event, compressOpt) {
    dialog.showOpenDialog({
        properties: [compressOpt.openMethod] 
    }, function (files) {
        if (files) {// 如果有选中
            // 发送选择的对象给子进程, files[0]是打开的文件绝对路径
            // console.log("path", files[0]);
            // TODO 文件操作,压缩打包
            event.sender.send('selectedItem', files[0])
        }
    })
});

  • node的文件操作:
  1. 路径是否存在: fs.existsSync(path)
  2. 文件信息: info = fs.statSync(path)
  3. 是目录:info.isDirectory()
  4. 是文件:info.isFile()
  5. 新建文件夹:fs.mkdirSync(path)
  6. 删除空文件夹,必须为空才能删除:fs.rmdirSync(path)
  7. 创建文件:fs.writeFileSync(filename,arraybuffer)
  8. 删除文件:fs.unlinkSync(path)
  9. 读取文件:content =fs.readFileSync(filePath)
  10. 打开文件夹,使用cmd调用explorer.exe:
const exec = require('child_process').exec
exec(`explorer.exe /e, /root,${dirPath}`)
  1. 删除目录代码(目录有文件也能删除)
function delPath(path) {
    if (!fs.existsSync(path)) {
        console.log("路径不存在");
        return "路径不存在";
    }
    var info = fs.statSync(path);
    if (info.isDirectory()) {//目录
        var data = fs.readdirSync(path);
        if (data.length > 0) {
            for (var i = 0; i < data.length; i++) {
                delPath(`${path}/${data[i]}`); //使用递归
                if (i == data.length - 1) { //删了目录里的内容就删掉这个目录
                    delPath(`${path}`);
                }
            }
        } else {
            fs.rmdirSync(path);//删除空目录
        }
    } else if (info.isFile()) {
        fs.unlinkSync(path);//删除文件
    }
}
  • 判断图片是否是jpg图片
    在实践中我发现单纯根据后缀名来判断并不准确,因为有相当多的jpg图片是被用户用其他类型图片(png,jfif等)改过来的,所以我编写了对jpg图片的校验函数。使用jpg文件头来校验
  const JPGBuffer = fs.readFileSync(JPGPath);
  testJPGHeader(JPGBuffer)

  function testJPGHeader(JPGBuffer) {
        const JPGHeader = JPGBuffer.slice(0, 3);
        const HT_STD_HEADER = [0xff, 0xd8, 0xff];
        let notJPG = false;
        JPGHeader.forEach((v, i) => {
            if (HT_STD_HEADER[i] !== v) notJPG = true;
        });
        return notJPG;
    }
  • ipc重复消息事件处理
    由于是通过点击调用进程通信,处理事件,所以我将ipc监听注册在了点击事件中。但是这样有一个弊端,当重复点击时,会造成ipc监听函数被反复注册,解决的方法一种是使用ipcRenderer.once使注册的监听函数只运行一次,即被释放,这适用于唯一的确定的消息,比如开始、结束等。另一种是使用ipcRenderer.on注册,并在注册之前使用ipcRenderer.removeAllListeners取消以往的注册,这适用于频繁的、不确定数量的消息,比如发送图片缩放失败的消息。
监听函数被反复注册,造成失败消息被重复发送.jpg
html:
 <button class="btn"
              @click="showModalHandler">选择要打包的文件夹</button>

js:
 showModalHandler() {
    const that = this;
    this.init();
    //取消失败消息监听,避免反复注册
    ipcRenderer.removeAllListeners("extension-error");
    //发送打开目录消息
    ipcRenderer.send("open-directory-dialog", {
      openMethod: "openDirectory",
      compress: this.compress
    });
     //打开文件消息
    ipcRenderer.once("selectedItem", function(e, path) {
      console.log("selectedItem", path);
      that.loading = true;
    });
     //完成压缩消息
    ipcRenderer.once("patchZip-Done", function(e, zipName) {
      console.log("zipName", zipName);
      if (zipName === "error") {
        that.zipFill = true;
      }
      that.loading = false;
    });
    //完成压缩消息
    ipcRenderer.on("extension-error", function(e, filename) {
      console.log("extension-error-filename", filename);
      that.errfiles.push(filename);
    });
  },
  • 图片压缩
    要对图像进行压缩,选用resize-optimize-images,能知道这个包还是非常巧合,网上的博客,适合node图片压缩的包有两个: gm和images, images在win10中有很大的兼容性问题,而gm在electron中编译一直失败. 所以我在github上以images和resize为关键字,搜索结果中排名前三十的包全部试用了一遍,大部分在electron都存在问题,node-OpenCV是一个万能的图形库, 但是十分臃肿, 正当我准备手撸插值压缩算法的时候,忽然发现了resize-optimize-images这个包非常小巧,而且正好能够满足要求

    压缩的基本思路是在图片中找到相对比较短的边, 以他为基准压缩到一个比较小的比例(如640*480),然后将原始图片复制出来,再将压缩后的图片写进去即可. 图像的大小最终取决于三个因素:图片像素、编码质量和位深度,图形像素在执行完这条个函数之后就达到一个比较小的范围了,一般不会超过100k,而编码质量是一种类似于PS中表面模糊的效果,使用的是插值算法,一般编程80-90区间内可以对图像进行进一步的压缩,但是不建议压的太多,会失真。真彩色的位深度一般是四个字节

    getPixels(fileAbPath, function (err, pixels) {
        if (err) {
            return
        }
        let compressRio = 0.9;
        let [imageWidth, imageHeight] = pixels.shape
         //找到比较小的边
        let loopVar = imageWidth > imageHeight ? imageHeight : imageWidth;
        while (loopVar >= 480) {
            loopVar = loopVar * compressRio;
            imageWidth = imageWidth * compressRio;
            imageHeight = imageHeight * compressRio;
        }
        imageWidth = Math.floor(imageWidth)
        imageHeight = Math.floor(imageHeight)
        const content = pixels.data
        fs.writeFile(outPath, content, async function (err) {
            const options = {
                images: [outPath],
                width: imageWidth,
                height: imageHeight,
                quality: 85
            };
            // 执行压缩.
            await resizeOptimizeImages(options);
        });
    })
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,718评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,683评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,207评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,755评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,862评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,050评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,136评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,882评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,330评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,651评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,789评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,477评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,135评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,864评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,099评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,598评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,697评论 2 351