/* -*- 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 namespace framework{ //***************************************************************************************************************** // XInterface, XTypeProvider DEFINE_XINTERFACE_4(PersistentWindowState , OWeakObject , DIRECT_INTERFACE (css::lang::XTypeProvider ), DIRECT_INTERFACE (css::lang::XInitialization ), DIRECT_INTERFACE (css::frame::XFrameActionListener ), DERIVED_INTERFACE(css::lang::XEventListener,css::frame::XFrameActionListener)) DEFINE_XTYPEPROVIDER_4(PersistentWindowState , css::lang::XTypeProvider , css::lang::XInitialization , css::frame::XFrameActionListener, css::lang::XEventListener ) //***************************************************************************************************************** PersistentWindowState::PersistentWindowState(const css::uno::Reference< css::lang::XMultiServiceFactory >& xSMGR) : ThreadHelpBase (&Application::GetSolarMutex()) , m_xSMGR (xSMGR ) , m_bWindowStateAlreadySet(sal_False ) { } //***************************************************************************************************************** PersistentWindowState::~PersistentWindowState() { } //***************************************************************************************************************** void SAL_CALL PersistentWindowState::initialize(const css::uno::Sequence< css::uno::Any >& lArguments) throw(css::uno::Exception , css::uno::RuntimeException) { // check arguments css::uno::Reference< css::frame::XFrame > xFrame; if (lArguments.getLength() < 1) throw css::lang::IllegalArgumentException( DECLARE_ASCII("Empty argument list!"), static_cast< ::cppu::OWeakObject* >(this), 1); lArguments[0] >>= xFrame; if (!xFrame.is()) throw css::lang::IllegalArgumentException( DECLARE_ASCII("No valid frame specified!"), static_cast< ::cppu::OWeakObject* >(this), 1); // SAFE -> ---------------------------------- WriteGuard aWriteLock(m_aLock); // hold the frame as weak reference(!) so it can die everytimes :-) m_xFrame = xFrame; aWriteLock.unlock(); // <- SAFE ---------------------------------- // start listening xFrame->addFrameActionListener(this); } //***************************************************************************************************************** void SAL_CALL PersistentWindowState::frameAction(const css::frame::FrameActionEvent& aEvent) throw(css::uno::RuntimeException) { // SAFE -> ---------------------------------- ReadGuard aReadLock(m_aLock); css::uno::Reference< css::uno::XComponentContext > xContext = comphelper::getComponentContext(m_xSMGR); css::uno::Reference< css::frame::XFrame > xFrame(m_xFrame.get(), css::uno::UNO_QUERY); sal_Bool bRestoreWindowState = !m_bWindowStateAlreadySet; aReadLock.unlock(); // <- SAFE ---------------------------------- // frame already gone ? We hold it weak only ... if (!xFrame.is()) return; // no window -> no position and size available css::uno::Reference< css::awt::XWindow > xWindow = xFrame->getContainerWindow(); if (!xWindow.is()) return; // unknown module -> no configuration available! ::rtl::OUString sModuleName = PersistentWindowState::implst_identifyModule(xContext, xFrame); if (sModuleName.isEmpty()) return; switch(aEvent.Action) { case css::frame::FrameAction_COMPONENT_ATTACHED : { if (bRestoreWindowState) { ::rtl::OUString sWindowState = PersistentWindowState::implst_getWindowStateFromConfig(xContext, sModuleName); PersistentWindowState::implst_setWindowStateOnWindow(xWindow,sWindowState); // SAFE -> ---------------------------------- WriteGuard aWriteLock(m_aLock); m_bWindowStateAlreadySet = sal_True; aWriteLock.unlock(); // <- SAFE ---------------------------------- } } break; case css::frame::FrameAction_COMPONENT_REATTACHED : { // nothing todo here, because its not allowed to change position and size // of an alredy existing frame! } break; case css::frame::FrameAction_COMPONENT_DETACHING : { ::rtl::OUString sWindowState = PersistentWindowState::implst_getWindowStateFromWindow(xWindow); PersistentWindowState::implst_setWindowStateOnConfig(xContext, sModuleName, sWindowState); } break; default: break; } } //***************************************************************************************************************** void SAL_CALL PersistentWindowState::disposing(const css::lang::EventObject&) throw(css::uno::RuntimeException) { // nothing todo here - because we hold the frame as weak reference only } //***************************************************************************************************************** ::rtl::OUString PersistentWindowState::implst_identifyModule(const css::uno::Reference< css::uno::XComponentContext >& rxContext, const css::uno::Reference< css::frame::XFrame >& xFrame) { ::rtl::OUString sModuleName; css::uno::Reference< css::frame::XModuleManager2 > xModuleManager = css::frame::ModuleManager::create( rxContext ); try { sModuleName = xModuleManager->identify(xFrame); } catch(const css::uno::RuntimeException&) { throw; } catch(const css::uno::Exception&) { sModuleName = ::rtl::OUString(); } return sModuleName; } //***************************************************************************************************************** ::rtl::OUString PersistentWindowState::implst_getWindowStateFromConfig(const css::uno::Reference< css::uno::XComponentContext >& rxContext, const ::rtl::OUString& sModuleName) { ::rtl::OUString sWindowState; ::rtl::OUStringBuffer sRelPathBuf(256); sRelPathBuf.appendAscii("Office/Factories/*[\""); sRelPathBuf.append (sModuleName ); sRelPathBuf.appendAscii("\"]" ); ::rtl::OUString sPackage("org.openoffice.Setup/"); ::rtl::OUString sRelPath = sRelPathBuf.makeStringAndClear(); ::rtl::OUString sKey("ooSetupFactoryWindowAttributes"); try { ::comphelper::ConfigurationHelper::readDirectKey(rxContext, sPackage, sRelPath, sKey, ::comphelper::ConfigurationHelper::E_READONLY) >>= sWindowState; } catch(const css::uno::RuntimeException&) { throw; } catch(const css::uno::Exception&) { sWindowState = ::rtl::OUString(); } return sWindowState; } //***************************************************************************************************************** void PersistentWindowState::implst_setWindowStateOnConfig(const css::uno::Reference< css::uno::XComponentContext >& rxContext, const ::rtl::OUString& sModuleName , const ::rtl::OUString& sWindowState) { ::rtl::OUStringBuffer sRelPathBuf(256); sRelPathBuf.appendAscii("Office/Factories/*[\""); sRelPathBuf.append (sModuleName ); sRelPathBuf.appendAscii("\"]" ); ::rtl::OUString sPackage("org.openoffice.Setup/"); ::rtl::OUString sRelPath = sRelPathBuf.makeStringAndClear(); ::rtl::OUString sKey("ooSetupFactoryWindowAttributes"); try { ::comphelper::ConfigurationHelper::writeDirectKey(rxContext, sPackage, sRelPath, sKey, css::uno::makeAny(sWindowState), ::comphelper::ConfigurationHelper::E_STANDARD); } catch(const css::uno::RuntimeException&) { throw; } catch(const css::uno::Exception&) {} } //***************************************************************************************************************** ::rtl::OUString PersistentWindowState::implst_getWindowStateFromWindow(const css::uno::Reference< css::awt::XWindow >& xWindow) { ::rtl::OUString sWindowState; if (xWindow.is()) { // SOLAR SAFE -> ------------------------ SolarMutexGuard aSolarGuard; Window* pWindow = VCLUnoHelper::GetWindow(xWindow); // check for system window is neccessary to guarantee correct pointer cast! if ( (pWindow ) && (pWindow->IsSystemWindow()) ) { sal_uLong nMask = WINDOWSTATE_MASK_ALL; nMask &= ~(WINDOWSTATE_MASK_MINIMIZED); sWindowState = rtl::OStringToOUString( ((SystemWindow*)pWindow)->GetWindowState(nMask), RTL_TEXTENCODING_UTF8); } // <- SOLAR SAFE ------------------------ } return sWindowState; } //********************************************************************************************************* void PersistentWindowState::implst_setWindowStateOnWindow(const css::uno::Reference< css::awt::XWindow >& xWindow , const ::rtl::OUString& sWindowState) { if ( (!xWindow.is() ) || ( sWindowState.isEmpty() ) ) return; // SOLAR SAFE -> ------------------------ SolarMutexGuard aSolarGuard; Window* pWindow = VCLUnoHelper::GetWindow(xWindow); if (!pWindow) return; // check for system and work window - its neccessary to guarantee correct pointer cast! sal_Bool bSystemWindow = pWindow->IsSystemWindow(); sal_Bool bWorkWindow = (pWindow->GetType() == WINDOW_WORKWINDOW); if (!bSystemWindow && !bWorkWindow) return; SystemWindow* pSystemWindow = (SystemWindow*)pWindow; WorkWindow* pWorkWindow = (WorkWindow* )pWindow; // dont save this special state! if (pWorkWindow->IsMinimized()) return; ::rtl::OUString sOldWindowState = ::rtl::OStringToOUString( pSystemWindow->GetWindowState(), RTL_TEXTENCODING_ASCII_US ); if ( sOldWindowState != sWindowState ) pSystemWindow->SetWindowState(rtl::OUStringToOString(sWindowState,RTL_TEXTENCODING_UTF8)); // <- SOLAR SAFE ------------------------ } } // namespace framework /* vim:set shiftwidth=4 softtabstop=4 expandtab: */