热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

实现带图片的Excel打印级别方案

导出二维数据excel,其实很简单,使用cvs就可以了。但是如果导出格式复杂带样式还带图片的怎么办?客户的要求有时就是这么变态。呵呵。如果使用.net,微软提供的有库,使用php好

导出二维数据excel,其实很简单,使用cvs就可以了。但是如果导出格式复杂带样式还带图片的怎么办?客户的要求有时就是这么变态。呵呵。如果使用.net,微软提供的有库,使用php好像也有现成的有库。我大致对这些库进行了解,都可以实现我们的需求。可惜我现在使用的golang没有那么库支持,所以只好裸搞了。

Excel格式分为两种,第一种是封闭式,不知道使用啥格式,比如office 2003使用的格式,扩展名为*.xsl。另一种是开放式的,使用的是Open XML技术,比如office 2007 以后的版本,好在现在已经2014了,7年过去,大部分人都已经用到office 2012,即使国产的wps也早已经完美支持Open XML 了。 所以不用考虑兼容问题了。

这次我的解决方案就是从Open XML 入手。通过对Open XML学习,解决思路大致可分为3步骤:

第一步:使用支持Open XML 的软件,比如 office 2010制作一个我们想要的Excel,保存扩展名名为xlsx。

,

第二步:把xlsx修改扩展名为 zip,解压后使用占位符,修改里面相应的XML文件,然后压缩,再把扩展名修改为xlsx。这个压缩文件我们可称为导出模板。

,

第三步:使用程序动态解压,替换占位符,再压缩。其中对图片的处理,是把图片保存到相应文件夹以及相应XML,千言万语,不如代码更加直接,实现代码如下:

func (c Order) Excel(pageIndex int, pageSize int, sortField string, sortOrder string, customId int64, state string, orderTime string) revel.Result {
    sql := "select a.name A,b.name B,e.name C,d.name D,c.name E,a.order_time F,a.money G,a.state H,a.image I,a.width J,a.height K,a.area L,a.unit M,a.amount N,a.price O,f.alias P,a.remarks Q from ad_order a,ad_custom b,ad_product c,ad_stuff d,ad_stuff_cat e,ad_user f where a.product_id=c.id and c.stuff_id=d.id and d.cat_id=e.id and a.custom_id=b.id and a.user_id=f.id and a.del_state=‘未删‘ %s %s %s order by a.id desc"
    sql = fmt.Sprintf(sql, fmt.Sprintf("and a.custom_id=%d", customId), "%s", "%s")
    if state != "" {
        sql = fmt.Sprintf(sql, fmt.Sprintf("and a.state=‘%s‘", state), "%s")
    } else {
        sql = fmt.Sprintf(sql, "", "%s")
    }
    if orderTime != "" {
        sql = fmt.Sprintf(sql, fmt.Sprintf("and a.order_time=‘%s‘", orderTime))
    } else {
        sql = fmt.Sprintf(sql, "")
    }
    orders, err := Orm.Query(sql)
    if err != nil {
        return c.RenderJson(models.Message{State: "failure", Msg: err.Error()})
    }
    rows := make([]Rows, 0)
    col := 17
    for i := 1; i <= len(orders); i++ {
        row := Rows{}
        row.RowId = i + 1
        for k := 0; k  {
            row.Columns = append(row.Columns, Columns{R: fmt.Sprintf("%s%d", string(ABC[k]), i+1), V: i*col + k})
        }
        rows = append(rows, row)
    }
    vmlDrawings := make([]VmlDrawing, 0)
    sharedStrings := make([]string, 0)
    sharedStrings = append(sharedStrings, "订单名称")
    sharedStrings = append(sharedStrings, "客户名称")
    sharedStrings = append(sharedStrings, "产品目录")
    sharedStrings = append(sharedStrings, "产品材料")
    sharedStrings = append(sharedStrings, "产品名称")
    sharedStrings = append(sharedStrings, "订单日期")
    sharedStrings = append(sharedStrings, "金额")
    sharedStrings = append(sharedStrings, "收费状态")
    sharedStrings = append(sharedStrings, "产品图片")
    sharedStrings = append(sharedStrings, "宽度(米)")
    sharedStrings = append(sharedStrings, "高度(米)")
    sharedStrings = append(sharedStrings, "面积")
    sharedStrings = append(sharedStrings, "单位")
    sharedStrings = append(sharedStrings, "数量")
    sharedStrings = append(sharedStrings, "单价")
    sharedStrings = append(sharedStrings, "经手人")
    sharedStrings = append(sharedStrings, "备注")
    for i, row := range orders {
        if string(row["I"]) != "" {
            img := string(row["I"])
            vmlDrawing := VmlDrawing{}
            vmlDrawing.Index = i
            vmlDrawing.Id = img[:strings.Index(img, ".")]
            vmlDrawing.Name = fmt.Sprintf("S%s", img)
            vmlDrawing.RowBegin = i + 1
            vmlDrawing.RowEnd = i + 2
            vmlDrawings = append(vmlDrawings, vmlDrawing)
        }
        sharedStrings = append(sharedStrings, string(row["A"])) //订单名称
        sharedStrings = append(sharedStrings, string(row["B"])) //客户名称
        sharedStrings = append(sharedStrings, string(row["C"])) //产品目录
        sharedStrings = append(sharedStrings, string(row["D"])) //产品材料
        sharedStrings = append(sharedStrings, string(row["E"])) //产品名称
        sharedStrings = append(sharedStrings, string(row["F"])) //订单日期

        f1, _ := strconv.ParseFloat(string(row["G"]), 32)
        sharedStrings = append(sharedStrings, fmt.Sprintf("%.2f", f1)) //金额
        sharedStrings = append(sharedStrings, string(row["H"]))        //收费状态
        sharedStrings = append(sharedStrings, "")                      //产品图片

        f2, _ := strconv.ParseFloat(string(row["J"]), 32)
        sharedStrings = append(sharedStrings, fmt.Sprintf("%.2f", f2)) //宽度(米)

        f3, _ := strconv.ParseFloat(string(row["K"]), 32)
        sharedStrings = append(sharedStrings, fmt.Sprintf("%.2f", f3)) //高度(米)

        f4, _ := strconv.ParseFloat(string(row["L"]), 32)
        sharedStrings = append(sharedStrings, fmt.Sprintf("%.4f", f4)) //面积
        sharedStrings = append(sharedStrings, string(row["M"]))        //单位
        sharedStrings = append(sharedStrings, string(row["N"]))        //数量

        f5, _ := strconv.ParseFloat(string(row["O"]), 32)
        sharedStrings = append(sharedStrings, fmt.Sprintf("%.2f", f5)) //单价
        sharedStrings = append(sharedStrings, string(row["P"]))        //经手人
        sharedStrings = append(sharedStrings, string(row["Q"]))        //备注
    }
    basePath := revel.BasePath
    basePathPrefix := fpath.Join(basePath, fpath.FromSlash("app/templates"))

    file, _ := os.Create(fpath.Join(basePathPrefix, fpath.FromSlash(fmt.Sprintf("%s.xlsx", "orders"))))
    w := zip.NewWriter(file)
    defer w.Close()
    r, _ := zip.OpenReader(fpath.Join(basePathPrefix, fpath.FromSlash(fmt.Sprintf("%s.xlsx", "order"))))
    defer r.Close()
    for _, f := range r.File {
        switch f.Name {
        case "xl/worksheets/sheet1.xml":
            buf := new(bytes.Buffer)
            rc, _ := f.Open()
            data, _ := ioutil.ReadAll(rc)
            rc.Close()
            tmpl, _ := template.New("sheet").Parse(string(data))
            tmpl.Execute(buf, rows)
            ff, _ := w.Create(f.Name)
            ff.Write(buf.Bytes())
            break
        case "xl/sharedStrings.xml":
            buf := new(bytes.Buffer)
            rc, _ := f.Open()
            data, _ := ioutil.ReadAll(rc)
            rc.Close()
            tmpl, _ := template.New("sharedStrings").Parse(string(data))
            tmpl.Execute(buf, sharedStrings)
            ff, _ := w.Create(f.Name)
            ff.Write(buf.Bytes())
            break
        case "xl/drawings/_rels/vmlDrawing1.vml.rels":
            buf := new(bytes.Buffer)
            rc, _ := f.Open()
            data, _ := ioutil.ReadAll(rc)
            rc.Close()
            tmpl, _ := template.New("vmlDrawing1.vml").Parse(string(data))
            tmpl.Execute(buf, vmlDrawings)
            ff, _ := w.Create(f.Name)
            ff.Write(buf.Bytes())
            break
        case "xl/drawings/vmlDrawing1.vml":
            buf := new(bytes.Buffer)
            rc, _ := f.Open()
            data, _ := ioutil.ReadAll(rc)
            rc.Close()
            tmpl, _ := template.New("vmlDrawing1").Parse(string(data))
            tmpl.Execute(buf, vmlDrawings)
            ff, _ := w.Create(f.Name)
            ff.Write(buf.Bytes())
            basePath := revel.BasePath
            basePathPrefix := fpath.Join(basePath, fpath.FromSlash("public/upload"))
            for _, v := range vmlDrawings {
                fsmall := fpath.Join(basePathPrefix, fpath.FromSlash(v.Name))
                file, _ := os.Open(fsmall)
                data, _ := ioutil.ReadAll(file)
                ff, _ := w.Create(fmt.Sprintf("xl/media/%s", v.Name))
                ff.Write(data)
            }
            break
        default:
            ff, _ := w.Create(f.Name)
            rc, _ := f.Open()
            data, _ := ioutil.ReadAll(rc)
            ff.Write(data)
            rc.Close()
        }
    }
    return c.RenderFile(file, revel.Attachment)
}

最后吐槽一下,博客园插入code,既然没有golang。

实现打印级别且带图片的Excel 方案


推荐阅读
  • Python自动化处理:从Word文档提取内容并生成带水印的PDF
    本文介绍如何利用Python实现从特定网站下载Word文档,去除水印并添加自定义水印,最终将文档转换为PDF格式。该方法适用于批量处理和自动化需求。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • PHP 5.2.5 安装与配置指南
    本文详细介绍了 PHP 5.2.5 的安装和配置步骤,帮助开发者解决常见的环境配置问题,特别是上传图片时遇到的错误。通过本教程,您可以顺利搭建并优化 PHP 运行环境。 ... [详细]
  • 1.如何在运行状态查看源代码?查看函数的源代码,我们通常会使用IDE来完成。比如在PyCharm中,你可以Ctrl+鼠标点击进入函数的源代码。那如果没有IDE呢?当我们想使用一个函 ... [详细]
  • andr ... [详细]
  • 本文介绍如何使用Python进行文本处理,包括分词和生成词云图。通过整合多个文本文件、去除停用词并生成词云图,展示文本数据的可视化分析方法。 ... [详细]
  • 本文探讨了如何在 PHP 的 Eloquent ORM 中实现数据表之间的关联查询,并通过具体示例详细解释了如何将关联数据嵌入到查询结果中。这不仅提高了数据查询的效率,还简化了代码逻辑。 ... [详细]
  • 三星W799在2011年的表现堪称经典,以其独特的双屏设计和强大的功能引领了双模手机的潮流。本文详细介绍其配置、功能及锁屏设置。 ... [详细]
  • 本文详细介绍了如何使用 Yii2 的 GridView 组件在列表页面实现数据的直接编辑功能。通过具体的代码示例和步骤,帮助开发者快速掌握这一实用技巧。 ... [详细]
  • 本文介绍如何使用 Python 提取和替换 .docx 文件中的图片。.docx 文件本质上是压缩文件,通过解压可以访问其中的图片资源。此外,我们还将探讨使用第三方库 docx 的方法来简化这一过程。 ... [详细]
  • 网络运维工程师负责确保企业IT基础设施的稳定运行,保障业务连续性和数据安全。他们需要具备多种技能,包括搭建和维护网络环境、监控系统性能、处理突发事件等。本文将探讨网络运维工程师的职业前景及其平均薪酬水平。 ... [详细]
  • 帝国CMS多图上传插件详解及使用指南
    本文介绍了一款用于帝国CMS的多图上传插件,该插件通过Flash技术实现批量图片上传功能,显著提升了多图上传效率。文章详细说明了插件的安装、配置和使用方法。 ... [详细]
  • PHP 5.5.0rc1 发布:深入解析 Zend OPcache
    2013年5月9日,PHP官方发布了PHP 5.5.0rc1和PHP 5.4.15正式版,这两个版本均支持64位环境。本文将详细介绍Zend OPcache的功能及其在Windows环境下的配置与测试。 ... [详细]
  • Win11扩展卷无法使用?解决扩展卷灰色问题的指南
    本文详细介绍了在Windows 11中遇到扩展卷灰色无法使用时的解决方案,帮助用户快速恢复磁盘扩展功能。 ... [详细]
  • 本文介绍如何在Linux服务器之间使用SCP命令进行文件传输。SCP(Secure Copy Protocol)是一种基于SSH的安全文件传输协议,支持从远程机器复制文件到本地服务器或反之。示例包括从192.168.45.147复制tomcat目录到本地/home路径。 ... [详细]
author-avatar
手机用户2502937767
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有