2010-05-17 3 views
6

apache poi를 사용하여 Excel 시트를 프레젠테이션 (PPT)에 포함하려고합니다. 우리가 어떻게 할 수 있니? 아무도 모른다면 제발 도와주세요.apache poi를 사용하여 HSLF (excel)를 HSLF (ppt)에 포함

+0

현재 내가 만이 [링크] (HTTP에 설명 된 것처럼 이미 포함 된 Excel 시트를 수정 관리 : //apache-poi.1045710.n5.nabble. co.kr/해결 방법 - 솔루션 - 포함 워크 시트 - 슬라이드 - POI-td2300252.html 사용). 비슷한 POI 버그 (https://issues.apache.org/bugzilla/show_bug.cgi?id=44939)도 있는데, 이는 올레 임베딩이 불완전한 지원이라는 것을 지적합니다. 다른 한편으로는 Libre Office가 그것을 지원하는 것 같습니다. POI를 사용하여 글을 쓰고 얼마나 커스터마이징했는지는 확실하지 않습니다 ... – kiwiwings

답변

5

이 부분이 함께 속해있는 방법을 알아낼 걸 렸어요 ...

매립은 두 가지 방법으로 수행 할 수 있습니다

  • by updating an already embedded worksheet
    • 프로 : 단지 ObjectData.get/setData()를 호출 귀하의 작업이 완료되었습니다
    • 단점 : 둘 이상의 OLE 객체가 포함되도록하려면 어떻게해야합니까?
  • 또는 당신은 내가 특정 POI 기능을 구현하는 방법을, 나는 리브레 오피스 파일과 결과를 비교하고 있습니다 파악하려고 할 때 평소처럼

(아래 참조) 처음부터 요소를 포함 할 수 있습니다 이 경우 몇 가지 부품/생성, 수정해야했다 다음 emebedded 객체의 파워 포인트 객체의

  • ...
    • 이진 데이터는 루트 수준의 기록으로 저장됩니다. 루트 레코드의 대부분은 position dependent입니다. 따라서 새 레코드 (예 : 레코드)가 삭제 될 때 모든 오프셋을 다시 계산해야합니다. 슬라이드는
    • 이진 데이터 레코드가 Document 기록
    • 안에 매립 레코드에서 참조 ... 그리고 좀더를 난독
    • 이 문서 참조 실제 형상 객체가 한번 더 참조 만들어
    Ole Stream 항목이
  • 작성해야합니다
    • 포함 된 워크 시트의 POIFS에서
    • ... 그리고 루트 노드가 포함 된 문서 유형의 클래스 ID를 가지고 있어야이 외에도에서, 포함 된 통합 문서 객체이 켜지지 및 데이터 자체에 아무런 변화가없고, 독립적 인 엑셀은 또한

파일 나는이 개 실제적인 정보 클래스를 사용했습니다입니다

  • : BiffViewerPOIFSLister.

    이것은 단지 개념 증명 일 뿐이므로 완전히 완료되지 않았습니다. 임베디드 요소의 표현을 수정하려면 the spec을 참조하십시오.

    아직 포함 된 개체에 대한 미리보기 이미지를 만드는 문제는 해결되지 않았습니다. 사용자가 ole 객체를 활성화 (더블 클릭)하자마자 대체되는 중립적 인 이미지를 사용하고 싶을 수도 있습니다 ... 대안은 jodconverter을 사용하는 것이지만 POI 접근 방식은 약간 무의미합니다.

    (POI3.9/Libre Office 4에서 테스트 됨.0 2003/MS 오피스/MS 엑셀 뷰어)

    import java.awt.geom.Rectangle2D; 
    import java.io.*; 
    import java.lang.reflect.Field; 
    
    import org.apache.poi.POIDocument; 
    import org.apache.poi.ddf.*; 
    import org.apache.poi.hpsf.ClassID; 
    import org.apache.poi.hslf.HSLFSlideShow; 
    import org.apache.poi.hslf.exceptions.HSLFException; 
    import org.apache.poi.hslf.model.*; 
    import org.apache.poi.hslf.model.Picture; 
    import org.apache.poi.hslf.model.Slide; 
    import org.apache.poi.hslf.record.*; 
    import org.apache.poi.hslf.usermodel.*; 
    import org.apache.poi.hssf.usermodel.*; 
    import org.apache.poi.hwpf.HWPFDocument; 
    import org.apache.poi.hwpf.usermodel.*; 
    import org.apache.poi.poifs.filesystem.*; 
    import org.apache.poi.util.*; 
    
    public class PoiOleXlsInPpt { 
        static final OleType EXCEL97  = new OleType("{00020820-0000-0000-C000-000000000046}"); 
        static final OleType EXCEL95  = new OleType("{00020810-0000-0000-C000-000000000046}"); 
        static final OleType WORD97  = new OleType("{00020906-0000-0000-C000-000000000046}"); 
        static final OleType WORD95  = new OleType("{00020900-0000-0000-C000-000000000046}"); 
        static final OleType POWERPOINT97 = new OleType("{64818D10-4F9B-11CF-86EA-00AA00B929E8}"); 
        static final OleType POWERPOINT95 = new OleType("{EA7BAE70-FB3B-11CD-A903-00AA00510EA3}"); 
    
        static class OleType { 
         final String classId; 
         OleType(String classId) { 
          this.classId = classId; 
         } 
         ClassID getClassID() { 
          ClassID cls = new ClassID(); 
          byte clsBytes[] = cls.getBytes(); 
          String clsStr = classId.replaceAll("[{}-]", ""); 
          for (int i=0; i<clsStr.length(); i+=2) { 
           clsBytes[i/2] = (byte)Integer.parseInt(clsStr.substring(i, i+2), 16); 
          } 
          return cls; 
         } 
        } 
    
        public static void main(String[] args) throws Exception { 
         HSLFSlideShow _hslfSlideShow = HSLFSlideShow.create(); 
         SlideShow ppt = new SlideShow(_hslfSlideShow); 
    
         OLEShape oleShape1 = createOLEShape(getSampleWorkbook1(), ppt, _hslfSlideShow, EXCEL97); 
         oleShape1.setAnchor(new Rectangle2D.Double(100,100,100,100)); 
         OLEShape oleShape2 = createOLEShape(getSampleWorkbook2(), ppt, _hslfSlideShow, EXCEL97); 
         oleShape2.setAnchor(new Rectangle2D.Double(300,300,100,100)); 
         OLEShape oleShape3 = createOLEShape(getSampleDocument(), ppt, _hslfSlideShow, WORD97); 
         oleShape3.setAnchor(new Rectangle2D.Double(300,100,100,100)); 
    
         // create and link visuals to the ole data 
         Slide slide = ppt.createSlide(); 
         slide.addShape(oleShape1); 
         slide.addShape(oleShape2); 
         slide.addShape(oleShape3); 
    
         FileOutputStream fos = new FileOutputStream("ole_xls_in_ppt_out2.ppt"); 
         ppt.write(fos); 
         fos.close(); 
        } 
    
        static OLEShape createOLEShape(
          POIDocument sample 
         , SlideShow ppt 
         , HSLFSlideShow _hslfSlideShow 
         , OleType oleType 
        ) throws IOException { 
         // generate a preview image 
         int prevIdx = generatePreview(ppt, sample); 
    
         // add the data to the SlideShow 
         ExEmbed eeEmbed = addOleDataToDocumentRecord(ppt); 
         ExOleObjStg exOleObjStg = addOleDataToRootRecords(_hslfSlideShow, sample, oleType); 
         eeEmbed.getExOleObjAtom().setObjStgDataRef(exOleObjStg.getPersistId()); 
    
         OLEShape oleShape = new OLEShape(prevIdx); 
         linkOleDataToShape(oleShape, eeEmbed); 
    
         return oleShape; 
        } 
    
        static POIDocument getSampleWorkbook1() { 
         HSSFWorkbook wb = new HSSFWorkbook(); 
         HSSFSheet sheet = wb.createSheet(); 
         sheet.createRow(1).createCell(1).setCellValue("First Workbook"); 
         return wb; 
        } 
    
        static POIDocument getSampleWorkbook2() { 
         HSSFWorkbook wb = new HSSFWorkbook(); 
         HSSFSheet sheet = wb.createSheet(); 
         sheet.createRow(1).createCell(1).setCellValue("Second Workbook"); 
         return wb; 
        } 
    
        // the sample document has apparently a problem, 
        // i.e. word inside ms powerpoint crashed, and libre office doesn't display the text 
        // it was just a test, if embedding elements != Excel works 
        // in case HWPF is interesting to you, you probably know anyway, where the error below is ... 
        static POIDocument getSampleDocument() throws IOException { 
         FileInputStream fis = new FileInputStream("src/test/resources/empty.doc"); 
         HWPFDocument doc = new HWPFDocument(fis); 
         fis.close(); 
         Range range = doc.getRange(); 
         CharacterRun run1 = range.insertAfter("Sample text");  
         run1.setFontSize(11);  
         return doc; 
        } 
    
        /** 
        * Generates a modified version of the sample element, which 
        * contains embedding informations 
        */ 
        static byte[] wrapOleData(POIDocument oleData, OleType oleType) { 
         try { 
          ByteArrayOutputStream bos = new ByteArrayOutputStream(); 
          oleData.write(bos); 
    
          ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray()); 
          bos.reset(); 
          POIFSFileSystem poifs = new POIFSFileSystem(bis); 
    
          final String OLESTREAM_NAME = "\u0001Ole"; 
          DirectoryNode root = poifs.getRoot(); 
          if (!root.hasEntry(OLESTREAM_NAME)) { 
           // the following data was taken from an example libre office document 
           // beside this "\u0001Ole" record there were several other records, e.g. CompObj, 
           // OlePresXXX, but it seems, that they aren't neccessary 
           byte oleBytes[] = { 1, 0, 0, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }; 
           poifs.createDocument(new ByteArrayInputStream(oleBytes), OLESTREAM_NAME); 
          } 
    
          // need to set storage clsid, otherwise embedded object is not recognized 
          root.setStorageClsid(oleType.getClassID()); 
    
          poifs.writeFilesystem(bos); 
          return bos.toByteArray(); 
         } catch (IOException e) { 
          throw new RuntimeException("wth?!", e); 
         } 
        } 
    
    
        /** 
        * to be defined, how to create a preview image 
        * for a start, I've taken just a dummy image, which will be 
        * replaced, when the user activates the ole object 
        * 
        * not really an alternativ: 
        * http://stackoverflow.com/questions/16704624/how-to-print-a-workbook-file-made-using-apache-poi-and-java 
        * 
        * @return image index of the preview image 
        */ 
        static int generatePreview(SlideShow ppt, POIDocument oleData) { 
         try { 
          FileInputStream fis = new FileInputStream("src/test/resources/dilbert-2011-09-28-powerpoint.jpg"); 
          byte previewImg[] = IOUtils.toByteArray(fis); 
          fis.close(); 
          return ppt.addPicture(previewImg, Picture.JPEG); 
         } catch (IOException e) { 
          throw new RuntimeException("not really?", e); 
         } 
        } 
    
        static ExEmbed addOleDataToDocumentRecord(SlideShow ppt) { 
         // taken from SlideShow.addControl() 
         Document _documentRecord = ppt.getDocumentRecord(); 
         ExObjList lst = _documentRecord.getExObjList(); 
         if (lst == null) { 
          lst = new ExObjList(); 
          _documentRecord.addChildAfter(lst, _documentRecord.getDocumentAtom()); 
          try { 
           Field f = Document.class.getDeclaredField("exObjList"); 
           f.setAccessible(true); 
           f.set(_documentRecord, lst); 
          } catch (Exception e) { 
           throw new RuntimeException("not here", e); 
          } 
         } 
         ExObjListAtom objAtom = lst.getExObjListAtom(); 
         // increment the object ID seed 
         int objectId = (int) objAtom.getObjectIDSeed() + 1; 
         objAtom.setObjectIDSeed(objectId); 
    
         ExEmbed exEmbed = new ExEmbed(); 
         // remove unneccessary infos, so we don't need to specify the type 
         // of the ole object multiple times 
         Record children[] = exEmbed.getChildRecords(); 
         exEmbed.removeChild(children[2]); 
         exEmbed.removeChild(children[3]); 
         exEmbed.removeChild(children[4]); 
    
         ExEmbedAtom eeEmbed = exEmbed.getExEmbedAtom(); 
         try { 
          Field f = ExEmbedAtom.class.getDeclaredField("_data"); 
          f.setAccessible(true); 
          f.set(eeEmbed, new byte[]{0,0,0,0,1/*CantLockServerB*/,0,0,0}); 
          // oops, there seems to be an error in the default constructor ... 
          // should be 8 and not 7 bytes 
          setRecordLength(eeEmbed, 8); 
         } catch (Exception e) { 
          throw new RuntimeException("trust me ;)", e); 
         } 
    
         ExOleObjAtom eeAtom = exEmbed.getExOleObjAtom(); 
         eeAtom.setObjID(objectId); 
         eeAtom.setDrawAspect(ExOleObjAtom.DRAW_ASPECT_VISIBLE); 
         eeAtom.setType(ExOleObjAtom.TYPE_EMBEDDED); 
         // eeAtom.setSubType(ExOleObjAtom.SUBTYPE_EXCEL); 
         // should be ignored?!?, see MS-PPT ExOleObjAtom, but Libre Office sets it ... 
         eeAtom.setOptions(1226240); 
    
         lst.addChildAfter(exEmbed, objAtom); 
    
         return exEmbed; 
        } 
    
        static ExOleObjStg addOleDataToRootRecords(
          HSLFSlideShow _hslfSlideShow 
         , POIDocument oleData 
         , OleType oleType 
        ) throws IOException { 
         ExOleObjStg exOleObjStg = new ExOleObjStg(); 
         int slideRecordPos = _hslfSlideShow.appendRootLevelRecord(exOleObjStg); 
         exOleObjStg.setPersistId(slideRecordPos); 
         exOleObjStg.setData(wrapOleData(oleData, oleType)); 
    
         // taken from SlideShow.createSlide 
         Record _records[] = _hslfSlideShow.getRecords(); 
    
         // Add the new OLE record into the PersistPtr stuff 
         int offset = 0; 
         int slideOffset = 0; 
         PersistPtrHolder ptr = null; 
         UserEditAtom usr = null; 
         for (int i = 0; i < _records.length; i++) { 
          Record record = _records[i]; 
          ByteArrayOutputStream out = new ByteArrayOutputStream(); 
          try { 
           record.writeOut(out); 
          } catch (IOException e) { 
           throw new HSLFException(e); 
          } 
    
          // Grab interesting records as they come past 
          if (_records[i].getRecordType() == RecordTypes.PersistPtrIncrementalBlock.typeID) { 
           ptr = (PersistPtrHolder) _records[i]; 
          } 
          if (_records[i].getRecordType() == RecordTypes.UserEditAtom.typeID) { 
           usr = (UserEditAtom) _records[i]; 
          } 
    
          if (i == slideRecordPos) { 
           slideOffset = offset; 
          } 
          offset += out.size(); 
         } 
    
         // the ole objects needs to know its position within 
         // the root records, because it will be later accessed 
         // via its index from the shape 
         int psrId = usr.getMaxPersistWritten() + 1; 
         exOleObjStg.setPersistId(psrId); 
    
         // Last view is now of the slide 
         usr.setLastViewType((short) UserEditAtom.LAST_VIEW_SLIDE_VIEW); 
         usr.setMaxPersistWritten(psrId); // increment the number of persit objects 
    
         // Add the new slide into the last PersistPtr 
         // (Also need to tell it where it is) 
         exOleObjStg.setLastOnDiskOffset(slideOffset); 
         ptr.addSlideLookup(psrId, slideOffset); 
    
         return exOleObjStg; 
        } 
    
        static void linkOleDataToShape(OLEShape oleShape, ExEmbed exEmbed) { 
         oleShape.setEscherProperty(EscherProperties.BLIP__PICTUREID, exEmbed.getExOleObjAtom().getObjID()); 
    
         EscherSpRecord spRecord = oleShape.getSpContainer().getChildById(EscherSpRecord.RECORD_ID); 
         spRecord.setFlags(spRecord.getFlags()|EscherSpRecord.FLAG_OLESHAPE); 
    
         // ExObjRefAtom is not set in OLEShape 
         UnknownEscherRecord uer = new UnknownEscherRecord(); 
         byte uerData[] = new byte[12]; 
         LittleEndian.putShort(uerData, 0, (short)0); // options = 0 
         LittleEndian.putShort(uerData, 2, (short)RecordTypes.ExObjRefAtom.typeID); // recordId 
         LittleEndian.putInt(uerData, 4, 4); // remaining bytes 
         LittleEndian.putInt(uerData, 8, exEmbed.getExOleObjAtom().getObjID()); // the data 
         uer.fillFields(uerData, 0, null); 
    
         EscherContainerRecord uerCont = new EscherContainerRecord(); 
         uerCont.setRecordId((short)RecordTypes.EscherClientData); 
         uerCont.setVersion((short)0x000F); // yes, we are still a container ... 
         uerCont.addChildRecord(uer); 
    
         oleShape.getSpContainer().addChildRecord(uerCont); 
        } 
    
        static void setRecordLength(Record record, int len) throws NoSuchFieldException, IllegalAccessException { 
         Field f = record.getClass().getDeclaredField("_header"); 
         f.setAccessible(true); 
         byte _header[] = (byte[])f.get(record); 
         LittleEndian.putInt(_header, 4, len); 
         f.set(record, _header); 
        } 
    } 
    
  • +1

    패치는 버그 보고서 [# 55579] (https : //issues.apache .org/bugzilla/show_bug.cgi? id = 55579) – kiwiwings