实现打印级别且带图片的Excel 方案
导出二维数据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 方案的更多相关文章
- 导出带图片的Excel——OOXML文件分析
需求: 普通js导出文件excel具有兼容性问题,通过js-xsl导出文件API未找到导出图片的方案,实例过少,因此针对07年后以.xlsx后缀的excel文件,通过修改后缀.zip参考文件模板来实现 ...
- Laravel Excel模板导出-带图片
Laravel Excel版本 3.1 1.数据准备 建个2个表,加点数据,控制器中查数据,给模板使用. 表1-order:id, order_no, img_path, note 表2-order_ ...
- 邮件实现详解(四)------JavaMail 发送(带图片和附件)和接收邮件
好了,进入这个系列教程最主要的步骤了,前面邮件的理论知识我们都了解了,那么这篇博客我们将用代码完成邮件的发送.这在实际项目中应用的非常广泛,比如注册需要发送邮件进行账号激活,再比如OA项目中利用邮件进 ...
- freemarker导出带图片的word文档
最近做一个关于文档导出功能, 顺便学习了下freemarker,做了个关于导出带图片的word文档,模板并没有写全,只是验证代码的正确性 这只是做一个小功能,故只做了后台代码关于导出的代码,并未与前台 ...
- 接口测试基础——第3篇smtplib发送带图片的邮件
smtplib发送邮件最后一篇,发送带图片的邮件: 大家可以去廖雪峰的网站看一下,下面的代码就是我跟着博客写的,哈哈,大家即使不明白为什么,也要多写两遍,记在心里,如果有不明白的地方可以留言,船长会第 ...
- 终端下更改printk打印级别
如何去更改printk的打印级别? 1.查看当前控制台的打印级别 # cat /proc/sys/kernel/printk 该文件有4个数字值,它们根据日志记录消息的重要性,定义将其发送到何处,上面 ...
- JAVA实现带图片的列表——JList
JList:显示对象列表并且允许用户选择一个或多个项的组件. JList的构造方法: 1.根据数组创建列表: JList(Object[] listData) 构造一个 JList,使其显示指定数组中 ...
- C# Qrcode生成二维码支持中文,带图片,带文字 2015-01-22 15:11 616人阅读 评论(1) 收藏
1.下载Qrcode库源码,下载地址:http://www.codeproject.com/Articles/20574/Open-Source-QRCode-Library 2.打开源码时,部分类库 ...
- 自定义带图片和文字的ImageTextButton
今天我们来讲一下有关自定义控件的问题,今天讲的这篇是从布局自定义开始的,难度不大,一看就明白,估计有的同学或者开发者看了说,这种方式多此一举,但是小编我不这么认为,多一种解决方式,就多一种举一反三的学 ...
随机推荐
- glRotate函数
void glRotate( GLdouble angle, GLdouble x, GLdouble y, GLdouble z ) API说明:angle为旋转角度,单位为度:x,y,z是旋转轴的 ...
- Excel表3级数据整理工具
前言 做专题经常会遇到做数据级联的需求,大部分需求都长一个模样.销售给你一个excel表,然后你做一个省市经销商的级联.不知道以前大家是怎么样做的,我之前是把excel复制到sublime中,然后使用 ...
- HTTP协议状态码详解(HTTP Status Code)
转自:http://www.cnblogs.com/shanyou/archive/2012/05/06/2486134.html 使用ASP.NET/PHP/JSP 或者javascript都会用到 ...
- Magicodes.WeiChat——WeChatOAuthTest(网页授权获取用户基本信息)
Demo访问地址:http://wechat.magicodes.net/app/AppDemo/WeChatOAuthTest?tenantId=1 关于公众号如何获取用户信息,请参考此文档:htt ...
- TypeScript 0.9.1 发布,新增 typeof 关键字
TypeScript 0.9.1 发布了,该版本提升了编译器和语言的性能,增加新的语言特性 typeof ,更好的 this 处理等.详细介绍请看发行说明. TypeScript 是微软新推出的一种语 ...
- ASP.NET的一次奇遇:UserControl写成Control引发的w3wp进程崩溃
昨天在写代码中一不小心将UserControl写成了Control,将原来应该继承自System.Web.UI.UserControl的用户控件,比如下面的BlogStats: <%@ Cont ...
- javascript和C#比较
C#和javascript有很多相似的地方,比如: 序列化 C#序列化 首先需要引用 using System.Web.Script.Serialization;//System.Web.Extens ...
- centos 关闭防火墙
在centos上搭建了个服务器,本机可以访问,局域网无法访问 解决方案:关闭防火墙 sudo systemctl stop firewalld.service 令人恼火的是stop iptables之 ...
- paip.输入法编程--英文ati化By音标原理与中文atiEn处理流程 python 代码为例
paip.输入法编程--英文ati化By音标原理与中文atiEn处理流程 python 代码为例 #---目标 1. en vs enPHati 2.en vs enPhAtiSmp 3.cn vs ...
- discuz x3在DIY模块中调用伪静态不成功,显示动态链接的解决办法
discuz x3在DIY模块中调用伪静态不成功,显示动态链接,然而其他的链接正常显示伪静态. 后台启用伪静态后,发现论坛版块.帖子点击链接,伪静态正常显示,然后在门户首页DIY显示的帖子,点进去后发 ...