/* * Copyright (C) 2013 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package foo.bar.print; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; import java.util.ArrayList; import java.util.List; import android.app.Activity; import android.content.Context; import android.graphics.pdf.PdfDocument.Page; import android.net.Uri; import android.os.AsyncTask; import android.os.Bundle; import android.os.CancellationSignal; import android.os.ParcelFileDescriptor; import android.print.PageRange; import android.print.PrintAttributes; import android.print.PrintDocumentAdapter; import android.print.PrintDocumentInfo; import android.print.PrintManager; import android.print.pdf.PrintedPdfDocument; import android.support.v4.print.PrintHelper; import android.util.SparseIntArray; import android.view.Menu; import android.view.MenuItem; import android.view.View; /** * Simple sample of how to use the print APIs. */ public class Main extends Activity { public static final String LOG_TAG = "PrintActivity"; private static final int PAGE_COUNT = 5; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); } @Override public boolean onCreateOptionsMenu(Menu menu) { super.onCreateOptionsMenu(menu); getMenuInflater().inflate(R.menu.activity_main, menu); return true; } @Override public boolean onOptionsItemSelected(MenuItem item) { switch (item.getItemId()) { case R.id.menu_print: printView(); return true; case R.id.menu_print_uri_portrait_force: case R.id.menu_print_uri_portrait: case R.id.menu_print_uri_landscape: try { PrintHelper ph = new PrintHelper(this); Uri uri = null; switch (item.getItemId()) { case R.id.menu_print_uri_portrait_force: ph.setOrientation(PrintHelper.ORIENTATION_PORTRAIT); /* fall through */ case R.id.menu_print_uri_portrait: uri = Uri.parse("android.resource://foo.bar.print/raw/portrait"); break; case R.id.menu_print_uri_landscape: uri = Uri.parse("android.resource://foo.bar.print/raw/landscape"); break; } ph.printBitmap("Print Uri", uri); } catch (FileNotFoundException e) { e.printStackTrace(); } return true; default: return super.onOptionsItemSelected(item); } } private void printView() { PrintManager printManager = (PrintManager) getSystemService(Context.PRINT_SERVICE); final View view = findViewById(R.id.content); printManager.print("Print_View", new PrintDocumentAdapter() { private static final int RESULT_LAYOUT_FAILED = 1; private static final int RESULT_LAYOUT_FINISHED = 2; private PrintAttributes mPrintAttributes; @Override public void onLayout(final PrintAttributes oldAttributes, final PrintAttributes newAttributes, final CancellationSignal cancellationSignal, final LayoutResultCallback callback, final Bundle metadata) { new AsyncTask<Void, Void, Integer>() { @Override protected void onPreExecute() { // First register for cancellation requests. cancellationSignal.setOnCancelListener(() -> cancel(true)); mPrintAttributes = newAttributes; } @Override protected Integer doInBackground(Void... params) { try { // Pretend we do some layout work. for (int i = 0; i < PAGE_COUNT; i++) { // Be nice and respond to cancellation. if (isCancelled()) { return null; } pretendDoingLayoutWork(); } return RESULT_LAYOUT_FINISHED; } catch (Exception e) { return RESULT_LAYOUT_FAILED; } } @Override protected void onPostExecute(Integer result) { // The task was not cancelled, so handle the layout result. switch (result) { case RESULT_LAYOUT_FINISHED: { PrintDocumentInfo info = new PrintDocumentInfo .Builder("print_view.pdf") .setContentType(PrintDocumentInfo .CONTENT_TYPE_DOCUMENT) .setPageCount(PAGE_COUNT) .build(); callback.onLayoutFinished(info, false); } break; case RESULT_LAYOUT_FAILED: { callback.onLayoutFailed(null); } break; } } @Override protected void onCancelled(Integer result) { // Task was cancelled, report that. callback.onLayoutCancelled(); } private void pretendDoingLayoutWork() throws Exception { } }.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR, (Void[]) null); } @Override public void onWrite(final PageRange[] pages, final ParcelFileDescriptor destination, final CancellationSignal canclleationSignal, final WriteResultCallback callback) { new AsyncTask<Void, Void, Integer>() { private static final int RESULT_WRITE_FAILED = 1; private static final int RESULT_WRITE_FINISHED = 2; private final SparseIntArray mWrittenPages = new SparseIntArray(); private final PrintedPdfDocument mPdfDocument = new PrintedPdfDocument( Main.this, mPrintAttributes); @Override protected void onPreExecute() { // First register for cancellation requests. canclleationSignal.setOnCancelListener(() -> cancel(true)); for (int i = 0; i < PAGE_COUNT; i++) { // Be nice and respond to cancellation. if (isCancelled()) { return; } // Write the page only if it was requested. if (containsPage(pages, i)) { mWrittenPages.append(mWrittenPages.size(), i); Page page = mPdfDocument.startPage(i); // The page of the PDF backed canvas size is in pixels (1/72") and // smaller that the view. We scale down the drawn content and to // fit. This does not lead to losing data as PDF is a vector format. final float scale = (float) Math.min(mPdfDocument.getPageWidth(), mPdfDocument.getPageHeight()) / Math.max(view.getWidth(), view.getHeight()); page.getCanvas().scale(scale, scale); view.draw(page.getCanvas()); mPdfDocument.finishPage(page); } } } @Override protected Integer doInBackground(Void... params) { // Write the data and return success or failure. try { mPdfDocument.writeTo(new FileOutputStream( destination.getFileDescriptor())); return RESULT_WRITE_FINISHED; } catch (IOException ioe) { return RESULT_WRITE_FAILED; } } @Override protected void onPostExecute(Integer result) { // The task was not cancelled, so handle the write result. switch (result) { case RESULT_WRITE_FINISHED: { PageRange[] pageRanges = computePageRanges(mWrittenPages); callback.onWriteFinished(pageRanges); } break; case RESULT_WRITE_FAILED: { callback.onWriteFailed(null); } break; } mPdfDocument.close(); } @Override protected void onCancelled(Integer result) { // Task was cancelled, report that. callback.onWriteCancelled(); mPdfDocument.close(); } }.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR, (Void[]) null); } private PageRange[] computePageRanges(SparseIntArray writtenPages) { List<PageRange> pageRanges = new ArrayList<>(); int start; int end; final int writtenPageCount = writtenPages.size(); for (int i = 0; i < writtenPageCount; i++) { start = writtenPages.valueAt(i); int oldEnd = end = start; while (i < writtenPageCount && (end - oldEnd) <= 1) { oldEnd = end; end = writtenPages.valueAt(i); i++; } PageRange pageRange = new PageRange(start, end); pageRanges.add(pageRange); } PageRange[] pageRangesArray = new PageRange[pageRanges.size()]; pageRanges.toArray(pageRangesArray); return pageRangesArray; } private boolean containsPage(PageRange[] pageRanges, int page) { final int pageRangeCount = pageRanges.length; for (int i = 0; i < pageRangeCount; i++) { if (pageRanges[i].getStart() <= page && pageRanges[i].getEnd() >= page) { return true; } } return false; } }, null); } }