需求場景:客戶根據前臺界面列表所選擇的數據,根據須要的信息批量生成二維碼並造成一張圖片,而且每張圖片顯示的二維碼數量是固定的,須要分頁(即總共生成的二維碼圖片超出每頁顯示的需另起一頁生成),並下載到客戶端。前端
實現思路與技術:服務器
主要用到技術:Qrcode.net(生成二維碼的庫) app
SharpZipLib(壓縮庫)async
.NET GDI繪圖ide
整體思路:根據相關的信息用Qrcode.net 與GDI繪圖先繪製成單張二維碼圖片(如圖1),而後再把多個單張生成的二維碼按照每頁需顯示的個數再繪製到一張圖上,可生成多張(如:圖二),而後服務器再把相似多張(圖二)的圖片打包到一塊兒,造成一個壓縮文件(圖三),下載到客戶端。ui
(圖1) (圖二) (圖三)this
服務器端主要代碼:url
//生成二維碼
public JsonResult MakeQrcodeImage([Json]List<EquipmentLedger_SheBei> list) { string virtual_Path="~/upload/EquipmentImage_Temp/"+Guid.NewGuid().ToString(); string folderName = Server.MapPath(virtual_Path); Directory.CreateDirectory(folderName); foreach (var li in list) { DrawQrCode(li, folderName); } int pageIndex = 1; int pageSize = 45; decimal totalCount =Convert.ToDecimal(list.Count); decimal pages =Math.Ceiling(totalCount / pageSize); List<string> fildes = new List<string>(); for (int i = 1; i <= pages; i++) { var list_new = (from s in list orderby s.Id descending select s).Skip((pageIndex - 1) * pageSize).Take(pageSize).ToList<EquipmentLedger_SheBei>(); CombineImage(list_new, pageIndex, folderName); pageIndex++; } for (int j = 1; j <= pages; j++){ fildes.Add(folderName + "/QrCodes-Page" + j + ".bmp"); } string saveFile = folderName + "/QrCodesTest.zip"; Zip(fildes.ToArray(),saveFile); return Json(new MyJsonResult() { success = true, data = virtual_Path + "/QrCodesTest" }, null, null); } public FilePathResult DownLoadImage(string fieldName) { return File(Server.MapPath(fieldName) + ".zip", "application/octet-stream", "EquipmentLedgerQrCodes.zip"); }
//刪除臨時文件夾 public JsonResult DleteTempMakeQrcodeImage(string fieldName) { Directory.Delete(Server.MapPath(fieldName.Replace("/QrCodesTest","")),true); return Json(new MyJsonResult() { success = true }, null, null); } //繪製二維碼 private void DrawQrCode(EquipmentLedger_SheBei eq,string folderName){ byte[] bytes = Encoding.Default.GetBytes(eq.Id.ToString()); string content = Convert.ToBase64String(bytes); ; int moduleSize = 30; var encoder = new QrEncoder(ErrorCorrectionLevel.M); QrCode qrCode = encoder.Encode(content); GraphicsRenderer render = new GraphicsRenderer(new FixedModuleSize(moduleSize, QuietZoneModules.Four), Brushes.Black, Brushes.White); DrawingSize dSize = render.SizeCalculator.GetSize(qrCode.Matrix.Width); RectangleF Rec_Top = new Rectangle(new Point(0, 0), new Size(dSize.CodeWidth, 120));//頂部文字框 RectangleF Rec_botom_Num = new Rectangle(new Point(0, dSize.CodeWidth - 120), new Size(dSize.CodeWidth, 120));//底部文字框 //建立畫布 using (Bitmap map = new Bitmap(dSize.CodeWidth, dSize.CodeWidth)) { Graphics g = Graphics.FromImage(map); g.Clear(Color.White); Font drawFont = new Font("宋體", 60, FontStyle.Bold); SolidBrush drawBrush = new SolidBrush(Color.Red); render.Draw(g, qrCode.Matrix, new Point(0, 0)); StringFormat sf = new StringFormat(); sf.Alignment = StringAlignment.Center; sf.LineAlignment = StringAlignment.Center; g.DrawString(eq.Name, drawFont, drawBrush, Rec_Top, sf); g.DrawString(eq.Number, drawFont, drawBrush, Rec_botom_Num, sf); map.Save(folderName + "/" + eq.Id + ".bmp", ImageFormat.Bmp);//fileName爲存放的圖片路徑 g.Dispose(); map.Dispose();//釋放bmp文件資源 } }
private void Zip(string[] files, string ZipedFileName) { Zip(files, ZipedFileName, string.Empty); } private void Zip(string[] files, string ZipedFileName, string Password) { files = files.Where(f => System.IO.File.Exists(f)).ToArray(); if (files.Length != 0) { ZipOutputStream s = new ZipOutputStream(System.IO.File.Create(ZipedFileName)); s.SetLevel(6); if (!string.IsNullOrEmpty(Password.Trim())) s.Password = Password.Trim(); ZipFileDictory(files, s); s.Finish(); s.Close(); } } /// <summary> /// 壓縮多個文件 /// </summary> /// <param name="files">文件名</param> /// <param name="ZipedFileName">壓縮包文件名</param> /// <returns></returns> private void ZipFileDictory(string[] files, ZipOutputStream s) { ZipEntry entry = null; FileStream fs = null; Crc32 crc = new Crc32(); try { //建立當前文件夾 entry = new ZipEntry("/"); s.PutNextEntry(entry); s.Flush(); foreach (string file in files) { //打開壓縮文件 fs = System.IO.File.OpenRead(file); byte[] buffer = new byte[fs.Length]; fs.Read(buffer, 0, buffer.Length); entry = new ZipEntry("/" + Path.GetFileName(file)); entry.DateTime = DateTime.Now; entry.Size = fs.Length; fs.Close(); crc.Reset(); crc.Update(buffer); entry.Crc = crc.Value; s.PutNextEntry(entry); s.Write(buffer, 0, buffer.Length); } } finally { if (fs != null) { fs.Close(); fs = null; } if (entry != null) entry = null; GC.Collect(); } } //生成每頁多張二維碼的圖片 private void CombineImage(List<EquipmentLedger_SheBei> list,int pageNum,string folderName) { int rows = 5; int cols = 9; int height = 330; int width = 330; int currentIndex = 1; using (Bitmap bt1 = new Bitmap(2970, 2100)) { List<Rectangle> recs = new List<Rectangle>(); Graphics g = Graphics.FromImage(bt1); g.Clear(Color.White); for (int col = 0; col < cols; col++) { for (int row = 0; row < rows; row++) { if (list.Count >= currentIndex) { using (Bitmap bm = new Bitmap(folderName + "/" + list[currentIndex - 1].Id + ".bmp")) { Rectangle rg = new Rectangle(col * height, row * width + row * 100+20, width, height); g.DrawImage(bm, rg); currentIndex++; } } } } //要繪製到的位圖 bt1.Save(folderName + "/QrCodes-Page" + pageNum + ".bmp", ImageFormat.Bmp);//fileName爲存放的圖片路徑 g.Dispose(); bt1.Dispose();//釋放bmp文件資源 } }
前端Extjs代碼:spa
makeQrcodeImage: function () { var me = this; var record = me.getList().getSelectionModel().getSelection()[0]; if (record == null) { Ext.Msg.alert("系統提示", "請選擇要生成二維碼圖片的設備"); return; } else { var maiView = Ext.ComponentQuery.query('MainView')[0]; me.PicMask = new Ext.LoadMask(maiView, { msg: "圖片生成中,請稍後..." }); me.PicMask.show(); var records = me.getList().getSelectionModel().getSelection(); var list = []; for (var i = 0; i < records.length; i++) { list.push(records[i].data); } Ext.Ajax.request({ url: "../../EquipmentLedgerSheBei/MakeQrcodeImage", async: true, scope: this, params: { list: Ext.util.base64.encode(Ext.JSON.encode(list)), }, success: function (Response) { var flag = Ext.JSON.decode(Response.responseText).success; if (flag) { //下載文件 window.open("../../EquipmentLedgerSheBei/DownLoadImage?fieldName=" + Ext.JSON.decode(Response.responseText).data, "_blank"); me.PicMask.hide(); Ext.Ajax.request({ url: "../../EquipmentLedgerSheBei/DleteTempMakeQrcodeImage", async: true, scope: this, params: { fieldName: Ext.JSON.decode(Response.responseText).data, } }); } } }) } }