19

我正在开发一个需要打印到打印机的 android 应用程序。我决定使用谷歌云打印,因为它看起来很容易设置。最初,我按照此处找到的步骤集成到 Android 中。这有效,因为它将打印到我想要​​的打印机。但是,这个过程对用户来说有点复杂。就我而言,过程如下:

  1. 用户选择了我在一些信息旁边显示的打印按钮。
  2. 将显示一个对话框,其中包含将要打印的内容的预览。ActionBar 中有一个按钮,上面写着“打印”。这开始了这个过程。
  3. 将显示一个新活动,其中显示了连接到该用户 Google 帐户的打印机列表。用户必须选择一个。
  4. 将显示一个新页面,提供打印作业的描述。
  5. 用户必须选择右上角的“打印”。
  6. 打印作业开始,打印机打印出图片。

不幸的是,我的客户不想要这个过程。他们希望用户在第二步中单击“打印”,然后打印图片(步骤 1、2 和 6)。因此,我不能使用谷歌提供的 Intent,我必须使用实际的 API。这需要我获取 Google Auth 令牌,获取所需的打印机,然后以这种方式提交打印作业。我执行以下操作:

  1. 使用 Google Play 服务检索用户 Gmail 帐户的 OAuth 令牌。
  2. 使用 /search API 调用获取打印机列表。
  3. 使用 /submit API 调用提交打印作业。

我已经完成了前两个。我只是在实际打印图片时遇到问题。不是打印图片,而是打印图片的字节数据(Base64 编码)。这是一些关于我如何发送请求的代码:

ContentResolver contentResolver = context.getContentResolver();
try {
    InputStream is = contentResolver.openInputStream(uri);
    ByteArrayOutputStream baos = new ByteArrayOutputStream();

    byte[] buffer = new byte[4096];
    int n = is.read(buffer);
    while (n >= 0) {
            baos.write(buffer, 0, n);
            n = is.read(buffer);
    }
    is.close();
    baos.flush();

    content = Base64.encodeToString(baos.toByteArray(), Base64.DEFAULT);
} catch (FileNotFoundException e) {
    Log.d(TAG, "File not found: " + uri.toString(), e);
} catch (IOException e) {
    e.printStackTrace();
}

此代码检索图片(变量“uri”是该文件的 URI),并将其转换为 Base64 编码字符串。这与 Google 云打印页面(链接到上面)上提供的 PrintDialogActivity 中使用的方法相同。以下是我发送它的方式:

据我所知,这就是它应该的样子。我在打印时收到 {"success":true} 的响应。但是,正如我上面所说,它会打印出实际的 Base64 数据字符串。任何帮助,将不胜感激。

编辑:使用 powerje 下面所说的,我设法解决了这个问题。我没有使用上面的代码,而是使用了以下代码:

public void submitPrintJobWithFile(String printerId, String title, String token, String filePath, String contentType){
    File file = new File(filePath);
    // Method that gets the correct headers
    List<Header> headers = getHeaders(contentType, token);
    // Method that gets the correct post parameters
    String url = CLOUDPRINT_URL + PATH_SUBMIT;
    List<NameValuePair> postParams = getParams(title, contentType);
    String params = "access_token=" + token + "&cookies=false" + "&printerid=" + printerId;
    url += params;
    response = sendMultipartData(url, file, postParams, headers);
}

private String sendMultipartData(String url, File file, List<NameValuePair> fields, List<Header> headers){
    HttpPost post = new HttpPost(url);
    MultipartEntity entity = new MultipartEntity();
    for(NameValuePair pair : fields){
        String name = pair.getName();
        String value = pair.getValue();
        try{
            entity.addPart(name, new StringBody(value));
        }catch (UnsupportedEncodingException e){
            Log.d(TAG, "Error turning pair (name=" + name + ", value=" + value + ") into StringBody.");
    }
    entity.addPart("content", new FileBody(file));
    post.setEntity(entity);
    // Finish HttpClient request here...
}
4

3 回答 3

2

看起来您需要使用多部分编码,例如:

http://blog.tacticalnuclearstrike.com/2010/01/using-multipartentity-in-android-applications/

自由贸易协定:

需要的文件是apache-mime4jhttpclient、httpcore 和 httpmime。都是Apache基金会构建的开源项目。

下载这 4 个文件并将它们添加到您的项目中,然后您应该能够使用以下代码将字符串和文件发布到页面。

HttpClient httpclient = new DefaultHttpClient();
HttpPost httppost = new HttpPost("http://www.tumblr.com/api/write");

try {
    MultipartEntity entity = new MultipartEntity();

    entity.addPart("type", new StringBody("photo"));
    entity.addPart("data", new FileBody(image));
    httppost.setEntity(entity);
    HttpResponse response = httpclient.execute(httppost);
} catch (ClientProtocolException e) {
} catch (IOException e) {
} 

在这种情况下,图像变量是一个文件,其中包含手机上的摄像头捕获的图像。

于 2013-01-24T16:44:37.320 回答
1

查看Python Sample Code SubmitJob 方法,似乎只有 PDF 类型需要用 Base64 编码。

于 2013-01-23T20:38:24.697 回答
0

用一点更新来回答这个问题。截至 2013 年 10 月,在 4.4 和支持库中,内置了处理打印的方法。请参阅以下文档以了解如何正确执行此操作:

于 2013-11-04T20:23:05.917 回答