鍍金池/ 教程/ Android/ 打印自定義文檔
檢測常用的手勢
優(yōu)化layout的層級
用戶輸入
管理應(yīng)用的內(nèi)存
聯(lián)系人信息
開發(fā)輔助程序
Android多媒體
添加語音功能
顯示位置地址
提供向下與橫向?qū)Ш?/span>
支持游戲控制器
訪問可穿戴數(shù)據(jù)層
處理多點觸控手勢
全屏沉浸式應(yīng)用
為多線程創(chuàng)建管理器
數(shù)據(jù)保存
Intent的發(fā)送
更新Notification
優(yōu)化下載以高效地訪問網(wǎng)絡(luò)
打印
打包可穿戴應(yīng)用
接收從其他App傳送來的數(shù)據(jù)
發(fā)送與接收消息
建立靈活動態(tài)的UI
處理鍵盤輸入
Building a Work Policy Controller
建立測試環(huán)境
創(chuàng)建表盤
分享文件
顯示Notification進度
實現(xiàn)自適應(yīng)UI流(Flows)
使用設(shè)備管理策略增強安全性
使用能感知版本的組件
執(zhí)行網(wǎng)絡(luò)操作
建立文件分享
添加移動
更新你的Security Provider來對抗SSL漏洞利用
支持鍵盤導(dǎo)航
創(chuàng)建和監(jiān)視地理圍欄
發(fā)送并同步數(shù)據(jù)
使用BigView樣式
無線連接設(shè)備
提供向上導(dǎo)航與歷史導(dǎo)航
最小化定期更新造成的影響
實現(xiàn)向下的導(dǎo)航
支持不同的屏幕大小
Android 可穿戴應(yīng)用
添加動畫
顯示聯(lián)系人頭像
使用OpenGL ES顯示圖像
處理輸入法可見性
分享文件
保持設(shè)備喚醒
淡化系統(tǒng)Bar
使用NFC分享文件
保存到Preference
Android聯(lián)系人信息與位置信息
創(chuàng)建標(biāo)準(zhǔn)的網(wǎng)絡(luò)請求
使用Drawables
管理Bitmap的內(nèi)存使用
管理Activity的生命周期
按需加載視圖
傳輸資源
為可穿戴設(shè)備創(chuàng)建自定義UI
在一個線程中執(zhí)行一段特定的代碼
性能優(yōu)化
隱藏導(dǎo)航欄
創(chuàng)建目錄瀏覽器
為多種大小的屏幕進行規(guī)劃
View間漸變
使用觸摸手勢
高效加載大圖
使用CursorLoader在后臺加載數(shù)據(jù)
創(chuàng)建抽屜式導(dǎo)航(navigation drawer)
管理音頻焦點
創(chuàng)建后臺服務(wù)
創(chuàng)建功能測試
創(chuàng)建使用Material Design的應(yīng)用
停止與重啟Activity
添加一個簡便的分享功能
啟動Activity時保留導(dǎo)航
TV應(yīng)用清單
創(chuàng)建向后兼容的UI
?# 優(yōu)化自定義View
創(chuàng)建單元測試
在UI上顯示Bitmap
建立OpenGL ES的環(huán)境
構(gòu)建表盤服務(wù)
JNI Tips
建立搜索界面
實現(xiàn)自定義View的繪制
使用HTTPS與SSL
按需操控BroadcastReceiver
分享簡單的數(shù)據(jù)
繪制形狀
Android位置信息
創(chuàng)建并運行可穿戴應(yīng)用
執(zhí)行 Sync Adpater
獲取最后可知位置
創(chuàng)建 Android 項目
實現(xiàn)高效的導(dǎo)航
退出全屏的Activity
創(chuàng)建Card
兼容音頻輸出設(shè)備
同步數(shù)據(jù)單元
傳輸數(shù)據(jù)時避免消耗大量電量
保存到文件
緩存Bitmap
提供配置 Activity
調(diào)度重復(fù)的鬧鐘
實現(xiàn)輔助功能
重復(fù)的下載是冗余的
隱藏狀態(tài)欄
實現(xiàn)自定義的網(wǎng)絡(luò)請求
規(guī)劃界面和他們之間的關(guān)系
使用Sync Adapter傳輸數(shù)據(jù)
TV應(yīng)用內(nèi)搜索
響應(yīng)觸摸事件
使用Google Cloud Messaging(已廢棄)
控制相機
Android網(wǎng)絡(luò)連接與云服務(wù)
請求分享一個文件
處理TV硬件
響應(yīng)UI可見性的變化
使用網(wǎng)絡(luò)服務(wù)發(fā)現(xiàn)
指定輸入法類型
優(yōu)化電池壽命
創(chuàng)建TV應(yīng)用
獲取聯(lián)系人列表
拖拽與縮放
啟動與停止線程池中的線程
創(chuàng)建 Sync Adpater
使用 WiFi P2P 服務(wù)發(fā)現(xiàn)
開始使用Material Design
代理至新的APIs
使用include標(biāo)簽重用layouts
使得View可交互
高效顯示Bitmap
創(chuàng)建企業(yè)級應(yīng)用
Fragments之間的交互
創(chuàng)建與執(zhí)行測試用例
綜合:設(shè)計我們的樣例 App
繪制表盤
建立簡單的用戶界面
自定義動畫
開發(fā)輔助服務(wù)
避免出現(xiàn)程序無響應(yīng)ANR(Keeping Your App Responsive)
使用ViewPager實現(xiàn)屏幕滑動
設(shè)計高效的導(dǎo)航
Android分享操作(Building Apps with Content Sharing)
提供向后的導(dǎo)航
保持向下兼容
創(chuàng)建TV播放應(yīng)用
縮放View
使用 WiFi 建立 P2P 連接
Android后臺任務(wù)
連接到網(wǎng)絡(luò)
為 Notification 添加頁面
使TV應(yīng)用是可被搜索的
添加Action Bar
使用Material的主題
啟動另一個Activity
顯示正在播放卡片
適配不同的系統(tǒng)版本
輕松錄制視頻
創(chuàng)建可穿戴的應(yīng)用
創(chuàng)建自定義的布局
重新創(chuàng)建Activity
使用CursorLoader執(zhí)行查詢?nèi)蝿?wù)
使用舊的APIs實現(xiàn)新API的效果
使用備份API
安全要點
Android入門基礎(chǔ):從這里開始
保存并搜索數(shù)據(jù)
根據(jù)網(wǎng)絡(luò)連接類型來調(diào)整下載模式
使用Tabs創(chuàng)建Swipe視圖
SMP(Symmetric Multi-Processor) Primer for Android
解析 XML 數(shù)據(jù)
使用 Volley 傳輸網(wǎng)絡(luò)數(shù)據(jù)
建立ActionBar
Android交互設(shè)計
使用Intent修改聯(lián)系人信息
增加搜索功能
輕松拍攝照片
定義形狀
測試你的Activity
在 Notifcation 中接收語音輸入
與其他應(yīng)用的交互
管理系統(tǒng)UI
追蹤手勢移動
Android界面設(shè)計
執(zhí)行 Android 程序
顯示確認界面
創(chuàng)建Lists與Cards
打印HTML文檔
創(chuàng)建TV應(yīng)用
為多屏幕設(shè)計
定義Shadows與Clipping視圖
使用Fragment建立動態(tài)UI
接收Activity返回的結(jié)果
布局變更動畫
定位常見的問題
自定義ActionBar的風(fēng)格
定義Layouts
發(fā)送簡單的網(wǎng)絡(luò)請求
啟動與銷毀Activity
與UI線程通信
非UI線程處理Bitmap
創(chuàng)建TV布局
提升Layout的性能
報告任務(wù)執(zhí)行狀態(tài)
判斷并監(jiān)測網(wǎng)絡(luò)連接狀態(tài)
兼容不同的設(shè)備
處理按鍵動作
優(yōu)化性能和電池使用時間
給其他App發(fā)送簡單的數(shù)據(jù)
Implementing App Restrictions
向后臺服務(wù)發(fā)送任務(wù)請求
展示Card翻轉(zhuǎn)動畫
管理ViewGroup中的觸摸事件
兼容不同的屏幕密度
通過藍牙進行調(diào)試
為可穿戴設(shè)備創(chuàng)建Notification
控制音量與音頻播放
獲取聯(lián)系人詳情
在表盤上顯示信息
提供向上的導(dǎo)航
滾動手勢動畫
幫助用戶在TV上找到內(nèi)容
創(chuàng)建TV導(dǎo)航
為索引指定App內(nèi)容
ActionBar的覆蓋疊加
Android Wear 上的位置檢測
保護安全與隱私的最佳策略
Ensuring Compatibility with Managed Profiles
解決云同步的保存沖突
獲取位置更新
創(chuàng)建List
測試程序
管理網(wǎng)絡(luò)的使用情況
為App內(nèi)容開啟深度鏈接
推薦TV內(nèi)容
建立一個Notification
管理音頻播放
設(shè)計表盤
拍照
處理控制器輸入動作
判斷并監(jiān)測設(shè)備的底座狀態(tài)與類型
處理查詢的結(jié)果
保存到數(shù)據(jù)庫
支持多個游戲控制器
創(chuàng)建 Stub Content Provider
使得ListView滑動順暢
處理數(shù)據(jù)層的事件
創(chuàng)建TV應(yīng)用的第一步
使得你的App內(nèi)容可被Google搜索
將 Notification 放成一疊
創(chuàng)建 Stub 授權(quán)器
暫停與恢復(fù)Activity
管理設(shè)備的喚醒狀態(tài)
Android圖像與動畫
打印照片
云同步
創(chuàng)建TV直播應(yīng)用
為Notification賦加可穿戴特性
提供一個Card視圖
建立請求隊列(RequestQueue)
適配不同的語言
創(chuàng)建詳情頁
測試UI組件
接收其他設(shè)備的文件
創(chuàng)建自定義View
建立第一個App
創(chuàng)建2D Picker
監(jiān)測電池的電量與充電狀態(tài)
打印自定義文檔
抽象出新的APIs
通知提示用戶
獲取文件信息
運用投影與相機視角
在IntentService中執(zhí)行后臺任務(wù)
多線程操作
創(chuàng)建一個Fragment
添加Action按鈕
在不同的 Android 系統(tǒng)版本支持控制器
維護兼容性
發(fā)送文件給其他設(shè)備
創(chuàng)建TV游戲應(yīng)用
創(chuàng)建自定義的View類
代碼性能優(yōu)化建議
Intent過濾
適配不同的屏幕

打印自定義文檔

編寫:jdneo - 原文:http://developer.android.com/training/printing/custom-docs.html

對于有些應(yīng)用,比如繪圖應(yīng)用,頁面布局應(yīng)用和其它一些關(guān)注于圖像輸出的應(yīng)用,創(chuàng)造出精美的打印頁面將是它的核心功能。在這種情況下,僅僅打印一幅圖片或一個HTML文檔就不夠了。這類應(yīng)用的打印輸出需要精確地控制每一個會在頁面中顯示的對象,包括字體,文本流,分頁符,頁眉,頁腳和一些圖像元素等等。

想要創(chuàng)建一個完全自定義的打印文檔,需要投入比之前討論的方法更多的編程精力。我們必須構(gòu)建可以和打印框架相互通信的組件,調(diào)整打印參數(shù),繪制頁面元素并管理多個頁面的打印。

這節(jié)課將展示如何連接打印管理器,創(chuàng)建一個打印適配器以及如何構(gòu)建出需要打印的內(nèi)容。

連接打印管理器

當(dāng)我們的應(yīng)用直接管理打印進程時,在收到來自用戶的打印請求后,第一步要做的是連接Android打印框架并獲取一個PrintManager類的實例。該類允許我們初始化一個打印任務(wù)并開始打印任務(wù)的生命周期。下面的代碼展示了如何獲得打印管理器并開始打印進程。

private void doPrint() {
    // Get a PrintManager instance
    PrintManager printManager = (PrintManager) getActivity()
            .getSystemService(Context.PRINT_SERVICE);

    // Set job name, which will be displayed in the print queue
    String jobName = getActivity().getString(R.string.app_name) + " Document";

    // Start a print job, passing in a PrintDocumentAdapter implementation
    // to handle the generation of a print document
    printManager.print(jobName, new MyPrintDocumentAdapter(getActivity()),
            null); //
}

上面的代碼展示了如何命名一個打印任務(wù)以及如何設(shè)置一個PrintDocumentAdapter類的實例,它負責(zé)處理打印生命周期的每一步。打印適配器的實現(xiàn)會在下一節(jié)中進行討論。

Note:print()方法的最后一個參數(shù)接收一個PrintAttributes對象。我們可以使用這個參數(shù)向打印框架進行一些打印設(shè)置,以及基于前一個打印周期的預(yù)設(shè),從而改善用戶體驗。我們也可以使用這個參數(shù)對打印內(nèi)容進行一些更符合實際情況的設(shè)置,比如當(dāng)打印一幅照片時,設(shè)置打印的方向與照片方向一致。

創(chuàng)建打印適配器

打印適配器負責(zé)與Android打印框架交互并處理打印過程的每一步。這個過程需要用戶在創(chuàng)建打印文檔前選擇打印機和打印選項。由于用戶可以選擇不同性能的打印機,不同的頁面尺寸或不同的頁面方向,因此這些選項可能會影響最終的打印效果。當(dāng)這些選項配置好之后,打印框架會尋求適配器進行布局并生成一個打印文檔,以此作為打印的前期準(zhǔn)備。一旦用戶點擊了打印按鈕,框架會將最終的打印文檔傳遞給Print Provider進行打印輸出。在打印過程中,用戶可以選擇取消打印,所以打印適配器必須監(jiān)聽并響應(yīng)取消打印的請求。

PrintDocumentAdapter抽象類負責(zé)處理打印的生命周期,它有四個主要的回調(diào)方法。我們必須在打印適配器中實現(xiàn)這些方法,以此來正確地和Android打印框架進行交互:

  • onStart():一旦打印進程開始,該方法就將被調(diào)用。如果我們的應(yīng)用有任何一次性的準(zhǔn)備任務(wù)要執(zhí)行,比如獲取一個要打印數(shù)據(jù)的快照,那么讓它們在此處執(zhí)行。在你的適配器中,這個回調(diào)方法不是必須實現(xiàn)的。
  • onLayout():每當(dāng)用戶改變了影響打印輸出的設(shè)置時(比如改變了頁面的尺寸,或者頁面的方向)該函數(shù)將會被調(diào)用,以此給我們的應(yīng)用一個機會去重新計算打印頁面的布局。另外,該方法必須返回打印文檔包含多少頁面。
  • onWrite():該方法調(diào)用后,會將打印頁面渲染成一個待打印的文件。該方法可以在onLayout()方法被調(diào)用后調(diào)用一次或多次。
  • onFinish():一旦打印進程結(jié)束后,該方法將會被調(diào)用。如果我們的應(yīng)用有任何一次性銷毀任務(wù)要執(zhí)行,讓這些任務(wù)在該方法內(nèi)執(zhí)行。這個回調(diào)方法不是必須實現(xiàn)的。

下面將介紹如何實現(xiàn)onLayout()以及onWrite()方法,他們是打印適配器的核心功能。

Note:這些適配器的回調(diào)方法會在應(yīng)用的主線程上被調(diào)用。如果這些方法的實現(xiàn)在執(zhí)行時可能需要花費大量的時間,那么應(yīng)該將他們放在另一個線程里執(zhí)行。例如:我們可以將布局或者寫入打印文檔的操作封裝在一個AsyncTask對象中。

計算打印文檔信息

在實現(xiàn)PrintDocumentAdapter類時,我們的應(yīng)用必須能夠指定出所創(chuàng)建文檔的類型,計算出打印任務(wù)所需要打印的總頁數(shù),并提供打印頁面的尺寸信息。在實現(xiàn)適配器的onLayout()方法時,我們執(zhí)行這些計算,并提供與理想的輸出相關(guān)的一些信息,這些信息可以在PrintDocumentInfo類中獲取,包括頁數(shù)和內(nèi)容類型。下面的例子展示了PrintDocumentAdapteronLayout()方法的基本實現(xiàn):

@Override
public void onLayout(PrintAttributes oldAttributes,
                     PrintAttributes newAttributes,
                     CancellationSignal cancellationSignal,
                     LayoutResultCallback callback,
                     Bundle metadata) {
    // Create a new PdfDocument with the requested page attributes
    mPdfDocument = new PrintedPdfDocument(getActivity(), newAttributes);

    // Respond to cancellation request
    if (cancellationSignal.isCancelled() ) {
        callback.onLayoutCancelled();
        return;
    }

    // Compute the expected number of printed pages
    int pages = computePageCount(newAttributes);

    if (pages > 0) {
        // Return print information to print framework
        PrintDocumentInfo info = new PrintDocumentInfo
                .Builder("print_output.pdf")
                .setContentType(PrintDocumentInfo.CONTENT_TYPE_DOCUMENT)
                .setPageCount(pages);
                .build();
        // Content layout reflow is complete
        callback.onLayoutFinished(info, true);
    } else {
        // Otherwise report an error to the print framework
        callback.onLayoutFailed("Page count calculation failed.");
    }
}

onLayout()方法的執(zhí)行結(jié)果有三種:完成,取消或失敗(計算布局無法順利完成時會失?。?。我們必須通過調(diào)用PrintDocumentAdapter.LayoutResultCallback對象中的適當(dāng)方法來指出這些結(jié)果中的一個。

Note:onLayoutFinished()方法的布爾類型參數(shù)明確了這個布局內(nèi)容是否和上一次打印請求相比發(fā)生了改變。恰當(dāng)?shù)卦O(shè)定了這個參數(shù)將避免打印框架不必要地調(diào)用onWrite()方法,緩存之前的打印文檔,提升執(zhí)行性能。

onLayout()的主要任務(wù)是計算打印文檔的頁數(shù),并將它作為打印參數(shù)交給打印機。如何計算頁數(shù)則高度依賴于應(yīng)用是如何對打印頁面進行布局的。下面的代碼展示了頁數(shù)是如何根據(jù)打印方向確定的:

private int computePageCount(PrintAttributes printAttributes) {
    int itemsPerPage = 4; // default item count for portrait mode

    MediaSize pageSize = printAttributes.getMediaSize();
    if (!pageSize.isPortrait()) {
        // Six items per page in landscape orientation
        itemsPerPage = 6;
    }

    // Determine number of print items
    int printItemCount = getPrintItemCount();

    return (int) Math.ceil(printItemCount / itemsPerPage);
}

將打印文檔寫入文件

當(dāng)需要將打印內(nèi)容輸出到一個文件時,Android打印框架會調(diào)用PrintDocumentAdapter類的onWrite()方法。這個方法的參數(shù)指定了哪些頁面要被寫入以及要使用的輸出文件。該方法的實現(xiàn)必須將每一個請求頁的內(nèi)容渲染成一個含有多個頁面的PDF文件。當(dāng)這個過程結(jié)束以后,你需要調(diào)用callback對象的onWriteFinished()方法。

Note: Android打印框架可能會在每次調(diào)用onLayout()后,調(diào)用onWrite()方法一次甚至更多次。請務(wù)必牢記:當(dāng)打印內(nèi)容的布局沒有變化時,可以將onLayoutFinished()方法的布爾參數(shù)設(shè)置為“false”,以此避免對打印文檔進行不必要的重寫操作。

Note:onLayoutFinished()方法的布爾類型參數(shù)明確了這個布局內(nèi)容是否和上一次打印請求相比發(fā)生了改變。恰當(dāng)?shù)卦O(shè)定了這個參數(shù)將避免打印框架不必要的調(diào)用onLayout()方法,緩存之前的打印文檔,提升執(zhí)行性能。

下面的代碼展示了使用PrintedPdfDocument類創(chuàng)建了PDF文件的基本原理:

@Override
public void onWrite(final PageRange[] pageRanges,
                    final ParcelFileDescriptor destination,
                    final CancellationSignal cancellationSignal,
                    final WriteResultCallback callback) {
    // Iterate over each page of the document,
    // check if it's in the output range.
    for (int i = 0; i < totalPages; i++) {
        // Check to see if this page is in the output range.
        if (containsPage(pageRanges, i)) {
            // If so, add it to writtenPagesArray. writtenPagesArray.size()
            // is used to compute the next output page index.
            writtenPagesArray.append(writtenPagesArray.size(), i);
            PdfDocument.Page page = mPdfDocument.startPage(i);

            // check for cancellation
            if (cancellationSignal.isCancelled()) {
                callback.onWriteCancelled();
                mPdfDocument.close();
                mPdfDocument = null;
                return;
            }

            // Draw page content for printing
            drawPage(page);

            // Rendering is complete, so page can be finalized.
            mPdfDocument.finishPage(page);
        }
    }

    // Write PDF document to file
    try {
        mPdfDocument.writeTo(new FileOutputStream(
                destination.getFileDescriptor()));
    } catch (IOException e) {
        callback.onWriteFailed(e.toString());
        return;
    } finally {
        mPdfDocument.close();
        mPdfDocument = null;
    }
    PageRange[] writtenPages = computeWrittenPages();
    // Signal the print framework the document is complete
    callback.onWriteFinished(writtenPages);

    ...
}

代碼中將PDF頁面遞交給了drawPage()方法,這個方法會在下一部分介紹。

就布局而言,onWrite()方法的執(zhí)行可以有三種結(jié)果:完成,取消或者失敗(內(nèi)容無法被寫入)。我們必須通過調(diào)用PrintDocumentAdapter.WriteResultCallback對象中的適當(dāng)方法來指明這些結(jié)果中的一個。

Note:渲染打印文檔是一個可能耗費大量資源的操作。為了避免阻塞應(yīng)用的主UI線程,我們應(yīng)該考慮將頁面的渲染和寫操作放在另一個線程中執(zhí)行,比如在AsyncTask中執(zhí)行。關(guān)于更多異步任務(wù)線程的知識,可以閱讀:Processes and Threads。

繪制PDF頁面內(nèi)容

當(dāng)我們的應(yīng)用進行打印時,應(yīng)用必須生成一個PDF文檔并將它傳遞給Android打印框架以進行打印。我們可以使用任何PDF生成庫來協(xié)助完成這個操作。本節(jié)將展示如何使用PrintedPdfDocument類將打印內(nèi)容生成為PDF頁面。

PrintedPdfDocument類使用一個Canvas對象來在PDF頁面上繪制元素,這一點和在activity布局上進行繪制很類似。我們可以在打印頁面上使用Canvas類提供的相關(guān)繪圖方法繪制頁面元素。下面的代碼展示了如何使用這些方法在PDF頁面上繪制一些簡單的元素:

private void drawPage(PdfDocument.Page page) {
    Canvas canvas = page.getCanvas();

    // units are in points (1/72 of an inch)
    int titleBaseLine = 72;
    int leftMargin = 54;

    Paint paint = new Paint();
    paint.setColor(Color.BLACK);
    paint.setTextSize(36);
    canvas.drawText("Test Title", leftMargin, titleBaseLine, paint);

    paint.setTextSize(11);
    canvas.drawText("Test paragraph", leftMargin, titleBaseLine + 25, paint);

    paint.setColor(Color.BLUE);
    canvas.drawRect(100, 100, 172, 172, paint);
}

當(dāng)使用Canvas在一個PDF頁面上繪圖時,元素通過單位“點(point)”來指定大小,一個點相當(dāng)于七十二分之一英寸。在編寫程序時,請確保使用該測量單位來指定頁面上的元素大小。在定位繪制的元素時,坐標(biāo)系的原點(即(0,0)點)在頁面的最左上角。

Tip:雖然Canvas對象允許我們將打印元素放置在一個PDF文檔的邊緣,但許多打印機無法在紙張的邊緣打印。所以當(dāng)我們使用這個類構(gòu)建一個打印文檔時,需要考慮到那些無法打印的邊緣區(qū)域。