xref: /AOO41X/main/sw/source/ui/app/applab.cxx (revision 8ef2f12b1aeba1404ab3c221e6e26281826cc4fc)
1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 // MARKER(update_precomp.py): autogen include statement, do not remove
25 #include "precompiled_sw.hxx"
26 
27 #include <cstdarg>
28 
29 #include <hintids.hxx>
30 
31 #include <vcl/svapp.hxx>
32 #include <vcl/wrkwin.hxx>
33 #include <vcl/msgbox.hxx>
34 #include <sfx2/app.hxx>
35 #include <sfx2/dispatch.hxx>
36 #include <sfx2/printer.hxx>
37 #include <sfx2/request.hxx>
38 #include <sfx2/linkmgr.hxx>
39 #include <editeng/pbinitem.hxx>
40 #include <editeng/ulspitem.hxx>
41 #include <editeng/lrspitem.hxx>
42 #include <editeng/boxitem.hxx>
43 #include <editeng/paperinf.hxx>
44 #include <editeng/protitem.hxx>
45 #include <com/sun/star/frame/XStorable.hpp>
46 #include <com/sun/star/frame/XModel.hpp>
47 #include <fmthdft.hxx>
48 #include <fmtanchr.hxx>
49 #include <fmtfsize.hxx>
50 #include <fmtornt.hxx>
51 #include <swwait.hxx>
52 #include <gloshdl.hxx>
53 #include <mdiexp.hxx>
54 #include <frmatr.hxx>
55 #include <paratr.hxx>
56 #include <swmodule.hxx>
57 #include <view.hxx>
58 #include <docsh.hxx>
59 #include <fldbas.hxx>
60 #include <swundo.hxx>
61 #include <wrtsh.hxx>
62 #include <cmdid.h>
63 #include <dbmgr.hxx>
64 #include <fmtcol.hxx>
65 #include <expfld.hxx>
66 #include <fldmgr.hxx>
67 #include <label.hxx>
68 #include <labimg.hxx>
69 #include <section.hxx>
70 #include <pagedesc.hxx>
71 #include <poolfmt.hxx>
72 
73 #ifndef _APP_HRC
74 #include <app.hrc>
75 #endif
76 #ifndef _POOLFMT_HRC
77 #include <poolfmt.hrc>
78 #endif
79 #include "swabstdlg.hxx"
80 #include "envelp.hrc"
81 #include <misc.hrc>
82 
83 #include <IDocumentDeviceAccess.hxx>
84 
85 using namespace ::com::sun::star;
86 using ::rtl::OUString;
87 
88 // steht im appenv.cxx
89 extern String InsertLabEnvText( SwWrtShell& , SwFldMgr& , const String& );
90 
91 const char __FAR_DATA MASTER_LABEL[] = "MasterLabel";
92 
93 // --------------------------------------------------------------------------
94 
lcl_InsertBCText(SwWrtShell & rSh,const SwLabItem & rItem,SwFrmFmt & rFmt,sal_uInt16 nCol,sal_uInt16 nRow,sal_Bool bPage)95 const SwFrmFmt *lcl_InsertBCText( SwWrtShell& rSh, const SwLabItem& rItem,
96                         SwFrmFmt &rFmt,
97                         sal_uInt16 nCol, sal_uInt16 nRow, sal_Bool bPage)
98 {
99     SfxItemSet aSet(rSh.GetAttrPool(), RES_ANCHOR, RES_ANCHOR,
100                         RES_VERT_ORIENT, RES_VERT_ORIENT, RES_HORI_ORIENT, RES_HORI_ORIENT, 0 );
101     sal_uInt16 nPhyPageNum, nVirtPageNum;
102     rSh.GetPageNum( nPhyPageNum, nVirtPageNum );
103 
104     aSet.Put(SwFmtAnchor(bPage ? FLY_AS_CHAR : FLY_AT_PAGE, nPhyPageNum));
105     if (!bPage)
106     {
107         aSet.Put(SwFmtHoriOrient(rItem.lLeft + nCol * rItem.lHDist,
108                                                     text::HoriOrientation::NONE, text::RelOrientation::PAGE_FRAME ));
109         aSet.Put(SwFmtVertOrient(rItem.lUpper + nRow * rItem.lVDist,
110                                                     text::VertOrientation::NONE, text::RelOrientation::PAGE_FRAME ));
111     }
112     const SwFrmFmt *pFmt = rSh.NewFlyFrm(aSet, sal_True,  &rFmt );  // Fly einfuegen
113     ASSERT( pFmt, "Fly not inserted" );
114 
115     rSh.UnSelectFrm();  //Rahmen wurde automatisch selektiert
116 
117     rSh.SetTxtFmtColl( rSh.GetTxtCollFromPool( RES_POOLCOLL_STANDARD ) );
118 
119     //
120     if(!rItem.bSynchron || !(nCol|nRow))
121     {
122         SwAbstractDialogFactory* pFact = SwAbstractDialogFactory::Create();
123         DBG_ASSERT(pFact, "Dialogdiet fail!");
124         ::GlossarySetActGroup fnSetActGroup = pFact->SetGlossaryActGroupFunc( DLG_RENAME_GLOS );
125         if ( fnSetActGroup )
126             (*fnSetActGroup)( rItem.sGlossaryGroup );
127         SwGlossaryHdl* pGlosHdl = rSh.GetView().GetGlosHdl();
128         pGlosHdl->SetCurGroup(rItem.sGlossaryGroup, sal_True);
129         pGlosHdl->InsertGlossary( rItem.sGlossaryBlockName );
130     }
131 
132     return pFmt;
133 }
134 
lcl_InsertLabText(SwWrtShell & rSh,const SwLabItem & rItem,SwFrmFmt & rFmt,SwFldMgr & rFldMgr,sal_uInt16 nCol,sal_uInt16 nRow,sal_Bool bLast,sal_Bool bPage)135 const SwFrmFmt *lcl_InsertLabText( SwWrtShell& rSh, const SwLabItem& rItem,
136                         SwFrmFmt &rFmt, SwFldMgr& rFldMgr,
137                         sal_uInt16 nCol, sal_uInt16 nRow, sal_Bool bLast, sal_Bool bPage)
138 {
139     SfxItemSet aSet(rSh.GetAttrPool(), RES_ANCHOR, RES_ANCHOR,
140                         RES_VERT_ORIENT, RES_VERT_ORIENT, RES_HORI_ORIENT, RES_HORI_ORIENT, 0 );
141     sal_uInt16 nPhyPageNum, nVirtPageNum;
142     rSh.GetPageNum( nPhyPageNum, nVirtPageNum );
143 
144     aSet.Put(SwFmtAnchor(bPage ? FLY_AS_CHAR : FLY_AT_PAGE, nPhyPageNum));
145     if (!bPage)
146     {
147         aSet.Put(SwFmtHoriOrient(rItem.lLeft + nCol * rItem.lHDist,
148                                                     text::HoriOrientation::NONE, text::RelOrientation::PAGE_FRAME ));
149         aSet.Put(SwFmtVertOrient(rItem.lUpper + nRow * rItem.lVDist,
150                                                     text::VertOrientation::NONE, text::RelOrientation::PAGE_FRAME ));
151     }
152     const SwFrmFmt *pFmt = rSh.NewFlyFrm(aSet, sal_True,  &rFmt );  // Fly einfuegen
153     ASSERT( pFmt, "Fly not inserted" );
154 
155     rSh.UnSelectFrm();  //Rahmen wurde automatisch selektiert
156 
157     rSh.SetTxtFmtColl( rSh.GetTxtCollFromPool( RES_POOLCOLL_STANDARD ) );
158 
159     // Ggf. "Naechster Datensatz"
160     String sDBName;
161     if( (!rItem.bSynchron || !(nCol|nRow)) && (sDBName = InsertLabEnvText( rSh, rFldMgr, rItem.aWriting )).Len() && !bLast )
162     {
163         sDBName.SetToken( 3, DB_DELIM, String::CreateFromAscii("True"));
164         SwInsertFld_Data aData(TYP_DBNEXTSETFLD, 0, sDBName, aEmptyStr, 0, &rSh );
165         rFldMgr.InsertFld( aData );
166     }
167 
168     return pFmt;
169 }
170 
171 // ----------------------------------------------------------------------------
172 
173 
InsertLab(SfxRequest & rReq,sal_Bool bLabel)174 void SwModule::InsertLab(SfxRequest& rReq, sal_Bool bLabel)
175 {
176 static sal_uInt16 nLabelTitleNo = 0;
177 static sal_uInt16 nBCTitleNo = 0;
178 
179     // DB-Manager anlegen
180     SwNewDBMgr* pNewDBMgr = new SwNewDBMgr;
181 
182     // SwLabItem aus Config lesen
183     SwLabCfgItem aLabCfg(bLabel);
184 
185     // Dialog hochfahren
186     SfxItemSet aSet( GetPool(), FN_LABEL, FN_LABEL, 0 );
187     aSet.Put( aLabCfg.GetItem() );
188 
189     SwAbstractDialogFactory* pDialogFactory = SwAbstractDialogFactory::Create();
190     DBG_ASSERT(pDialogFactory, "SwAbstractDialogFactory fail!");
191 
192     AbstarctSwLabDlg* pDlg = pDialogFactory->CreateSwLabDlg( 0, aSet, pNewDBMgr, bLabel, DLG_LAB );
193     DBG_ASSERT(pDlg, "Dialogdiet fail!");
194 
195     if ( RET_OK == pDlg->Execute() )
196     {
197         // Dialog auslesen, Item in Config speichern
198         const SwLabItem& rItem = (const SwLabItem&) pDlg->
199                                             GetOutputItemSet()->Get(FN_LABEL);
200         aLabCfg.GetItem() = rItem;
201         aLabCfg.Commit();
202 
203         // Neues Dokument erzeugen.
204         SfxObjectShellLock xDocSh( new SwDocShell( SFX_CREATE_MODE_STANDARD));
205         xDocSh->DoInitNew( 0 );
206 
207         // Drucker
208         Printer *pPrt = pDlg->GetPrt();
209         if (pPrt)
210         {
211             SwDocShell *pDocSh = (SwDocShell*)(&*xDocSh);
212             pDocSh->getIDocumentDeviceAccess()->setJobsetup(pPrt->GetJobSetup());
213         }
214 
215         SfxViewFrame* pViewFrame = SfxViewFrame::DisplayNewDocument( *xDocSh, rReq );
216 
217         SwView      *pNewView = (SwView*) pViewFrame->GetViewShell();
218         pNewView->AttrChangedNotify( &pNewView->GetWrtShell() );//Damit SelectShell gerufen wird.
219 
220         // Dokumenttitel setzen
221         String aTmp;
222         if(bLabel)
223         {
224             aTmp = String(SW_RES( STR_LAB_TITLE));
225             aTmp += String::CreateFromInt32(++nLabelTitleNo );
226         }
227         else
228         {
229             aTmp = pDlg->GetBusinessCardStr();
230             aTmp += String::CreateFromInt32( ++nBCTitleNo );
231         }
232         xDocSh->SetTitle( aTmp );
233 
234         pViewFrame->GetFrame().Appear();
235 
236         // Shell ermitteln
237         SwWrtShell *pSh = pNewView->GetWrtShellPtr();
238         ASSERT( pSh, "missing WrtShell" );
239 
240         {   // block for locks the dispatcher!!
241 
242             SwWait aWait( (SwDocShell&)*xDocSh, true );
243 
244             SET_CURR_SHELL(pSh);
245             pSh->SetLabelDoc(rItem.bSynchron);
246             pSh->DoUndo( sal_False );
247             pSh->StartAllAction();
248 
249             pSh->SetNewDoc();       // Performanceprobleme vermeiden
250 
251             SwPageDesc aDesc = pSh->GetPageDesc( 0 );
252             SwFrmFmt&  rFmt  = aDesc.GetMaster();
253 
254             // Raender
255             SvxLRSpaceItem aLRMargin( RES_LR_SPACE );
256             SvxULSpaceItem aULMargin( RES_UL_SPACE );
257             aLRMargin.SetLeft ((sal_uInt16) rItem.lLeft );
258             aULMargin.SetUpper((sal_uInt16) rItem.lUpper);
259             aLRMargin.SetRight(MINLAY/2);
260             aULMargin.SetLower(MINLAY/2);
261             rFmt.SetFmtAttr(aLRMargin);
262             rFmt.SetFmtAttr(aULMargin);
263 
264             // Kopf- und Fusszeilen
265             rFmt.SetFmtAttr(SwFmtHeader(sal_Bool(sal_False)));
266             aDesc.ChgHeaderShare(sal_False);
267             rFmt.SetFmtAttr(SwFmtFooter(sal_Bool(sal_False)));
268             aDesc.ChgFooterShare(sal_False);
269 
270 
271             aDesc.SetUseOn(nsUseOnPage::PD_ALL);                // Seitennumerierung
272 
273             // Einstellen der Seitengroesse
274             rFmt.SetFmtAttr(SwFmtFrmSize(ATT_FIX_SIZE,
275                                         rItem.lLeft  + rItem.nCols * rItem.lHDist + MINLAY,
276                                         rItem.lUpper + rItem.nRows * rItem.lVDist + MINLAY));
277 
278             // Numerierungsart
279             SvxNumberType aType;
280             aType.SetNumberingType(SVX_NUM_NUMBER_NONE);
281             aDesc.SetNumType( aType );
282 
283             // Folgevorlage
284             const SwPageDesc &rFollow = pSh->GetPageDesc( pSh->GetCurPageDesc() );
285             aDesc.SetFollow( &rFollow );
286 
287             pPrt = pSh->getIDocumentDeviceAccess()->getPrinter( true );
288             SvxPaperBinItem aItem( RES_PAPER_BIN );
289             aItem.SetValue((sal_Int8)pPrt->GetPaperBin());
290             rFmt.SetFmtAttr(aItem);
291 
292             //determine orientation by calculating the width and height of the resulting page
293             const int nResultWidth = rItem.lHDist * (rItem.nCols - 1) + rItem.lWidth + rItem.lLeft;
294             const int nResultHeight = rItem.lVDist * (rItem.nRows - 1) + rItem.lHeight + rItem.lUpper;
295             aDesc.SetLandscape(nResultWidth > nResultHeight);
296 
297             pSh->ChgPageDesc( 0, aDesc );
298 
299             // Rahmen einfuegen
300             SwFldMgr*        pFldMgr = new SwFldMgr;
301             pFldMgr->SetEvalExpFlds(sal_False);
302 
303             //fix(24446): Damit der Text der Ettiketten nicht im unbedruckbaren
304             //Bereich landet stellen wir entsprechende Raender ein. Um das Handling
305             //so Optimal wie moeglich zu halten stellen wir zunaechst an der
306             //aktuellen Absatzvorlage keinen Rand als hartes Attribut ein (Damit die
307             //Formatierung wg. der Zeichengeb. Rahmen passt. Dann stellen wir die
308             //Standarabsatzvorlage anhand des unbedruckbaren Bereiches ein.
309             const long nMin = pPrt->GetPageOffset().X() - rItem.lLeft;
310             if ( nMin > 0 )
311             {
312                 SvxLRSpaceItem aLR( RES_LR_SPACE );
313                 pSh->SetAttrItem( aLR );
314                 SwFmt *pStandard = pSh->GetTxtCollFromPool( RES_POOLCOLL_STANDARD );
315                 aLR.SetLeft ( sal_uInt16(nMin) );
316                 aLR.SetRight( sal_uInt16(nMin) );
317                 pStandard->SetFmtAttr( aLR );
318             }
319 
320             // Rahmenvorlage vorbereiten
321             SwFrmFmt* pFmt = pSh->GetFrmFmtFromPool( RES_POOLFRM_LABEL );
322             SwFmtFrmSize aFrmSize(  ATT_FIX_SIZE,
323                                     rItem.lHDist - (rItem.lHDist-rItem.lWidth),
324                                     rItem.lVDist - (rItem.lVDist-rItem.lHeight));
325             pFmt->SetFmtAttr(aFrmSize);
326 
327             SvxLRSpaceItem aFrmLRSpace( 0, (sal_uInt16)(rItem.lHDist - rItem.lWidth),
328                                         0, 0,
329                                         RES_LR_SPACE);
330             pFmt->SetFmtAttr(aFrmLRSpace);
331 
332             SvxULSpaceItem aFrmULSpace( 0, (sal_uInt16)(rItem.lVDist - rItem.lHeight),
333                                         RES_UL_SPACE);
334             pFmt->SetFmtAttr(aFrmULSpace);
335 
336             const SwFrmFmt *pFirstFlyFmt = 0;
337             if ( rItem.bPage )
338             {
339                 SwFmtVertOrient aFrmVertOrient( pFmt->GetVertOrient() );
340                 aFrmVertOrient.SetVertOrient( text::VertOrientation::TOP );
341                 pFmt->SetFmtAttr(aFrmVertOrient);
342 
343                 for ( sal_uInt16 i = 0; i < rItem.nRows; ++i )
344                 {
345                     for ( sal_uInt16 j = 0; j < rItem.nCols; ++j )
346                     {
347                         pSh->Push();
348                         const SwFrmFmt *pTmp =
349                                 bLabel ?
350                                 lcl_InsertLabText( *pSh, rItem, *pFmt, *pFldMgr, j, i,
351                                     i == rItem.nRows - 1 && j == rItem.nCols - 1,
352                                     sal_True ) :
353                                 lcl_InsertBCText(*pSh, rItem, *pFmt, j, i, sal_True);
354                         if (!(i|j))
355                         {
356                             pFirstFlyFmt = pTmp;
357 
358                             if (rItem.bSynchron)
359                             {
360                                 // if there is no content in the fly then
361                                 // dont leave the fly!!!
362                                 pSh->Push();
363                                 pSh->SttDoc();
364                                 sal_Bool bInFly = 0 != pSh->WizzardGetFly();
365                                 pSh->Pop( bInFly );
366 
367                                 if( bInFly )
368                                     pSh->EndDoc(sal_True);  // select all content
369                                                         // in the fly
370                                 else
371                                     pSh->SetMark();     // set only the mark
372 
373                                 SwSectionData aSect(CONTENT_SECTION,
374                                     String::CreateFromAscii(MASTER_LABEL));
375                                 pSh->InsertSection(aSect);
376                             }
377                         }
378                         else if (rItem.bSynchron)
379                         {
380                             SwSectionData aSect(FILE_LINK_SECTION,
381                                     pSh->GetUniqueSectionName());
382                             String sLinkName(sfx2::cTokenSeperator);
383                             sLinkName += sfx2::cTokenSeperator;
384                             sLinkName += String::CreateFromAscii(MASTER_LABEL);
385                             aSect.SetLinkFileName(sLinkName);
386                             aSect.SetProtectFlag(true);
387                             pSh->Insert(aDotStr);   // Dummytext zum Zuweisen der Section
388                             pSh->SttDoc();
389                             pSh->EndDoc(sal_True);  // Alles im Rahmen selektieren
390                             pSh->InsertSection(aSect);
391                         }
392                         pSh->Pop( sal_False );
393                     }
394                     if ( i + 1 != rItem.nRows )
395                         pSh->SplitNode(); // Kleine Optimierung
396                 }
397             }
398             else
399             {
400                 pFirstFlyFmt = bLabel ?
401                     lcl_InsertLabText( *pSh, rItem, *pFmt, *pFldMgr,
402                             static_cast< sal_uInt16 >(rItem.nCol - 1),
403                             static_cast< sal_uInt16 >(rItem.nRow - 1), sal_True, sal_False ) :
404                     lcl_InsertBCText(*pSh, rItem, *pFmt,
405                             static_cast< sal_uInt16 >(rItem.nCol - 1),
406                             static_cast< sal_uInt16 >(rItem.nRow - 1), sal_False);
407             }
408 
409             //fill the user fields
410             if(!bLabel)
411             {
412                 uno::Reference< frame::XModel >  xModel = pSh->GetView().GetDocShell()->GetBaseModel();
413                 DBG_ASSERT(pDialogFactory, "SwAbstractDialogFactory fail!");
414                 SwLabDlgMethod SwLabDlgUpdateFieldInformation = pDialogFactory->GetSwLabDlgStaticMethod ();
415                 SwLabDlgUpdateFieldInformation(xModel, rItem);
416             }
417 
418             pFldMgr->SetEvalExpFlds(sal_True);
419             pFldMgr->EvalExpFlds(pSh);
420 
421             delete pFldMgr;
422 
423             pSh->GotoFly(pFirstFlyFmt->GetName(), FLYCNTTYPE_ALL, sal_False);
424 
425             pSh->EndAllAction();
426             pSh->DoUndo( sal_True );
427         }
428 
429         if( rItem.aWriting.indexOf( '<' ) >= 0 )
430         {
431             // Datenbankbrowser mit zuletzt verwendeter Datenbank oeffnen
432             ShowDBObj( *pNewView, pSh->GetDBData() );
433         }
434 
435         if( rItem.bSynchron )
436         {
437             SfxDispatcher* pDisp = pViewFrame->GetDispatcher();
438             ASSERT(pDisp, "Heute kein Dispatcher am Frame?");
439             pDisp->Execute(FN_SYNC_LABELS, SFX_CALLMODE_ASYNCHRON);
440         }
441         rReq.SetReturnValue(SfxVoidItem(bLabel ? FN_LABEL : FN_BUSINESS_CARD));
442     }
443     delete pDlg;
444 
445     if( pNewDBMgr )
446         delete pNewDBMgr;
447 }
448 
449 
450