/* * 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.permission2; import java.io.FileOutputStream; import java.io.IOException; import java.util.ArrayList; import java.util.Arrays; import java.util.List; import android.app.Activity; import android.content.Context; import android.graphics.Color; import android.graphics.Paint; import android.graphics.pdf.PdfDocument.Page; import android.os.AsyncTask; import android.os.Bundle; import android.os.CancellationSignal; import android.os.CancellationSignal.OnCancelListener; 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.util.Log; import android.util.SparseIntArray; import android.view.Menu; import android.view.MenuItem; import android.view.View; import foo.bar.print.R; /** * Simple sample of how to use the print APIs. */ public class PrintActivity extends Activity { public static final String LOG_TAG = "PrintActivity"; private static final int PAGE_COUNT = 50; @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) { if (item.getItemId() == R.id.menu_print) { printView(); return true; } 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 onStart() { Log.i(LOG_TAG, "onStart"); } @Override public void onFinish() { Log.i(LOG_TAG, "onFinish"); } @Override public void onLayout(final PrintAttributes oldAttributes, final PrintAttributes newAttributes, final CancellationSignal cancellationSignal, final LayoutResultCallback callback, final Bundle metadata) { Log.i(LOG_TAG, "onLayout() oldAttrs:" + oldAttributes + "\n" + "newAttrs:" + newAttributes + "\n" + "preview:" + metadata.getBoolean( PrintDocumentAdapter.EXTRA_PRINT_PREVIEW) ); new AsyncTask<Void, Void, Integer>() { @Override protected void onPreExecute() { // First register for cancellation requests. cancellationSignal.setOnCancelListener(new OnCancelListener() { @Override public void onCancel() { 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 cancellationSignal, final WriteResultCallback callback) { Log.i(LOG_TAG, "onWrite() pages:" + Arrays.toString(pages)); 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( PrintActivity.this, mPrintAttributes); @Override protected void onPreExecute() { // First register for cancellation requests. cancellationSignal.setOnCancelListener(new OnCancelListener() { @Override public void onCancel() { 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()); Paint paint = new Paint(); paint.setTextSize(100); paint.setColor(Color.RED); final int x = page.getCanvas().getWidth() / 2; final int y = page.getCanvas().getHeight() / 2; page.getCanvas().drawText(String.valueOf(i), x, y, paint); 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<PageRange>(); int start = -1; int end = -1; final int writtenPageCount = writtenPages.size(); for (int i = 0; i < writtenPageCount; i++) { if (start < 0) { 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); start = end = -1; } 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); } }