/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you 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 . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "mathtype.hxx" #include "ooxmlexport.hxx" #include "ooxmlimport.hxx" #include "rtfexport.hxx" #include "mathmlimport.hxx" #include "mathmlexport.hxx" #include #include #include "cursor.hxx" #include #include "visitors.hxx" #include "accessibility.hxx" #include "cfgitem.hxx" #include using namespace ::com::sun::star; using namespace ::com::sun::star::accessibility; using namespace ::com::sun::star::uno; #define SmDocShell #include "smslots.hxx" SFX_IMPL_SUPERCLASS_INTERFACE(SmDocShell, SfxObjectShell) void SmDocShell::InitInterface_Impl() { GetStaticInterface()->RegisterPopupMenu("view"); } SFX_IMPL_OBJECTFACTORY(SmDocShell, SvGlobalName(SO3_SM_CLASSID), SfxObjectShellFlags::STD_NORMAL, "smath" ) void SmDocShell::Notify(SfxBroadcaster&, const SfxHint& rHint) { switch (static_cast(rHint).GetId()) { case HINT_FORMATCHANGED: SetFormulaArranged(false); mnModifyCount++; //! see comment for SID_GAPHIC_SM in SmDocShell::GetState Repaint(); break; } } void SmDocShell::LoadSymbols() { SmModule *pp = SM_MOD(); pp->GetSymbolManager().Load(); } const OUString SmDocShell::GetComment() const { uno::Reference xDPS( GetModel(), uno::UNO_QUERY_THROW); uno::Reference xDocProps( xDPS->getDocumentProperties()); return xDocProps->getDescription(); } void SmDocShell::SetText(const OUString& rBuffer) { if (rBuffer != maText) { bool bIsEnabled = IsEnableSetModified(); if( bIsEnabled ) EnableSetModified( false ); maText = rBuffer; SetFormulaArranged( false ); Parse(); SmViewShell *pViewSh = SmGetActiveView(); if( pViewSh ) { pViewSh->GetViewFrame()->GetBindings().Invalidate(SID_TEXT); if ( SfxObjectCreateMode::EMBEDDED == GetCreateMode() ) { // have SwOleClient::FormatChanged() to align the modified formula properly // even if the vis area does not change (e.g. when formula text changes from // "{a over b + c} over d" to "d over {a over b + c}" SfxGetpApp()->NotifyEvent(SfxEventHint( SFX_EVENT_VISAREACHANGED, GlobalEventConfig::GetEventName(GlobalEventId::VISAREACHANGED), this)); Repaint(); } else pViewSh->GetGraphicWindow().Invalidate(); } if ( bIsEnabled ) EnableSetModified( bIsEnabled ); SetModified(); // launch accessible event if necessary SmGraphicAccessible *pAcc = pViewSh ? pViewSh->GetGraphicWindow().GetAccessible_Impl() : nullptr; if (pAcc) { Any aOldValue, aNewValue; if ( comphelper::OCommonAccessibleText::implInitTextChangedEvent( maText, rBuffer, aOldValue, aNewValue ) ) { pAcc->LaunchEvent( AccessibleEventId::TEXT_CHANGED, aOldValue, aNewValue ); } } if ( GetCreateMode() == SfxObjectCreateMode::EMBEDDED ) OnDocumentPrinterChanged(nullptr); } } void SmDocShell::SetFormat(SmFormat& rFormat) { maFormat = rFormat; SetFormulaArranged( false ); SetModified(); mnModifyCount++; //! see comment for SID_GAPHIC_SM in SmDocShell::GetState // don't use SmGetActiveView since the view shell might not be active (0 pointer) // if for example the Basic Macro dialog currently has the focus. Thus: SfxViewFrame* pFrm = SfxViewFrame::GetFirst( this ); while (pFrm) { pFrm->GetBindings().Invalidate(SID_GAPHIC_SM); pFrm = SfxViewFrame::GetNext( *pFrm, this ); } } OUString const & SmDocShell::GetAccessibleText() { ArrangeFormula(); if (maAccText.isEmpty()) { OSL_ENSURE( mpTree, "Tree missing" ); if (mpTree) { OUStringBuffer aBuf; mpTree->GetAccessibleText(aBuf); maAccText = aBuf.makeStringAndClear(); } } return maAccText; } void SmDocShell::Parse() { delete mpTree; ReplaceBadChars(); mpTree = maParser.Parse(maText); mnModifyCount++; //! see comment for SID_GAPHIC_SM in SmDocShell::GetState SetFormulaArranged( false ); InvalidateCursor(); maUsedSymbols = maParser.GetUsedSymbols(); } void SmDocShell::ArrangeFormula() { if (mbFormulaArranged) return; // Only for the duration of the existence of this object the correct settings // at the printer are guaranteed! SmPrinterAccess aPrtAcc(*this); OutputDevice* pOutDev = aPrtAcc.GetRefDev(); SAL_WARN_IF( !pOutDev, "starmath", "!! SmDocShell::ArrangeFormula: reference device missing !!"); // if necessary get another OutputDevice for which we format if (!pOutDev) { SmViewShell *pView = SmGetActiveView(); if (pView) pOutDev = &pView->GetGraphicWindow(); else { pOutDev = &SM_MOD()->GetDefaultVirtualDev(); pOutDev->SetMapMode( MapMode(MAP_100TH_MM) ); } } OSL_ENSURE(pOutDev->GetMapMode().GetMapUnit() == MAP_100TH_MM, "Sm : falscher MapMode"); const SmFormat &rFormat = GetFormat(); mpTree->Prepare(rFormat, *this); // format/draw formulas always from left to right, // and numbers should not be converted ComplexTextLayoutFlags nLayoutMode = pOutDev->GetLayoutMode(); pOutDev->SetLayoutMode( ComplexTextLayoutFlags::Default ); sal_Int16 nDigitLang = pOutDev->GetDigitLanguage(); pOutDev->SetDigitLanguage( LANGUAGE_ENGLISH ); mpTree->Arrange(*pOutDev, rFormat); pOutDev->SetLayoutMode( nLayoutMode ); pOutDev->SetDigitLanguage( nDigitLang ); SetFormulaArranged(true); // invalidate accessible text maAccText.clear(); } void SetEditEngineDefaultFonts(SfxItemPool &rEditEngineItemPool) { // set fonts to be used SvtLinguOptions aOpt; SvtLinguConfig().GetOptions( aOpt ); struct FontDta { sal_Int16 nFallbackLang; sal_Int16 nLang; DefaultFontType nFontType; sal_uInt16 nFontInfoId; } aTable[3] = { // info to get western font to be used { LANGUAGE_ENGLISH_US, LANGUAGE_NONE, DefaultFontType::FIXED, EE_CHAR_FONTINFO }, // info to get CJK font to be used { LANGUAGE_JAPANESE, LANGUAGE_NONE, DefaultFontType::CJK_TEXT, EE_CHAR_FONTINFO_CJK }, // info to get CTL font to be used { LANGUAGE_ARABIC_SAUDI_ARABIA, LANGUAGE_NONE, DefaultFontType::CTL_TEXT, EE_CHAR_FONTINFO_CTL } }; aTable[0].nLang = aOpt.nDefaultLanguage; aTable[1].nLang = aOpt.nDefaultLanguage_CJK; aTable[2].nLang = aOpt.nDefaultLanguage_CTL; for (FontDta & rFntDta : aTable) { LanguageType nLang = (LANGUAGE_NONE == rFntDta.nLang) ? rFntDta.nFallbackLang : rFntDta.nLang; vcl::Font aFont = OutputDevice::GetDefaultFont( rFntDta.nFontType, nLang, GetDefaultFontFlags::OnlyOne ); rEditEngineItemPool.SetPoolDefaultItem( SvxFontItem( aFont.GetFamilyType(), aFont.GetFamilyName(), aFont.GetStyleName(), aFont.GetPitch(), aFont.GetCharSet(), rFntDta.nFontInfoId ) ); } // set font heights SvxFontHeightItem aFontHeigt( Application::GetDefaultDevice()->LogicToPixel( Size( 0, 11 ), MapMode( MAP_POINT ) ).Height(), 100, EE_CHAR_FONTHEIGHT ); rEditEngineItemPool.SetPoolDefaultItem( aFontHeigt ); aFontHeigt.SetWhich( EE_CHAR_FONTHEIGHT_CJK ); rEditEngineItemPool.SetPoolDefaultItem( aFontHeigt ); aFontHeigt.SetWhich( EE_CHAR_FONTHEIGHT_CTL ); rEditEngineItemPool.SetPoolDefaultItem( aFontHeigt ); } EditEngine& SmDocShell::GetEditEngine() { if (!mpEditEngine) { //! //! see also SmEditWindow::DataChanged ! //! mpEditEngineItemPool = EditEngine::CreatePool(); SetEditEngineDefaultFonts(*mpEditEngineItemPool); mpEditEngine = new EditEngine( mpEditEngineItemPool ); mpEditEngine->EnableUndo( true ); mpEditEngine->SetDefTab( sal_uInt16( Application::GetDefaultDevice()->GetTextWidth("XXXX")) ); mpEditEngine->SetControlWord( (mpEditEngine->GetControlWord() | EEControlBits::AUTOINDENTING) & EEControlBits(~EEControlBits::UNDOATTRIBS) & EEControlBits(~EEControlBits::PASTESPECIAL) ); mpEditEngine->SetWordDelimiters(" .=+-*/(){}[];\""); mpEditEngine->SetRefMapMode( MAP_PIXEL ); mpEditEngine->SetPaperSize( Size( 800, 0 ) ); mpEditEngine->EraseVirtualDevice(); // set initial text if the document already has some... // (may be the case when reloading a doc) OUString aTxt( GetText() ); if (!aTxt.isEmpty()) mpEditEngine->SetText( aTxt ); mpEditEngine->ClearModifyFlag(); } return *mpEditEngine; } SfxItemPool& SmDocShell::GetEditEngineItemPool() { if (!mpEditEngineItemPool) GetEditEngine(); assert(mpEditEngineItemPool && "EditEngineItemPool missing"); return *mpEditEngineItemPool; } void SmDocShell::DrawFormula(OutputDevice &rDev, Point &rPosition, bool bDrawSelection) { if (!mpTree) Parse(); OSL_ENSURE(mpTree, "Sm : NULL pointer"); ArrangeFormula(); // Problem: What happens to WYSIWYG? While we're active inplace, we don't have a reference // device and aren't aligned to that either. So now there can be a difference between the // VisArea (i.e. the size within the client) and the current size. // Idea: The difference could be adapted with SmNod::SetSize (no long-term solution) rPosition.X() += maFormat.GetDistance( DIS_LEFTSPACE ); rPosition.Y() += maFormat.GetDistance( DIS_TOPSPACE ); //! in case of high contrast-mode (accessibility option!) //! the draw mode needs to be set to default, because when imbedding //! Math for example in Calc in "a over b" the fraction bar may not //! be visible else. More generally: the FillColor may have been changed. DrawModeFlags nOldDrawMode = DrawModeFlags::Default; bool bRestoreDrawMode = false; if (OUTDEV_WINDOW == rDev.GetOutDevType() && static_cast(rDev).GetSettings().GetStyleSettings().GetHighContrastMode()) { nOldDrawMode = rDev.GetDrawMode(); rDev.SetDrawMode( DrawModeFlags::Default ); bRestoreDrawMode = true; } // format/draw formulas always from left to right // and numbers should not be converted ComplexTextLayoutFlags nLayoutMode = rDev.GetLayoutMode(); rDev.SetLayoutMode( ComplexTextLayoutFlags::Default ); sal_Int16 nDigitLang = rDev.GetDigitLanguage(); rDev.SetDigitLanguage( LANGUAGE_ENGLISH ); //Set selection if any if(mpCursor && bDrawSelection){ mpCursor->AnnotateSelection(); SmSelectionDrawingVisitor(rDev, mpTree, rPosition); } //Drawing using visitor SmDrawingVisitor(rDev, rPosition, mpTree); rDev.SetLayoutMode( nLayoutMode ); rDev.SetDigitLanguage( nDigitLang ); if (bRestoreDrawMode) rDev.SetDrawMode( nOldDrawMode ); } Size SmDocShell::GetSize() { Size aRet; if (!mpTree) Parse(); if (mpTree) { ArrangeFormula(); aRet = mpTree->GetSize(); if ( !aRet.Width() ) aRet.Width() = 2000; else aRet.Width() += maFormat.GetDistance( DIS_LEFTSPACE ) + maFormat.GetDistance( DIS_RIGHTSPACE ); if ( !aRet.Height() ) aRet.Height() = 1000; else aRet.Height() += maFormat.GetDistance( DIS_TOPSPACE ) + maFormat.GetDistance( DIS_BOTTOMSPACE ); } return aRet; } void SmDocShell::InvalidateCursor(){ mpCursor.reset(); } SmCursor& SmDocShell::GetCursor(){ if(!mpCursor) mpCursor.reset(new SmCursor(mpTree, this)); return *mpCursor; } bool SmDocShell::HasCursor() { return mpCursor.get() != nullptr; } SmPrinterAccess::SmPrinterAccess( SmDocShell &rDocShell ) { pPrinter = rDocShell.GetPrt(); if ( pPrinter ) { pPrinter->Push( PushFlags::MAPMODE ); if ( SfxObjectCreateMode::EMBEDDED == rDocShell.GetCreateMode() ) { // if it is an embedded object (without it's own printer) // we change the MapMode temporarily. //!If it is a document with its own printer the MapMode should //!be set correct (once) elsewhere(!), in order to avoid numerous //!superfluous pushing and poping of the MapMode when using //!this class. const MapUnit eOld = pPrinter->GetMapMode().GetMapUnit(); if ( MAP_100TH_MM != eOld ) { MapMode aMap( pPrinter->GetMapMode() ); aMap.SetMapUnit( MAP_100TH_MM ); Point aTmp( aMap.GetOrigin() ); aTmp.X() = OutputDevice::LogicToLogic( aTmp.X(), eOld, MAP_100TH_MM ); aTmp.Y() = OutputDevice::LogicToLogic( aTmp.Y(), eOld, MAP_100TH_MM ); aMap.SetOrigin( aTmp ); pPrinter->SetMapMode( aMap ); } } } if ( !!(pRefDev = rDocShell.GetRefDev()) && pPrinter.get() != pRefDev.get() ) { pRefDev->Push( PushFlags::MAPMODE ); if ( SfxObjectCreateMode::EMBEDDED == rDocShell.GetCreateMode() ) { // if it is an embedded object (without it's own printer) // we change the MapMode temporarily. //!If it is a document with its own printer the MapMode should //!be set correct (once) elsewhere(!), in order to avoid numerous //!superfluous pushing and poping of the MapMode when using //!this class. const MapUnit eOld = pRefDev->GetMapMode().GetMapUnit(); if ( MAP_100TH_MM != eOld ) { MapMode aMap( pRefDev->GetMapMode() ); aMap.SetMapUnit( MAP_100TH_MM ); Point aTmp( aMap.GetOrigin() ); aTmp.X() = OutputDevice::LogicToLogic( aTmp.X(), eOld, MAP_100TH_MM ); aTmp.Y() = OutputDevice::LogicToLogic( aTmp.Y(), eOld, MAP_100TH_MM ); aMap.SetOrigin( aTmp ); pRefDev->SetMapMode( aMap ); } } } } SmPrinterAccess::~SmPrinterAccess() { if ( pPrinter ) pPrinter->Pop(); if ( pRefDev && pRefDev != pPrinter ) pRefDev->Pop(); } Printer* SmDocShell::GetPrt() { if (SfxObjectCreateMode::EMBEDDED == GetCreateMode()) { // Normally the server provides the printer. But if it doesn't provide one (e.g. because // there is no connection) it still can be the case that we know the printer because it // has been passed on by the server in OnDocumentPrinterChanged and being kept temporarily. Printer* pPrt = GetDocumentPrinter(); if (!pPrt && mpTmpPrinter) pPrt = mpTmpPrinter; return pPrt; } else if (!mpPrinter) { SfxItemSet* pOptions = new SfxItemSet(GetPool(), SID_PRINTSIZE, SID_PRINTSIZE, SID_PRINTZOOM, SID_PRINTZOOM, SID_PRINTTITLE, SID_PRINTTITLE, SID_PRINTTEXT, SID_PRINTTEXT, SID_PRINTFRAME, SID_PRINTFRAME, SID_NO_RIGHT_SPACES, SID_NO_RIGHT_SPACES, SID_SAVE_ONLY_USED_SYMBOLS, SID_SAVE_ONLY_USED_SYMBOLS, SID_AUTO_CLOSE_BRACKETS, SID_AUTO_CLOSE_BRACKETS, 0); SmModule *pp = SM_MOD(); pp->GetConfig()->ConfigToItemSet(*pOptions); mpPrinter = VclPtr::Create(pOptions); mpPrinter->SetMapMode(MapMode(MAP_100TH_MM)); } return mpPrinter; } OutputDevice* SmDocShell::GetRefDev() { if (SfxObjectCreateMode::EMBEDDED == GetCreateMode()) { OutputDevice* pOutDev = GetDocumentRefDev(); if (pOutDev) return pOutDev; } return GetPrt(); } void SmDocShell::SetPrinter( SfxPrinter *pNew ) { mpPrinter.disposeAndClear(); mpPrinter = pNew; //Transfer ownership mpPrinter->SetMapMode( MapMode(MAP_100TH_MM) ); SetFormulaArranged(false); Repaint(); } void SmDocShell::OnDocumentPrinterChanged( Printer *pPrt ) { mpTmpPrinter = pPrt; SetFormulaArranged(false); Size aOldSize = GetVisArea().GetSize(); Repaint(); if( aOldSize != GetVisArea().GetSize() && !maText.isEmpty() ) SetModified(); mpTmpPrinter = nullptr; } void SmDocShell::Repaint() { bool bIsEnabled = IsEnableSetModified(); if (bIsEnabled) EnableSetModified( false ); SetFormulaArranged(false); Size aVisSize = GetSize(); SetVisAreaSize(aVisSize); SmViewShell* pViewSh = SmGetActiveView(); if (pViewSh) pViewSh->GetGraphicWindow().Invalidate(); if (bIsEnabled) EnableSetModified(bIsEnabled); } SmDocShell::SmDocShell( SfxModelFlags i_nSfxCreationFlags ) : SfxObjectShell(i_nSfxCreationFlags) , mpTree(nullptr) , mpEditEngineItemPool(nullptr) , mpEditEngine(nullptr) , mpPrinter(nullptr) , mpTmpPrinter(nullptr) , mnModifyCount(0) , mbFormulaArranged(false) { SetPool(&SfxGetpApp()->GetPool()); SmModule *pp = SM_MOD(); maFormat = pp->GetConfig()->GetStandardFormat(); StartListening(maFormat); StartListening(*pp->GetConfig()); SetBaseModel(new SmModel(this)); } SmDocShell::~SmDocShell() { SmModule *pp = SM_MOD(); EndListening(maFormat); EndListening(*pp->GetConfig()); mpCursor.reset(); delete mpEditEngine; SfxItemPool::Free(mpEditEngineItemPool); delete mpTree; mpPrinter.disposeAndClear(); } bool SmDocShell::ConvertFrom(SfxMedium &rMedium) { bool bSuccess = false; const OUString& rFltName = rMedium.GetFilter()->GetFilterName(); OSL_ENSURE( rFltName != STAROFFICE_XML, "Wrong filter!"); if ( rFltName == MATHML_XML ) { if (mpTree) { delete mpTree; mpTree = nullptr; InvalidateCursor(); } Reference xModel(GetModel()); SmXMLImportWrapper aEquation(xModel); bSuccess = ( ERRCODE_NONE == aEquation.Import(rMedium) ); } else { SvStream *pStream = rMedium.GetInStream(); if ( pStream ) { if ( SotStorage::IsStorageFile( pStream ) ) { tools::SvRef aStorage = new SotStorage( pStream, false ); if ( aStorage->IsStream("Equation Native") ) { // is this a MathType Storage? MathType aEquation( maText ); bSuccess = aEquation.Parse( aStorage ); if ( bSuccess ) Parse(); } } } } if ( GetCreateMode() == SfxObjectCreateMode::EMBEDDED ) { SetFormulaArranged( false ); Repaint(); } FinishedLoading(); return bSuccess; } bool SmDocShell::InitNew( const uno::Reference < embed::XStorage >& xStorage ) { bool bRet = false; if ( SfxObjectShell::InitNew( xStorage ) ) { bRet = true; SetVisArea(Rectangle(Point(0, 0), Size(2000, 1000))); } return bRet; } bool SmDocShell::Load( SfxMedium& rMedium ) { bool bRet = false; if( SfxObjectShell::Load( rMedium )) { uno::Reference < embed::XStorage > xStorage = GetMedium()->GetStorage(); uno::Reference < container::XNameAccess > xAccess (xStorage, uno::UNO_QUERY); if ( ( xAccess->hasByName( "content.xml" ) && xStorage->isStreamElement( "content.xml" ) ) || ( xAccess->hasByName( "Content.xml" ) && xStorage->isStreamElement( "Content.xml" ) ) ) { // is this a fabulous math package ? Reference xModel(GetModel()); SmXMLImportWrapper aEquation(xModel); sal_uLong nError = aEquation.Import(rMedium); bRet = 0 == nError; SetError( nError, OSL_LOG_PREFIX ); } } if ( GetCreateMode() == SfxObjectCreateMode::EMBEDDED ) { SetFormulaArranged( false ); Repaint(); } FinishedLoading(); return bRet; } bool SmDocShell::Save() { //! apply latest changes if necessary UpdateText(); if ( SfxObjectShell::Save() ) { if (!mpTree) Parse(); if( mpTree ) ArrangeFormula(); Reference xModel(GetModel()); SmXMLExportWrapper aEquation(xModel); aEquation.SetFlat(false); return aEquation.Export(*GetMedium()); } return false; } /* * replace bad characters that can not be saved. (#i74144) * */ void SmDocShell::ReplaceBadChars() { bool bReplace = false; if (mpEditEngine) { OUStringBuffer aBuf( mpEditEngine->GetText() ); for (sal_Int32 i = 0; i < aBuf.getLength(); ++i) { if (aBuf[i] < ' ' && aBuf[i] != '\r' && aBuf[i] != '\n' && aBuf[i] != '\t') { aBuf[i] = ' '; bReplace = true; } } if (bReplace) maText = aBuf.makeStringAndClear(); } } void SmDocShell::UpdateText() { if (mpEditEngine && mpEditEngine->IsModified()) { OUString aEngTxt( mpEditEngine->GetText() ); if (GetText() != aEngTxt) SetText( aEngTxt ); } } bool SmDocShell::SaveAs( SfxMedium& rMedium ) { bool bRet = false; //! apply latest changes if necessary UpdateText(); if ( SfxObjectShell::SaveAs( rMedium ) ) { if (!mpTree) Parse(); if( mpTree ) ArrangeFormula(); Reference xModel(GetModel()); SmXMLExportWrapper aEquation(xModel); aEquation.SetFlat(false); bRet = aEquation.Export(rMedium); } return bRet; } bool SmDocShell::ConvertTo( SfxMedium &rMedium ) { bool bRet = false; std::shared_ptr pFlt = rMedium.GetFilter(); if( pFlt ) { if( !mpTree ) Parse(); if( mpTree ) ArrangeFormula(); const OUString& rFltName = pFlt->GetFilterName(); if(rFltName == STAROFFICE_XML) { Reference xModel(GetModel()); SmXMLExportWrapper aEquation(xModel); aEquation.SetFlat(false); bRet = aEquation.Export(rMedium); } else if(rFltName == MATHML_XML) { Reference xModel(GetModel()); SmXMLExportWrapper aEquation(xModel); aEquation.SetFlat(true); bRet = aEquation.Export(rMedium); } else if (pFlt->GetFilterName() == "MathType 3.x") bRet = WriteAsMathType3( rMedium ); } return bRet; } bool SmDocShell::writeFormulaOoxml( ::sax_fastparser::FSHelperPtr const& pSerializer, oox::core::OoxmlVersion const version, oox::drawingml::DocumentType const documentType) { if( !mpTree ) Parse(); if( mpTree ) ArrangeFormula(); SmOoxmlExport aEquation(mpTree, version, documentType); return aEquation.ConvertFromStarMath( pSerializer ); } void SmDocShell::writeFormulaRtf(OStringBuffer& rBuffer, rtl_TextEncoding nEncoding) { if (!mpTree) Parse(); if (mpTree) ArrangeFormula(); SmRtfExport aEquation(mpTree); aEquation.ConvertFromStarMath(rBuffer, nEncoding); } void SmDocShell::readFormulaOoxml( oox::formulaimport::XmlStream& stream ) { SmOoxmlImport aEquation( stream ); SetText( aEquation.ConvertToStarMath()); } bool SmDocShell::SaveCompleted( const css::uno::Reference< css::embed::XStorage >& xStorage ) { if( SfxObjectShell::SaveCompleted( xStorage )) return true; return false; } void SmDocShell::Execute(SfxRequest& rReq) { switch (rReq.GetSlot()) { case SID_TEXTMODE: { SmFormat aOldFormat = GetFormat(); SmFormat aNewFormat( aOldFormat ); aNewFormat.SetTextmode(!aOldFormat.IsTextmode()); ::svl::IUndoManager *pTmpUndoMgr = GetUndoManager(); if (pTmpUndoMgr) pTmpUndoMgr->AddUndoAction( new SmFormatAction(this, aOldFormat, aNewFormat)); SetFormat( aNewFormat ); Repaint(); } break; case SID_AUTO_REDRAW : { SmModule *pp = SM_MOD(); bool bRedraw = pp->GetConfig()->IsAutoRedraw(); pp->GetConfig()->SetAutoRedraw(!bRedraw); } break; case SID_LOADSYMBOLS: LoadSymbols(); break; case SID_SAVESYMBOLS: SaveSymbols(); break; case SID_FONT: { // get device used to retrieve the FontList OutputDevice *pDev = GetPrinter(); if (!pDev || pDev->GetDevFontCount() == 0) pDev = &SM_MOD()->GetDefaultVirtualDev(); OSL_ENSURE (pDev, "device for font list missing" ); VclPtrInstance< SmFontTypeDialog > xFontTypeDialog( nullptr, pDev ); SmFormat aOldFormat = GetFormat(); xFontTypeDialog->ReadFrom( aOldFormat ); if (xFontTypeDialog->Execute() == RET_OK) { SmFormat aNewFormat( aOldFormat ); xFontTypeDialog->WriteTo(aNewFormat); ::svl::IUndoManager *pTmpUndoMgr = GetUndoManager(); if (pTmpUndoMgr) pTmpUndoMgr->AddUndoAction( new SmFormatAction(this, aOldFormat, aNewFormat)); SetFormat( aNewFormat ); Repaint(); } } break; case SID_FONTSIZE: { VclPtrInstance< SmFontSizeDialog > xFontSizeDialog(nullptr); SmFormat aOldFormat = GetFormat(); xFontSizeDialog->ReadFrom( aOldFormat ); if (xFontSizeDialog->Execute() == RET_OK) { SmFormat aNewFormat( aOldFormat ); xFontSizeDialog->WriteTo(aNewFormat); ::svl::IUndoManager *pTmpUndoMgr = GetUndoManager(); if (pTmpUndoMgr) pTmpUndoMgr->AddUndoAction( new SmFormatAction(this, aOldFormat, aNewFormat)); SetFormat( aNewFormat ); Repaint(); } } break; case SID_DISTANCE: { VclPtrInstance< SmDistanceDialog > xDistanceDialog(nullptr); SmFormat aOldFormat = GetFormat(); xDistanceDialog->ReadFrom( aOldFormat ); if (xDistanceDialog->Execute() == RET_OK) { SmFormat aNewFormat( aOldFormat ); xDistanceDialog->WriteTo(aNewFormat); ::svl::IUndoManager *pTmpUndoMgr = GetUndoManager(); if (pTmpUndoMgr) pTmpUndoMgr->AddUndoAction( new SmFormatAction(this, aOldFormat, aNewFormat)); SetFormat( aNewFormat ); Repaint(); } } break; case SID_ALIGN: { VclPtrInstance< SmAlignDialog > xAlignDialog(nullptr); SmFormat aOldFormat = GetFormat(); xAlignDialog->ReadFrom( aOldFormat ); if (xAlignDialog->Execute() == RET_OK) { SmFormat aNewFormat( aOldFormat ); xAlignDialog->WriteTo(aNewFormat); SmModule *pp = SM_MOD(); SmFormat aFmt( pp->GetConfig()->GetStandardFormat() ); xAlignDialog->WriteTo( aFmt ); pp->GetConfig()->SetStandardFormat( aFmt ); ::svl::IUndoManager *pTmpUndoMgr = GetUndoManager(); if (pTmpUndoMgr) pTmpUndoMgr->AddUndoAction( new SmFormatAction(this, aOldFormat, aNewFormat)); SetFormat( aNewFormat ); Repaint(); } } break; case SID_TEXT: { const SfxStringItem& rItem = static_cast(rReq.GetArgs()->Get(SID_TEXT)); if (GetText() != OUString(rItem.GetValue())) SetText(rItem.GetValue()); } break; case SID_UNDO: case SID_REDO: { ::svl::IUndoManager* pTmpUndoMgr = GetUndoManager(); if( pTmpUndoMgr ) { sal_uInt16 nId = rReq.GetSlot(), nCnt = 1; const SfxItemSet* pArgs = rReq.GetArgs(); const SfxPoolItem* pItem; if( pArgs && SfxItemState::SET == pArgs->GetItemState( nId, false, &pItem )) nCnt = static_cast(pItem)->GetValue(); bool (::svl::IUndoManager:: *fnDo)(); std::size_t nCount; if( SID_UNDO == rReq.GetSlot() ) { nCount = pTmpUndoMgr->GetUndoActionCount(); fnDo = &::svl::IUndoManager::Undo; } else { nCount = pTmpUndoMgr->GetRedoActionCount(); fnDo = &::svl::IUndoManager::Redo; } try { for( ; nCnt && nCount; --nCnt, --nCount ) (pTmpUndoMgr->*fnDo)(); } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } } Repaint(); UpdateText(); SfxViewFrame* pFrm = SfxViewFrame::GetFirst( this ); while( pFrm ) { SfxBindings& rBind = pFrm->GetBindings(); rBind.Invalidate(SID_UNDO); rBind.Invalidate(SID_REDO); rBind.Invalidate(SID_REPEAT); rBind.Invalidate(SID_CLEARHISTORY); pFrm = SfxViewFrame::GetNext( *pFrm, this ); } } break; } rReq.Done(); } void SmDocShell::GetState(SfxItemSet &rSet) { SfxWhichIter aIter(rSet); for (sal_uInt16 nWh = aIter.FirstWhich(); 0 != nWh; nWh = aIter.NextWhich()) { switch (nWh) { case SID_TEXTMODE: rSet.Put(SfxBoolItem(SID_TEXTMODE, GetFormat().IsTextmode())); break; case SID_DOCTEMPLATE : rSet.DisableItem(SID_DOCTEMPLATE); break; case SID_AUTO_REDRAW : { SmModule *pp = SM_MOD(); bool bRedraw = pp->GetConfig()->IsAutoRedraw(); rSet.Put(SfxBoolItem(SID_AUTO_REDRAW, bRedraw)); } break; case SID_MODIFYSTATUS: { sal_Unicode cMod = ' '; if (IsModified()) cMod = '*'; rSet.Put(SfxStringItem(SID_MODIFYSTATUS, OUString(cMod))); } break; case SID_TEXT: rSet.Put(SfxStringItem(SID_TEXT, GetText())); break; case SID_GAPHIC_SM: //! very old (pre UNO) and ugly hack to invalidate the SmGraphicWindow. //! If mnModifyCount gets changed then the call below will implicitly notify //! SmGraphicController::StateChanged and there the window gets invalidated. //! Thus all the 'mnModifyCount++' before invalidating this slot. rSet.Put(SfxInt16Item(SID_GAPHIC_SM, mnModifyCount)); break; case SID_UNDO: case SID_REDO: { SfxViewFrame* pFrm = SfxViewFrame::GetFirst( this ); if( pFrm ) pFrm->GetSlotState( nWh, nullptr, &rSet ); else rSet.DisableItem( nWh ); } break; case SID_GETUNDOSTRINGS: case SID_GETREDOSTRINGS: { ::svl::IUndoManager* pTmpUndoMgr = GetUndoManager(); if( pTmpUndoMgr ) { OUString(::svl::IUndoManager:: *fnGetComment)( size_t, bool const ) const; sal_uInt16 nCount; if( SID_GETUNDOSTRINGS == nWh ) { nCount = pTmpUndoMgr->GetUndoActionCount(); fnGetComment = &::svl::IUndoManager::GetUndoActionComment; } else { nCount = pTmpUndoMgr->GetRedoActionCount(); fnGetComment = &::svl::IUndoManager::GetRedoActionComment; } if( nCount ) { OUStringBuffer aBuf; for( sal_uInt16 n = 0; n < nCount; ++n ) { aBuf.append((pTmpUndoMgr->*fnGetComment)( n, ::svl::IUndoManager::TopLevel )); aBuf.append('\n'); } SfxStringListItem aItem( nWh ); aItem.SetString( aBuf.makeStringAndClear() ); rSet.Put( aItem ); } } else rSet.DisableItem( nWh ); } break; } } } ::svl::IUndoManager *SmDocShell::GetUndoManager() { if (!mpEditEngine) GetEditEngine(); return &mpEditEngine->GetUndoManager(); } void SmDocShell::SaveSymbols() { SmModule *pp = SM_MOD(); pp->GetSymbolManager().Save(); } void SmDocShell::Draw(OutputDevice *pDevice, const JobSetup &, sal_uInt16 /*nAspect*/) { pDevice->IntersectClipRegion(GetVisArea()); Point atmppoint; DrawFormula(*pDevice, atmppoint); } SfxItemPool& SmDocShell::GetPool() { return SfxGetpApp()->GetPool(); } void SmDocShell::SetVisArea(const Rectangle & rVisArea) { Rectangle aNewRect(rVisArea); aNewRect.SetPos(Point()); if (! aNewRect.Right()) aNewRect.Right() = 2000; if (! aNewRect.Bottom()) aNewRect.Bottom() = 1000; bool bIsEnabled = IsEnableSetModified(); if ( bIsEnabled ) EnableSetModified( false ); //TODO/LATER: it's unclear how this interacts with the SFX code // If outplace editing, then don't resize the OutplaceWindow. But the // ObjectShell has to resize. Bug 56470 bool bUnLockFrame; if( GetCreateMode() == SfxObjectCreateMode::EMBEDDED && !IsInPlaceActive() && GetFrame() ) { GetFrame()->LockAdjustPosSizePixel(); bUnLockFrame = true; } else bUnLockFrame = false; SfxObjectShell::SetVisArea( aNewRect ); if( bUnLockFrame ) GetFrame()->UnlockAdjustPosSizePixel(); if ( bIsEnabled ) EnableSetModified( bIsEnabled ); } void SmDocShell::FillClass(SvGlobalName* pClassName, SotClipboardFormatId* pFormat, OUString* /*pAppName*/, OUString* pFullTypeName, OUString* pShortTypeName, sal_Int32 nFileFormat, bool bTemplate /* = false */) const { if (nFileFormat == SOFFICE_FILEFORMAT_60 ) { *pClassName = SvGlobalName(SO3_SM_CLASSID_60); *pFormat = SotClipboardFormatId::STARMATH_60; *pFullTypeName = SM_RESSTR(STR_MATH_DOCUMENT_FULLTYPE_CURRENT); *pShortTypeName = SM_RESSTR(RID_DOCUMENTSTR); } else if (nFileFormat == SOFFICE_FILEFORMAT_8 ) { *pClassName = SvGlobalName(SO3_SM_CLASSID_60); *pFormat = bTemplate ? SotClipboardFormatId::STARMATH_8_TEMPLATE : SotClipboardFormatId::STARMATH_8; *pFullTypeName = SM_RESSTR(STR_MATH_DOCUMENT_FULLTYPE_CURRENT); *pShortTypeName = SM_RESSTR(RID_DOCUMENTSTR); } } sal_uLong SmDocShell::GetMiscStatus() const { return SfxObjectShell::GetMiscStatus() | SVOBJ_MISCSTATUS_NOTRESIZEABLE | SVOBJ_MISCSTATUS_RESIZEONPRINTERCHANGE; } void SmDocShell::SetModified(bool bModified) { if( IsEnableSetModified() ) { SfxObjectShell::SetModified( bModified ); Broadcast(SfxSimpleHint(SFX_HINT_DOCCHANGED)); } } bool SmDocShell::WriteAsMathType3( SfxMedium& rMedium ) { MathType aEquation( maText, mpTree ); return aEquation.ConvertFromStarMath( rMedium ); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */