00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021 #include <qdir.h>
00022
00023 #include <klocale.h>
00024 #include <kapplication.h>
00025 #include <kbookmarkmanager.h>
00026 #include <kdebug.h>
00027 #include <krun.h>
00028 #include <kprotocolinfo.h>
00029 #include <kiconloader.h>
00030 #include <kinputdialog.h>
00031 #include <kglobalsettings.h>
00032 #include <kstandarddirs.h>
00033 #include <kxmlguifactory.h>
00034 #include <kxmlguibuilder.h>
00035 #include <kparts/componentfactory.h>
00036
00037 #include <assert.h>
00038
00039 #include <kfileshare.h>
00040 #include <kprocess.h>
00041
00042 #include "kpropertiesdialog.h"
00043 #include "knewmenu.h"
00044 #include "konq_popupmenu.h"
00045 #include "konq_operations.h"
00046 #include <dcopclient.h>
00047
00048
00049
00050
00051
00052
00053
00054
00055
00056
00057
00058
00059
00060
00061
00062
00063
00064
00065
00066
00067
00068
00069
00070 class KonqPopupMenuGUIBuilder : public KXMLGUIBuilder
00071 {
00072 public:
00073 KonqPopupMenuGUIBuilder( QPopupMenu *menu )
00074 : KXMLGUIBuilder( 0 )
00075 {
00076 m_menu = menu;
00077 }
00078 virtual ~KonqPopupMenuGUIBuilder()
00079 {
00080 }
00081
00082 virtual QWidget *createContainer( QWidget *parent, int index,
00083 const QDomElement &element,
00084 int &id )
00085 {
00086 if ( !parent && element.attribute( "name" ) == "popupmenu" )
00087 return m_menu;
00088
00089 return KXMLGUIBuilder::createContainer( parent, index, element, id );
00090 }
00091
00092 QPopupMenu *m_menu;
00093 };
00094
00095 class KonqPopupMenu::KonqPopupMenuPrivate
00096 {
00097 public:
00098 KonqPopupMenuPrivate() : m_parentWidget( 0 ),
00099 m_itemFlags( KParts::BrowserExtension::DefaultPopupItems )
00100 {
00101 }
00102 QString m_urlTitle;
00103 QWidget *m_parentWidget;
00104 KParts::BrowserExtension::PopupFlags m_itemFlags;
00105 };
00106
00107 KonqPopupMenu::ProtocolInfo::ProtocolInfo()
00108 {
00109 m_Reading = false;
00110 m_Writing = false;
00111 m_Deleting = false;
00112 m_Moving = false;
00113 m_TrashIncluded = false;
00114 }
00115
00116 bool KonqPopupMenu::ProtocolInfo::supportsReading() const
00117 {
00118 return m_Reading;
00119 }
00120
00121 bool KonqPopupMenu::ProtocolInfo::supportsWriting() const
00122 {
00123 return m_Writing;
00124 }
00125
00126 bool KonqPopupMenu::ProtocolInfo::supportsDeleting() const
00127 {
00128 return m_Deleting;
00129 }
00130
00131 bool KonqPopupMenu::ProtocolInfo::supportsMoving() const
00132 {
00133 return m_Moving;
00134 }
00135
00136 bool KonqPopupMenu::ProtocolInfo::trashIncluded() const
00137 {
00138 return m_TrashIncluded;
00139 }
00140
00141
00142
00143 class PopupServices
00144 {
00145 public:
00146 ServiceList* selectList( const QString& priority, const QString& submenuName );
00147
00148 ServiceList builtin;
00149 ServiceList user, userToplevel, userPriority;
00150 QMap<QString, ServiceList> userSubmenus, userToplevelSubmenus, userPrioritySubmenus;
00151 };
00152
00153 ServiceList* PopupServices::selectList( const QString& priority, const QString& submenuName )
00154 {
00155
00156
00157 if (submenuName.isEmpty())
00158 {
00159 if (priority == "TopLevel")
00160 {
00161 return &userToplevel;
00162 }
00163 else if (priority == "Important")
00164 {
00165 return &userPriority;
00166 }
00167 }
00168 else if (priority == "TopLevel")
00169 {
00170 return &(userToplevelSubmenus[submenuName]);
00171 }
00172 else if (priority == "Important")
00173 {
00174 return &(userPrioritySubmenus[submenuName]);
00175 }
00176 else
00177 {
00178 return &(userSubmenus[submenuName]);
00179 }
00180 return &user;
00181 }
00182
00184
00185 KonqPopupMenu::KonqPopupMenu( KBookmarkManager *mgr, const KFileItemList &items,
00186 KURL viewURL,
00187 KActionCollection & actions,
00188 KNewMenu * newMenu,
00189 bool showProperties )
00190 : QPopupMenu( 0L, "konq_popupmenu" ),
00191 m_actions( actions ), m_ownActions( static_cast<QWidget *>( 0 ), "KonqPopupMenu::m_ownActions" ),
00192 m_pMenuNew( newMenu ), m_sViewURL(viewURL), m_lstItems(items), m_pManager(mgr)
00193 {
00194 KonqPopupFlags kpf = ( showProperties ? ShowProperties : IsLink ) | ShowNewWindow;
00195 init(0, kpf, KParts::BrowserExtension::DefaultPopupItems);
00196 }
00197
00198 KonqPopupMenu::KonqPopupMenu( KBookmarkManager *mgr, const KFileItemList &items,
00199 KURL viewURL,
00200 KActionCollection & actions,
00201 KNewMenu * newMenu,
00202 QWidget * parentWidget,
00203 bool showProperties )
00204 : QPopupMenu( parentWidget, "konq_popupmenu" ), m_actions( actions ), m_ownActions( static_cast<QWidget *>( 0 ), "KonqPopupMenu::m_ownActions" ), m_pMenuNew( newMenu ), m_sViewURL(viewURL), m_lstItems(items), m_pManager(mgr)
00205 {
00206 KonqPopupFlags kpf = ( showProperties ? ShowProperties : IsLink ) | ShowNewWindow;
00207 init(parentWidget, kpf, KParts::BrowserExtension::DefaultPopupItems);
00208 }
00209
00210 KonqPopupMenu::KonqPopupMenu( KBookmarkManager *mgr, const KFileItemList &items,
00211 const KURL& viewURL,
00212 KActionCollection & actions,
00213 KNewMenu * newMenu,
00214 QWidget * parentWidget,
00215 KonqPopupFlags kpf,
00216 KParts::BrowserExtension::PopupFlags flags)
00217 : QPopupMenu( parentWidget, "konq_popupmenu" ), m_actions( actions ), m_ownActions( static_cast<QWidget *>( 0 ), "KonqPopupMenu::m_ownActions" ), m_pMenuNew( newMenu ), m_sViewURL(viewURL), m_lstItems(items), m_pManager(mgr)
00218 {
00219 init(parentWidget, kpf, flags);
00220 }
00221
00222 void KonqPopupMenu::init (QWidget * parentWidget, KonqPopupFlags kpf, KParts::BrowserExtension::PopupFlags flags)
00223 {
00224 d = new KonqPopupMenuPrivate;
00225 d->m_parentWidget = parentWidget;
00226 d->m_itemFlags = flags;
00227 setup(kpf);
00228 }
00229
00230 int KonqPopupMenu::insertServicesSubmenus(const QMap<QString, ServiceList>& submenus,
00231 QDomElement& menu,
00232 bool isBuiltin)
00233 {
00234 int count = 0;
00235 QMap<QString, ServiceList>::ConstIterator it;
00236
00237 for (it = submenus.begin(); it != submenus.end(); ++it)
00238 {
00239 if (it.data().isEmpty())
00240 {
00241
00242 continue;
00243 }
00244
00245 QDomElement actionSubmenu = m_doc.createElement( "menu" );
00246 actionSubmenu.setAttribute( "name", "actions " + it.key() );
00247 menu.appendChild( actionSubmenu );
00248 QDomElement subtext = m_doc.createElement( "text" );
00249 actionSubmenu.appendChild( subtext );
00250 subtext.appendChild( m_doc.createTextNode( it.key() ) );
00251 count += insertServices(it.data(), actionSubmenu, isBuiltin);
00252 }
00253
00254 return count;
00255 }
00256
00257 int KonqPopupMenu::insertServices(const ServiceList& list,
00258 QDomElement& menu,
00259 bool isBuiltin)
00260 {
00261 static int id = 1000;
00262 int count = 0;
00263
00264 ServiceList::const_iterator it = list.begin();
00265 for( ; it != list.end(); ++it )
00266 {
00267 if ((*it).isEmpty())
00268 {
00269 if (!menu.firstChild().isNull() &&
00270 menu.lastChild().toElement().tagName().lower() != "separator")
00271 {
00272 QDomElement separator = m_doc.createElement( "separator" );
00273 menu.appendChild(separator);
00274 }
00275 continue;
00276 }
00277
00278 if (isBuiltin || (*it).m_display == true)
00279 {
00280 QCString name;
00281 name.setNum( id );
00282 name.prepend( isBuiltin ? "builtinservice_" : "userservice_" );
00283 KAction * act = new KAction( QString((*it).m_strName).replace('&',"&&"), 0,
00284 this, SLOT( slotRunService() ),
00285 &m_ownActions, name );
00286
00287 if ( !(*it).m_strIcon.isEmpty() )
00288 {
00289 QPixmap pix = SmallIcon( (*it).m_strIcon );
00290 act->setIconSet( pix );
00291 }
00292
00293 addAction( act, menu );
00294
00295 m_mapPopupServices[ id++ ] = *it;
00296 ++count;
00297 }
00298 }
00299
00300 return count;
00301 }
00302
00303 bool KonqPopupMenu::KIOSKAuthorizedAction(KConfig& cfg)
00304 {
00305 if ( !cfg.hasKey( "X-KDE-AuthorizeAction") )
00306 {
00307 return true;
00308 }
00309
00310 QStringList list = cfg.readListEntry("X-KDE-AuthorizeAction");
00311 if (kapp && !list.isEmpty())
00312 {
00313 for(QStringList::ConstIterator it = list.begin();
00314 it != list.end();
00315 ++it)
00316 {
00317 if (!kapp->authorize((*it).stripWhiteSpace()))
00318 {
00319 return false;
00320 }
00321 }
00322 }
00323
00324 return true;
00325 }
00326
00327
00328 void KonqPopupMenu::setup(KonqPopupFlags kpf)
00329 {
00330 assert( m_lstItems.count() >= 1 );
00331
00332 m_ownActions.setWidget( this );
00333
00334 const bool bIsLink = (kpf & IsLink);
00335 bool currentDir = false;
00336 bool sReading = true;
00337 bool sDeleting = ( d->m_itemFlags & KParts::BrowserExtension::NoDeletion ) == 0;
00338 bool sMoving = sDeleting;
00339 bool sWriting = sDeleting && m_lstItems.first()->isWritable();
00340 m_sMimeType = m_lstItems.first()->mimetype();
00341 QString mimeGroup = m_sMimeType.left(m_sMimeType.find('/'));
00342 mode_t mode = m_lstItems.first()->mode();
00343 bool isDirectory = S_ISDIR(mode);
00344 bool bTrashIncluded = false;
00345 bool mediaFiles = false;
00346 bool isLocal = m_lstItems.first()->isLocalFile()
00347 || m_lstItems.first()->url().protocol()=="media"
00348 || m_lstItems.first()->url().protocol()=="system";
00349 bool isTrashLink = false;
00350 m_lstPopupURLs.clear();
00351 int id = 0;
00352 setFont(KGlobalSettings::menuFont());
00353 m_pluginList.setAutoDelete( true );
00354 m_ownActions.setHighlightingEnabled( true );
00355
00356 attrName = QString::fromLatin1( "name" );
00357
00358 prepareXMLGUIStuff();
00359 m_builder = new KonqPopupMenuGUIBuilder( this );
00360 m_factory = new KXMLGUIFactory( m_builder );
00361
00362 KURL url;
00363 KFileItemListIterator it ( m_lstItems );
00364 QStringList mimeTypeList;
00365
00366 for ( ; it.current(); ++it )
00367 {
00368 url = (*it)->url();
00369
00370
00371 m_lstPopupURLs.append( url );
00372
00373
00374 if ( mode != (*it)->mode() )
00375 mode = 0;
00376
00377
00378 if ( m_sMimeType != (*it)->mimetype() )
00379 {
00380 m_sMimeType = QString::null;
00381
00382 if ( mimeGroup != (*it)->mimetype().left((*it)->mimetype().find('/')))
00383 mimeGroup = QString::null;
00384 }
00385
00386 if ( mimeTypeList.findIndex( (*it)->mimetype() ) == -1 )
00387 mimeTypeList << (*it)->mimetype();
00388
00389 if ( isLocal && !url.isLocalFile() && url.protocol() != "media" && url.protocol() != "system" )
00390 isLocal = false;
00391
00392 if ( !bTrashIncluded && (
00393 ( url.protocol() == "trash" && url.path().length() <= 1 )
00394 || url.url() == "system:/trash" || url.url() == "system:/trash/" ) ) {
00395 bTrashIncluded = true;
00396 isLocal = false;
00397 }
00398
00399 if ( sReading )
00400 sReading = KProtocolInfo::supportsReading( url );
00401
00402 if ( sWriting )
00403 sWriting = KProtocolInfo::supportsWriting( url ) && (*it)->isWritable();
00404
00405 if ( sDeleting )
00406 sDeleting = KProtocolInfo::supportsDeleting( url );
00407
00408 if ( sMoving )
00409 sMoving = KProtocolInfo::supportsMoving( url );
00410 if ( (*it)->mimetype().startsWith("media/") )
00411 mediaFiles = true;
00412 }
00413 url = m_sViewURL;
00414 url.cleanPath();
00415
00416
00417 if ( m_lstItems.count() == 1 )
00418 {
00419 KURL firstPopupURL( m_lstItems.first()->url() );
00420 firstPopupURL.cleanPath();
00421
00422
00423 currentDir = firstPopupURL.equals( url, true );
00424 if ( isLocal && m_sMimeType == "application/x-desktop" ) {
00425 KSimpleConfig cfg( firstPopupURL.path(), true );
00426 cfg.setDesktopGroup();
00427 isTrashLink = ( cfg.readEntry("Type") == "Link" && cfg.readEntry("URL") == "trash:/" );
00428 }
00429
00430 if ( isTrashLink ) {
00431 sDeleting = false;
00432 }
00433 }
00434 const bool isSingleLocal = m_lstItems.count() == 1 && isLocal;
00435
00436 m_info.m_Reading = sReading;
00437 m_info.m_Writing = sWriting;
00438 m_info.m_Deleting = sDeleting;
00439 m_info.m_Moving = sMoving;
00440 m_info.m_TrashIncluded = bTrashIncluded;
00441
00442
00443 bool isCurrentTrash = ( m_lstItems.count() == 1 && bTrashIncluded ) || isTrashLink;
00444 bool isIntoTrash = ( url.protocol() == "trash" || url.url().startsWith( "system:/trash" ) ) && !isCurrentTrash;
00445
00446 bool isSingleMedium = m_lstItems.count() == 1 && mediaFiles;
00447 clear();
00448
00450
00451 KAction * act;
00452
00453 if (!isCurrentTrash)
00454 addMerge( "konqueror" );
00455
00456 bool isKDesktop = QCString( kapp->name() ) == "kdesktop";
00457 KAction *actNewWindow = 0;
00458
00459 if (( kpf & ShowProperties ) && isKDesktop &&
00460 !kapp->authorize("editable_desktop_icons"))
00461 {
00462 kpf &= ~ShowProperties;
00463 }
00464
00465
00466
00467 if ( ((kpf & ShowNewWindow) != 0) && sReading )
00468 {
00469 QString openStr = isKDesktop ? i18n( "&Open" ) : i18n( "Open in New &Window" );
00470 actNewWindow = new KAction( openStr, "window_new", 0, this, SLOT( slotPopupNewView() ), &m_ownActions, "newview" );
00471 }
00472
00473 if ( actNewWindow && !isKDesktop )
00474 {
00475 if (isCurrentTrash)
00476 actNewWindow->setToolTip( i18n( "Open the trash in a new window" ) );
00477 else if (isSingleMedium)
00478 actNewWindow->setToolTip( i18n( "Open the medium in a new window") );
00479 else
00480 actNewWindow->setToolTip( i18n( "Open the document in a new window" ) );
00481 }
00482
00483 if ( S_ISDIR(mode) && sWriting && !isCurrentTrash )
00484 {
00485 if ( currentDir && m_pMenuNew )
00486 {
00487
00488 m_pMenuNew->slotCheckUpToDate();
00489 m_pMenuNew->setPopupFiles( m_lstPopupURLs );
00490
00491 addAction( m_pMenuNew );
00492
00493 addSeparator();
00494 }
00495 else
00496 {
00497 if (d->m_itemFlags & KParts::BrowserExtension::ShowCreateDirectory)
00498 {
00499 KAction *actNewDir = new KAction( i18n( "Create &Folder..." ), "folder_new", 0, this, SLOT( slotPopupNewDir() ), &m_ownActions, "newdir" );
00500 addAction( actNewDir );
00501 addSeparator();
00502 }
00503 }
00504 } else if ( isIntoTrash ) {
00505
00506 act = new KAction( i18n( "&Restore" ), 0, this, SLOT( slotPopupRestoreTrashedItems() ), &m_ownActions, "restore" );
00507 addAction( act );
00508 }
00509
00510 if (d->m_itemFlags & KParts::BrowserExtension::ShowNavigationItems)
00511 {
00512 if (d->m_itemFlags & KParts::BrowserExtension::ShowUp)
00513 addAction( "up" );
00514 addAction( "back" );
00515 addAction( "forward" );
00516 if (d->m_itemFlags & KParts::BrowserExtension::ShowReload)
00517 addAction( "reload" );
00518 addSeparator();
00519 }
00520
00521
00522 if (actNewWindow)
00523 {
00524 addAction( actNewWindow );
00525 addSeparator();
00526 }
00527 addGroup( "tabhandling" );
00528
00529 if ( !bIsLink )
00530 {
00531 if ( !currentDir && sReading ) {
00532 if ( sDeleting ) {
00533 addAction( "cut" );
00534 }
00535 addAction( "copy" );
00536 }
00537
00538 if ( S_ISDIR(mode) && sWriting ) {
00539 if ( currentDir )
00540 addAction( "paste" );
00541 else
00542 addAction( "pasteto" );
00543 }
00544 if ( !currentDir )
00545 {
00546 if ( m_lstItems.count() == 1 && sMoving )
00547 addAction( "rename" );
00548
00549 bool addTrash = false;
00550 bool addDel = false;
00551
00552 if ( sMoving && !isIntoTrash && !isTrashLink )
00553 addTrash = true;
00554
00555 if ( sDeleting ) {
00556 if ( !isLocal )
00557 addDel = true;
00558 else if (KApplication::keyboardMouseState() & Qt::ShiftButton) {
00559 addTrash = false;
00560 addDel = true;
00561 }
00562 else {
00563 KConfigGroup configGroup( kapp->config(), "KDE" );
00564 if ( configGroup.readBoolEntry( "ShowDeleteCommand", false ) )
00565 addDel = true;
00566 }
00567 }
00568
00569 if ( addTrash )
00570 addAction( "trash" );
00571 if ( addDel )
00572 addAction( "del" );
00573 }
00574 }
00575 if ( isCurrentTrash )
00576 {
00577 act = new KAction( i18n( "&Empty Trash Bin" ), "emptytrash", 0, this, SLOT( slotPopupEmptyTrashBin() ), &m_ownActions, "empytrash" );
00578 addAction( act );
00579 }
00580 addGroup( "editactions" );
00581
00582 if (d->m_itemFlags & KParts::BrowserExtension::ShowTextSelectionItems) {
00583 addMerge( 0 );
00584 m_factory->addClient( this );
00585 return;
00586 }
00587
00588 if ( !isCurrentTrash && !isIntoTrash && (d->m_itemFlags & KParts::BrowserExtension::ShowBookmark))
00589 {
00590 addSeparator();
00591 QString caption;
00592 if (currentDir)
00593 {
00594 bool httpPage = (m_sViewURL.protocol().find("http", 0, false) == 0);
00595 if (httpPage)
00596 caption = i18n("&Bookmark This Page");
00597 else
00598 caption = i18n("&Bookmark This Location");
00599 }
00600 else if (S_ISDIR(mode))
00601 caption = i18n("&Bookmark This Folder");
00602 else if (bIsLink)
00603 caption = i18n("&Bookmark This Link");
00604 else
00605 caption = i18n("&Bookmark This File");
00606
00607 act = new KAction( caption, "bookmark_add", 0, this, SLOT( slotPopupAddToBookmark() ), &m_ownActions, "bookmark_add" );
00608 if (m_lstItems.count() > 1)
00609 act->setEnabled(false);
00610 if (kapp->authorizeKAction("bookmarks"))
00611 addAction( act );
00612 if (bIsLink)
00613 addGroup( "linkactions" );
00614 }
00615
00617
00618 PopupServices s;
00619 KURL urlForServiceMenu( m_lstItems.first()->url() );
00620
00621
00622 if ( m_sMimeType == "application/x-desktop" && isSingleLocal )
00623 {
00624
00625 s.builtin = KDEDesktopMimeType::builtinServices( m_lstItems.first()->url() );
00626 const QString path = m_lstItems.first()->url().path();
00627 KSimpleConfig cfg( path, true );
00628 cfg.setDesktopGroup();
00629 const QString priority = cfg.readEntry("X-KDE-Priority");
00630 const QString submenuName = cfg.readEntry( "X-KDE-Submenu" );
00631 if ( cfg.readEntry("Type") == "Link" ) {
00632 urlForServiceMenu = cfg.readEntry("URL");
00633
00634
00635 }
00636 ServiceList* list = s.selectList( priority, submenuName );
00637 (*list) = KDEDesktopMimeType::userDefinedServices( path, cfg, url.isLocalFile() );
00638 }
00639
00640 if ( sReading )
00641 {
00642
00643
00644
00645
00646 if (isDirectory && isSingleLocal)
00647 {
00648 QString dotDirectoryFile = m_lstItems.first()->url().path(1).append(".directory");
00649 KSimpleConfig cfg( dotDirectoryFile, true );
00650 cfg.setDesktopGroup();
00651
00652 if (KIOSKAuthorizedAction(cfg))
00653 {
00654 const QString priority = cfg.readEntry("X-KDE-Priority");
00655 const QString submenuName = cfg.readEntry( "X-KDE-Submenu" );
00656 ServiceList* list = s.selectList( priority, submenuName );
00657 (*list) += KDEDesktopMimeType::userDefinedServices( dotDirectoryFile, cfg, true );
00658 }
00659 }
00660
00661
00662 const QStringList entries = KGlobal::dirs()->findAllResources("data",
00663 "konqueror/servicemenus/*.desktop",
00664 false ,
00665 true );
00666 QStringList::ConstIterator eIt = entries.begin();
00667 const QStringList::ConstIterator eEnd = entries.end();
00668 for (; eIt != eEnd; ++eIt )
00669 {
00670 KSimpleConfig cfg( *eIt, true );
00671 cfg.setDesktopGroup();
00672
00673 if (!KIOSKAuthorizedAction(cfg))
00674 {
00675 continue;
00676 }
00677
00678 if ( cfg.hasKey( "X-KDE-ShowIfRunning" ) )
00679 {
00680 const QString app = cfg.readEntry( "X-KDE-ShowIfRunning" );
00681 if ( !kapp->dcopClient()->isApplicationRegistered( app.utf8() ) )
00682 continue;
00683 }
00684 if ( cfg.hasKey( "X-KDE-ShowIfDcopCall" ) )
00685 {
00686 QString dcopcall = cfg.readEntry( "X-KDE-ShowIfDcopCall" );
00687 const QCString app = dcopcall.section(' ', 0,0).utf8();
00688
00689
00690
00691
00692 QByteArray dataToSend;
00693 QDataStream dataStream(dataToSend, IO_WriteOnly);
00694 dataStream << m_lstPopupURLs;
00695
00696 QCString replyType;
00697 QByteArray replyData;
00698 QCString object = dcopcall.section(' ', 1,-2).utf8();
00699 QString function = dcopcall.section(' ', -1);
00700 if(!function.endsWith("(KURL::List)")) {
00701 kdWarning() << "Desktop file " << *eIt << " contains an invalid X-KDE-ShowIfDcopCall - the function must take the exact parameter (KURL::List) and must be specified." << endl;
00702 continue;
00703 }
00704
00705 if(!kapp->dcopClient()->call( app, object,
00706 function.utf8(),
00707 dataToSend, replyType, replyData, true, 1000))
00708 continue;
00709 if(replyType != "bool" || !replyData[0])
00710 continue;
00711
00712 }
00713 if ( cfg.hasKey( "X-KDE-Protocol" ) )
00714 {
00715 const QString protocol = cfg.readEntry( "X-KDE-Protocol" );
00716 if ( protocol != urlForServiceMenu.protocol() )
00717 continue;
00718 }
00719 else if ( cfg.hasKey( "X-KDE-Protocols" ) )
00720 {
00721 QStringList protocols = QStringList::split( "," , cfg.readEntry( "X-KDE-Protocols" ) );
00722 if ( !protocols.contains( urlForServiceMenu.protocol() ) )
00723 continue;
00724 }
00725 else if ( urlForServiceMenu.protocol() == "trash" || urlForServiceMenu.url().startsWith( "system:/trash" ) )
00726 {
00727
00728
00729
00730 continue;
00731 }
00732
00733 if ( cfg.hasKey( "X-KDE-Require" ) )
00734 {
00735 const QStringList capabilities = cfg.readListEntry( "X-KDE-Require" );
00736 if ( capabilities.contains( "Write" ) && !sWriting )
00737 continue;
00738 }
00739 if ( (cfg.hasKey( "Actions" ) || cfg.hasKey( "X-KDE-GetActionMenu") ) && cfg.hasKey( "ServiceTypes" ) )
00740 {
00741 const QStringList types = cfg.readListEntry( "ServiceTypes" );
00742 const QStringList excludeTypes = cfg.readListEntry( "ExcludeServiceTypes" );
00743 bool ok = false;
00744
00745
00746 for (QStringList::ConstIterator it = types.begin();
00747 it != types.end() && !ok;
00748 ++it)
00749 {
00750
00751 bool checkTheMimetypes = false;
00752 if (*it == "all/all" ||
00753 *it == "allfiles" )
00754 {
00755 checkTheMimetypes = true;
00756 }
00757
00758
00759 if (!ok &&
00760 !isDirectory &&
00761 *it == "all/allfiles")
00762 {
00763 checkTheMimetypes = true;
00764 }
00765
00766
00767 if (!ok &&
00768 (!m_sMimeType.isEmpty() &&
00769 *it == m_sMimeType) ||
00770 (!mimeGroup.isEmpty() &&
00771 ((*it).right(1) == "*" &&
00772 (*it).left((*it).find('/')) == mimeGroup)))
00773 {
00774 checkTheMimetypes = true;
00775 }
00776
00777 if (checkTheMimetypes)
00778 {
00779 ok = true;
00780 for (QStringList::ConstIterator itex = excludeTypes.begin(); itex != excludeTypes.end(); ++itex)
00781 {
00782 if( ((*itex).right(1) == "*" && (*itex).left((*itex).find('/')) == mimeGroup) ||
00783 ((*itex) == m_sMimeType) )
00784 {
00785 ok = false;
00786 break;
00787 }
00788 }
00789 }
00790 }
00791
00792 if ( ok )
00793 {
00794 const QString priority = cfg.readEntry("X-KDE-Priority");
00795 const QString submenuName = cfg.readEntry( "X-KDE-Submenu" );
00796
00797 ServiceList* list = s.selectList( priority, submenuName );
00798 (*list) += KDEDesktopMimeType::userDefinedServices( *eIt, cfg, url.isLocalFile(), m_lstPopupURLs );
00799 }
00800 }
00801 }
00802
00803 KTrader::OfferList offers;
00804
00805 if (kapp->authorizeKAction("openwith"))
00806 {
00807 QString constraint = "Type == 'Application' and DesktopEntryName != 'kfmclient' and DesktopEntryName != 'kfmclient_dir' and DesktopEntryName != 'kfmclient_html'";
00808 QString subConstraint = " and '%1' in ServiceTypes";
00809
00810 QStringList::ConstIterator it = mimeTypeList.begin();
00811 QStringList::ConstIterator end = mimeTypeList.end();
00812 Q_ASSERT( it != end );
00813 QString first = *it;
00814 ++it;
00815 while ( it != end ) {
00816 constraint += subConstraint.arg( *it );
00817 ++it;
00818 }
00819
00820 offers = KTrader::self()->query( first, constraint );
00821 }
00822
00824
00825 m_mapPopup.clear();
00826 m_mapPopupServices.clear();
00827
00828
00829 if ( !isDirectory || isLocal )
00830 {
00831 if ( hasAction() )
00832 addSeparator();
00833
00834 if ( !offers.isEmpty() )
00835 {
00836
00837 id = 1;
00838
00839 QDomElement menu = m_menuElement;
00840
00841 if ( offers.count() > 1 )
00842 {
00843 menu = m_doc.createElement( "menu" );
00844 menu.setAttribute( "name", "openwith submenu" );
00845 m_menuElement.appendChild( menu );
00846 QDomElement text = m_doc.createElement( "text" );
00847 menu.appendChild( text );
00848 text.appendChild( m_doc.createTextNode( i18n("&Open With") ) );
00849 }
00850
00851 KTrader::OfferList::ConstIterator it = offers.begin();
00852 for( ; it != offers.end(); it++ )
00853 {
00854 if ((*it)->noDisplay())
00855 continue;
00856
00857 QCString nam;
00858 nam.setNum( id );
00859
00860 QString actionName( (*it)->name().replace("&", "&&") );
00861 if ( menu == m_menuElement )
00862 actionName = i18n( "Open with %1" ).arg( actionName );
00863
00864 act = new KAction( actionName, (*it)->pixmap( KIcon::Small ), 0,
00865 this, SLOT( slotRunService() ),
00866 &m_ownActions, nam.prepend( "appservice_" ) );
00867 addAction( act, menu );
00868
00869 m_mapPopup[ id++ ] = *it;
00870 }
00871
00872 QString openWithActionName;
00873 if ( menu != m_menuElement )
00874 {
00875 addSeparator( menu );
00876 openWithActionName = i18n( "&Other..." );
00877 }
00878 else
00879 {
00880 openWithActionName = i18n( "&Open With..." );
00881 }
00882 KAction *openWithAct = new KAction( openWithActionName, 0, this, SLOT( slotPopupOpenWith() ), &m_ownActions, "openwith" );
00883 addAction( openWithAct, menu );
00884 }
00885 else
00886 {
00887 act = new KAction( i18n( "&Open With..." ), 0, this, SLOT( slotPopupOpenWith() ), &m_ownActions, "openwith" );
00888 addAction( act );
00889 }
00890
00891 }
00892 addGroup( "preview" );
00893 }
00894
00895
00896 QDomElement actionMenu = m_menuElement;
00897 int userItemCount = 0;
00898 if (s.user.count() + s.userSubmenus.count() +
00899 s.userPriority.count() + s.userPrioritySubmenus.count() > 1)
00900 {
00901
00902 actionMenu = m_doc.createElement( "menu" );
00903 actionMenu.setAttribute( "name", "actions submenu" );
00904 m_menuElement.appendChild( actionMenu );
00905 QDomElement text = m_doc.createElement( "text" );
00906 actionMenu.appendChild( text );
00907 text.appendChild( m_doc.createTextNode( i18n("Ac&tions") ) );
00908 }
00909
00910 userItemCount += insertServicesSubmenus(s.userPrioritySubmenus, actionMenu, false);
00911 userItemCount += insertServices(s.userPriority, actionMenu, false);
00912
00913
00914 if (userItemCount > 0 &&
00915 (s.user.count() > 0 ||
00916 s.userSubmenus.count() > 0 ||
00917 s.builtin.count() > 0) &&
00918 actionMenu.lastChild().toElement().tagName().lower() != "separator")
00919 {
00920 QDomElement separator = m_doc.createElement( "separator" );
00921 actionMenu.appendChild(separator);
00922 }
00923
00924 userItemCount += insertServicesSubmenus(s.userSubmenus, actionMenu, false);
00925 userItemCount += insertServices(s.user, actionMenu, false);
00926 userItemCount += insertServices(s.builtin, m_menuElement, true);
00927
00928 userItemCount += insertServicesSubmenus(s.userToplevelSubmenus, m_menuElement, false);
00929 userItemCount += insertServices(s.userToplevel, m_menuElement, false);
00930
00931 if ( userItemCount > 0 )
00932 {
00933 addPendingSeparator();
00934 }
00935
00936 if ( !isCurrentTrash && !isIntoTrash && !mediaFiles && sReading )
00937 addPlugins();
00938
00939 if ( KPropertiesDialog::canDisplay( m_lstItems ) && (kpf & ShowProperties) )
00940 {
00941 act = new KAction( i18n( "&Properties" ), 0, this, SLOT( slotPopupProperties() ),
00942 &m_ownActions, "properties" );
00943 addAction( act );
00944 }
00945
00946 while ( !m_menuElement.lastChild().isNull() &&
00947 m_menuElement.lastChild().toElement().tagName().lower() == "separator" )
00948 m_menuElement.removeChild( m_menuElement.lastChild() );
00949
00950 if ( isDirectory && isLocal )
00951 {
00952 if ( KFileShare::authorization() == KFileShare::Authorized )
00953 {
00954 addSeparator();
00955 act = new KAction( i18n("Share"), 0, this, SLOT( slotOpenShareFileDialog() ),
00956 &m_ownActions, "sharefile" );
00957 addAction( act );
00958 }
00959 }
00960
00961 addMerge( 0 );
00962
00963
00964 m_factory->addClient( this );
00965 }
00966
00967 void KonqPopupMenu::slotOpenShareFileDialog()
00968 {
00969 KPropertiesDialog* dlg = showPropertiesDialog();
00970 dlg->showFileSharingPage();
00971 }
00972
00973 KonqPopupMenu::~KonqPopupMenu()
00974 {
00975 m_pluginList.clear();
00976 delete m_factory;
00977 delete m_builder;
00978 delete d;
00979
00980 }
00981
00982 void KonqPopupMenu::setURLTitle( const QString& urlTitle )
00983 {
00984 d->m_urlTitle = urlTitle;
00985 }
00986
00987 void KonqPopupMenu::slotPopupNewView()
00988 {
00989 KURL::List::ConstIterator it = m_lstPopupURLs.begin();
00990 for ( ; it != m_lstPopupURLs.end(); it++ )
00991 (void) new KRun(*it);
00992 }
00993
00994 void KonqPopupMenu::slotPopupNewDir()
00995 {
00996 if (m_lstPopupURLs.empty())
00997 return;
00998
00999 KonqOperations::newDir(d->m_parentWidget, m_lstPopupURLs.first());
01000 }
01001
01002 void KonqPopupMenu::slotPopupEmptyTrashBin()
01003 {
01004 KonqOperations::emptyTrash();
01005 }
01006
01007 void KonqPopupMenu::slotPopupRestoreTrashedItems()
01008 {
01009 KonqOperations::restoreTrashedItems( m_lstPopupURLs );
01010 }
01011
01012 void KonqPopupMenu::slotPopupOpenWith()
01013 {
01014 KRun::displayOpenWithDialog( m_lstPopupURLs );
01015 }
01016
01017 void KonqPopupMenu::slotPopupAddToBookmark()
01018 {
01019 KBookmarkGroup root;
01020 if ( m_lstPopupURLs.count() == 1 ) {
01021 KURL url = m_lstPopupURLs.first();
01022 QString title = d->m_urlTitle.isEmpty() ? url.prettyURL() : d->m_urlTitle;
01023 root = m_pManager->addBookmarkDialog( url.prettyURL(), title );
01024 }
01025 else
01026 {
01027 root = m_pManager->root();
01028 KURL::List::ConstIterator it = m_lstPopupURLs.begin();
01029 for ( ; it != m_lstPopupURLs.end(); it++ )
01030 root.addBookmark( m_pManager, (*it).prettyURL(), (*it) );
01031 }
01032 m_pManager->emitChanged( root );
01033 }
01034
01035 void KonqPopupMenu::slotRunService()
01036 {
01037 QCString senderName = sender()->name();
01038 int id = senderName.mid( senderName.find( '_' ) + 1 ).toInt();
01039
01040
01041 QMap<int,KService::Ptr>::Iterator it = m_mapPopup.find( id );
01042 if ( it != m_mapPopup.end() )
01043 {
01044 KRun::run( **it, m_lstPopupURLs );
01045 return;
01046 }
01047
01048
01049 QMap<int,KDEDesktopMimeType::Service>::Iterator it2 = m_mapPopupServices.find( id );
01050 if ( it2 != m_mapPopupServices.end() )
01051 {
01052 KDEDesktopMimeType::executeService( m_lstPopupURLs, it2.data() );
01053 }
01054
01055 return;
01056 }
01057
01058 void KonqPopupMenu::slotPopupMimeType()
01059 {
01060 KonqOperations::editMimeType( m_sMimeType );
01061 }
01062
01063 void KonqPopupMenu::slotPopupProperties()
01064 {
01065 (void)showPropertiesDialog();
01066 }
01067
01068 KPropertiesDialog* KonqPopupMenu::showPropertiesDialog()
01069 {
01070
01071
01072
01073
01074 if ( m_lstItems.count() == 1 )
01075 {
01076 KFileItem * item = m_lstItems.first();
01077 if (item->entry().count() == 0)
01078 {
01079
01080 return new KPropertiesDialog( item->url(), d->m_parentWidget );
01081 }
01082 }
01083 return new KPropertiesDialog( m_lstItems, d->m_parentWidget );
01084 }
01085
01086 KAction *KonqPopupMenu::action( const QDomElement &element ) const
01087 {
01088 QCString name = element.attribute( attrName ).ascii();
01089 KAction *res = m_ownActions.action( name );
01090
01091 if ( !res )
01092 res = m_actions.action( name );
01093
01094 if ( !res && m_pMenuNew && strcmp( name, m_pMenuNew->name() ) == 0 )
01095 return m_pMenuNew;
01096
01097 return res;
01098 }
01099
01100 KActionCollection *KonqPopupMenu::actionCollection() const
01101 {
01102 return const_cast<KActionCollection *>( &m_ownActions );
01103 }
01104
01105 QString KonqPopupMenu::mimeType() const
01106 {
01107 return m_sMimeType;
01108 }
01109
01110 KonqPopupMenu::ProtocolInfo KonqPopupMenu::protocolInfo() const
01111 {
01112 return m_info;
01113 }
01114
01115 void KonqPopupMenu::addPlugins()
01116 {
01117
01118
01119 KTrader::OfferList plugin_offers;
01120 unsigned int pluginCount = 0;
01121 plugin_offers = KTrader::self()->query( m_sMimeType.isNull() ? QString::fromLatin1( "all/all" ) : m_sMimeType, "'KonqPopupMenu/Plugin' in ServiceTypes");
01122 if ( plugin_offers.isEmpty() )
01123 return;
01124
01125 KTrader::OfferList::ConstIterator iterator = plugin_offers.begin();
01126 KTrader::OfferList::ConstIterator end = plugin_offers.end();
01127
01128 addGroup( "plugins" );
01129
01130 for(; iterator != end; ++iterator, ++pluginCount ) {
01131
01132 KonqPopupMenuPlugin *plugin =
01133 KParts::ComponentFactory::
01134 createInstanceFromLibrary<KonqPopupMenuPlugin>( QFile::encodeName( (*iterator)->library() ),
01135 this,
01136 (*iterator)->name().latin1() );
01137 if ( !plugin )
01138 continue;
01139 QString pluginClientName = QString::fromLatin1( "Plugin%1" ).arg( pluginCount );
01140 addMerge( pluginClientName );
01141 plugin->domDocument().documentElement().setAttribute( "name", pluginClientName );
01142 m_pluginList.append( plugin );
01143 insertChildClient( plugin );
01144 }
01145
01146
01147 addMerge( "plugins" );
01148 }
01149
01150 KURL KonqPopupMenu::url() const
01151 {
01152 return m_sViewURL;
01153 }
01154
01155 KFileItemList KonqPopupMenu::fileItemList() const
01156 {
01157 return m_lstItems;
01158 }
01159
01160 KURL::List KonqPopupMenu::popupURLList() const
01161 {
01162 return m_lstPopupURLs;
01163 }
01164
01169 KonqPopupMenuPlugin::KonqPopupMenuPlugin( KonqPopupMenu *parent, const char *name )
01170 : QObject( parent, name )
01171 {
01172 }
01173
01174 KonqPopupMenuPlugin::~KonqPopupMenuPlugin()
01175 {
01176 }
01177
01178 #include "konq_popupmenu.moc"