导出二维数据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 < col; 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 方案的更多相关文章

  1. 导出带图片的Excel——OOXML文件分析

    需求: 普通js导出文件excel具有兼容性问题,通过js-xsl导出文件API未找到导出图片的方案,实例过少,因此针对07年后以.xlsx后缀的excel文件,通过修改后缀.zip参考文件模板来实现 ...

  2. Laravel Excel模板导出-带图片

    Laravel Excel版本 3.1 1.数据准备 建个2个表,加点数据,控制器中查数据,给模板使用. 表1-order:id, order_no, img_path, note 表2-order_ ...

  3. 邮件实现详解(四)------JavaMail 发送(带图片和附件)和接收邮件

    好了,进入这个系列教程最主要的步骤了,前面邮件的理论知识我们都了解了,那么这篇博客我们将用代码完成邮件的发送.这在实际项目中应用的非常广泛,比如注册需要发送邮件进行账号激活,再比如OA项目中利用邮件进 ...

  4. freemarker导出带图片的word文档

    最近做一个关于文档导出功能, 顺便学习了下freemarker,做了个关于导出带图片的word文档,模板并没有写全,只是验证代码的正确性 这只是做一个小功能,故只做了后台代码关于导出的代码,并未与前台 ...

  5. 接口测试基础——第3篇smtplib发送带图片的邮件

    smtplib发送邮件最后一篇,发送带图片的邮件: 大家可以去廖雪峰的网站看一下,下面的代码就是我跟着博客写的,哈哈,大家即使不明白为什么,也要多写两遍,记在心里,如果有不明白的地方可以留言,船长会第 ...

  6. 终端下更改printk打印级别

    如何去更改printk的打印级别? 1.查看当前控制台的打印级别 # cat /proc/sys/kernel/printk 该文件有4个数字值,它们根据日志记录消息的重要性,定义将其发送到何处,上面 ...

  7. JAVA实现带图片的列表——JList

    JList:显示对象列表并且允许用户选择一个或多个项的组件. JList的构造方法: 1.根据数组创建列表: JList(Object[] listData) 构造一个 JList,使其显示指定数组中 ...

  8. C# Qrcode生成二维码支持中文,带图片,带文字 2015-01-22 15:11 616人阅读 评论(1) 收藏

    1.下载Qrcode库源码,下载地址:http://www.codeproject.com/Articles/20574/Open-Source-QRCode-Library 2.打开源码时,部分类库 ...

  9. 自定义带图片和文字的ImageTextButton

    今天我们来讲一下有关自定义控件的问题,今天讲的这篇是从布局自定义开始的,难度不大,一看就明白,估计有的同学或者开发者看了说,这种方式多此一举,但是小编我不这么认为,多一种解决方式,就多一种举一反三的学 ...

随机推荐

  1. Linux下paste命令

    paste 用于将多个文件按照列队列进行合并. 该命令主要用来将多个文件的内容合并,与cut命令完成的功能刚好相反. 1.原文件: 1>a.txt [root@localhost home]# ...

  2. iOS 直播(一)

    由于业务需求,需要从腾讯直播sdk要迅速转移到自主开发(先让我默哀三分钟).不多说,直接开始唠嗑! 这个项目超级简单,简单到只能一个推流,一个拉流的功能.需求多的再另谈. 前期准备:推流用优酷开源的L ...

  3. Replication--复制延迟的诊断和解决

    要解决复制延迟问题,需要首先定位复制延迟发生点,再找出复制延迟的原因,再做相应处理. 复制延迟发生点:1. 发布服务器2. 分发服务器3. 订阅服务器4. 发布服务器与分发服务器和分发服务器与订阅服务 ...

  4. Queue插入的时候报错:源数组长度不足。请检查 srcIndex 和长度以及数组的下限。

    异常问题记录: 本想自己手动实现一个日志记录功能.使用Queue队列集合来实现多线程的日志记录. 测试 一个线程写入数据Enqueue和一个线程读取数据Dequeue ,直接用的无休眠死循环. 终于抛 ...

  5. C#与数据库访问技术总结(十八)

    ADO.NET 代码综合示例 前面已经介绍过OLE DB.NET和SQL Server.NET数据提供者可以用来连接不同的数据源. 以下代码不仅综合演示了使用ADO.NET的这两种数据提供者访问数据库 ...

  6. 编码剖析Spring管理bean的原理

    project目录 MyClassPathXMLApplicationContext读取xml,以及实例化bean. 因为是一开始实例化配置文件所有bean,所以需要构造器完成这些工作. packag ...

  7. paip.花生壳 服务启动失败 以及不能安装服务,权限失败的解决

    paip.花生壳 服务启动失败 以及不能安装服务,权限失败的解决 系统win7 NewPhDDNS_1.0.0.30166.exe  作者Attilax  艾龙,  EMAIL:1466519819@ ...

  8. Leetcode 6 ZigZag Conversion 字符串处理

    题意:将字符串排成Z字形. PAHNAPLSIIGYIR 如果是5的话,是这样排的 P     I AP   YR H L G N  SI A    I 于是,少年少女们,自己去找规律吧 提示:每个Z ...

  9. Java插件开发-取插件下的某个文件

    //找到插件所在处 Bundle bundle = Activator.getDefault().getBundle(); //根据插件转义成URL路径 URL url = FileLocator.t ...

  10. .NET开发中经常用到的扩展方法

    整理一下自己经常用到的几个扩展方法,在实际项目中确实好用,节省了不少的工作量. 1  匿名对象转化 在WinForm中,如果涉及较长时间的操作,我们一般会用一个BackgroundWorker来做封装 ...