基於C#在WPF中使用斑馬打印機進行打印


    最近在項目中接手了一個比較有挑戰性的模塊——用斑馬打印機將需要打印的內容打印出來。苦苦折騰了兩天,總算有所收獲,就發到網上來騙騙分數-_-||

    項目中使用的打印機型號為GX430t的打印機,接手的時候,自己對於打印機這塊兒是眼前一抹黑,啥都不知道。沒辦法一步步來。

    首先嘗試使用WPF中的PrintDialog里面的PrintVisual和PrintDocument方法,打印機是一點反應都沒有,最后得到的結論是:斑馬打印機不支持MS的XPS文檔格式,所以使用WPF來排版后進行驅動打印就不要想了,不可能!!!這條路到這里就斷了。

    然后就想到有沒有SDK可以直接進行打印,就找到了斑馬打印機的技術支持,還是個妹子,我提了一下,妹子說沒有開發包可以用,然后就貼了一個網址給我,網址就是這條:http://blog.csdn.net/ldljlq/article/details/7338772。可能我的水平有限,反正我感覺啰啰嗦嗦一大堆,實際的東西沒多少,不過關鍵點倒是提了出來使用圖像或者指令進行打印。我首先想到的是使用指令進行打印,就去找妹子要了Zebra的技術手冊,一打開就嚇尿了,尼瑪1000+頁的東西,我只是用一下打印機又不是去幫你們打印機開發驅動,當時心里那個抵觸啊。但是沒辦法,得看吶,就仔細看了一些,找到打印的指令試了試,有東西能打出來,當時感覺挺滿足的。唯一比較糾結的就是打印機支持的字體和字體本身的一些設置,比如粗體、斜體等等,資料里面沒找到,然后去問了下斑馬的技術支持,得知斜體字可以打,但是不清楚有沒有對應的指令。沒辦法,項目里面用的字體多,還有各種斜體神馬的,玩WPF的都知道,那么多屬性一個個設置下來,光字體類型斑馬就搞不定。

    既然直接使用指令打印行不通就考慮使用圖像打印,圖像又跟多媒體掛鈎了,尼瑪真是夠了。因為玩過鏈接里面的仁兄提到的獲取打印模板的命令的方法。就是在安裝好打印機驅動后,手動創建一個新的本地端口並在打印機設置中將打印機端口設為新建的端口。使用Zebra的創建模板的軟件創建好你想要的東西,然后打印,就能在你創建的端口文件中得到你想要的指令序列(其實,模板里面使用的就是圖像打印)。指令序列有了,對照手冊查詢相應的指令就能得到你想要的東西。

    這里說的圖像打印並不是我們平時說的位圖或者矢量圖,手冊里面說是叫GRF格式的圖像,仔細研究了一下,其實就是綴着這么個名字而已,里面需要的數據其實就是圖像矩陣。而且圖像矩陣中的像素表示法是:一個字節表示8個像素,也就是一個bit位(0或1)表示一個像素的顏色(黑或白)。看到這里腦子里有了思路:將要打印的內容進行排版->將排版好的數據轉換成位圖->將位圖中的數據,根據需要轉換成指令中要求的格式->交給打印機打印。這樣一來就沒有什么打印機對字體本身的限制了。思路有了,剩下的就是方法。

    排版比較簡單,這個玩過自定義控件的人都知道,使用DrawingVisual可以構建自己想要的Visual。然而將Visual轉換成位圖就難住我了,糾結了一個下午終於從網上找到了一個東西----RenderTargetBitmap。這個類可以將你的Visual轉換成位圖。

    下面就是將位圖數據轉換成指令中的圖像數據,咳咳,數學不夠好,在分析數據的時候搞錯了一個地方讓我糾結了好長時間,不過總體來說還是解決了。說一下思路:

  1. 通過RenderTargetBitmap類的CopyPixels方法將像素數據拷貝出來。因為這個位圖創建的時候只是作為一個中間的過程,所以格式可以隨便選,我是選擇了PixelFormats.Pbgra32格式,比較簡單。這個格式的圖像像素是用4個字節表示,依次為:Blue、Green、Red、Alpha。拷貝的時候,作為緩沖區的數組需要將長度設為像素數的4倍。
  2. 像素拷貝的時候會有一個“跨距”的東西。這個表示的是圖像中一行中數據的字節數,必須為4的倍數。也就是取大於或等於真實值的最小的能夠被4整除的數值。
  3. 獲取到數據就可以對數據進行整理了,遍歷整個數組,如果當前像素的顏色值不為白色或者透明色就將目標數組中的bit位之一(目標數組中用bit位表示對應像素的值)
  4. 將獲得的數組轉換成string串,然后將該串插入到指令序列中相應的位置就得到對應的指令。

    說到這里其實說的也差不多了,順便說下,WPF里面的打印支持真的很強大,給打印機傳遞指令的操作也很簡單,具體見下面的代碼。

這里是源代碼:

        /// <summary>
        /// 獲取繪制Visual的命令
        /// </summary>
        /// <param name="visual">要獲取的Visual</param>
        /// <param name="pixelWidth">像素寬度</param>
        /// <param name="pixelHeight">像素高度</param>
        /// <param name="dpiX">橫向dpi</param>
        /// <param name="dpiY">縱向dpi</param>
        /// <param name="offsetX">橫坐標偏移量,單位為像素數</param>
        /// <param name="offsetY">縱坐標偏移量,單位為像素數</param>
        /// <returns></returns>
        private string GetPrintZPL(Visual visual, int pixelWidth, int pixelHeight, double dpiX, double dpiY, int offsetX, int offsetY)
        {
            string ret = string.Empty;
            //構建圖片
            RenderTargetBitmap bmp = new RenderTargetBitmap(pixelWidth, pixelHeight, dpiX, dpiY, PixelFormats.Pbgra32);
#if TEST //test
            DrawingVisual newVisual = new DrawingVisual();
            DrawingContext dc = newVisual.RenderOpen();
            dc.DrawEllipse(Brushes.Black, new Pen(), new Point(bmp.Width / 2, bmp.Height / 2), bmp.Width / 2, bmp.Height / 2);
            dc.Close();
            visual = null;
            bmp.Render(newVisual);
#else
            bmp.Render(visual);
#endif
            byte[] datas = new byte[bmp.PixelWidth * bmp.PixelHeight * 4];
            bmp.CopyPixels(datas, bmp.PixelWidth * 4, 0);//獲取圖像數據
            int rowBytes = (pixelWidth + 7) / 8;
            byte[] targetDatas = new byte[rowBytes * bmp.PixelHeight];
            for (int i = 0; i < bmp.PixelHeight; i++) //數據調整,並將數據
            {
                for (int j = 0; j < bmp.PixelWidth; j++)
                {
                    byte blue = datas[i * bmp.PixelWidth * 4 + j * 4 + 0];
                    byte green = datas[i * bmp.PixelWidth * 4 + j * 4 + 1];
                    byte red = datas[i * bmp.PixelWidth * 4 + j * 4 + 2];
                    byte alpha = datas[i * bmp.PixelWidth * 4 + j * 4 + 3];
                    if (blue == 0 && green == 0 && red == 0)
                    {
                        if (alpha == 255)//alpha也是0則為透明色
                        {
                            byte cur = 1;
                            cur = (byte)(cur << (7 - j % 8));
                            targetDatas[i * rowBytes + j / 8] |= cur;
                        }
                    }
                    else
                    {
                        if (!(blue == 255 && green == 255 && red == 255 && alpha == 255))//全為255則表示白色
                        {
                            byte cur = 1;
                            cur = (byte)(cur << (7 - j % 8));
                            targetDatas[i * rowBytes + j / 8] |= cur;
                        }
                    }
                }
            }
            ret = string.Format("^XA~TA000~JSN^LT0^MNW^MTT^PON^PMN^LH0,0^JMA^PR3,3~SD29^JUS^LRN^CI0^XZ~DG000.GRF,{0},{1},{2}^XA^MMT^PW260^LL0189^LS0^FT0,192^FO{3},{4},^XG000.GRF,1,1^FS^PQ1,0,1,Y^XZ^XA^ID000.GRF^FS^XZ", targetDatas.Length, rowBytes, BitConverter.ToString(targetDatas).Replace("-", string.Empty), offsetX, offsetY);
            return ret;
        }

轉載請注明出處


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM