1.网上找了好多没有显示出来效果不错,后来看到调用手机打印预览,看了效果还不错,就打算使用系统打印服务预览下载
2.‘webView.createPrintDocumentAdapter()’得到打印的PrintDocumentAdapter有了该类就可以使用onWrite方法写入制定的文件,但是这个方法需要传入回调这个悲剧的是这个回调方法是hiden的我们没办法调用
3,字怎么解决呢,有连个方法
3.1 使用此开源库替换自己的sdk 中的android.jar文件,就可以使用了
https://github.com/anggrayudi/android-hidden-api
3.2 使用dexmaker生成动态代理代理PrintDocumentAdapter.WriteResultCallback和PrintDocumentAdapter.LayoutResultCallback方法依赖地址
compile 'org.droidparts.dexmaker:dexmaker-mockito:1.5'
4.完整代码如下:
File file = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DCIM + "/PDFTest.pdf");
File dexCacheFile;
// 获取需要打印的webview适配器
PrintDocumentAdapter printAdapter;
PageRange[] ranges;
ParcelFileDescriptor descriptor;
/**
a* @param webView
*/
private void printPDFFile(WebView webView) {
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.KITKAT) {
/**
* android 5.0之后,出于对动态注入字节码安全性德考虑,已经不允许随意指定字节码的保存路径了,需要放在应用自己的包名文件夹下。
*/
//新的创建DexMaker缓存目录的方式,直接通过context获取路径
dexCacheFile = getDir("dex", 0);
if (!dexCacheFile.exists()) {
dexCacheFile.mkdir();
}
try {
//创建待写入的PDF文件,pdfFilePath为自行指定的PDF文件路径
if (file.exists()) {
file.delete();
}
file.createNewFile();
descriptor = ParcelFileDescriptor.open(file, ParcelFileDescriptor.MODE_READ_WRITE);
// 设置打印参数
PrintAttributes attributes = new PrintAttributes.Builder()
.setMediaSize(PrintAttributes.MediaSize.ISO_A4)
.setResolution(new PrintAttributes.Resolution("id", Context.PRINT_SERVICE, 300, 300))
.setColorMode(PrintAttributes.COLOR_MODE_COLOR)
.setMinMargins(PrintAttributes.Margins.NO_MARGINS)
.build();
//打印所有界面
ranges = new PageRange[]{PageRange.ALL_PAGES};
printAdapter = webView.createPrintDocumentAdapter();
// 开始打印
printAdapter.onStart();
printAdapter.onLayout(attributes, attributes, new CancellationSignal(), getLayoutResultCallback(new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
if (method.getName().equals("onLayoutFinished")) {
// 监听到内部调用了onLayoutFinished()方法,即打印成功
onLayoutSuccess();
} else {
// 监听到打印失败或者取消了打印
}
return null;
}
}, dexCacheFile.getAbsoluteFile()), new Bundle());
} catch (IOException e) {
e.printStackTrace();
}
}
}
/**
* @throws IOException
*/
private void onLayoutSuccess() throws IOException {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
PrintDocumentAdapter.WriteResultCallback callback = getWriteResultCallback(new InvocationHandler() {
@Override
public Object invoke(Object o, Method method, Object[] objects) throws Throwable {
if (method.getName().equals("onWriteFinished")) {
Toast.makeText(MainActivity.this,"Success",Toast.LENGTH_SHORT).show();
// PDF文件写入本地完成,导出成功
Log.e("onLayoutSuccess","onLayoutSuccess");
} else {
Toast.makeText(MainActivity.this,"导出失败",Toast.LENGTH_SHORT).show();
}
return null;
}
}, dexCacheFile.getAbsoluteFile());
//写入文件到本地
printAdapter.onWrite(ranges, descriptor, new CancellationSignal(), callback);
}else {
Toast.makeText(MainActivity.this,"不支持4.4.以下",Toast.LENGTH_SHORT).show();
}
}
@SuppressLint("NewApi")
public static PrintDocumentAdapter.LayoutResultCallback getLayoutResultCallback(InvocationHandler invocationHandler, File dexCacheDir) throws IOException {
return ProxyBuilder.forClass(PrintDocumentAdapter.LayoutResultCallback.class)
.dexCache(dexCacheDir)
.handler(invocationHandler)
.build();
}
@SuppressLint("NewApi")
public static PrintDocumentAdapter.WriteResultCallback getWriteResultCallback(InvocationHandler invocationHandler, File dexCacheDir) throws IOException {
return ProxyBuilder.forClass(PrintDocumentAdapter.WriteResultCallback.class)
.dexCache(dexCacheDir)
.handler(invocationHandler)
.build();
}