- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我有一个 Apex3,我已经能够关注 the documentation 的大部分内容没问题,但是当涉及到图像时,事情变得非常奇怪(缺乏示例 + 在操作方法上缺乏一致性)。
首先,我尝试使用以下命令尝试通过 JPEG 和 0 质量压缩的位图字节 [] 数组压缩的天真方法:
ESC V n1 n2数据
结果不太好。
然后我发现有一个 apex3 的 android 库接受一个位图并打算打印它但是它不起作用只是打印像这样的奇怪符号:
我尝试使用 JD gui 解码 jar 源代码,他们似乎对位图字节做了一些工作,这是他们的代码(建议代码,如 addToDoc(m_Document, ESC + "B"); 只需将代码放入一个 ByteArrayOutputStream 数据),(反编译 source from here ):
public void writeImage(Bitmap imageObject, int printHeadWidth)
throws IllegalArgumentException
{
if (imageObject == null) {
throw new IllegalArgumentException("Parameter 'imageObject' was null.");
}
if (printHeadWidth < 1) {
throw new IllegalArgumentException("Parameter 'printHeadWidth' must be greater than 0.");
}
int height = imageObject.getHeight();
int width = imageObject.getWidth();
byte blanklineCount = 0;
byte[] dataline = new byte[printHeadWidth + 7 >> 3];
int[] imageData = new int[height * width];
imageObject.getPixels(imageData, 0, width, 0, 0, width, height);
addToDoc(m_Document, ESC + "B");
for (int row = 0; row < height; row++)
{
boolean blankLine = true;
for (int index = 0; index < width; index += 8)
{
byte currentByte = 0;
int offset = row * width + index;
if (index >= printHeadWidth) {
break;
}
int value = index + 0 < width ? imageData[(offset + 0)] & 0xFFFFFF : 16777215;
boolean set = (value >> 0 & 0xFF) + (value >> 8 & 0xFF) + (value >> 16 & 0xFF) < 384;
currentByte = (byte)(currentByte | (set ? -128 : 0));
value = index + 1 < width ? imageData[(offset + 1)] & 0xFFFFFF : 16777215;
set = (value >> 0 & 0xFF) + (value >> 8 & 0xFF) + (value >> 16 & 0xFF) < 384;
currentByte = (byte)(currentByte | (set ? 64 : 0));
value = index + 2 < width ? imageData[(offset + 2)] & 0xFFFFFF : 16777215;
set = (value >> 0 & 0xFF) + (value >> 8 & 0xFF) + (value >> 16 & 0xFF) < 384;
currentByte = (byte)(currentByte | (set ? 32 : 0));
value = index + 3 < width ? imageData[(offset + 3)] & 0xFFFFFF : 16777215;
set = (value >> 0 & 0xFF) + (value >> 8 & 0xFF) + (value >> 16 & 0xFF) < 384;
currentByte = (byte)(currentByte | (set ? 16 : 0));
value = index + 4 < width ? imageData[(offset + 4)] & 0xFFFFFF : 16777215;
set = (value >> 0 & 0xFF) + (value >> 8 & 0xFF) + (value >> 16 & 0xFF) < 384;
currentByte = (byte)(currentByte | (set ? 8 : 0));
value = index + 5 < width ? imageData[(offset + 5)] & 0xFFFFFF : 16777215;
set = (value >> 0 & 0xFF) + (value >> 8 & 0xFF) + (value >> 16 & 0xFF) < 384;
currentByte = (byte)(currentByte | (set ? 4 : 0));
value = index + 6 < width ? imageData[(offset + 6)] & 0xFFFFFF : 16777215;
set = (value >> 0 & 0xFF) + (value >> 8 & 0xFF) + (value >> 16 & 0xFF) < 384;
currentByte = (byte)(currentByte | (set ? 2 : 0));
value = index + 7 < width ? imageData[(offset + 7)] & 0xFFFFFF : 16777215;
set = (value >> 0 & 0xFF) + (value >> 8 & 0xFF) + (value >> 16 & 0xFF) < 384;
currentByte = (byte)(currentByte | (set ? 1 : 0));
dataline[(index >> 3)] = currentByte;
blankLine &= currentByte == 0;
}
if (!blankLine)
{
if (blanklineCount > 0)
{
addToDoc(m_Document, "A");
addToDoc(m_Document, blanklineCount);
blanklineCount = 0;
}
addToDoc(m_Document, compressGraphicLine(dataline));
}
else
{
blanklineCount = (byte)(blanklineCount + 1);
if (blanklineCount == 255)
{
addToDoc(m_Document, "A");
addToDoc(m_Document, blanklineCount);
blanklineCount = 0;
}
}
}
if (blanklineCount > 0)
{
addToDoc(m_Document, "A");
addToDoc(m_Document, blanklineCount);
blanklineCount = 0;
}
addToDoc(m_Document, ESC + "E");
}
private byte[] compressGraphicLine(byte[] dataline)
{
byte count = 0;
byte currentByte = 0;
ByteArrayOutputStream rleString = new ByteArrayOutputStream(128);
addToDoc(rleString, "G");
for (int index = 0; index < dataline.length; index++) {
if (count == 0)
{
currentByte = dataline[index];
addToDoc(rleString, currentByte);
count = (byte)(count + 1);
}
else if ((count < 255) && (currentByte == dataline[index]))
{
count = (byte)(count + 1);
}
else
{
addToDoc(rleString, count);
count = 0;
currentByte = dataline[index];
addToDoc(rleString, currentByte);
count = (byte)(count + 1);
}
}
if (count > 0) {
addToDoc(rleString, count);
}
if (rleString.size() > dataline.length + 1)
{
rleString.reset();
addToDoc(rleString, "U");
for (int item = 0; item < dataline.length; item++) {
addToDoc(rleString, dataline[item]);
}
}
return rleString.toByteArray();
}
但我不明白为什么它不起作用。
最后我尝试使用 How can I print an image on a Bluetooth printer in Android? 使用与指南相同的算法,但仍然打印随机的奇怪符号。
最佳答案
与其浪费时间去反编译一些 apk,不如看看官方的 SDK?在制造商网页上 Downloads & Drivers有一个链接到 Java SDK其中包括源 Sample.java
。在源代码中创建了一个 BufferedImage
,所以我猜(我没有这样的打印机)这将为您提供解决问题的切入点。他们很可能在同一页面上提供了 Android 演示的源代码 Printer Demo Source code for Android
编辑 好的。让我们总结一下:您有一张图像并想打印它。在示例 Sample.java 中涵盖了这种情况
BufferedImage newImage = new BufferedImage(1024, 512, BufferedImage.TYPE_4BYTE_ABGR);
// some lines and rectangles are drawn in the image
...
// the image is printed, following the SDK javadoc for DocumentLP.writeImage
// "This will cause the image specified to be printed. Images will be expanded to occupy
// the entire width of the printer, so the correct current width of the printer must be
// specified. Images that are too wide will be cropped, and images that are too narrow
// will be padded on the right."
testDoc.writeImage(newImage, m_PrinterWidth);
对我来说,你唯一需要做的事情:
编辑2伪代码 fragment
// taken from SDK javadoc
DocumentLP docLP;
docLP = new DocumentLP("$");
// own code
BufferedInputStream bis = new BufferedInputStream(--from your image--);
BufferedImage bufImage = ImageIO.read(bis);
// have a look into Sample.java for the expected value of m_PrinterWidth
testDoc.writeImage(bufImage, m_PrinterWidth);
编辑 3 Android 代码 fragment (取自 datamax o´neil Android SDK
提供的 DO_AndroidSDKDemo_MainActivity.javaFile file = new File(selectedPath);
byte[] readBuffer = new byte[(int)file.length()];
InputStream inputStream= new BufferedInputStream(new FileInputStream(file));
inputStream.read(readBuffer);
inputStream.close();
fileData = readBuffer;
Bitmap m_imageObject = BitmapFactory.decodeByteArray(fileData, 0, fileData.length);
documentLP.clear();
ocumentLP.writeImage(m_imageObject, m_printHeadWidth);
关于java - 打印 POS/ESC Apex3 图像 S.O.S,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19918941/
这个问题在这里已经有了答案: Why don't Java's +=, -=, *=, /= compound assignment operators require casting? (11 个
我搜索了很多,但没有一个链接能帮助我解决这个问题。我得到了 ORA-21500: internal error code, arguments: [%s], [%s], [%s], [%s], [%s
我正在做 RegexOne 正则表达式教程,它有一个 question关于编写正则表达式以删除不必要的空格。 教程中提供的解决方案是 We can just skip all the starting
([\s\S]+|\s?) 中 |\s? 的目的或作用是什么?如果没有它,表达式会不会与 ([\s\S]+) 相同? 最佳答案 这不是完全相同的。 ([\s\S]+|\s?) 会匹配空字符串,而 ([
这个正则表达式有一组还是两组? 我正在尝试使用第二组访问 bookTitle 但出现错误: Pattern pattern = Pattern.compile("^\\s*(.*?)\\s+-\\s+
在 C 中给定一个字符串指针 s,下面的迭代会做什么?即它以什么方式遍历字符串? for (++s ; *s; ++s); 最佳答案 for (++s ; *s;++s) 表示 将指针 s 递增到字符
我正在用一个 node.js 应用程序解析一个大列表并有这段代码 sizeCode = dbfr.CN_DESC.split('\s+-\s*|\s*-\s+') 这似乎不起作用,因为它返回了 [ '
我正在编写一个简单的字符串连接程序。 该程序按照我发布的方式运行。但是,我首先使用以下代码编写它来查找字符串的结尾: while (*s++) ; 但是,这个方法并没有奏效。我传递给它的字符串
这个问题已经有答案了: What does (?和aramchand来自Mohandas Karamchand G 因此,在使用这些匹配来分割字符串后,您最终会得到 {"M", "K", "G"} 注
我正在尝试转换 Map到 List使用 lambda。 本质上,我想将键和值与 '=' 连接起来之间。这看起来微不足道,但我找不到如何去做。 例如 Map map = new HashMap<>();
我正在经历 K & R,并且在递增指针时遇到困难。练习 5.3(第 107 页)要求您使用指针编写一个 strcat 函数。 在伪代码中,该函数执行以下操作: 将 2 个字符串作为输入。 找到字符串
在下面的代码中,pS 和 s.pS 在最后一行是否保证相等?也就是说,在语句S s = S();中,是否可以确定不会构造一个临时的S? #include using namespace std; s
演示示例代码: public void ReverseString(char[] s) { for(int i = 0, j = s.Length-1; i < j; i++, j--){
我一直在寻找类似于 .NET examples 中的示例的 PowerShell 脚本.取一个 New-TimeSpan 并显示为 1 天 2 小时 3 分钟 4 秒。排除其零的地方,在需要的地方添加
def func(s): s = s + " is corrected" return s string_list = ["She", "He"] for s in string_li
我是 python 的新手。当我在互联网上搜索 lambda 时。我在 lambda_functions 中找到了这个声明. processFunc = collapse and (lambda s:
我最近开始学习正则表达式,并试图为上面的问题写一个正则表达式。如果限制只放在一个字母上(例如不超过 2 个“b”),这并不困难。 那么答案就是:a* c*(b|ε)a* c*(b|ε)a* c* 但是
当我运行 npm install 时出现以下错误,但我无法修复它。 我试过:npm install -g windows-build-tools 也没有修复这个错误 ERR! configure
有很多有趣的haskell网上可以找到片段。 This post可以在 this (awesome) Stack Overflow question 下找到. The author写道: discou
我知道以下三行代码旨在将字符串提取到$ value中并将其存储在$ header中。但是我不知道$value =~ s/^\s+//;和$value =~ s/\s+$//;之间有什么区别。 $val
我是一名优秀的程序员,十分优秀!