2014-09-18 3 views
4

사용자가 내 Android 앱에 표시되는 양식을 기반으로 PDF 문서를 만들려고합니다. 나는 다음 이메일에 첨부 된 PDF 파일을 만들려면 다음 코드를 사용PrintedPdfDocument는 항상 손상된 PDF 파일을 만듭니다.

public static File generate(View salesFragmentTableLayout, Context context) throws KingdomSpasException { 
     Builder printAttrsBuilder = new Builder(); 
     printAttrsBuilder.setMediaSize(PrintAttributes.MediaSize.ISO_A4); 
     printAttrsBuilder.setMinMargins(new Margins(5, 5, 5, 5)); 

     PrintedPdfDocument document = new PrintedPdfDocument(context, printAttrsBuilder.build()); 

     PageInfo pageInfo = new PageInfo.Builder(150, 150, 1).create(); 
     Page page = document.startPage(pageInfo); 
     salesFragmentTableLayout.draw(page.getCanvas()); 
     document.finishPage(page); 
     File result = null; 
     try { 
      result = File.createTempFile("Kingdom Spas Agreement", ".pdf", context.getCacheDir()); 
      document.writeTo(new BufferedOutputStream(new FileOutputStream(result))); 
     } catch (FileNotFoundException e) { 
      throw new KingdomSpasException("Failed to find relevent file", e); 
     } catch (IOException e) { 
      throw new KingdomSpasException("IO Problem occured while creatin the PDF", e); 
     } 
     document.close(); 
     return result; 
    } 

결과 PDF는 항상 손상되어 하나 어도비 경우 Acroread 또는 GS 열 수 없습니다. 나는 시도하고 다음 명령을 사용하여 GS에서 열 때

There was an error opening this document. The file is damaged and could not be repaired.

:

**** Error: Cannot find a 'startxref' anywhere in the file.
**** Warning: An error occurred while reading an XREF table.
**** The file has been damaged. This may have been caused
**** by a problem while converting or transfering the file.
**** Ghostscript will attempt to recover the data.
**** Error: Trailer is not found. No pages will be processed (FirstPage > LastPage).

**** This file had errors that were repaired or ignored.
**** Please notify the author of the software that produced this
**** file that it does not conform to Adobe's published PDF
**** specification.

: 나는 다음과 같은 출력을 얻을

gs \ 
    -o repaired.pdf \ 
    -sDEVICE=pdfwrite \ 
    -dPDFSETTINGS=/prepress \ 
    KingdomSpasAgreement.pdf 

을 내가 acroread로 그것을 열 때 나는 오류

나는 내가 잘못하고있는 것이 무엇인지 이해하지 못한다 - 모든 과정은 매우 간단하지만 항상 실패한다.

편집 :

<TableLayout 
      android:id="@+id/salesAgreementTableLayout" 
      android:layout_width="wrap_content" 
      android:layout_height="wrap_content" 
      android:orientation="vertical" > 

      <TableRow 
       android:layout_width="fill_parent" 
       android:layout_height="wrap_content" > 

       <ImageView 
        android:id="@+id/companyLogo" 
        android:layout_width="wrap_content" 
        android:layout_height="wrap_content" 
        android:layout_weight="1" 
        android:contentDescription="@string/logo_description" 
        android:src="@drawable/company_logo" /> 

       <ImageView 
        android:id="@+id/companyAddress" 
        android:layout_width="wrap_content" 
        android:layout_height="wrap_content" 
        android:layout_weight="1" 
        android:contentDescription="@string/address_description" 
        android:src="@drawable/company_address" /> 
      </TableRow> 

      <TableRow 
       android:layout_width="fill_parent" 
       android:layout_height="wrap_content" 
       android:orientation="vertical" > 

       <FrameLayout 
        android:layout_width="match_parent" 
        android:layout_height="wrap_content" > 

        <EditText 
         android:id="@+id/salesExecInitials" 
         android:layout_width="fill_parent" 
         android:layout_height="wrap_content" 
         android:background="@drawable/round" 
         android:inputType="text" 
         android:maxLines="1" 
         android:paddingLeft="130dp" 
         android:singleLine="true" > 

         <requestFocus /> 
        </EditText> 

        <TextView 
         android:layout_width="wrap_content" 
         android:layout_height="wrap_content" 
         android:layout_gravity="center_vertical" 
         android:paddingLeft="10dp" 
         android:text="@string/sales_exec_initials" /> 
       </FrameLayout> 
      </TableRow> 
     </TableLayout> 

편집 # 2 : 내가 직접 사용하여 이메일로 전송 된 후 파일을 확인했지만, 지금 확인했다 "ADB 풀 여기

완전성에 대한 관련 레이아웃입니다 "그리고 그것은 여전히 ​​같은 방식으로 손상되었습니다.

편집 # 3 : 나는 dropox에 손상된 PDF의 예를 업로드 한 : 그것은 실제로 훨씬 더 단순화 된보기를 사용하고 Example of Corrupted PDF

,하지만 여전히

편집 # 4 손상 : 나는했습니다 이제는 getFD()에서 sync()를 호출하고 아래에서 CommonsWare가 제안한 것처럼 스트림을 닫으려고하면 예외가 발생합니다. 흥미 롭습니다. 그래서 왜 당신이 할 수있는 설명

12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): com.kingdomspas.android.kingdomspasforms.exceptions.KingdomSpasException: Failed to correctly clean up streams 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at com.kingdomspas.android.kingdomspasforms.utils.KingdomSpasFormsPDFGenerator.generate(KingdomSpasFormsPDFGenerator.java:69) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at com.kingdomspas.android.kingdomspasforms.listeners.KingdomSpasSubmitButtonListener.onClick(KingdomSpasSubmitButtonListener.java:25) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at android.view.View.performClick(View.java:4438) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at android.view.View$PerformClick.run(View.java:18439) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at android.os.Handler.handleCallback(Handler.java:733) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at android.os.Handler.dispatchMessage(Handler.java:95) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at android.os.Looper.loop(Looper.java:136) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at android.app.ActivityThread.main(ActivityThread.java:5147) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at java.lang.reflect.Method.invokeNative(Native Method) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at java.lang.reflect.Method.invoke(Method.java:515) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:795) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:611) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at dalvik.system.NativeStart.main(Native Method) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): Caused by: java.io.IOException: write failed: EBADF (Bad file number) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at libcore.io.IoBridge.write(IoBridge.java:455) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at java.io.FileOutputStream.write(FileOutputStream.java:187) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at java.io.BufferedOutputStream.flushInternal(BufferedOutputStream.java:185) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at java.io.BufferedOutputStream.flush(BufferedOutputStream.java:85) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at com.kingdomspas.android.kingdomspasforms.utils.KingdomSpasFormsPDFGenerator.generate(KingdomSpasFormsPDFGenerator.java:63) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): ... 12 more 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): Caused by: libcore.io.ErrnoException: write failed: EBADF (Bad file number) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at libcore.io.Posix.writeBytes(Native Method) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at libcore.io.Posix.write(Posix.java:202) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at libcore.io.BlockGuardOs.write(BlockGuardOs.java:197) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): at libcore.io.IoBridge.write(IoBridge.java:450) 
12-16 16:39:05.675: E/KingdomSpasSubmitButtonListener(17950): ... 16 more 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): Failed to generate the agreement PDF 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): com.kingdomspas.android.kingdomspasforms.exceptions.KingdomSpasException: Failed to correctly clean up streams 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at com.kingdomspas.android.kingdomspasforms.utils.KingdomSpasFormsPDFGenerator.generate(KingdomSpasFormsPDFGenerator.java:69) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at com.kingdomspas.android.kingdomspasforms.listeners.KingdomSpasSubmitButtonListener.onClick(KingdomSpasSubmitButtonListener.java:25) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at android.view.View.performClick(View.java:4438) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at android.view.View$PerformClick.run(View.java:18439) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at android.os.Handler.handleCallback(Handler.java:733) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at android.os.Handler.dispatchMessage(Handler.java:95) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at android.os.Looper.loop(Looper.java:136) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at android.app.ActivityThread.main(ActivityThread.java:5147) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at java.lang.reflect.Method.invokeNative(Native Method) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at java.lang.reflect.Method.invoke(Method.java:515) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:795) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:611) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at dalvik.system.NativeStart.main(Native Method) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): Caused by: java.io.IOException: write failed: EBADF (Bad file number) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at libcore.io.IoBridge.write(IoBridge.java:455) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at java.io.FileOutputStream.write(FileOutputStream.java:187) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at java.io.BufferedOutputStream.flushInternal(BufferedOutputStream.java:185) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at java.io.BufferedOutputStream.flush(BufferedOutputStream.java:85) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at com.kingdomspas.android.kingdomspasforms.utils.KingdomSpasFormsPDFGenerator.generate(KingdomSpasFormsPDFGenerator.java:63) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): ... 12 more 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): Caused by: libcore.io.ErrnoException: write failed: EBADF (Bad file number) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at libcore.io.Posix.writeBytes(Native Method) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at libcore.io.Posix.write(Posix.java:202) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at libcore.io.BlockGuardOs.write(BlockGuardOs.java:197) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): at libcore.io.IoBridge.write(IoBridge.java:450) 
12-16 16:39:13.845: E/KingdomSpasSubmitButtonListener(17950): ... 16 more 
+0

언제 파일을 확인하고 있습니까? 이메일로받은 후? – greenapps

+0

예 - 이전에 어떻게 확인할 수 있습니까? –

+0

당신은 정말로 저에게 묻지 않습니다. – greenapps

답변

5

당신의 스트림은 flush()도 아니고 close()도 아닙니다. FileOutputStream에서 sync()getFD()으로 호출하여 모든 바이트가 디스크에 커밋되었는지 (파일 시스템에 의해 쓰기 캐시 된 것보다) 확인해야합니다. 파일로 무언가를하려고 할 때의 타이밍에 따라 이런 종류의 결과물은 잘려나갑니다.

+0

나는 이것을 시험해 보았다 - 새 코드로 해결할 몇 가지 버그가 있었기 때문에 나는 잠시 논평을했다! 불행히도 손상된 PDF에는 아무런 차이가 없습니다. 내 FileOutputStream에서 flush() 및 close()를 호출하고 FileDescriptor에서도 동기화합니다. –

+0

사과! 내가 말했듯이 FileOutputStream에서 모든 것을 시도했지만 BufferedOutputStream에서 시도해 보았습니다. –

0

내가 코드를 테스트 한, 그것은 항상 (0 콘텐츠 바이트),하지 손상 한 빈 문서를 만듭니다 : 스택 추적은 어떤 빛을 비춰 경우에 여기에있다 gs으로 수정하십시오. Android 인쇄 프레임 워크에 대해서는 PrintManager 클래스를 호출하지 않고 직접 문서를 인쇄 할 수 있는지 잘 모르겠습니다. 내가 읽은 모든 문서는 그것을 언급하고, 내가 본 모든 샘플 코드는 그것을 사용합니다.

Android 개발자 웹 사이트에서 사용자 정의 문서 인쇄에 대해 good tutorial이 있으며, 사용자는이를 확인할 수 있습니다. 완전한 예제는 here입니다. 이 마지막 샘플 코드를 테스트했으며 잘 작동합니다! 유일한 차이점은 인쇄하기 전에 옵션을 선택하기위한 UI를 일부 표시한다는 것입니다.

희망은이 방법으로 만들 수 있습니다!

+0

테스트 할 때 나는 0 바이트 문서를 얻곤 했었습니다. 전화 또는 에뮬레이터에서 실행할 때 손상된 PDF가 표시됩니다. 인쇄용으로이 PDF를 만들지는 않습니다. 최소한 이메일은 보내지 마세요. PDFDocument로 동일한 코드를 실제로 사용해 보았습니다. 동일한 결과를 얻었습니다. –

관련 문제