[{"id":"ff052704-1389-4029-bcdd-73c6cbe9f807","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"8e72e33e-b4ab-4fb3-98fc-a0b148134aed","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"c90d5fdf-420f-4978-8bf5-c9a2bb4334b3","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"900e6cab-065d-4f1c-844d-efa3c074e270","tags":[{"product":null,"links":null,"id":"a2f84374-4a3f-4d22-96fd-765e9de495bf","name":"\u66F4\u65B0","color":"Coral","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"760b37c7-c713-4b24-b9ba-4bfe7d8437a6","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"89c707ed-9841-4e53-96fb-940cc3214804","tags":[{"product":null,"links":null,"id":"a2f84374-4a3f-4d22-96fd-765e9de495bf","name":"\u66F4\u65B0","color":"Coral","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"64e2f4a3-2303-4bfa-8a93-6c23ef01de58","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"aa71a884-c5bc-4842-8d6a-873dfd645167","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"2605431b-dc80-491d-886e-28981595d277","tags":[{"product":null,"links":null,"id":"a2f84374-4a3f-4d22-96fd-765e9de495bf","name":"\u66F4\u65B0","color":"Coral","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"69e79655-e015-4f9a-a230-2a25c988c926","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"a570c8e7-07a2-47da-965b-da44fd1fa5cf","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"86662220-9b9e-4940-9ced-d22642ea49a8","tags":[{"product":null,"links":null,"id":"a2f84374-4a3f-4d22-96fd-765e9de495bf","name":"\u66F4\u65B0","color":"Coral","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"24769ecd-2b08-4a85-b318-4f533bbf8393","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"a244ead7-a2c6-47a3-ac17-c5dbfa337362","tags":[{"product":null,"links":null,"id":"a2f84374-4a3f-4d22-96fd-765e9de495bf","name":"\u66F4\u65B0","color":"Coral","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"04e73782-aa78-4dfe-a4f9-e72ed4c78a11","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"4f41e22e-eb51-49e5-aeae-a42dd6bf352c","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"790bc77a-9216-48fd-b8f9-fbc374ebb155","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"21c6131f-0f2d-41d1-9284-6ad9ee803c1f","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"7ded3a22-15eb-49b8-a488-e83c2cd872eb","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"33638a1c-7196-42c1-a96d-38b2d9ba8ac4","tags":[{"product":null,"links":null,"id":"a2f84374-4a3f-4d22-96fd-765e9de495bf","name":"\u66F4\u65B0","color":"Coral","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"4a0842a3-20b1-40c3-8e00-cd5941ffdf53","tags":[{"product":null,"links":null,"id":"a2f84374-4a3f-4d22-96fd-765e9de495bf","name":"\u66F4\u65B0","color":"Coral","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"6e6ac5b1-1501-4e28-89cc-525139488537","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"b0576ca2-cb84-4390-9f95-9354ec20eda5","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"55fd3bb8-18d4-4edb-9640-ca3a365b798f","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"dacc7931-6785-4675-be31-80930403cf7b","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"63c577e6-6cf9-497e-94e1-2307f7d3f498","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"1b88a165-2563-437e-99bb-ae30bd4b56db","tags":[{"product":null,"links":null,"id":"a2f84374-4a3f-4d22-96fd-765e9de495bf","name":"\u66F4\u65B0","color":"Coral","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"adf3817f-7667-4a4c-8a5f-767b0b7e1e3e","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"3a083583-1d5f-492b-b450-34b2b5c775b8","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"5d37413b-3600-4da9-9700-feea54355f59","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"a5a73576-16ea-4cbc-925c-ef547389eaa5","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"1126cfab-6210-4e28-bee0-02c113fb7a0c","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"09a66339-64c0-415c-b142-0691587a8e4a","tags":[{"product":null,"links":null,"id":"9e117e35-984a-4c14-95ca-ef0ec7b9fb60","name":"\u65B0\u589E","color":"DarkGreen","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"0baaf91d-84f6-404c-a487-735226b6d5b6","tags":[{"product":null,"links":null,"id":"a2f84374-4a3f-4d22-96fd-765e9de495bf","name":"\u66F4\u65B0","color":"Coral","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"5b7aeac5-c755-426a-95c7-1ae8e547179a","tags":[{"product":null,"links":null,"id":"a2f84374-4a3f-4d22-96fd-765e9de495bf","name":"\u66F4\u65B0","color":"Coral","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]},{"id":"57c94653-8893-403b-a5b1-0d1e33a0bd0f","tags":[{"product":null,"links":null,"id":"a2f84374-4a3f-4d22-96fd-765e9de495bf","name":"\u66F4\u65B0","color":"Coral","productId":"098be112-50ec-44e4-b746-6bc8bf76af97"}]}]
        
(Showing Draft Content)

附件的上传下载修改

需求:在SpreadJS中添加附件


背景:

在本地使用Excel时,经常会有需要在Excel中添加一些附件文件的需求,例如在Excel中附带一些Word,CAD图等等。Excel能支持这些文件的添加,本质上并不是Excel可以直接打开这些文件。而是由于我们本机已经安装了能打开这些文件的软件。关于Excel中上传附件文件可以参考:在Excel电子表格中插入对象。

类比到Web端,也会有不少客户,希望能够在Spread中添加一个附件。但是Web端不是本地环境,无法直接打开Word等其它附件。但是在Excel中,也有一种通过链接的形式来显示附件的方式。而SpreadJS中也是提供超链接的形式,利用这一点,或许可以对SpreadJS添加附件展开探索。

解决方案:

SpreadJS中提供了三种超链接的形式,分别是:

(1)超链接单元格

(2)超链接类型

(3)HYPERLINK函数

超链接类型与HYPERLINK函数与Excel兼容,而超链接单元格从SpreadJS导出Excel时,由于实现机制有差异,Excel无法识别,所以实现附件需求尽量使用超链接类型或HYPERLINK函数,我在这里使用的是超链接类型。

在需要在某个单元格中上传附件时,我们可以弹出一个模态框,在模态框中上传文件,点击提交之后,可以对文件做一个暂存,将文件信息存储在单元格的Tag中。Tag在SpreadJS当中可以理解为是对单元格的一种属性标记,可以支持复杂类型。在这里我存储的是一个json,json结构如下:

{
    type: 'attachfile',
    fileInfo: object file
}

其中type表示当前是否为附件,fileInfo代表的是文件对象。**由于我演示的是纯前端的demo,所以fileInfo中传递的是File对象,File对象在实际使用时需要从os中在我读取文件,这种形式在传前端的demo演示下可行。但在实际项目中,fileInfo应该使用的是文件上传完成后的路径地址,这一点在实际项目中一定要注意!**同时需要将单元格设置为超链接,超链接的url为一个相对路径。至于为何是相对路径,这一点是为了后续导出文件后能正常访问到附件文件,与Excel中的机制相同。对同一个单元格重复上传附件时会做文件的替换,此时我们只需要对应的去修改Tag及超链接的信息即可。与此对应,清除附件文件,只需要清除对应的Tag和值信息即可。接下来会将一些核心的代码逻辑。


1. 创建页面的HTML结构

<body>
    <button id="uploadAttach">上传附件</button>
    <div id="fileOperate"
        style="visibility: hidden;position: absolute;top: 100px;left: 300px;z-index: 10; background-color: #eee;padding: 16px">
        <label for="choseFile">选择文件</label>
        <input type="file" id="choseFile" name="choseFile" />
        <button id="submit">提交</button>
        <button id="cancel">取消</button>
    </div>
    <div id="ss" style="width:100%;height:98vh;border:1px solid darkgray"></div>
</body>

点击上传附件文件,id为fileOperate的div会显示出来,在这里可以完成文件的上传及暂存。


2. 暂存附件信息

function hasAttachFile(sheet,row,col,file){
    /**
     * 附件文件暂存
     * 这里由于没有服务端,所以我直接存了File对象,但File对象只有在实际使用时才会去获取实际的文件内容。在demo中可行
     * 在实际项目中,需要将file对象上传到文件服务器中
     * 上传完成后tag中的fileInfo应该代表的是文件的访问地址,而不能再是File对象。
     */
    sheet.setValue(row,col,file.name)
    sheet.setTag(row,col,{
        type: hyerlinkType,
        fileInfo: file        // 实际项目中fileInfo应该为上传完成文件的访问路径
    })
    sheet.setHyperlink(row, col, {
        url: file.name,
        linkColor: '#0066cc',
        visitedLinkColor: '#3399ff',
        drawUnderline: true,
        command:'downloadAttachFile',
     }, GC.Spread.Sheets.SheetArea.viewport);
}

这一步起始主要用来设置文件上传之后单元格超链接及tag信息。细心的同学会注意到,这里我注册了一个命令,超链接本身会有一个跳转的行为,写command之后,会阻止这个默认跳转,转去执行对应的命令。注册的命令主要就是用来做附件文件的下载。关于SpreadJS中如何注册命令可以在学习指南或API中检索。

 // 下载文件
    spread.commandManager().register("downloadAttachFile",{
        canUndo: false,
        execute: function(context,options,isUndo){
            let sheet = context.getActiveSheet()
            let row = sheet.getActiveRowIndex()
            let col = sheet.getActiveColumnIndex()
            let cellTag = sheet.getTag(row,col)
            console.log(sheet,row,col,cellTag)
            if(cellTag && cellTag.type==hyerlinkType){
                /***
                 * 纯前端demo,文件存在于本地,fileInfo中存储的是File对象,可以直接获取到文件
                 * 实际项目中,fileInfo应该是上传到文件服务器上的文件访问地址。
                 * 因此这里需要发送请求,先获取文件blob,将获取的blob传递到saveAs的第二个参数中。
                 */
                saveAs(cellTag.fileInfo,cellTag.fileInfo.name)
            }
        }
    })

在这里,我引入了三方组件库FileSaver,在点击超链接单元格时,可以支持当前附件文件的下载。


3. 附件文件清除

document.getElementById("removeAttach").onclick = function(){
    /***
     * 清除附件
     * 清除附件需要先删除远程文件服务器的文件,之后清除单元格的Tag信息。
     * 这里前端演示demo,只删除了tag。
     * 实际项目中tag中的fileInfo应该是文件上传后的路径
     */
    let sheet = spread.getActiveSheet()
    let row = sheet.getActiveRowIndex()
    let col = sheet.getActiveColumnIndex()
     spread.commandManager().execute({
            cmd:"removeAttachFile",
            sheet,row,col
        })
}

4. 文件保存

SpreadJS提供了一种结构叫做SSJSON,用来描述一个Excel文件的全部信息。也提供了文件blob的形式,但由于导出文件流之后Tag信息会清除,因为Excel中并没有Tag的概念。所以对于当前的形式,需要将文件保存成json结构。如果有必须保存excel文件的需求,我们可以将文件信息保存在Excel与SpreadJS都能兼容的属性中。

document.getElementById("fileSaver").onclick = function(){
    // 保存文件
    submitFile = spread.toJSON()
    spread.clearSheets()
    spread.addSheet(0)
}

因为是一个纯前端的demo,这里我用一个全局变量来表示当前保存的文件。


5. 文件加载

document.getElementById("loadSubmitFile").onclick = function(){
    // 加载已保存文件
    spread.fromJSON(submitFile)
}

详细的Demo可以参考:https://gcdn.grapecity.com.cn/forum.php?mod=viewthread&tid=151235&extra=page%3D8%26filter%3Dauthor%26orderby%3Ddateline

接下来还有最后一步,就是实现文件的下载。在Excel中,如果我们想要保证附件文件能访问到,需要保证链接所指向的地址能够找到文件。按照之前添加的超链接的形式,附件文件需要和宿主文件处于同一个层级。那最简单的就是我们依次去下载宿主文件及附件文件即可。但是为了易用性考虑,我们尽量将宿主文件与附件文件做一个打包下载,这样时比较直观易用的。


实现方案:

关于打包下载的实现,这里我采用了两个开源的组件,分别是FileSaver以及JSZIP,关于这两个开源组件的介绍,大家自行搜索即可(用好搜索引擎是一个优秀程序员的必备素养)。

在点击打包下载时,我们先依次遍历当前spread中时附件文件的单元格,依次将文件信息添加至实例化的zip中。**需要注意的是demo与实际项目存储在Tag中的fileInfo是不同的。demo中fileInfo存储的是File对象,但是File对象并不包含文件的实际内容,因此只能在demo中如此使用。在实际项目中,fileInfo应该是附件文件上传完成之后的访问地址。在添加到zip示例中,是需要发送请求,拿到具体的文件blob的。**除了附件文件之外,还需要利用SpreadJS中提供的ExcelIO能力,将Spread对象转换成一个文件blob,并添加到实例化zip中。所有文件添加完成之后,调用jszip打包下载即可。核心代码如下:

function loadAllFiles(){
    // 打包下载所有文件
    let zip = new JSZip()
    let io = new ExcelIO.IO()
    // 先获取附件文件
    let sheetCount = spread.getSheetCount()
    for(let i=0;i<sheetCount;i++){
        let sheet = spread.getSheet(i)
        let rowCount = sheet.getRowCount()
        let colCount = sheet.getColumnCount()
        for(let row=0;row<rowCount;row++){
            for(let col=0;col<colCount;col++){
                let cellTag = sheet.getTag(row,col)
                if(sheet.getHyperlink(row,col) && cellTag && cellTag.type==hyerlinkType){
                    // 实际项目中,单元格Tag中的fileInfo应该为文件路径,这里需要发送请求拿到文件的blob,demo里边直接从os里边读本地文件。
                    zip.file(cellTag.fileInfo.name,cellTag.fileInfo,{binary: true})
                }
            }
        }
    }
    io.save(spread.toJSON(),(blob)=>{
        console.log(blob)
        zip.file("主文件.xlsx",blob,{binary: true})
        zip.generateAsync({type:"blob"}).then((content)=>{
            saveAs(content,"download.zip")
        })
    })
}

具体的Demo实现可以参考: https://gcdn.grapecity.com.cn/forum.php?mod=viewthread&tid=151259&extra=page%3D8%26filter%3Dauthor%26orderby%3Ddateline

作者: Grapecity.China | 审核:Clark.Pan | 更新时间:2023.12.24