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_framework.hxx" 26 27 #include <uielement/recentfilesmenucontroller.hxx> 28 #include <threadhelp/resetableguard.hxx> 29 #include <classes/resource.hrc> 30 #include <classes/fwkresid.hxx> 31 32 #include <com/sun/star/util/XStringWidth.hpp> 33 34 #include <cppuhelper/implbase1.hxx> 35 #include <dispatch/uieventloghelper.hxx> 36 #include <osl/file.hxx> 37 #include <tools/urlobj.hxx> 38 #include <unotools/historyoptions.hxx> 39 #include <vcl/menu.hxx> 40 #include <vcl/svapp.hxx> 41 #include <vos/mutex.hxx> 42 43 using namespace com::sun::star::uno; 44 using namespace com::sun::star::lang; 45 using namespace com::sun::star::frame; 46 using namespace com::sun::star::beans; 47 using namespace com::sun::star::util; 48 49 #define MAX_STR_WIDTH 46 50 #define MAX_MENU_ITEMS 99 51 52 static const char SFX_REFERER_USER[] = "private:user"; 53 static const char CMD_CLEAR_LIST[] = ".uno:ClearRecentFileList"; 54 static const char CMD_PREFIX[] = "vnd.sun.star.popup:RecentFileList?entry="; 55 static const char MENU_SHOTCUT[] = "~N: "; 56 57 namespace framework 58 { 59 60 class RecentFilesStringLength : public ::cppu::WeakImplHelper1< ::com::sun::star::util::XStringWidth > 61 { 62 public: 63 RecentFilesStringLength() {} 64 virtual ~RecentFilesStringLength() {} 65 66 // XStringWidth 67 sal_Int32 SAL_CALL queryStringWidth( const ::rtl::OUString& aString ) 68 throw (::com::sun::star::uno::RuntimeException) 69 { 70 return aString.getLength(); 71 } 72 }; 73 74 DEFINE_XSERVICEINFO_MULTISERVICE ( RecentFilesMenuController , 75 OWeakObject , 76 SERVICENAME_POPUPMENUCONTROLLER , 77 IMPLEMENTATIONNAME_RECENTFILESMENUCONTROLLER 78 ) 79 80 DEFINE_INIT_SERVICE ( RecentFilesMenuController, {} ) 81 82 RecentFilesMenuController::RecentFilesMenuController( const ::com::sun::star::uno::Reference< ::com::sun::star::lang::XMultiServiceFactory >& xServiceManager ) : 83 svt::PopupMenuControllerBase( xServiceManager ), 84 m_bDisabled( sal_False ) 85 { 86 } 87 88 RecentFilesMenuController::~RecentFilesMenuController() 89 { 90 } 91 92 // private function 93 void RecentFilesMenuController::fillPopupMenu( Reference< css::awt::XPopupMenu >& rPopupMenu ) 94 { 95 VCLXPopupMenu* pPopupMenu = (VCLXPopupMenu *)VCLXMenu::GetImplementation( rPopupMenu ); 96 PopupMenu* pVCLPopupMenu = 0; 97 98 vos::OGuard aSolarMutexGuard( Application::GetSolarMutex() ); 99 100 resetPopupMenu( rPopupMenu ); 101 if ( pPopupMenu ) 102 pVCLPopupMenu = (PopupMenu *)pPopupMenu->GetMenu(); 103 104 if ( pVCLPopupMenu ) 105 { 106 Sequence< Sequence< PropertyValue > > aHistoryList = SvtHistoryOptions().GetList( ePICKLIST ); 107 Reference< XStringWidth > xStringLength( new RecentFilesStringLength ); 108 109 int nPickListMenuItems = ( aHistoryList.getLength() > MAX_MENU_ITEMS ) ? MAX_MENU_ITEMS : aHistoryList.getLength(); 110 111 m_aRecentFilesItems.clear(); 112 if (( nPickListMenuItems > 0 ) && !m_bDisabled ) 113 { 114 for ( int i = 0; i < nPickListMenuItems; i++ ) 115 { 116 Sequence< PropertyValue >& rPickListEntry = aHistoryList[i]; 117 RecentFile aRecentFile; 118 119 for ( int j = 0; j < rPickListEntry.getLength(); j++ ) 120 { 121 Any a = rPickListEntry[j].Value; 122 123 if ( rPickListEntry[j].Name == HISTORY_PROPERTYNAME_URL ) 124 a >>= aRecentFile.aURL; 125 else if ( rPickListEntry[j].Name == HISTORY_PROPERTYNAME_FILTER ) 126 a >>= aRecentFile.aFilter; 127 else if ( rPickListEntry[j].Name == HISTORY_PROPERTYNAME_TITLE ) 128 a >>= aRecentFile.aTitle; 129 else if ( rPickListEntry[j].Name == HISTORY_PROPERTYNAME_PASSWORD ) 130 a >>= aRecentFile.aPassword; 131 } 132 133 m_aRecentFilesItems.push_back( aRecentFile ); 134 } 135 } 136 137 if ( !m_aRecentFilesItems.empty() ) 138 { 139 const sal_uInt32 nCount = m_aRecentFilesItems.size(); 140 for ( sal_uInt32 i = 0; i < nCount; i++ ) 141 { 142 rtl::OUStringBuffer aMenuShortCut; 143 if ( i <= 9 ) 144 { 145 if ( i == 9 ) 146 aMenuShortCut.appendAscii( RTL_CONSTASCII_STRINGPARAM( "1~0: " ) ); 147 else 148 { 149 aMenuShortCut.appendAscii( RTL_CONSTASCII_STRINGPARAM( MENU_SHOTCUT ) ); 150 aMenuShortCut.setCharAt( 1, sal_Unicode( i + '1' ) ); 151 } 152 } 153 else 154 { 155 aMenuShortCut.append( sal_Int32( i + 1 ) ); 156 aMenuShortCut.appendAscii( RTL_CONSTASCII_STRINGPARAM( ": " ) ); 157 } 158 159 rtl::OUStringBuffer aStrBuffer; 160 aStrBuffer.appendAscii( RTL_CONSTASCII_STRINGPARAM( CMD_PREFIX ) ); 161 aStrBuffer.append( sal_Int32( i ) ); 162 rtl::OUString aURLString( aStrBuffer.makeStringAndClear() ); 163 164 // Abbreviate URL 165 rtl::OUString aTipHelpText; 166 rtl::OUString aMenuTitle; 167 INetURLObject aURL( m_aRecentFilesItems[i].aURL ); 168 169 if ( aURL.GetProtocol() == INET_PROT_FILE ) 170 { 171 // Do handle file URL differently => convert it to a system 172 // path and abbreviate it with a special function: 173 rtl::OUString aSystemPath( aURL.getFSysPath( INetURLObject::FSYS_DETECT ) ); 174 aTipHelpText = aSystemPath; 175 176 ::rtl::OUString aCompactedSystemPath; 177 if ( osl_abbreviateSystemPath( aSystemPath.pData, &aCompactedSystemPath.pData, MAX_STR_WIDTH, NULL ) == osl_File_E_None ) 178 aMenuTitle = aCompactedSystemPath; 179 else 180 aMenuTitle = aSystemPath; 181 } 182 else 183 { 184 // Use INetURLObject to abbreviate all other URLs 185 aMenuTitle = aURL.getAbbreviated( xStringLength, MAX_STR_WIDTH, INetURLObject::DECODE_UNAMBIGUOUS ); 186 aTipHelpText = aURLString; 187 } 188 189 aMenuShortCut.append( aMenuTitle ); 190 191 pVCLPopupMenu->InsertItem( sal_uInt16( i+1 ), aMenuShortCut.makeStringAndClear() ); 192 pVCLPopupMenu->SetTipHelpText( sal_uInt16( i+1 ), aTipHelpText ); 193 pVCLPopupMenu->SetItemCommand( sal_uInt16( i+1 ), aURLString ); 194 } 195 196 pVCLPopupMenu->InsertSeparator(); 197 // Clear List menu entry 198 pVCLPopupMenu->InsertItem( sal_uInt16( nCount + 1 ), 199 String( FwkResId( STR_CLEAR_RECENT_FILES ) ) ); 200 pVCLPopupMenu->SetItemCommand( sal_uInt16( nCount + 1 ), 201 rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( CMD_CLEAR_LIST ) ) ); 202 pVCLPopupMenu->SetHelpText( sal_uInt16( nCount + 1 ), 203 String( FwkResId( STR_CLEAR_RECENT_FILES_HELP ) ) ); 204 } 205 else 206 { 207 // No recent documents => insert "no document" string 208 pVCLPopupMenu->InsertItem( 1, String( FwkResId( STR_NODOCUMENT ) ) ); 209 pVCLPopupMenu->EnableItem( 1, sal_False ); 210 } 211 } 212 } 213 214 void RecentFilesMenuController::executeEntry( sal_Int32 nIndex ) 215 { 216 static int NUM_OF_PICKLIST_ARGS = 3; 217 218 Reference< XDispatch > xDispatch; 219 Reference< XDispatchProvider > xDispatchProvider; 220 css::util::URL aTargetURL; 221 Sequence< PropertyValue > aArgsList; 222 223 osl::ClearableMutexGuard aLock( m_aMutex ); 224 xDispatchProvider = Reference< XDispatchProvider >( m_xFrame, UNO_QUERY ); 225 aLock.clear(); 226 227 if (( nIndex >= 0 ) && 228 ( nIndex < sal::static_int_cast<sal_Int32>( m_aRecentFilesItems.size() ))) 229 { 230 const RecentFile& rRecentFile = m_aRecentFilesItems[ nIndex ]; 231 232 aTargetURL.Complete = rRecentFile.aURL; 233 m_xURLTransformer->parseStrict( aTargetURL ); 234 235 aArgsList.realloc( NUM_OF_PICKLIST_ARGS ); 236 aArgsList[0].Name = ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "Referer" )); 237 aArgsList[0].Value = makeAny( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( SFX_REFERER_USER ))); 238 239 // documents in the picklist will never be opened as templates 240 aArgsList[1].Name = ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "AsTemplate" )); 241 aArgsList[1].Value = makeAny( (sal_Bool) sal_False ); 242 243 ::rtl::OUString aFilter( rRecentFile.aFilter ); 244 sal_Int32 nPos = aFilter.indexOf( '|' ); 245 if ( nPos >= 0 ) 246 { 247 ::rtl::OUString aFilterOptions; 248 249 if ( nPos < ( aFilter.getLength() - 1 ) ) 250 aFilterOptions = aFilter.copy( nPos+1 ); 251 252 aArgsList[2].Name = ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "FilterOptions" )); 253 aArgsList[2].Value <<= aFilterOptions; 254 255 aFilter = aFilter.copy( 0, nPos-1 ); 256 aArgsList.realloc( ++NUM_OF_PICKLIST_ARGS ); 257 } 258 259 aArgsList[NUM_OF_PICKLIST_ARGS-1].Name = ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "FilterName" )); 260 aArgsList[NUM_OF_PICKLIST_ARGS-1].Value <<= aFilter; 261 262 xDispatch = xDispatchProvider->queryDispatch( aTargetURL, ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "_default" ) ), 0 ); 263 } 264 265 if ( xDispatch.is() ) 266 { 267 // Call dispatch asychronously as we can be destroyed while dispatch is 268 // executed. VCL is not able to survive this as it wants to call listeners 269 // after select!!! 270 LoadRecentFile* pLoadRecentFile = new LoadRecentFile; 271 pLoadRecentFile->xDispatch = xDispatch; 272 pLoadRecentFile->aTargetURL = aTargetURL; 273 pLoadRecentFile->aArgSeq = aArgsList; 274 275 if(::comphelper::UiEventsLogger::isEnabled()) //#i88653# 276 UiEventLogHelper(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("RecentFilesMenuController"))).log(m_xServiceManager, m_xFrame, aTargetURL, aArgsList); 277 278 Application::PostUserEvent( STATIC_LINK(0, RecentFilesMenuController, ExecuteHdl_Impl), pLoadRecentFile ); 279 } 280 } 281 282 // XEventListener 283 void SAL_CALL RecentFilesMenuController::disposing( const EventObject& ) throw ( RuntimeException ) 284 { 285 Reference< css::awt::XMenuListener > xHolder(( OWeakObject *)this, UNO_QUERY ); 286 287 osl::MutexGuard aLock( m_aMutex ); 288 m_xFrame.clear(); 289 m_xDispatch.clear(); 290 m_xServiceManager.clear(); 291 292 if ( m_xPopupMenu.is() ) 293 m_xPopupMenu->removeMenuListener( Reference< css::awt::XMenuListener >(( OWeakObject *)this, UNO_QUERY )); 294 m_xPopupMenu.clear(); 295 } 296 297 // XStatusListener 298 void SAL_CALL RecentFilesMenuController::statusChanged( const FeatureStateEvent& Event ) throw ( RuntimeException ) 299 { 300 osl::MutexGuard aLock( m_aMutex ); 301 m_bDisabled = !Event.IsEnabled; 302 } 303 304 void SAL_CALL RecentFilesMenuController::itemSelected( const css::awt::MenuEvent& rEvent ) throw (RuntimeException) 305 { 306 Reference< css::awt::XPopupMenu > xPopupMenu; 307 308 osl::ClearableMutexGuard aLock( m_aMutex ); 309 xPopupMenu = m_xPopupMenu; 310 aLock.clear(); 311 312 if ( xPopupMenu.is() ) 313 { 314 const rtl::OUString aCommand( xPopupMenu->getCommand( rEvent.MenuId ) ); 315 OSL_TRACE( "RecentFilesMenuController::itemSelected() - Command : %s", 316 rtl::OUStringToOString( aCommand, RTL_TEXTENCODING_UTF8 ).getStr() ); 317 318 if ( aCommand.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM( CMD_CLEAR_LIST ) ) ) 319 SvtHistoryOptions().Clear( ePICKLIST ); 320 else 321 executeEntry( rEvent.MenuId-1 ); 322 } 323 } 324 325 void SAL_CALL RecentFilesMenuController::itemActivated( const css::awt::MenuEvent& ) throw (RuntimeException) 326 { 327 osl::MutexGuard aLock( m_aMutex ); 328 impl_setPopupMenu(); 329 } 330 331 // XPopupMenuController 332 void RecentFilesMenuController::impl_setPopupMenu() 333 { 334 if ( m_xPopupMenu.is() ) 335 fillPopupMenu( m_xPopupMenu ); 336 } 337 338 // XDispatchProvider 339 Reference< XDispatch > SAL_CALL RecentFilesMenuController::queryDispatch( 340 const URL& aURL, 341 const ::rtl::OUString& /*sTarget*/, 342 sal_Int32 /*nFlags*/ ) 343 throw( RuntimeException ) 344 { 345 osl::MutexGuard aLock( m_aMutex ); 346 347 throwIfDisposed(); 348 349 if ( aURL.Complete.indexOf( m_aBaseURL ) == 0 ) 350 return Reference< XDispatch >( static_cast< OWeakObject* >( this ), UNO_QUERY ); 351 else 352 return Reference< XDispatch >(); 353 } 354 355 // XDispatch 356 void SAL_CALL RecentFilesMenuController::dispatch( 357 const URL& aURL, 358 const Sequence< PropertyValue >& /*seqProperties*/ ) 359 throw( RuntimeException ) 360 { 361 osl::MutexGuard aLock( m_aMutex ); 362 363 throwIfDisposed(); 364 365 if ( aURL.Complete.indexOf( m_aBaseURL ) == 0 ) 366 { 367 // Parse URL to retrieve entry argument and its value 368 sal_Int32 nQueryPart = aURL.Complete.indexOf( '?', m_aBaseURL.getLength() ); 369 if ( nQueryPart > 0 ) 370 { 371 const rtl::OUString aEntryArgStr( RTL_CONSTASCII_USTRINGPARAM( "entry=" )); 372 sal_Int32 nEntryArg = aURL.Complete.indexOf( aEntryArgStr, nQueryPart ); 373 sal_Int32 nEntryPos = nEntryArg + aEntryArgStr.getLength(); 374 if (( nEntryArg > 0 ) && ( nEntryPos < aURL.Complete.getLength() )) 375 { 376 sal_Int32 nAddArgs = aURL.Complete.indexOf( '&', nEntryPos ); 377 rtl::OUString aEntryArg; 378 379 if ( nAddArgs < 0 ) 380 aEntryArg = aURL.Complete.copy( nEntryPos ); 381 else 382 aEntryArg = aURL.Complete.copy( nEntryPos, nAddArgs-nEntryPos ); 383 384 sal_Int32 nEntry = aEntryArg.toInt32(); 385 executeEntry( nEntry ); 386 } 387 } 388 } 389 } 390 391 IMPL_STATIC_LINK_NOINSTANCE( RecentFilesMenuController, ExecuteHdl_Impl, LoadRecentFile*, pLoadRecentFile ) 392 { 393 try 394 { 395 // Asynchronous execution as this can lead to our own destruction! 396 // Framework can recycle our current frame and the layout manager disposes all user interface 397 // elements if a component gets detached from its frame! 398 pLoadRecentFile->xDispatch->dispatch( pLoadRecentFile->aTargetURL, pLoadRecentFile->aArgSeq ); 399 } 400 catch ( Exception& ) 401 { 402 } 403 404 delete pLoadRecentFile; 405 return 0; 406 } 407 408 } 409