跳至主要内容

[Widget]Android小票打印,蓝牙打印、固定IP打印、黑白图片打印

Printer


标准ES-POS命令打印,固定IP或蓝牙打印,支持黑白图片打印

预览

screenshot
printer_example

要求

  • minSdkVersion 5
  • <uses-permission android:name="android.permission.INTERNET" />
  • <uses-permission android:name="android.permission.BLUETOOTH" />

引用

dependencies {
    ⋯
    compile 'am.util:printer:1.1.3'
    ⋯
}

详情

  • 继承PrintTask来实现打印任务
  • 继承PrinterWriter来实现更多纸张类型的打印
  • PrinterUtils包含了众多打印指令

使用

1.添加蓝牙权限<uses-permission android:name="android.permission.BLUETOOTH" />或者网络请求权限<uses-permission android:name="android.permission.INTERNET" />
2.继承PrintTask类,实现具体打印任务:
private class TestPrintTask extends PrintTask {

    public TestPrintTask(BluetoothDevice device, int type) {
        super(device, type);
    }

    public TestPrintTask(String ip, int port, int type) {
        super(ip, port, type);
    }

    @Override
    protected void onPreExecute() {
        super.onPreExecute();
        //打印前的准备工作,比如显示对话框
    }

    @Override
    protected byte[] getPrintData(int type) throws Exception {
        //实现打印数据的排版,生成字节流
        //根据不同的type创建不同的PrinterWriter,包内提供PrinterWriter80mm和PrinterWriter58mm,对应的是80mm小票打印机及58mm小票打印机。也可以自己创建PrinterWriter
        PrinterWriter80mm printer = new PrinterWriter80mm();
        printer.setAlignCenter();
        printer.printDrawable(res, R.drawable.ic_printer_logo);
        printer.setAlignLeft();
        printer.printLine();
        printer.printLineFeed();
        printer.setLineHeight(80);
        printer.print("最时尚的明星餐厅");
        printer.printLineFeed();
        printer.print("客服电话:400-8008800");
        printer.printLineFeed();
        ...
        printer.feedPaperCutPartial();
        return printer.getData();
    }

    @Override
    protected void onPrinterStateChanged(int state) {
        super.onPrinterStateChanged(state);
        //这些状态的变更是在主线程内
        switch (state) {
            case PrintRequest.STATE_0:
                //生成打印页面数据...
                break;
            case PrintRequest.STATE_1:
                ///生成数据成功,开始创建Socket连接...
                break;
            case PrintRequest.STATE_2:
                //创建Socket成功,开始发送测试数据...
                break;
            case PrintRequest.STATE_3:
                //获取输出流成功,开始写入打印页面数据...
                break;
            case PrintRequest.STATE_4:
                //写入打印页面数据成功,正在完成打印...
                break;
        }
    }

    @Override
    protected void onResult(int errorCode) {
        super.onResult(errorCode);
        switch (errorCode) {
            case PrintRequest.ERROR_0:
                //打印成功完成!
                break;
            case PrintRequest.ERROR_1:
                //生成打印页面数据失败!
                break;
            case PrintRequest.ERROR_2:
                //创建Socket失败!
                break;
            case PrintRequest.ERROR_3:
                //获取输出流失败!
                break;
            case PrintRequest.ERROR_4:
                //写入打印页面数据失败!
                break;
            case PrintRequest.ERROR_5:
                //必要的参数不能为空!
                break;
        }
    }
}
3.执行打印:
new TestPrintTask(device, type).execute();
new TestPrintTask(ip, port, type).execute();
如果你要实现自己的打印机PrinterWriter,那么你需要继承
public class PrinterWriter80mm extends PrinterWriter{

    public static final int TYPE_80 = 80;// 纸宽80mm

    public PrinterWriter80mm() throws IOException {
    }

    @Override
    protected int getLineWidth() {
        //一行能够放下多少个“-”
        return 24;
    }

    @Override
    protected int getLineStringWidth(int textSize) {
        //根据字体的大小,一行可以放下多少个英文字符
        switch (textSize) {
            default:
            case 0:
                return 47;
            case 1:
                return 23;
        }
    }

    @Override
    protected int getDrawableMaxWidth() {
        //图片能够全部打印在纸上的最大宽度
        return 500;
    }
}

注意

  • 仅提供建立蓝牙连接打印,不包括蓝牙搜索及配对功能
  • 不包含二维码生成功能

评论

此博客中的热门博文

[Widget]StateFrameLayout-状态帧布局

StateFrameLayout icon 一般网络交互的状态提示及处理大多数情况下考虑使用Dialog,在一切状态处理理想状态下时,使用Dialog进行交互是可行的。但稍微一不注意,使用Dialog则会出现一系列隐藏的Bug。为节省用户时间怎加体验感觉,数据的载入可以在onCreate时候就进行,甚至可以在Activity构造函数里面启动网络请求,因为Activity还没有建立窗口(onAttachedToWindow),而Dialog必须附着在Activity的Window上,显然这时候不能弹出Dialog;网络交互并非即时,也就是在交互过程中用户可能进行任何操作,多数情况下,应用并不允许用户中断网络交互,而将Dialog设置为不可取消的话,用户体验是很差的,因为你同时阻止了用户退出当前Activity的操作,若用户仅仅是误点了进来,那么必须等待交互结束才能退出,而如果不讲Dialog设置为不可取消的话,那么用户进行了取消操作,但实际是并没有取消,这又会让用户很困惑,如果交互是更新当前页面的数据,当用户取消以后就可以进行旧数据操作,但其实这时候数据已过时,操作是不应该的;当网络交互已完成时,若交互结果需要告知用户时,此时又得注意Activity的状态,也许Activity已经关闭了Window(用户进行了返回操作,Activity在销毁;或者用户点按了Home键,设备内存不够,Activity在进行保存并关闭Window)。操控好Window,则使用Dialog并无任何问题,但是这就会怎加代码复杂度。其实我们的目的就是告知用户在进行网络请求,阻止用户对未载入或旧页面进行操作,网络交互结束后有必要时告知用户;使用StateFrameLayout则能轻松达到效果。 状态帧布局,通常用于网络请求的四种状态,普通、载入、错误、空白。支持Drawable或者View来展示,也可以混搭。 预览 screenshots 要求 minSdkVersion 4 链接 Github Bintray 使用 基本布局 < am .widget.stateframelayout.StateFrameLayout xmlns : app = " http://schemas.androi...

[Widget]GradientTabStrip-微信式底部渐变栏

GradientTabStrip icon 继承自BaseTabStrip,实现微信式渐变底部Tab效果,为ViewPager添加如PagerTitleStrip一样的Tab,但支持更多自定义功能,并支持为Tab增加标记点功能,并可以自定义标记点各自的位置及显示状态以及背景等。 预览 screenshots 要求 minSdkVersion 9 保持跟其他官方支持库版本一致(如:com.android.support:appcompat-v7) 链接 Github Bintray 使用 基本布局 < am .widget.gradienttabstrip.GradientTabStrip android : id = " @+id/gts_gts_tabs " android : layout_width = " match_parent " android : layout_height = " 64dp " android : textColor = " @color/color_gradienttabstrip_tab " android : textSize = " 12sp " app : gtsBackground = " @drawable/bg_common_press " /> 基本代码 GradientTabStrip tabStrip = ( GradientTabStrip ) findViewById(id); GradientTabStrip . GradientTabAdapter adapter = new GradientTabStrip . GradientTabAdapter () { @Override public Drawable getNormalDrawable ( int position , Context context ) { return null ; } @...

[Widget]IndicatorTabStrip-移动式下标渐变缩放Tab

IndicatorTabStrip 继承自BaseTabStrip,移动式下标渐变缩放Tab,Item不建议超过5个,为ViewPager添加如PagerTitleStrip一样的Tab,但支持更多自定义功能,并支持为Tab增加标记点功能,并可以自定义标记点各自的位置及显示状态以及背景等。 预览 要求 minSdkVersion 9 保持跟其他官方支持库版本一致(如:com.android.support:appcompat-v7) 链接 Github Bintray 使用 基本布局 < am .widget.indicatortabstrip.IndicatorTabStrip xmlns : app = " http://schemas.android.com/apk/res-auto " android : id = " @+id/its_its_tabs " android : layout_width = " match_parent " android : layout_height = " ?attr/actionBarSize " android : padding = " 5dp " android : textColor = " @color/color_main_tabs " android : textSize = " 16sp " app : ttsTextScale = " 1.2 " app : ttsDivider = " @drawable/divider_indicator_under " app : ttsInterval = " @drawable/divider_indicator_interval " app : ttsIndicator = " @drawable/ic_indicator_indicator " ...