/* -*- 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 using namespace css; using namespace com::sun::star::uno; using namespace com::sun::star::lang; using namespace com::sun::star::frame; using namespace com::sun::star::beans; using namespace com::sun::star::util; using namespace framework; #define MAX_MENU_ITEMS 99 namespace { static const char CMD_CLEAR_LIST[] = ".uno:ClearRecentFileList"; static const char CMD_OPEN_REMOTE[] = ".uno:OpenRemote"; static const char CMD_PREFIX[] = "vnd.sun.star.popup:RecentFileList?entry="; static const char MENU_SHORTCUT[] = "~N. "; struct LoadRecentFile { util::URL aTargetURL; uno::Sequence< beans::PropertyValue > aArgSeq; uno::Reference< frame::XDispatch > xDispatch; }; class RecentFilesMenuController : public svt::PopupMenuControllerBase { using svt::PopupMenuControllerBase::disposing; public: RecentFilesMenuController( const uno::Reference< uno::XComponentContext >& xContext, const uno::Sequence< uno::Any >& args ); virtual ~RecentFilesMenuController(); // XServiceInfo virtual OUString SAL_CALL getImplementationName() throw (css::uno::RuntimeException, std::exception) override { return OUString("com.sun.star.comp.framework.RecentFilesMenuController"); } virtual sal_Bool SAL_CALL supportsService(OUString const & ServiceName) throw (css::uno::RuntimeException, std::exception) override { return cppu::supportsService(this, ServiceName); } virtual css::uno::Sequence SAL_CALL getSupportedServiceNames() throw (css::uno::RuntimeException, std::exception) override { css::uno::Sequence< OUString > aSeq { "com.sun.star.frame.PopupMenuController" }; return aSeq; } // XStatusListener virtual void SAL_CALL statusChanged( const frame::FeatureStateEvent& Event ) throw ( uno::RuntimeException, std::exception ) override; // XMenuListener virtual void SAL_CALL itemSelected( const awt::MenuEvent& rEvent ) throw (uno::RuntimeException, std::exception) override; virtual void SAL_CALL itemActivated( const awt::MenuEvent& rEvent ) throw (uno::RuntimeException, std::exception) override; // XDispatchProvider virtual uno::Reference< frame::XDispatch > SAL_CALL queryDispatch( const util::URL& aURL, const OUString& sTarget, sal_Int32 nFlags ) throw( uno::RuntimeException, std::exception ) override; // XDispatch virtual void SAL_CALL dispatch( const util::URL& aURL, const uno::Sequence< beans::PropertyValue >& seqProperties ) throw( uno::RuntimeException, std::exception ) override; // XEventListener virtual void SAL_CALL disposing( const css::lang::EventObject& Source ) throw ( uno::RuntimeException, std::exception ) override; DECL_STATIC_LINK_TYPED( RecentFilesMenuController, ExecuteHdl_Impl, void*, void ); private: virtual void impl_setPopupMenu() override; struct RecentFile { OUString aURL; OUString aTitle; }; void fillPopupMenu( css::uno::Reference< css::awt::XPopupMenu >& rPopupMenu ); void executeEntry( sal_Int32 nIndex ); std::vector< RecentFile > m_aRecentFilesItems; bool m_bDisabled : 1; bool m_bShowRemote; }; RecentFilesMenuController::RecentFilesMenuController( const uno::Reference< uno::XComponentContext >& xContext, const uno::Sequence< uno::Any >& args ) : svt::PopupMenuControllerBase( xContext ), m_bDisabled( false ), m_bShowRemote( false ) { css::beans::PropertyValue aPropValue; for ( sal_Int32 i = 0; i < args.getLength(); ++i ) { args[i] >>= aPropValue; if ( aPropValue.Name == "InToolbar" ) { aPropValue.Value >>= m_bShowRemote; break; } } } RecentFilesMenuController::~RecentFilesMenuController() { } // private function void RecentFilesMenuController::fillPopupMenu( Reference< css::awt::XPopupMenu >& rPopupMenu ) { VCLXPopupMenu* pPopupMenu = static_cast(VCLXMenu::GetImplementation( rPopupMenu )); PopupMenu* pVCLPopupMenu = nullptr; SolarMutexGuard aSolarMutexGuard; resetPopupMenu( rPopupMenu ); if ( pPopupMenu ) pVCLPopupMenu = static_cast(pPopupMenu->GetMenu()); if ( pVCLPopupMenu ) { Sequence< Sequence< PropertyValue > > aHistoryList = SvtHistoryOptions().GetList( ePICKLIST ); int nPickListMenuItems = ( aHistoryList.getLength() > MAX_MENU_ITEMS ) ? MAX_MENU_ITEMS : aHistoryList.getLength(); m_aRecentFilesItems.clear(); if (( nPickListMenuItems > 0 ) && !m_bDisabled ) { for ( int i = 0; i < nPickListMenuItems; i++ ) { Sequence< PropertyValue >& rPickListEntry = aHistoryList[i]; RecentFile aRecentFile; for ( int j = 0; j < rPickListEntry.getLength(); j++ ) { Any a = rPickListEntry[j].Value; if ( rPickListEntry[j].Name == HISTORY_PROPERTYNAME_URL ) a >>= aRecentFile.aURL; else if ( rPickListEntry[j].Name == HISTORY_PROPERTYNAME_TITLE ) a >>= aRecentFile.aTitle; } m_aRecentFilesItems.push_back( aRecentFile ); } } if ( !m_aRecentFilesItems.empty() ) { const sal_uInt32 nCount = m_aRecentFilesItems.size(); for ( sal_uInt32 i = 0; i < nCount; i++ ) { OUStringBuffer aMenuShortCut; if ( i <= 9 ) { if ( i == 9 ) aMenuShortCut.append( "1~0. " ); else { aMenuShortCut.append( MENU_SHORTCUT ); aMenuShortCut[ 1 ] = sal_Unicode( i + '1' ); } } else { aMenuShortCut.append( sal_Int32( i + 1 ) ); aMenuShortCut.append( ". " ); } OUStringBuffer aStrBuffer; aStrBuffer.append( CMD_PREFIX ); aStrBuffer.append( sal_Int32( i ) ); OUString aURLString( aStrBuffer.makeStringAndClear() ); // Abbreviate URL OUString aMenuTitle; INetURLObject aURL( m_aRecentFilesItems[i].aURL ); OUString aTipHelpText( aURL.getFSysPath( INetURLObject::FSYS_DETECT ) ); if ( aURL.GetProtocol() == INetProtocol::File ) { // Do handle file URL differently: don't show the protocol, just the file name aMenuTitle = aURL.GetLastName(INetURLObject::DECODE_WITH_CHARSET); } else { // In all other URLs show the protocol name before the file name aMenuTitle = INetURLObject::GetSchemeName(aURL.GetProtocol()) + ": " + aURL.getName(); } aMenuShortCut.append( aMenuTitle ); pVCLPopupMenu->InsertItem( sal_uInt16( i+1 ), aMenuShortCut.makeStringAndClear() ); pVCLPopupMenu->SetTipHelpText( sal_uInt16( i+1 ), aTipHelpText ); pVCLPopupMenu->SetItemCommand( sal_uInt16( i+1 ), aURLString ); } pVCLPopupMenu->InsertSeparator(); // Clear List menu entry pVCLPopupMenu->InsertItem( sal_uInt16( nCount + 1 ), FWK_RESSTR(STR_CLEAR_RECENT_FILES) ); pVCLPopupMenu->SetItemCommand( sal_uInt16( nCount + 1 ), CMD_CLEAR_LIST ); pVCLPopupMenu->SetHelpText( sal_uInt16( nCount + 1 ), FWK_RESSTR(STR_CLEAR_RECENT_FILES_HELP) ); // Open remote menu entry if ( m_bShowRemote ) { pVCLPopupMenu->InsertItem( sal_uInt16( nCount + 2 ), FWK_RESSTR(STR_OPEN_REMOTE) ); pVCLPopupMenu->SetItemCommand( sal_uInt16( nCount + 2 ), CMD_OPEN_REMOTE ); } } else { if ( m_bShowRemote ) { // Open remote menu entry pVCLPopupMenu->InsertItem( 1, FWK_RESSTR(STR_OPEN_REMOTE) ); pVCLPopupMenu->SetItemCommand( 1, CMD_OPEN_REMOTE ); } else { // No recent documents => insert "no document" string pVCLPopupMenu->InsertItem( 1, FWK_RESSTR(STR_NODOCUMENT) ); // Do not disable it, otherwise the Toolbar controller and MenuButton // will display SV_RESID_STRING_NOSELECTIONPOSSIBLE instead of STR_NODOCUMENT pVCLPopupMenu->SetItemBits( 1, pVCLPopupMenu->GetItemBits( 1 ) | MenuItemBits::NOSELECT ); } } } } void RecentFilesMenuController::executeEntry( sal_Int32 nIndex ) { Reference< XDispatch > xDispatch; Reference< XDispatchProvider > xDispatchProvider; css::util::URL aTargetURL; Sequence< PropertyValue > aArgsList; osl::ClearableMutexGuard aLock( m_aMutex ); xDispatchProvider.set( m_xFrame, UNO_QUERY ); aLock.clear(); if (( nIndex >= 0 ) && ( nIndex < sal::static_int_cast( m_aRecentFilesItems.size() ))) { const RecentFile& rRecentFile = m_aRecentFilesItems[ nIndex ]; aTargetURL.Complete = rRecentFile.aURL; m_xURLTransformer->parseStrict( aTargetURL ); sal_Int32 nSize = 2; aArgsList.realloc( nSize ); aArgsList[0].Name = "Referer"; aArgsList[0].Value = makeAny( OUString( "private:user" ) ); // documents in the picklist will never be opened as templates aArgsList[1].Name = "AsTemplate"; aArgsList[1].Value = makeAny( sal_False ); if (!m_aModuleName.isEmpty()) { // Type detection needs to know which app we are opening it from. aArgsList.realloc(++nSize); aArgsList[nSize-1].Name = "DocumentService"; aArgsList[nSize-1].Value <<= m_aModuleName; } xDispatch = xDispatchProvider->queryDispatch( aTargetURL, "_default", 0 ); } if ( xDispatch.is() ) { // Call dispatch asynchronously as we can be destroyed while dispatch is // executed. VCL is not able to survive this as it wants to call listeners // after select!!! LoadRecentFile* pLoadRecentFile = new LoadRecentFile; pLoadRecentFile->xDispatch = xDispatch; pLoadRecentFile->aTargetURL = aTargetURL; pLoadRecentFile->aArgSeq = aArgsList; Application::PostUserEvent( LINK(nullptr, RecentFilesMenuController, ExecuteHdl_Impl), pLoadRecentFile ); } } // XEventListener void SAL_CALL RecentFilesMenuController::disposing( const EventObject& ) throw ( RuntimeException, std::exception ) { Reference< css::awt::XMenuListener > xHolder(static_cast(this), UNO_QUERY ); osl::MutexGuard aLock( m_aMutex ); m_xFrame.clear(); m_xDispatch.clear(); if ( m_xPopupMenu.is() ) m_xPopupMenu->removeMenuListener( Reference< css::awt::XMenuListener >(static_cast(this), UNO_QUERY )); m_xPopupMenu.clear(); } // XStatusListener void SAL_CALL RecentFilesMenuController::statusChanged( const FeatureStateEvent& Event ) throw ( RuntimeException, std::exception ) { osl::MutexGuard aLock( m_aMutex ); m_bDisabled = !Event.IsEnabled; } void SAL_CALL RecentFilesMenuController::itemSelected( const css::awt::MenuEvent& rEvent ) throw (RuntimeException, std::exception) { Reference< css::awt::XPopupMenu > xPopupMenu; osl::ClearableMutexGuard aLock( m_aMutex ); xPopupMenu = m_xPopupMenu; aLock.clear(); if ( xPopupMenu.is() ) { const OUString aCommand( xPopupMenu->getCommand( rEvent.MenuId ) ); OSL_TRACE( "RecentFilesMenuController::itemSelected() - Command : %s", OUStringToOString( aCommand, RTL_TEXTENCODING_UTF8 ).getStr() ); if ( aCommand == CMD_CLEAR_LIST ) { SvtHistoryOptions().Clear( ePICKLIST ); dispatchCommand( "vnd.org.libreoffice.recentdocs:ClearRecentFileList", css::uno::Sequence< css::beans::PropertyValue >() ); } else if ( aCommand == CMD_OPEN_REMOTE ) { Sequence< PropertyValue > aArgsList( 0 ); dispatchCommand( CMD_OPEN_REMOTE, aArgsList ); } else executeEntry( rEvent.MenuId-1 ); } } void SAL_CALL RecentFilesMenuController::itemActivated( const css::awt::MenuEvent& ) throw (RuntimeException, std::exception) { osl::MutexGuard aLock( m_aMutex ); impl_setPopupMenu(); } // XPopupMenuController void RecentFilesMenuController::impl_setPopupMenu() { if ( m_xPopupMenu.is() ) fillPopupMenu( m_xPopupMenu ); } // XDispatchProvider Reference< XDispatch > SAL_CALL RecentFilesMenuController::queryDispatch( const URL& aURL, const OUString& /*sTarget*/, sal_Int32 /*nFlags*/ ) throw( RuntimeException, std::exception ) { osl::MutexGuard aLock( m_aMutex ); throwIfDisposed(); if ( aURL.Complete.startsWith( m_aBaseURL ) ) return Reference< XDispatch >( static_cast< OWeakObject* >( this ), UNO_QUERY ); else return Reference< XDispatch >(); } // XDispatch void SAL_CALL RecentFilesMenuController::dispatch( const URL& aURL, const Sequence< PropertyValue >& /*seqProperties*/ ) throw( RuntimeException, std::exception ) { osl::MutexGuard aLock( m_aMutex ); throwIfDisposed(); if ( aURL.Complete.startsWith( m_aBaseURL ) ) { // Parse URL to retrieve entry argument and its value sal_Int32 nQueryPart = aURL.Complete.indexOf( '?', m_aBaseURL.getLength() ); if ( nQueryPart > 0 ) { const OUString aEntryArgStr( "entry=" ); sal_Int32 nEntryArg = aURL.Complete.indexOf( aEntryArgStr, nQueryPart ); sal_Int32 nEntryPos = nEntryArg + aEntryArgStr.getLength(); if (( nEntryArg > 0 ) && ( nEntryPos < aURL.Complete.getLength() )) { sal_Int32 nAddArgs = aURL.Complete.indexOf( '&', nEntryPos ); OUString aEntryArg; if ( nAddArgs < 0 ) aEntryArg = aURL.Complete.copy( nEntryPos ); else aEntryArg = aURL.Complete.copy( nEntryPos, nAddArgs-nEntryPos ); sal_Int32 nEntry = aEntryArg.toInt32(); executeEntry( nEntry ); } } } } IMPL_STATIC_LINK_TYPED( RecentFilesMenuController, ExecuteHdl_Impl, void*, p, void ) { LoadRecentFile* pLoadRecentFile = static_cast(p); try { // Asynchronous execution as this can lead to our own destruction! // Framework can recycle our current frame and the layout manager disposes all user interface // elements if a component gets detached from its frame! pLoadRecentFile->xDispatch->dispatch( pLoadRecentFile->aTargetURL, pLoadRecentFile->aArgSeq ); } catch ( const Exception& ) { } delete pLoadRecentFile; } } extern "C" SAL_DLLPUBLIC_EXPORT css::uno::XInterface * SAL_CALL com_sun_star_comp_framework_RecentFilesMenuController_get_implementation( css::uno::XComponentContext *context, css::uno::Sequence const &args) { return cppu::acquire(new RecentFilesMenuController(context, args)); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */