summaryrefslogtreecommitdiffstats
path: root/kmenuedit
diff options
context:
space:
mode:
Diffstat (limited to 'kmenuedit')
-rw-r--r--kmenuedit/CMakeLists.txt58
-rw-r--r--kmenuedit/Makefile.am52
-rw-r--r--kmenuedit/basictab.cpp505
-rw-r--r--kmenuedit/basictab.h87
-rw-r--r--kmenuedit/hi16-app-kmenuedit.pngbin0 -> 679 bytes
-rw-r--r--kmenuedit/hi22-app-kmenuedit.pngbin0 -> 1485 bytes
-rw-r--r--kmenuedit/hi32-app-kmenuedit.pngbin0 -> 2481 bytes
-rw-r--r--kmenuedit/hi48-app-kmenuedit.pngbin0 -> 1963 bytes
-rw-r--r--kmenuedit/kcontrol_main.cpp58
-rw-r--r--kmenuedit/kcontroleditui.rc41
-rw-r--r--kmenuedit/khotkeys.cpp148
-rw-r--r--kmenuedit/khotkeys.h44
-rw-r--r--kmenuedit/kmenuedit.cpp194
-rw-r--r--kmenuedit/kmenuedit.desktop93
-rw-r--r--kmenuedit/kmenuedit.h63
-rw-r--r--kmenuedit/kmenueditui.rc45
-rw-r--r--kmenuedit/main.cpp88
-rw-r--r--kmenuedit/menufile.cpp552
-rw-r--r--kmenuedit/menufile.h114
-rw-r--r--kmenuedit/menuinfo.cpp502
-rw-r--r--kmenuedit/menuinfo.h192
-rw-r--r--kmenuedit/pixmaps/CMakeLists.txt13
-rw-r--r--kmenuedit/pixmaps/Makefile.am5
-rw-r--r--kmenuedit/pixmaps/cr22-action-filesave_and_close.pngbin0 -> 3263 bytes
-rw-r--r--kmenuedit/pixmaps/cr22-action-menu_new.pngbin0 -> 1247 bytes
-rw-r--r--kmenuedit/pixmaps/cr22-action-menu_new_sep.pngbin0 -> 162 bytes
-rw-r--r--kmenuedit/pixmaps/cr32-action-menu_new.pngbin0 -> 2481 bytes
-rw-r--r--kmenuedit/pixmaps/cr32-action-menu_new_sep.pngbin0 -> 596 bytes
-rw-r--r--kmenuedit/pixmaps/lo16-action-menu_new.pngbin0 -> 679 bytes
-rw-r--r--kmenuedit/treeview.cpp1581
-rw-r--r--kmenuedit/treeview.h179
-rw-r--r--kmenuedit/uninstall.desktop2
32 files changed, 4616 insertions, 0 deletions
diff --git a/kmenuedit/CMakeLists.txt b/kmenuedit/CMakeLists.txt
new file mode 100644
index 000000000..e864464f8
--- /dev/null
+++ b/kmenuedit/CMakeLists.txt
@@ -0,0 +1,58 @@
+#################################################
+#
+# (C) 2010-2011 Serghei Amelian
+# serghei (DOT) amelian (AT) gmail.com
+#
+# Improvements and feedback are welcome
+#
+# This file is released under GPL >= 2
+#
+#################################################
+
+add_subdirectory( pixmaps )
+
+include_directories(
+ ${CMAKE_CURRENT_BINARY_DIR}
+ ${CMAKE_CURRENT_SOURCE_DIR}
+ ${TDE_INCLUDE_DIR}
+ ${TQT_INCLUDE_DIRS}
+)
+
+link_directories(
+ ${TQT_LIBRARY_DIRS}
+)
+
+
+##### other data ################################
+
+tde_install_icons( kmenuedit )
+install( FILES kmenuedit.desktop DESTINATION ${XDG_APPS_INSTALL_DIR} )
+install( FILES uninstall.desktop DESTINATION ${APPS_INSTALL_DIR}/System RENAME kmenuedit.desktop )
+install( FILES kmenueditui.rc DESTINATION ${DATA_INSTALL_DIR}/kmenuedit )
+install( FILES kcontroleditui.rc DESTINATION ${DATA_INSTALL_DIR}/kcontroledit )
+
+
+##### kmenuedit (tdeinit) #######################
+
+tde_add_tdeinit_executable( kmenuedit AUTOMOC
+ SOURCES main.cpp
+ LINK ${TQT_LIBRARIES} kmenueditcommon-static
+)
+
+
+##### kcontroledit (tdeinit) ####################
+
+tde_add_tdeinit_executable( kcontroledit AUTOMOC
+ SOURCES kcontrol_main.cpp
+ LINK ${TQT_LIBRARIES} kmenueditcommon-static
+)
+
+
+##### kmenueditcommon (static) ##################
+
+tde_add_library( kmenueditcommon STATIC_PIC AUTOMOC
+ SOURCES
+ basictab.cpp treeview.cpp kmenuedit.cpp
+ khotkeys.cpp menufile.cpp menuinfo.cpp
+ LINK tdeio-shared
+)
diff --git a/kmenuedit/Makefile.am b/kmenuedit/Makefile.am
new file mode 100644
index 000000000..d385186e7
--- /dev/null
+++ b/kmenuedit/Makefile.am
@@ -0,0 +1,52 @@
+INCLUDES = $(all_includes)
+
+bin_PROGRAMS =
+lib_LTLIBRARIES =
+tdeinit_LTLIBRARIES = kmenuedit.la kcontroledit.la
+
+noinst_LTLIBRARIES = libkmenueditcommon.la
+
+CLEANFILES = dummy.cpp
+
+libkmenueditcommon_la_SOURCES = basictab.cpp treeview.cpp kmenuedit.cpp \
+ khotkeys.cpp menufile.cpp menuinfo.cpp
+
+libkmenueditcommon_la_LIBADD = $(LIB_TDEUI) $(LIB_TDEIO)
+libkmenueditcommon_la_LDFLAGS = $(all_libraries) -module -avoid-version
+
+kmenuedit_la_SOURCES = main.cpp
+kmenuedit_la_LIBADD = libkmenueditcommon.la
+kmenuedit_la_LDFLAGS = $(all_libraries) -module -avoid-version
+
+kcontroledit_la_SOURCES = kcontrol_main.cpp
+kcontroledit_la_LIBADD = libkmenueditcommon.la
+kcontroledit_la_LDFLAGS = $(all_libraries) -module -avoid-version
+
+noinst_HEADERS = kmenuedit.h treeview.h basictab.h khotkeys.h \
+ menufile.h menuinfo.h
+
+METASOURCES = AUTO
+
+xdg_apps_DATA = kmenuedit.desktop
+
+install-data-local: uninstall.desktop
+ $(mkinstalldirs) $(DESTDIR)$(kde_appsdir)/System
+ $(INSTALL_DATA) $(srcdir)/uninstall.desktop $(DESTDIR)$(kde_appsdir)/System/kmenuedit.desktop
+
+KDE_ICON = kmenuedit
+
+EXTRA_DIST = $(xdg_apps_DATA)
+
+rcdir = $(kde_datadir)/kmenuedit
+rc_DATA = kmenueditui.rc
+
+rc2dir = $(kde_datadir)/kcontroledit
+rc2_DATA = kcontroleditui.rc
+
+messages: rc.cpp
+ $(XGETTEXT) *.cpp -o $(podir)/kmenuedit.pot
+
+dummy.cpp:
+ echo > dummy.cpp
+
+SUBDIRS = pixmaps
diff --git a/kmenuedit/basictab.cpp b/kmenuedit/basictab.cpp
new file mode 100644
index 000000000..32dba3f3e
--- /dev/null
+++ b/kmenuedit/basictab.cpp
@@ -0,0 +1,505 @@
+/*
+ * Copyright (C) 2000 Matthias Elter <[email protected]>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ */
+
+
+#include <tqcheckbox.h>
+#include <tqlabel.h>
+#include <tqlayout.h>
+#include <tqfileinfo.h>
+#include <tqgroupbox.h>
+#include <tqhbox.h>
+#include <tqwhatsthis.h>
+
+#include <tdelocale.h>
+#include <kstandarddirs.h>
+#include <tdeglobal.h>
+#include <kdialog.h>
+#include <kkeybutton.h>
+#include <klineedit.h>
+#include <tdemessagebox.h>
+#include <kicondialog.h>
+#include <kdesktopfile.h>
+#include <kurlrequester.h>
+#include <tdefiledialog.h>
+#include <kcombobox.h>
+#include <kkeydialog.h>
+#include <kprocess.h>
+#include "khotkeys.h"
+
+#include "menuinfo.h"
+
+#include "basictab.h"
+#include "basictab.moc"
+
+BasicTab::BasicTab( TQWidget *parent, const char *name )
+ : TQWidget(parent, name)
+{
+ _menuFolderInfo = 0;
+ _menuEntryInfo = 0;
+
+ TQGridLayout *layout = new TQGridLayout(this, 6, 2,
+ KDialog::marginHint(),
+ KDialog::spacingHint());
+
+ // general group
+ TQGroupBox *general_group = new TQGroupBox(this);
+ TQGridLayout *grid = new TQGridLayout(general_group, 5, 2,
+ KDialog::marginHint(),
+ KDialog::spacingHint());
+
+ general_group->setAcceptDrops(false);
+
+ // setup line inputs
+ _nameEdit = new KLineEdit(general_group);
+ _nameEdit->setAcceptDrops(false);
+ _descriptionEdit = new KLineEdit(general_group);
+ _descriptionEdit->setAcceptDrops(false);
+ _commentEdit = new KLineEdit(general_group);
+ _commentEdit->setAcceptDrops(false);
+ _execEdit = new KURLRequester(general_group);
+ _execEdit->lineEdit()->setAcceptDrops(false);
+ TQWhatsThis::add(_execEdit,i18n(
+ "Following the command, you can have several place holders which will be replaced "
+ "with the actual values when the actual program is run:\n"
+ "%f - a single file name\n"
+ "%F - a list of files; use for applications that can open several local files at once\n"
+ "%u - a single URL\n"
+ "%U - a list of URLs\n"
+ "%d - the folder of the file to open\n"
+ "%D - a list of folders\n"
+ "%i - the icon\n"
+ "%m - the mini-icon\n"
+ "%c - the caption"));
+
+ _launchCB = new TQCheckBox(i18n("Enable &launch feedback"), general_group);
+ _systrayCB = new TQCheckBox(i18n("&Place in system tray"), general_group);
+
+ // setup labels
+ _nameLabel = new TQLabel(_nameEdit, i18n("&Name:"), general_group);
+ _descriptionLabel = new TQLabel(_descriptionEdit, i18n("&Description:"), general_group);
+ _commentLabel = new TQLabel(_commentEdit, i18n("&Comment:"), general_group);
+ _execLabel = new TQLabel(_execEdit, i18n("Co&mmand:"), general_group);
+ grid->addWidget(_nameLabel, 0, 0);
+ grid->addWidget(_descriptionLabel, 1, 0);
+ grid->addWidget(_commentLabel, 2, 0);
+ grid->addWidget(_execLabel, 3, 0);
+
+ // connect line inputs
+ connect(_nameEdit, TQT_SIGNAL(textChanged(const TQString&)),
+ TQT_SLOT(slotChanged()));
+ connect(_descriptionEdit, TQT_SIGNAL(textChanged(const TQString&)),
+ TQT_SLOT(slotChanged()));
+ connect(_commentEdit, TQT_SIGNAL(textChanged(const TQString&)),
+ TQT_SLOT(slotChanged()));
+ connect(_execEdit, TQT_SIGNAL(textChanged(const TQString&)),
+ TQT_SLOT(slotChanged()));
+ connect(_execEdit, TQT_SIGNAL(urlSelected(const TQString&)),
+ TQT_SLOT(slotExecSelected()));
+ connect(_launchCB, TQT_SIGNAL(clicked()), TQT_SLOT(launchcb_clicked()));
+ connect(_systrayCB, TQT_SIGNAL(clicked()), TQT_SLOT(systraycb_clicked()));
+
+ // add line inputs to the grid
+ grid->addMultiCellWidget(_nameEdit, 0, 0, 1, 1);
+ grid->addMultiCellWidget(_descriptionEdit, 1, 1, 1, 1);
+ grid->addMultiCellWidget(_commentEdit, 2, 2, 1, 2);
+ grid->addMultiCellWidget(_execEdit, 3, 3, 1, 2);
+ grid->addMultiCellWidget(_launchCB, 4, 4, 0, 2);
+ grid->addMultiCellWidget(_systrayCB, 5, 5, 0, 2);
+
+ // setup icon button
+ _iconButton = new TDEIconButton(general_group);
+ _iconButton->setFixedSize(56,56);
+ _iconButton->setIconSize(48);
+ connect(_iconButton, TQT_SIGNAL(iconChanged(TQString)), TQT_SLOT(slotChanged()));
+ grid->addMultiCellWidget(_iconButton, 0, 1, 2, 2);
+
+ // add the general group to the main layout
+ layout->addMultiCellWidget(general_group, 0, 0, 0, 1);
+
+ // path group
+ _path_group = new TQGroupBox(this);
+ TQVBoxLayout *vbox = new TQVBoxLayout(_path_group, KDialog::marginHint(),
+ KDialog::spacingHint());
+
+ TQHBox *hbox = new TQHBox(_path_group);
+ hbox->setSpacing(KDialog::spacingHint());
+
+ _pathLabel = new TQLabel(i18n("&Work path:"), hbox);
+
+ _pathEdit = new KURLRequester(hbox);
+ _pathEdit->setMode(KFile::Directory | KFile::LocalOnly);
+ _pathEdit->lineEdit()->setAcceptDrops(false);
+
+ _pathLabel->setBuddy(_pathEdit);
+
+ connect(_pathEdit, TQT_SIGNAL(textChanged(const TQString&)),
+ TQT_SLOT(slotChanged()));
+ vbox->addWidget(hbox);
+ layout->addMultiCellWidget(_path_group, 1, 1, 0, 1);
+
+ // terminal group
+ _term_group = new TQGroupBox(this);
+ vbox = new TQVBoxLayout(_term_group, KDialog::marginHint(),
+ KDialog::spacingHint());
+
+ _terminalCB = new TQCheckBox(i18n("Run in term&inal"), _term_group);
+ connect(_terminalCB, TQT_SIGNAL(clicked()), TQT_SLOT(termcb_clicked()));
+ vbox->addWidget(_terminalCB);
+
+ hbox = new TQHBox(_term_group);
+ hbox->setSpacing(KDialog::spacingHint());
+ _termOptLabel = new TQLabel(i18n("Terminal &options:"), hbox);
+ _termOptEdit = new KLineEdit(hbox);
+ _termOptEdit->setAcceptDrops(false);
+ _termOptLabel->setBuddy(_termOptEdit);
+
+ connect(_termOptEdit, TQT_SIGNAL(textChanged(const TQString&)),
+ TQT_SLOT(slotChanged()));
+ vbox->addWidget(hbox);
+ layout->addMultiCellWidget(_term_group, 2, 2, 0, 1);
+
+ _termOptEdit->setEnabled(false);
+
+ // uid group
+ _uid_group = new TQGroupBox(this);
+ vbox = new TQVBoxLayout(_uid_group, KDialog::marginHint(),
+ KDialog::spacingHint());
+
+ _uidCB = new TQCheckBox(i18n("&Run as a different user"), _uid_group);
+ connect(_uidCB, TQT_SIGNAL(clicked()), TQT_SLOT(uidcb_clicked()));
+ vbox->addWidget(_uidCB);
+
+ hbox = new TQHBox(_uid_group);
+ hbox->setSpacing(KDialog::spacingHint());
+ _uidLabel = new TQLabel(i18n("&Username:"), hbox);
+ _uidEdit = new KLineEdit(hbox);
+ _uidEdit->setAcceptDrops(false);
+ _uidLabel->setBuddy(_uidEdit);
+
+ connect(_uidEdit, TQT_SIGNAL(textChanged(const TQString&)),
+ TQT_SLOT(slotChanged()));
+ vbox->addWidget(hbox);
+ layout->addMultiCellWidget(_uid_group, 3, 3, 0, 1);
+
+ _uidEdit->setEnabled(false);
+
+ layout->setRowStretch(0, 2);
+
+ // key binding group
+ general_group_keybind = new TQGroupBox(this);
+ layout->addMultiCellWidget( general_group_keybind, 4, 4, 0, 1 );
+ // dummy widget in order to make it look a bit better
+ layout->addWidget( new TQWidget(this), 5, 0 );
+ layout->setRowStretch( 5, 4 );
+ TQGridLayout *grid_keybind = new TQGridLayout(general_group_keybind, 3, 1,
+ KDialog::marginHint(),
+ KDialog::spacingHint());
+
+ //_keyEdit = new KLineEdit(general_group_keybind);
+ //_keyEdit->setReadOnly( true );
+ //_keyEdit->setText( "" );
+ //TQPushButton* _keyButton = new TQPushButton( i18n( "Change" ),
+ // general_group_keybind );
+ //connect( _keyButton, TQT_SIGNAL( clicked()), this, TQT_SLOT( keyButtonPressed()));
+ _keyEdit = new KKeyButton(general_group_keybind);
+ grid_keybind->addWidget(new TQLabel(_keyEdit, i18n("Current shortcut &key:"), general_group_keybind), 0, 0);
+ connect( _keyEdit, TQT_SIGNAL(capturedShortcut(const TDEShortcut&)),
+ this, TQT_SLOT(slotCapturedShortcut(const TDEShortcut&)));
+ grid_keybind->addWidget(_keyEdit, 0, 1);
+ //grid_keybind->addWidget(_keyButton, 0, 2 );
+
+ if (!KHotKeys::present())
+ general_group_keybind->hide();
+
+ slotDisableAction();
+}
+
+void BasicTab::slotDisableAction()
+{
+ //disable all group at the begining.
+ //because there is not file selected.
+ _nameEdit->setEnabled(false);
+ _descriptionEdit->setEnabled(false);
+ _commentEdit->setEnabled(false);
+ _execEdit->setEnabled(false);
+ _launchCB->setEnabled(false);
+ _systrayCB->setEnabled(false);
+ _nameLabel->setEnabled(false);
+ _descriptionLabel->setEnabled(false);
+ _commentLabel->setEnabled(false);
+ _execLabel->setEnabled(false);
+ _path_group->setEnabled(false);
+ _term_group->setEnabled(false);
+ _uid_group->setEnabled(false);
+ _iconButton->setEnabled(false);
+ // key binding part
+ general_group_keybind->setEnabled( false );
+}
+
+void BasicTab::enableWidgets(bool isDF, bool isDeleted)
+{
+ // set only basic attributes if it is not a .desktop file
+ _nameEdit->setEnabled(!isDeleted);
+ _descriptionEdit->setEnabled(!isDeleted);
+ _commentEdit->setEnabled(!isDeleted);
+ _iconButton->setEnabled(!isDeleted);
+ _execEdit->setEnabled(isDF && !isDeleted);
+ _launchCB->setEnabled(isDF && !isDeleted);
+ _systrayCB->setEnabled(isDF && !isDeleted);
+ _nameLabel->setEnabled(!isDeleted);
+ _descriptionLabel->setEnabled(!isDeleted);
+ _commentLabel->setEnabled(!isDeleted);
+ _execLabel->setEnabled(isDF && !isDeleted);
+
+ _path_group->setEnabled(isDF && !isDeleted);
+ _term_group->setEnabled(isDF && !isDeleted);
+ _uid_group->setEnabled(isDF && !isDeleted);
+ general_group_keybind->setEnabled( isDF && !isDeleted );
+
+ _termOptEdit->setEnabled(isDF && !isDeleted && _terminalCB->isChecked());
+ _termOptLabel->setEnabled(isDF && !isDeleted && _terminalCB->isChecked());
+
+ _uidEdit->setEnabled(isDF && !isDeleted && _uidCB->isChecked());
+ _uidLabel->setEnabled(isDF && !isDeleted && _uidCB->isChecked());
+}
+
+void BasicTab::setFolderInfo(MenuFolderInfo *folderInfo)
+{
+ blockSignals(true);
+ _menuFolderInfo = folderInfo;
+ _menuEntryInfo = 0;
+
+ _nameEdit->setText(folderInfo->caption);
+ _descriptionEdit->setText(folderInfo->genericname);
+ _descriptionEdit->setCursorPosition(0);
+ _commentEdit->setText(folderInfo->comment);
+ _commentEdit->setCursorPosition(0);
+ _iconButton->setIcon(folderInfo->icon);
+
+ // clean all disabled fields and return
+ _execEdit->lineEdit()->setText("");
+ _pathEdit->lineEdit()->setText("");
+ _termOptEdit->setText("");
+ _uidEdit->setText("");
+ _launchCB->setChecked(false);
+ _systrayCB->setChecked(false);
+ _terminalCB->setChecked(false);
+ _uidCB->setChecked(false);
+ _keyEdit->setShortcut(0, false);
+
+ enableWidgets(false, folderInfo->hidden);
+ blockSignals(false);
+}
+
+void BasicTab::setEntryInfo(MenuEntryInfo *entryInfo)
+{
+ blockSignals(true);
+ _menuFolderInfo = 0;
+ _menuEntryInfo = entryInfo;
+
+ if (!entryInfo)
+ {
+ _nameEdit->setText(TQString::null);
+ _descriptionEdit->setText(TQString::null);
+ _commentEdit->setText(TQString::null);
+ _iconButton->setIcon(TQString::null);
+
+ // key binding part
+ _keyEdit->setShortcut( TDEShortcut(), false );
+ _execEdit->lineEdit()->setText(TQString::null);
+ _systrayCB->setChecked(false);
+
+ _pathEdit->lineEdit()->setText(TQString::null);
+ _termOptEdit->setText(TQString::null);
+ _uidEdit->setText(TQString::null);
+
+ _launchCB->setChecked(false);
+ _terminalCB->setChecked(false);
+ _uidCB->setChecked(false);
+ enableWidgets(true, true);
+ blockSignals(false);
+ return;
+ }
+
+ KDesktopFile *df = entryInfo->desktopFile();
+
+ _nameEdit->setText(df->readName());
+ _descriptionEdit->setText(df->readGenericName());
+ _descriptionEdit->setCursorPosition(0);
+ _commentEdit->setText(df->readComment());
+ _commentEdit->setCursorPosition(0);
+ _iconButton->setIcon(df->readIcon());
+
+ // key binding part
+ if( KHotKeys::present())
+ {
+ _keyEdit->setShortcut( entryInfo->shortcut(), false );
+ }
+
+ TQString temp = df->readPathEntry("Exec");
+ if (temp.left(12) == "ksystraycmd ")
+ {
+ _execEdit->lineEdit()->setText(temp.right(temp.length()-12));
+ _systrayCB->setChecked(true);
+ }
+ else
+ {
+ _execEdit->lineEdit()->setText(temp);
+ _systrayCB->setChecked(false);
+ }
+
+ _pathEdit->lineEdit()->setText(df->readPath());
+ _termOptEdit->setText(df->readEntry("TerminalOptions"));
+ if( df->hasKey( "X-TDE-Username" )) {
+ _uidEdit->setText(df->readEntry("X-TDE-Username"));
+ }
+ else {
+ _uidEdit->setText(df->readEntry("X-KDE-Username"));
+ }
+
+ if( df->hasKey( "StartupNotify" ))
+ _launchCB->setChecked(df->readBoolEntry("StartupNotify", true));
+ else // backwards comp.
+ _launchCB->setChecked(df->readBoolEntry("X-TDE-StartupNotify", true));
+
+ if(df->readNumEntry("Terminal", 0) == 1)
+ _terminalCB->setChecked(true);
+ else
+ _terminalCB->setChecked(false);
+
+ _uidCB->setChecked(df->readBoolEntry("X-TDE-SubstituteUID", false) || df->readBoolEntry("X-KDE-SubstituteUID", false));
+
+ enableWidgets(true, entryInfo->hidden);
+ blockSignals(false);
+}
+
+void BasicTab::apply()
+{
+ if (_menuEntryInfo)
+ {
+ _menuEntryInfo->setDirty();
+ _menuEntryInfo->setCaption(_nameEdit->text());
+ _menuEntryInfo->setDescription(_descriptionEdit->text());
+ _menuEntryInfo->setIcon(_iconButton->icon());
+
+ KDesktopFile *df = _menuEntryInfo->desktopFile();
+ df->writeEntry("Comment", _commentEdit->text());
+ if (_systrayCB->isChecked())
+ df->writePathEntry("Exec", _execEdit->lineEdit()->text().prepend("ksystraycmd "));
+ else
+ df->writePathEntry("Exec", _execEdit->lineEdit()->text());
+
+ df->writePathEntry("Path", _pathEdit->lineEdit()->text());
+
+ if (_terminalCB->isChecked())
+ df->writeEntry("Terminal", 1);
+ else
+ df->writeEntry("Terminal", 0);
+
+ df->writeEntry("TerminalOptions", _termOptEdit->text());
+ df->writeEntry("X-TDE-SubstituteUID", _uidCB->isChecked());
+ df->writeEntry("X-TDE-Username", _uidEdit->text());
+ df->writeEntry("StartupNotify", _launchCB->isChecked());
+ }
+ else
+ {
+ _menuFolderInfo->setCaption(_nameEdit->text());
+ _menuFolderInfo->setGenericName(_descriptionEdit->text());
+ _menuFolderInfo->setComment(_commentEdit->text());
+ _menuFolderInfo->setIcon(_iconButton->icon());
+ }
+}
+
+void BasicTab::slotChanged()
+{
+ if (signalsBlocked())
+ return;
+ apply();
+ if (_menuEntryInfo)
+ emit changed( _menuEntryInfo );
+ else
+ emit changed( _menuFolderInfo );
+}
+
+void BasicTab::launchcb_clicked()
+{
+ slotChanged();
+}
+
+void BasicTab::systraycb_clicked()
+{
+ slotChanged();
+}
+
+void BasicTab::termcb_clicked()
+{
+ _termOptEdit->setEnabled(_terminalCB->isChecked());
+ _termOptLabel->setEnabled(_terminalCB->isChecked());
+ slotChanged();
+}
+
+void BasicTab::uidcb_clicked()
+{
+ _uidEdit->setEnabled(_uidCB->isChecked());
+ _uidLabel->setEnabled(_uidCB->isChecked());
+ slotChanged();
+}
+
+void BasicTab::slotExecSelected()
+{
+ TQString path = _execEdit->lineEdit()->text();
+ if (!path.startsWith("'"))
+ _execEdit->lineEdit()->setText(TDEProcess::quote(path));
+}
+
+void BasicTab::slotCapturedShortcut(const TDEShortcut& cut)
+{
+ if (signalsBlocked())
+ return;
+
+ if( KKeyChooser::checkGlobalShortcutsConflict( cut, true, topLevelWidget())
+ || KKeyChooser::checkStandardShortcutsConflict( cut, true, topLevelWidget()))
+ return;
+
+ if ( KHotKeys::present() )
+ {
+ if (!_menuEntryInfo->isShortcutAvailable( cut ) )
+ {
+ KService::Ptr service;
+ emit findServiceShortcut(cut, service);
+ if (!service)
+ service = KHotKeys::findMenuEntry(cut.toString());
+ if (service)
+ {
+ KMessageBox::sorry(this, i18n("<qt>The key <b>%1</b> can not be used here because it is already used to activate <b>%2</b>.").arg(cut.toString(), service->name()));
+ return;
+ }
+ else
+ {
+ KMessageBox::sorry(this, i18n("<qt>The key <b>%1</b> can not be used here because it is already in use.").arg(cut.toString()));
+ return;
+ }
+ }
+ _menuEntryInfo->setShortcut( cut );
+ }
+ _keyEdit->setShortcut(cut, false);
+ if (_menuEntryInfo)
+ emit changed( _menuEntryInfo );
+}
+
diff --git a/kmenuedit/basictab.h b/kmenuedit/basictab.h
new file mode 100644
index 000000000..724f1f339
--- /dev/null
+++ b/kmenuedit/basictab.h
@@ -0,0 +1,87 @@
+/*
+ * Copyright (C) 2000 Matthias Elter <[email protected]>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ */
+
+#ifndef __basictab_h__
+#define __basictab_h__
+
+#include <tqwidget.h>
+#include <tqstring.h>
+
+#include <klineedit.h>
+
+class KKeyButton;
+class KLineEdit;
+class TDEIconButton;
+class TQCheckBox;
+class TQGroupBox;
+class TQLabel;
+class KURLRequester;
+class KComboBox;
+class KService;
+
+class MenuFolderInfo;
+class MenuEntryInfo;
+
+class BasicTab : public TQWidget
+{
+ Q_OBJECT
+
+public:
+ BasicTab( TQWidget *parent=0, const char *name=0 );
+
+ void apply();
+signals:
+ void changed( MenuFolderInfo * );
+ void changed( MenuEntryInfo * );
+ void findServiceShortcut(const TDEShortcut&, KService::Ptr &);
+
+public slots:
+ void setFolderInfo(MenuFolderInfo *folderInfo);
+ void setEntryInfo(MenuEntryInfo *entryInfo);
+ void slotDisableAction();
+protected slots:
+ void slotChanged();
+ void launchcb_clicked();
+ void systraycb_clicked();
+ void termcb_clicked();
+ void uidcb_clicked();
+ void slotCapturedShortcut(const TDEShortcut&);
+ void slotExecSelected();
+
+protected:
+ void enableWidgets(bool isDF, bool isDeleted);
+
+protected:
+ KLineEdit *_nameEdit, *_commentEdit;
+ KLineEdit *_descriptionEdit;
+ KKeyButton *_keyEdit;
+ KURLRequester *_execEdit, *_pathEdit;
+ KLineEdit *_termOptEdit, *_uidEdit;
+ TQCheckBox *_terminalCB, *_uidCB, *_launchCB, *_systrayCB;
+ TDEIconButton *_iconButton;
+ TQGroupBox *_path_group, *_term_group, *_uid_group, *general_group_keybind;
+ TQLabel *_termOptLabel, *_uidLabel, *_pathLabel, *_nameLabel, *_commentLabel, *_execLabel;
+ TQLabel *_descriptionLabel;
+
+ MenuFolderInfo *_menuFolderInfo;
+ MenuEntryInfo *_menuEntryInfo;
+ bool _isDeleted;
+};
+
+#endif
diff --git a/kmenuedit/hi16-app-kmenuedit.png b/kmenuedit/hi16-app-kmenuedit.png
new file mode 100644
index 000000000..b121605f7
--- /dev/null
+++ b/kmenuedit/hi16-app-kmenuedit.png
Binary files differ
diff --git a/kmenuedit/hi22-app-kmenuedit.png b/kmenuedit/hi22-app-kmenuedit.png
new file mode 100644
index 000000000..65509221a
--- /dev/null
+++ b/kmenuedit/hi22-app-kmenuedit.png
Binary files differ
diff --git a/kmenuedit/hi32-app-kmenuedit.png b/kmenuedit/hi32-app-kmenuedit.png
new file mode 100644
index 000000000..d1e532be1
--- /dev/null
+++ b/kmenuedit/hi32-app-kmenuedit.png
Binary files differ
diff --git a/kmenuedit/hi48-app-kmenuedit.png b/kmenuedit/hi48-app-kmenuedit.png
new file mode 100644
index 000000000..e6b483a7f
--- /dev/null
+++ b/kmenuedit/hi48-app-kmenuedit.png
Binary files differ
diff --git a/kmenuedit/kcontrol_main.cpp b/kmenuedit/kcontrol_main.cpp
new file mode 100644
index 000000000..cd580e02e
--- /dev/null
+++ b/kmenuedit/kcontrol_main.cpp
@@ -0,0 +1,58 @@
+/*
+ * Copyright (C) 2000 Matthias Elter <[email protected]>
+ * Copyright (C) 2001-2002 Raffaele Sandrini <[email protected]>
+ * Copyright (C) 2004 Waldo Bastian <[email protected]>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ */
+
+#include <unistd.h>
+
+#include <kuniqueapplication.h>
+#include <tdelocale.h>
+#include <tdecmdlineargs.h>
+#include <tdeaboutdata.h>
+#include <kstandarddirs.h>
+
+#include "kmenuedit.h"
+
+static const char description[] = I18N_NOOP("TDE control center editor");
+static const char version[] = "1.0";
+
+extern "C" int KDE_EXPORT kdemain( int argc, char **argv )
+{
+ TDELocale::setMainCatalogue("kmenuedit");
+ TDEAboutData aboutData("kcontroledit", I18N_NOOP("Trinity Control Center Editor"),
+ version, description, TDEAboutData::License_GPL,
+ "(C) 2000-2004, Waldo Bastian, Raffaele Sandrini, Matthias Elter");
+ aboutData.addAuthor("Waldo Bastian", I18N_NOOP("Maintainer"), "[email protected]");
+ aboutData.addAuthor("Raffaele Sandrini", I18N_NOOP("Previous Maintainer"), "[email protected]");
+ aboutData.addAuthor("Matthias Elter", I18N_NOOP("Original Author"), "[email protected]");
+
+ TDECmdLineArgs::init( argc, argv, &aboutData );
+ KUniqueApplication::addCmdLineOptions();
+
+ if (!KUniqueApplication::start())
+ return 1;
+
+ KUniqueApplication app;
+
+ KMenuEdit *menuEdit = new KMenuEdit(true);
+ menuEdit->show();
+
+ app.setMainWidget(menuEdit);
+ return app.exec();
+}
diff --git a/kmenuedit/kcontroleditui.rc b/kmenuedit/kcontroleditui.rc
new file mode 100644
index 000000000..8e621a98a
--- /dev/null
+++ b/kmenuedit/kcontroleditui.rc
@@ -0,0 +1,41 @@
+<!DOCTYPE kpartgui SYSTEM "kpartgui.dtd">
+<kpartgui name="kcontroledit" version="4">
+
+<MenuBar>
+
+<Menu name="file" noMerge="1"><text>&amp;File</text>
+ <Action name="newitem"/>
+ <Action name="newsubmenu" />
+ <Separator/>
+ <Action name="file_save"/>
+ <Action name="file_save_and_quit"/>
+ <Separator/>
+ <Action name="file_quit"/>
+</Menu>
+
+<Menu name="edit" noMerge="1"><text>&amp;Edit</text>
+ <Action name="edit_cut"/>
+ <Action name="edit_copy"/>
+ <Action name="edit_paste"/>
+ <Separator/>
+ <Action name="delete"/>
+</Menu>
+
+<Menu name="settings">
+ <Action name="show_removed"/>
+ <Action name="show_hidden"/>
+</Menu>
+</MenuBar>
+
+<ToolBar name="mainToolBar" noMerge="1" fullWidth="true"><text>Main Toolbar</text>
+ <Action name="newitem"/>
+ <Action name="newsubmenu"/>
+ <Separator/>
+ <Action name="edit_cut"/>
+ <Action name="edit_copy"/>
+ <Action name="edit_paste"/>
+ <Separator/>
+ <Action name="delete"/>
+</ToolBar>
+
+</kpartgui>
diff --git a/kmenuedit/khotkeys.cpp b/kmenuedit/khotkeys.cpp
new file mode 100644
index 000000000..9b0ead3c7
--- /dev/null
+++ b/kmenuedit/khotkeys.cpp
@@ -0,0 +1,148 @@
+/*
+ * Copyright (C) 2000 Matthias Elter <[email protected]>
+ * Lubos Lunak <[email protected]>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ */
+
+#include <klibloader.h>
+#include "khotkeys.h"
+
+extern "C"
+{
+ static void (*khotkeys_init_2)( void );
+ static void (*khotkeys_cleanup_2)( void );
+ static TQString (*khotkeys_get_menu_entry_shortcut_2)( const TQString& entry_P );
+ static TQString (*khotkeys_change_menu_entry_shortcut_2)( const TQString& entry_P,
+ const TQString& shortcut_P );
+ static bool (*khotkeys_menu_entry_moved_2)( const TQString& new_P, const TQString& old_P );
+ static void (*khotkeys_menu_entry_deleted_2)( const TQString& entry_P );
+ static TQStringList (*khotkeys_get_all_shortcuts_2)( );
+ static KService::Ptr (*khotkeys_find_menu_entry_2)( const TQString& shortcut_P );
+}
+
+static bool khotkeys_present = false;
+static bool khotkeys_inited = false;
+
+bool KHotKeys::init()
+{
+ khotkeys_inited = true;
+ KLibrary* lib = KLibLoader::self()->library( "kcm_khotkeys.la" );
+ if( lib == NULL ) return false;
+
+ khotkeys_init_2 = ( void (*)(void)) ( lib->symbol( "khotkeys_init" ));
+ khotkeys_cleanup_2 = ( void (*)(void)) ( lib->symbol( "khotkeys_cleanup" ));
+ khotkeys_get_menu_entry_shortcut_2 =
+ ( TQString (*)( const TQString& ))
+ ( lib->symbol( "khotkeys_get_menu_entry_shortcut" ));
+ khotkeys_change_menu_entry_shortcut_2 =
+ ( TQString (*)( const TQString&, const TQString& ))
+ ( lib->symbol( "khotkeys_change_menu_entry_shortcut" ));
+ khotkeys_menu_entry_moved_2 =
+ ( bool (*)( const TQString&, const TQString& ))
+ ( lib->symbol( "khotkeys_menu_entry_moved" ));
+ khotkeys_menu_entry_deleted_2 =
+ ( void (*)( const TQString& ))
+ ( lib->symbol( "khotkeys_menu_entry_deleted" ));
+ khotkeys_get_all_shortcuts_2 =
+ ( TQStringList (*)( ))
+ ( lib->symbol( "khotkeys_get_all_shortcuts" ));
+ khotkeys_find_menu_entry_2 =
+ ( KService::Ptr (*)( const TQString& ))
+ ( lib->symbol( "khotkeys_find_menu_entry" ));
+
+ if( khotkeys_init_2
+ && khotkeys_cleanup_2
+ && khotkeys_get_menu_entry_shortcut_2
+ && khotkeys_change_menu_entry_shortcut_2
+ && khotkeys_menu_entry_moved_2
+ && khotkeys_menu_entry_deleted_2 )
+ {
+ khotkeys_init_2();
+ khotkeys_present = true;
+ return true;
+ }
+ return false;
+}
+
+void KHotKeys::cleanup()
+{
+ if( khotkeys_inited && khotkeys_present )
+ khotkeys_cleanup_2();
+ khotkeys_inited = false;
+}
+
+bool KHotKeys::present()
+{
+ if( !khotkeys_inited )
+ init();
+ return khotkeys_present;
+}
+
+TQString KHotKeys::getMenuEntryShortcut( const TQString& entry_P )
+{
+ if( !khotkeys_inited )
+ init();
+ if( !khotkeys_present )
+ return "";
+ return khotkeys_get_menu_entry_shortcut_2( entry_P );
+}
+
+TQString KHotKeys::changeMenuEntryShortcut( const TQString& entry_P,
+ const TQString shortcut_P )
+ {
+ if( !khotkeys_inited )
+ init();
+ if( !khotkeys_present )
+ return "";
+ return khotkeys_change_menu_entry_shortcut_2( entry_P, shortcut_P );
+ }
+
+bool KHotKeys::menuEntryMoved( const TQString& new_P, const TQString& old_P )
+{
+ if( !khotkeys_inited )
+ init();
+ if( !khotkeys_present )
+ return "";
+ return khotkeys_menu_entry_moved_2( new_P, old_P );
+}
+
+void KHotKeys::menuEntryDeleted( const TQString& entry_P )
+{
+ if( !khotkeys_inited )
+ init();
+ if( !khotkeys_present )
+ return;
+ khotkeys_menu_entry_deleted_2( entry_P );
+}
+
+TQStringList KHotKeys::allShortCuts( )
+{
+ if( !khotkeys_inited )
+ init();
+ if (!khotkeys_get_all_shortcuts_2)
+ return TQStringList();
+ return khotkeys_get_all_shortcuts_2();
+}
+
+KService::Ptr KHotKeys::findMenuEntry( const TQString &shortcut_P )
+{
+ if( !khotkeys_inited )
+ init();
+ if (!khotkeys_find_menu_entry_2)
+ return 0;
+ return khotkeys_find_menu_entry_2(shortcut_P);
+}
diff --git a/kmenuedit/khotkeys.h b/kmenuedit/khotkeys.h
new file mode 100644
index 000000000..eb9a0844b
--- /dev/null
+++ b/kmenuedit/khotkeys.h
@@ -0,0 +1,44 @@
+/*
+ * Copyright (C) 2000 Matthias Elter <[email protected]>
+ * Lubos Lunak <[email protected]>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ */
+
+#ifndef __khotkeys_public_h__
+#define __khotkeys_public_h__
+
+#include <tqstring.h>
+#include <kservice.h>
+
+// see tdebase/khotkeys/kcontrol for info on these
+
+class KHotKeys
+{
+public:
+ static bool init();
+ static void cleanup();
+ static bool present();
+ static TQString getMenuEntryShortcut( const TQString& entry_P );
+ static TQString changeMenuEntryShortcut( const TQString& entry_P,
+ const TQString shortcut_P );
+ static bool menuEntryMoved( const TQString& new_P, const TQString& old_P );
+ static void menuEntryDeleted( const TQString& entry_P );
+ static TQStringList allShortCuts( );
+ static KService::Ptr findMenuEntry( const TQString &shortcut_P );
+};
+
+#endif
diff --git a/kmenuedit/kmenuedit.cpp b/kmenuedit/kmenuedit.cpp
new file mode 100644
index 000000000..8a90cfcd0
--- /dev/null
+++ b/kmenuedit/kmenuedit.cpp
@@ -0,0 +1,194 @@
+/*
+ * Copyright (C) 2000 Matthias Elter <[email protected]>
+ * Copyright (C) 2001-2002 Raffaele Sandrini <[email protected])
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ */
+
+#include <tqsplitter.h>
+
+#include <tdeaction.h>
+#include <tdeapplication.h>
+#include <tdeconfig.h>
+#include <kdebug.h>
+#include <kedittoolbar.h>
+#include <tdeglobal.h>
+#include <tdelocale.h>
+#include <tdemessagebox.h>
+#include <kservice.h>
+#include <kstdaction.h>
+#include <tdestdaccel.h>
+
+#include "treeview.h"
+#include "basictab.h"
+#include "kmenuedit.h"
+#include "kmenuedit.moc"
+
+KMenuEdit::KMenuEdit (bool controlCenter, TQWidget *, const char *name)
+ : TDEMainWindow (0, name), m_tree(0), m_basicTab(0), m_splitter(0), m_controlCenter(controlCenter)
+{
+#if 0
+ m_showHidden = config->readBoolEntry("ShowHidden");
+#else
+ m_showHidden = false;
+#endif
+
+ // setup GUI
+ setupActions();
+ slotChangeView();
+}
+
+KMenuEdit::~KMenuEdit()
+{
+ TDEConfig *config = TDEGlobal::config();
+ config->setGroup("General");
+ config->writeEntry("SplitterSizes", m_splitter->sizes());
+
+ config->sync();
+}
+
+void KMenuEdit::setupActions()
+{
+ (void)new TDEAction(i18n("&New Submenu..."), "menu_new", 0, actionCollection(), "newsubmenu");
+ (void)new TDEAction(i18n("New &Item..."), "document-new", TDEStdAccel::openNew(), actionCollection(), "newitem");
+ if (!m_controlCenter)
+ (void)new TDEAction(i18n("New S&eparator"), "menu_new_sep", 0, actionCollection(), "newsep");
+
+ (void)new TDEAction(i18n("Save && Quit"), "filesave_and_close", 0, TQT_TQOBJECT(this), TQT_SLOT( slotSave_and_close()), actionCollection(), "file_save_and_quit");
+
+ m_actionDelete = 0;
+
+ KStdAction::save(TQT_TQOBJECT(this), TQT_SLOT( slotSave() ), actionCollection());
+ KStdAction::quit(TQT_TQOBJECT(this), TQT_SLOT( close() ), actionCollection());
+ KStdAction::cut(0, 0, actionCollection());
+ KStdAction::copy(0, 0, actionCollection());
+ KStdAction::paste(0, 0, actionCollection());
+}
+
+void KMenuEdit::setupView()
+{
+ m_splitter = new TQSplitter(Qt::Horizontal, this);
+ m_tree = new TreeView(m_controlCenter, actionCollection(), m_splitter);
+ m_basicTab = new BasicTab(m_splitter);
+
+ connect(m_tree, TQT_SIGNAL(entrySelected(MenuFolderInfo *)),
+ m_basicTab, TQT_SLOT(setFolderInfo(MenuFolderInfo *)));
+ connect(m_tree, TQT_SIGNAL(entrySelected(MenuEntryInfo *)),
+ m_basicTab, TQT_SLOT(setEntryInfo(MenuEntryInfo *)));
+ connect(m_tree, TQT_SIGNAL(disableAction()),
+ m_basicTab, TQT_SLOT(slotDisableAction() ) );
+
+ connect(m_basicTab, TQT_SIGNAL(changed(MenuFolderInfo *)),
+ m_tree, TQT_SLOT(currentChanged(MenuFolderInfo *)));
+
+ connect(m_basicTab, TQT_SIGNAL(changed(MenuEntryInfo *)),
+ m_tree, TQT_SLOT(currentChanged(MenuEntryInfo *)));
+
+ connect(m_basicTab, TQT_SIGNAL(findServiceShortcut(const TDEShortcut&, KService::Ptr &)),
+ m_tree, TQT_SLOT(findServiceShortcut(const TDEShortcut&, KService::Ptr &)));
+
+ // restore splitter sizes
+ TDEConfig* config = TDEGlobal::config();
+ TQValueList<int> sizes = config->readIntListEntry("SplitterSizes");
+
+ if (sizes.isEmpty())
+ sizes << 1 << 3;
+ m_splitter->setSizes(sizes);
+ m_tree->setFocus();
+
+ setCentralWidget(m_splitter);
+}
+
+void KMenuEdit::slotChangeView()
+{
+#if 0
+ m_showHidden = m_actionShowHidden->isChecked();
+#else
+ m_showHidden = false;
+#endif
+
+ // disabling the updates prevents unnecessary redraws
+ setUpdatesEnabled( false );
+ guiFactory()->removeClient( this );
+
+ delete m_actionDelete;
+
+ m_actionDelete = new TDEAction(i18n("&Delete"), "edit-delete", Key_Delete, actionCollection(), "delete");
+
+ if (!m_splitter)
+ setupView();
+ if (m_controlCenter)
+ setupGUI(TDEMainWindow::ToolBar|Keys|Save|Create, "kcontroleditui.rc");
+ else
+ setupGUI(TDEMainWindow::ToolBar|Keys|Save|Create, "kmenueditui.rc");
+
+ m_tree->setViewMode(m_showHidden);
+}
+
+void KMenuEdit::slotSave()
+{
+ m_tree->save();
+}
+
+void KMenuEdit::slotSave_and_close()
+{
+ if (m_tree->save())
+ close();
+}
+
+bool KMenuEdit::queryClose()
+{
+ if (!m_tree->dirty()) return true;
+
+
+ int result;
+ if (m_controlCenter)
+ {
+ result = KMessageBox::warningYesNoCancel(this,
+ i18n("You have made changes to the Control Center.\n"
+ "Do you want to save the changes or discard them?"),
+ i18n("Save Control Center Changes?"),
+ KStdGuiItem::save(), KStdGuiItem::discard() );
+ }
+ else
+ {
+ result = KMessageBox::warningYesNoCancel(this,
+ i18n("You have made changes to the menu.\n"
+ "Do you want to save the changes or discard them?"),
+ i18n("Save Menu Changes?"),
+ KStdGuiItem::save(), KStdGuiItem::discard() );
+ }
+
+ switch(result)
+ {
+ case KMessageBox::Yes:
+ return m_tree->save();
+
+ case KMessageBox::No:
+ return true;
+
+ default:
+ break;
+ }
+ return false;
+}
+
+void KMenuEdit::slotConfigureToolbars()
+{
+ KEditToolbar dlg( factory() );
+
+ dlg.exec();
+}
diff --git a/kmenuedit/kmenuedit.desktop b/kmenuedit/kmenuedit.desktop
new file mode 100644
index 000000000..e95403953
--- /dev/null
+++ b/kmenuedit/kmenuedit.desktop
@@ -0,0 +1,93 @@
+[Desktop Entry]
+Exec=kmenuedit
+Icon=kmenuedit
+X-DocPath=kmenuedit/index.html
+Type=Application
+X-TDE-StartupNotify=true
+OnlyShowIn=TDE;
+
+Name=KMenuEdit
+GenericName=Menu Editor
+GenericName[af]=Kieslys Redigeerder
+GenericName[ar]=محرر القوائم
+GenericName[az]=Menyu Editoru
+GenericName[be]=Рэдактар меню
+GenericName[bg]=Редактор на системното меню
+GenericName[bn]=মেনু সম্পাদক
+GenericName[br]=Aozer lañserioù
+GenericName[bs]=Editor menija
+GenericName[ca]=Editor del menú
+GenericName[cs]=Editor nabídek
+GenericName[csb]=Editora menu
+GenericName[cy]=Golygydd Dewislen
+GenericName[da]=Menueditor
+GenericName[de]=Menü-Editor
+GenericName[el]=Επεξεργαστής μενού
+GenericName[eo]=Menuredaktilo
+GenericName[es]=Editor de menús
+GenericName[et]=Menüü redaktor
+GenericName[eu]=Menu editorea
+GenericName[fa]=ویرایشگر گزینگان
+GenericName[fi]=Valikon asetukset
+GenericName[fr]=KMenuEdit
+GenericName[fy]=Menubewurker
+GenericName[ga]=Eagarthóir Roghchláir
+GenericName[gl]=Editor do Menu
+GenericName[he]=עורך התפריטים
+GenericName[hi]=मेन्यू संपादक
+GenericName[hr]=Uređivač izbornika
+GenericName[hu]=Menüszerkesztő
+GenericName[id]=Editor Menu
+GenericName[is]=Sýsla með valmyndir
+GenericName[it]=Editor dei menu
+GenericName[ja]=メニューエディタ
+GenericName[ka]=მენიუს რედაქტორი
+GenericName[kk]=Мәзір редакторы
+GenericName[km]=កម្មវិធី​និពន្ធ​ម៉ឺនុយ
+GenericName[ko]=메뉴 편집기
+GenericName[lo]=ເຄື່ອງມືແກ້ໄຂເມນູ
+GenericName[lt]=Meniu redaktorius
+GenericName[lv]=Izvēlnes Redaktors
+GenericName[mk]=Уредувач на мени
+GenericName[mn]=Цэс-Боловсруулагч
+GenericName[ms]=Editor Menu
+GenericName[mt]=Editur tal-menu
+GenericName[nb]=Menyredigering
+GenericName[nds]=Menüeditor
+GenericName[ne]=मेनु सम्पादक
+GenericName[nl]=Menubewerker
+GenericName[nn]=Menyredigering
+GenericName[nso]=Mofetosi wa Menu
+GenericName[oc]=Editor de menu
+GenericName[pa]=ਮੇਨੂ ਸੰਪਾਦਕ
+GenericName[pl]=Edytor menu
+GenericName[pt]=Editor de Menus
+GenericName[pt_BR]=Editor de Menus
+GenericName[ro]=Editor de meniuri
+GenericName[ru]=Редактор меню
+GenericName[rw]=Muhinduzi Ibikubiyemo
+GenericName[se]=Fállodoaimmaheaddji
+GenericName[sk]=Editor menu
+GenericName[sl]=Urejevalnik menijev
+GenericName[sr]=Уређивач менија
+GenericName[sr@Latn]=Uređivač menija
+GenericName[sv]=Menyeditor
+GenericName[ta]=பட்டியல் திருத்துபவர்
+GenericName[te]=పట్టి ఎడిటర్
+GenericName[tg]=Муҳаррири меню
+GenericName[th]=ตัวแก้ไขเมนู
+GenericName[tr]=Menü Düzenleyicisi
+GenericName[tt]=Saylaq Tözätü
+GenericName[uk]=Редактор меню
+GenericName[uz]=Menyu tahrirchi
+GenericName[uz@cyrillic]=Меню таҳрирчи
+GenericName[ven]=Musengulusi wa Menu
+GenericName[vi]=Biên soạn Thực đơn
+GenericName[wa]=Aspougneu di menus
+GenericName[xh]=Umhleli we Menu
+GenericName[zh_CN]=菜单编辑器
+GenericName[zh_TW]=選單編輯器
+GenericName[zu]=Umlungisi wemenu
+
+X-DCOP-ServiceType=Unique
+Categories=Qt;TDE;Settings;
diff --git a/kmenuedit/kmenuedit.h b/kmenuedit/kmenuedit.h
new file mode 100644
index 000000000..15e5c797f
--- /dev/null
+++ b/kmenuedit/kmenuedit.h
@@ -0,0 +1,63 @@
+/*
+ * Copyright (C) 2000 Matthias Elter <[email protected]>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ */
+
+#ifndef __kmenuedit_h__
+#define __kmenuedit_h__
+
+#include <tdemainwindow.h>
+#include <treeview.h>
+
+class BasicTab;
+class TQSplitter;
+class TDEAction;
+class TDEToggleAction;
+
+class KMenuEdit : public TDEMainWindow
+{
+ Q_OBJECT
+
+public:
+ KMenuEdit( bool controlCenter, TQWidget *parent=0, const char *name=0 );
+ ~KMenuEdit();
+
+ void selectMenu(const TQString &menu) { m_tree->selectMenu(menu); }
+ void selectMenuEntry(const TQString &menuEntry) { m_tree->selectMenuEntry(menuEntry); }
+
+protected:
+ void setupView();
+ void setupActions();
+ bool queryClose();
+
+protected slots:
+ void slotSave();
+ void slotSave_and_close();
+ void slotChangeView();
+ void slotConfigureToolbars();
+protected:
+ TreeView *m_tree;
+ BasicTab *m_basicTab;
+ TQSplitter *m_splitter;
+
+ TDEAction *m_actionDelete;
+ TDEToggleAction *m_actionShowHidden;
+ bool m_showHidden;
+ bool m_controlCenter;
+};
+
+#endif
diff --git a/kmenuedit/kmenueditui.rc b/kmenuedit/kmenueditui.rc
new file mode 100644
index 000000000..eb8c3ae00
--- /dev/null
+++ b/kmenuedit/kmenueditui.rc
@@ -0,0 +1,45 @@
+<!DOCTYPE kpartgui SYSTEM "kpartgui.dtd">
+<kpartgui name="kmenuedit" version="6">
+
+<MenuBar>
+
+<Menu name="file" noMerge="1"><text>&amp;File</text>
+ <Action name="newitem"/>
+ <Action name="newsubmenu" />
+ <Action name="newsep" />
+ <Separator/>
+ <Action name="file_save"/>
+ <Action name="file_save_and_quit"/>
+ <Separator/>
+ <Action name="file_quit"/>
+</Menu>
+
+<Menu name="edit" noMerge="1"><text>&amp;Edit</text>
+ <Action name="edit_cut"/>
+ <Action name="edit_copy"/>
+ <Action name="edit_paste"/>
+ <Separator/>
+ <Action name="delete"/>
+</Menu>
+
+<Menu name="settings">
+ <Action name="show_removed"/>
+ <Action name="show_hidden"/>
+</Menu>
+</MenuBar>
+
+<ToolBar name="mainToolBar" noMerge="1" fullWidth="true"><text>Main Toolbar</text>
+ <Action name="file_save"/>
+ <Separator/>
+ <Action name="newitem"/>
+ <Action name="newsubmenu"/>
+ <Action name="newsep" />
+ <Separator/>
+ <Action name="edit_cut"/>
+ <Action name="edit_copy"/>
+ <Action name="edit_paste"/>
+ <Separator/>
+ <Action name="delete"/>
+</ToolBar>
+
+</kpartgui>
diff --git a/kmenuedit/main.cpp b/kmenuedit/main.cpp
new file mode 100644
index 000000000..b78a71ef0
--- /dev/null
+++ b/kmenuedit/main.cpp
@@ -0,0 +1,88 @@
+/*
+ * Copyright (C) 2000 Matthias Elter <[email protected]>
+ * Copyright (C) 2001-2002 Raffaele Sandrini <[email protected])
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ */
+
+#include <unistd.h>
+
+#include <kuniqueapplication.h>
+#include <tdelocale.h>
+#include <tdecmdlineargs.h>
+#include <tdeaboutdata.h>
+
+#include "kmenuedit.h"
+#include "khotkeys.h"
+
+static const char description[] = I18N_NOOP("TDE menu editor");
+static const char version[] = "0.7";
+
+static const TDECmdLineOptions options[] =
+{
+ { "+[menu]", I18N_NOOP("Sub menu to pre-select"), 0 },
+ { "+[menu-id]", I18N_NOOP("Menu entry to pre-select"), 0 },
+ TDECmdLineLastOption
+};
+
+static KMenuEdit *menuEdit = 0;
+
+class KMenuApplication : public KUniqueApplication
+{
+public:
+ KMenuApplication() { }
+ virtual ~KMenuApplication() { KHotKeys::cleanup(); }
+
+ virtual int newInstance()
+ {
+ TDECmdLineArgs *args = TDECmdLineArgs::parsedArgs();
+ if (args->count() > 0)
+ {
+ menuEdit->selectMenu(TQString::fromLocal8Bit(args->arg(0)));
+ if (args->count() > 1)
+ {
+ menuEdit->selectMenuEntry(TQString::fromLocal8Bit(args->arg(1)));
+ }
+ }
+ return KUniqueApplication::newInstance();
+ }
+};
+
+
+extern "C" int KDE_EXPORT kdemain( int argc, char **argv )
+{
+ TDEAboutData aboutData("kmenuedit", I18N_NOOP("TDE Menu Editor"),
+ version, description, TDEAboutData::License_GPL,
+ "(C) 2000-2003, Waldo Bastian, Raffaele Sandrini, Matthias Elter");
+ aboutData.addAuthor("Waldo Bastian", I18N_NOOP("Maintainer"), "[email protected]");
+ aboutData.addAuthor("Raffaele Sandrini", I18N_NOOP("Previous Maintainer"), "[email protected]");
+ aboutData.addAuthor("Matthias Elter", I18N_NOOP("Original Author"), "[email protected]");
+
+ TDECmdLineArgs::init( argc, argv, &aboutData );
+ KUniqueApplication::addCmdLineOptions();
+ TDECmdLineArgs::addCmdLineOptions( options );
+
+ if (!KUniqueApplication::start())
+ return 1;
+
+ KMenuApplication app;
+
+ menuEdit = new KMenuEdit(false);
+ menuEdit->show();
+
+ app.setMainWidget(menuEdit);
+ return app.exec();
+}
diff --git a/kmenuedit/menufile.cpp b/kmenuedit/menufile.cpp
new file mode 100644
index 000000000..2322cc31d
--- /dev/null
+++ b/kmenuedit/menufile.cpp
@@ -0,0 +1,552 @@
+/*
+ * Copyright (C) 2003 Waldo Bastian <[email protected]>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 as
+ * published by the Free Software Foundation.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ */
+
+#include <tqfile.h>
+#include <tqtextstream.h>
+#include <tqregexp.h>
+
+#include <kdebug.h>
+#include <tdeglobal.h>
+#include <tdelocale.h>
+#include <kstandarddirs.h>
+
+#include "menufile.h"
+
+
+#define MF_MENU "Menu"
+#define MF_PUBLIC_ID "-//freedesktop//DTD Menu 1.0//EN"
+#define MF_SYSTEM_ID "http://www.freedesktop.org/standards/menu-spec/1.0/menu.dtd"
+#define MF_NAME "Name"
+#define MF_INCLUDE "Include"
+#define MF_EXCLUDE "Exclude"
+#define MF_FILENAME "Filename"
+#define MF_DELETED "Deleted"
+#define MF_NOTDELETED "NotDeleted"
+#define MF_MOVE "Move"
+#define MF_OLD "Old"
+#define MF_NEW "New"
+#define MF_DIRECTORY "Directory"
+#define MF_LAYOUT "Layout"
+#define MF_MENUNAME "Menuname"
+#define MF_SEPARATOR "Separator"
+#define MF_MERGE "Merge"
+
+MenuFile::MenuFile(const TQString &file)
+ : m_fileName(file), m_bDirty(false)
+{
+ load();
+}
+
+MenuFile::~MenuFile()
+{
+}
+
+bool MenuFile::load()
+{
+ if (m_fileName.isEmpty())
+ return false;
+
+ TQFile file( m_fileName );
+ if (!file.open( IO_ReadOnly ))
+ {
+ kdWarning() << "Could not read " << m_fileName << endl;
+ create();
+ return false;
+ }
+
+ TQString errorMsg;
+ int errorRow;
+ int errorCol;
+ if ( !m_doc.setContent( &file, &errorMsg, &errorRow, &errorCol ) ) {
+ kdWarning() << "Parse error in " << m_fileName << ", line " << errorRow << ", col " << errorCol << ": " << errorMsg << endl;
+ file.close();
+ create();
+ return false;
+ }
+ file.close();
+
+ return true;
+}
+
+void MenuFile::create()
+{
+ TQDomImplementation impl;
+ TQDomDocumentType docType = impl.createDocumentType( MF_MENU, MF_PUBLIC_ID, MF_SYSTEM_ID );
+ m_doc = impl.createDocument(TQString::null, MF_MENU, docType);
+}
+
+bool MenuFile::save()
+{
+ TQFile file( m_fileName );
+
+ if (!file.open( IO_WriteOnly ))
+ {
+ kdWarning() << "Could not write " << m_fileName << endl;
+ m_error = i18n("Could not write to %1").arg(m_fileName);
+ return false;
+ }
+ TQTextStream stream( &file );
+ stream.setEncoding(TQTextStream::UnicodeUTF8);
+
+ stream << m_doc.toString();
+
+ file.close();
+
+ if (file.status() != IO_Ok)
+ {
+ kdWarning() << "Could not close " << m_fileName << endl;
+ m_error = i18n("Could not write to %1").arg(m_fileName);
+ return false;
+ }
+
+ m_bDirty = false;
+
+ return true;
+}
+
+TQDomElement MenuFile::findMenu(TQDomElement elem, const TQString &menuName, bool create)
+{
+ TQString menuNodeName;
+ TQString subMenuName;
+ int i = menuName.find('/');
+ if (i >= 0)
+ {
+ menuNodeName = menuName.left(i);
+ subMenuName = menuName.mid(i+1);
+ }
+ else
+ {
+ menuNodeName = menuName;
+ }
+ if (i == 0)
+ return findMenu(elem, subMenuName, create);
+
+ if (menuNodeName.isEmpty())
+ return elem;
+
+ TQDomNode n = elem.firstChild();
+ while( !n.isNull() )
+ {
+ TQDomElement e = n.toElement(); // try to convert the node to an element.
+ if (e.tagName() == MF_MENU)
+ {
+ TQString name;
+
+ TQDomNode n2 = e.firstChild();
+ while ( !n2.isNull() )
+ {
+ TQDomElement e2 = n2.toElement();
+ if (!e2.isNull() && e2.tagName() == MF_NAME)
+ {
+ name = e2.text();
+ break;
+ }
+ n2 = n2.nextSibling();
+ }
+
+ if (name == menuNodeName)
+ {
+ if (subMenuName.isEmpty())
+ return e;
+ else
+ return findMenu(e, subMenuName, create);
+ }
+ }
+ n = n.nextSibling();
+ }
+
+ if (!create)
+ return TQDomElement();
+
+ // Create new node.
+ TQDomElement newElem = m_doc.createElement(MF_MENU);
+ TQDomElement newNameElem = m_doc.createElement(MF_NAME);
+ newNameElem.appendChild(m_doc.createTextNode(menuNodeName));
+ newElem.appendChild(newNameElem);
+ elem.appendChild(newElem);
+
+ if (subMenuName.isEmpty())
+ return newElem;
+ else
+ return findMenu(newElem, subMenuName, create);
+}
+
+static TQString entryToDirId(const TQString &path)
+{
+ // See also KDesktopFile::locateLocal
+ TQString local;
+ if (path.startsWith("/"))
+ {
+ // XDG Desktop menu items come with absolute paths, we need to
+ // extract their relative path and then build a local path.
+ local = TDEGlobal::dirs()->relativeLocation("xdgdata-dirs", path);
+ }
+
+ if (local.isEmpty() || local.startsWith("/"))
+ {
+ // What now? Use filename only and hope for the best.
+ local = path.mid(path.findRev('/')+1);
+ }
+ return local;
+}
+
+static void purgeIncludesExcludes(TQDomElement elem, const TQString &appId, TQDomElement &excludeNode, TQDomElement &includeNode)
+{
+ // Remove any previous includes/excludes of appId
+ TQDomNode n = elem.firstChild();
+ while( !n.isNull() )
+ {
+ TQDomElement e = n.toElement(); // try to convert the node to an element.
+ bool bIncludeNode = (e.tagName() == MF_INCLUDE);
+ bool bExcludeNode = (e.tagName() == MF_EXCLUDE);
+ if (bIncludeNode)
+ includeNode = e;
+ if (bExcludeNode)
+ excludeNode = e;
+ if (bIncludeNode || bExcludeNode)
+ {
+ TQDomNode n2 = e.firstChild();
+ while ( !n2.isNull() )
+ {
+ TQDomNode next = n2.nextSibling();
+ TQDomElement e2 = n2.toElement();
+ if (!e2.isNull() && e2.tagName() == MF_FILENAME)
+ {
+ if (e2.text() == appId)
+ {
+ e.removeChild(e2);
+ break;
+ }
+ }
+ n2 = next;
+ }
+ }
+ n = n.nextSibling();
+ }
+}
+
+static void purgeDeleted(TQDomElement elem)
+{
+ // Remove any previous includes/excludes of appId
+ TQDomNode n = elem.firstChild();
+ while( !n.isNull() )
+ {
+ TQDomNode next = n.nextSibling();
+ TQDomElement e = n.toElement(); // try to convert the node to an element.
+ if ((e.tagName() == MF_DELETED) ||
+ (e.tagName() == MF_NOTDELETED))
+ {
+ elem.removeChild(e);
+ }
+ n = next;
+ }
+}
+
+static void purgeLayout(TQDomElement elem)
+{
+ // Remove any previous includes/excludes of appId
+ TQDomNode n = elem.firstChild();
+ while( !n.isNull() )
+ {
+ TQDomNode next = n.nextSibling();
+ TQDomElement e = n.toElement(); // try to convert the node to an element.
+ if (e.tagName() == MF_LAYOUT)
+ {
+ elem.removeChild(e);
+ }
+ n = next;
+ }
+}
+
+void MenuFile::addEntry(const TQString &menuName, const TQString &menuId)
+{
+ m_bDirty = true;
+
+ m_removedEntries.remove(menuId);
+
+ TQDomElement elem = findMenu(m_doc.documentElement(), menuName, true);
+
+ TQDomElement excludeNode;
+ TQDomElement includeNode;
+
+ purgeIncludesExcludes(elem, menuId, excludeNode, includeNode);
+
+ if (includeNode.isNull())
+ {
+ includeNode = m_doc.createElement(MF_INCLUDE);
+ elem.appendChild(includeNode);
+ }
+
+ TQDomElement fileNode = m_doc.createElement(MF_FILENAME);
+ fileNode.appendChild(m_doc.createTextNode(menuId));
+ includeNode.appendChild(fileNode);
+}
+
+void MenuFile::setLayout(const TQString &menuName, const TQStringList &layout)
+{
+ m_bDirty = true;
+
+ TQDomElement elem = findMenu(m_doc.documentElement(), menuName, true);
+
+ purgeLayout(elem);
+
+ TQDomElement layoutNode = m_doc.createElement(MF_LAYOUT);
+ elem.appendChild(layoutNode);
+
+ for(TQStringList::ConstIterator it = layout.begin();
+ it != layout.end(); ++it)
+ {
+ TQString li = *it;
+ if (li == ":S")
+ {
+ layoutNode.appendChild(m_doc.createElement(MF_SEPARATOR));
+ }
+ else if (li == ":M")
+ {
+ TQDomElement mergeNode = m_doc.createElement(MF_MERGE);
+ mergeNode.setAttribute("type", "menus");
+ layoutNode.appendChild(mergeNode);
+ }
+ else if (li == ":F")
+ {
+ TQDomElement mergeNode = m_doc.createElement(MF_MERGE);
+ mergeNode.setAttribute("type", "files");
+ layoutNode.appendChild(mergeNode);
+ }
+ else if (li == ":A")
+ {
+ TQDomElement mergeNode = m_doc.createElement(MF_MERGE);
+ mergeNode.setAttribute("type", "all");
+ layoutNode.appendChild(mergeNode);
+ }
+ else if (li.endsWith("/"))
+ {
+ li.truncate(li.length()-1);
+ TQDomElement menuNode = m_doc.createElement(MF_MENUNAME);
+ menuNode.appendChild(m_doc.createTextNode(li));
+ layoutNode.appendChild(menuNode);
+ }
+ else
+ {
+ TQDomElement fileNode = m_doc.createElement(MF_FILENAME);
+ fileNode.appendChild(m_doc.createTextNode(li));
+ layoutNode.appendChild(fileNode);
+ }
+ }
+}
+
+
+void MenuFile::removeEntry(const TQString &menuName, const TQString &menuId)
+{
+ m_bDirty = true;
+
+ m_removedEntries.append(menuId);
+
+ TQDomElement elem = findMenu(m_doc.documentElement(), menuName, true);
+
+ TQDomElement excludeNode;
+ TQDomElement includeNode;
+
+ purgeIncludesExcludes(elem, menuId, excludeNode, includeNode);
+
+ if (excludeNode.isNull())
+ {
+ excludeNode = m_doc.createElement(MF_EXCLUDE);
+ elem.appendChild(excludeNode);
+ }
+
+ TQDomElement fileNode = m_doc.createElement(MF_FILENAME);
+ fileNode.appendChild(m_doc.createTextNode(menuId));
+ excludeNode.appendChild(fileNode);
+}
+
+void MenuFile::addMenu(const TQString &menuName, const TQString &menuFile)
+{
+ m_bDirty = true;
+ TQDomElement elem = findMenu(m_doc.documentElement(), menuName, true);
+
+ TQDomElement dirElem = m_doc.createElement(MF_DIRECTORY);
+ dirElem.appendChild(m_doc.createTextNode(entryToDirId(menuFile)));
+ elem.appendChild(dirElem);
+}
+
+void MenuFile::moveMenu(const TQString &oldMenu, const TQString &newMenu)
+{
+ m_bDirty = true;
+
+ // Undelete the new menu
+ TQDomElement elem = findMenu(m_doc.documentElement(), newMenu, true);
+ purgeDeleted(elem);
+ elem.appendChild(m_doc.createElement(MF_NOTDELETED));
+
+// TODO: GET RID OF COMMON PART, IT BREAKS STUFF
+ // Find common part
+ TQStringList oldMenuParts = TQStringList::split('/', oldMenu);
+ TQStringList newMenuParts = TQStringList::split('/', newMenu);
+ TQString commonMenuName;
+ uint max = TQMIN(oldMenuParts.count(), newMenuParts.count());
+ uint i = 0;
+ for(; i < max; i++)
+ {
+ if (oldMenuParts[i] != newMenuParts[i])
+ break;
+ commonMenuName += '/' + oldMenuParts[i];
+ }
+ TQString oldMenuName;
+ for(uint j = i; j < oldMenuParts.count(); j++)
+ {
+ if (i != j)
+ oldMenuName += '/';
+ oldMenuName += oldMenuParts[j];
+ }
+ TQString newMenuName;
+ for(uint j = i; j < newMenuParts.count(); j++)
+ {
+ if (i != j)
+ newMenuName += '/';
+ newMenuName += newMenuParts[j];
+ }
+
+ if (oldMenuName == newMenuName) return; // Can happen
+
+ elem = findMenu(m_doc.documentElement(), commonMenuName, true);
+
+ // Add instructions for moving
+ TQDomElement moveNode = m_doc.createElement(MF_MOVE);
+ TQDomElement node = m_doc.createElement(MF_OLD);
+ node.appendChild(m_doc.createTextNode(oldMenuName));
+ moveNode.appendChild(node);
+ node = m_doc.createElement(MF_NEW);
+ node.appendChild(m_doc.createTextNode(newMenuName));
+ moveNode.appendChild(node);
+ elem.appendChild(moveNode);
+}
+
+void MenuFile::removeMenu(const TQString &menuName)
+{
+ m_bDirty = true;
+
+ TQDomElement elem = findMenu(m_doc.documentElement(), menuName, true);
+
+ purgeDeleted(elem);
+ elem.appendChild(m_doc.createElement(MF_DELETED));
+}
+
+ /**
+ * Returns a unique menu-name for a new menu under @p menuName
+ * inspired by @p newMenu
+ */
+TQString MenuFile::uniqueMenuName(const TQString &menuName, const TQString &newMenu, const TQStringList & excludeList)
+{
+ TQDomElement elem = findMenu(m_doc.documentElement(), menuName, false);
+
+ TQString result = newMenu;
+ if (result.endsWith("/"))
+ result.truncate(result.length()-1);
+
+ TQRegExp r("(.*)(?=-\\d+)");
+ result = (r.search(result) > -1) ? r.cap(1) : result;
+
+ int trunc = result.length(); // Position of trailing '/'
+
+ result.append("/");
+
+ for(int n = 1; ++n; )
+ {
+ if (findMenu(elem, result, false).isNull() && !excludeList.contains(result))
+ return result;
+
+ result.truncate(trunc);
+ result.append(TQString("-%1/").arg(n));
+ }
+ return TQString::null; // Never reached
+}
+
+void MenuFile::performAction(const ActionAtom *atom)
+{
+ switch(atom->action)
+ {
+ case ADD_ENTRY:
+ addEntry(atom->arg1, atom->arg2);
+ return;
+ case REMOVE_ENTRY:
+ removeEntry(atom->arg1, atom->arg2);
+ return;
+ case ADD_MENU:
+ addMenu(atom->arg1, atom->arg2);
+ return;
+ case REMOVE_MENU:
+ removeMenu(atom->arg1);
+ return;
+ case MOVE_MENU:
+ moveMenu(atom->arg1, atom->arg2);
+ return;
+ }
+}
+
+MenuFile::ActionAtom *MenuFile::pushAction(MenuFile::ActionType action, const TQString &arg1, const TQString &arg2)
+{
+ ActionAtom *atom = new ActionAtom;
+ atom->action = action;
+ atom->arg1 = arg1;
+ atom->arg2 = arg2;
+ m_actionList.append(atom);
+ return atom;
+}
+
+void MenuFile::popAction(ActionAtom *atom)
+{
+ if (m_actionList.getLast() != atom)
+ {
+ tqWarning("MenuFile::popAction Error, action not last in list.");
+ return;
+ }
+ m_actionList.removeLast();
+ delete atom;
+}
+
+bool MenuFile::performAllActions()
+{
+ for(ActionAtom *atom; (atom = m_actionList.getFirst()); m_actionList.removeFirst())
+ {
+ performAction(atom);
+ delete atom;
+ }
+
+ // Entries that have been removed from the menu are added to .hidden
+ // so that they don't re-appear in Lost & Found
+ TQStringList removed = m_removedEntries;
+ m_removedEntries.clear();
+ for(TQStringList::ConstIterator it = removed.begin();
+ it != removed.end(); ++it)
+ {
+ addEntry("/.hidden/", *it);
+ }
+
+ m_removedEntries.clear();
+
+ if (!m_bDirty)
+ return true;
+
+ return save();
+}
+
+bool MenuFile::dirty()
+{
+ return (m_actionList.count() != 0) || m_bDirty;
+}
diff --git a/kmenuedit/menufile.h b/kmenuedit/menufile.h
new file mode 100644
index 000000000..c216d486c
--- /dev/null
+++ b/kmenuedit/menufile.h
@@ -0,0 +1,114 @@
+/*
+ * Copyright (C) 2003 Waldo Bastian <[email protected]>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 as
+ * published by the Free Software Foundation.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ */
+
+#ifndef __menufile_h__
+#define __menufile_h__
+
+#include <tqdom.h>
+#include <tqstring.h>
+
+class MenuFile
+{
+public:
+ MenuFile(const TQString &file);
+ ~MenuFile();
+
+ bool load();
+ bool save();
+ void create();
+ TQString error() { return m_error; } // Returns the last error message
+
+ enum ActionType {
+ ADD_ENTRY = 0,
+ REMOVE_ENTRY,
+ ADD_MENU,
+ REMOVE_MENU,
+ MOVE_MENU
+ };
+
+ struct ActionAtom
+ {
+ ActionType action;
+ TQString arg1;
+ TQString arg2;
+ };
+
+ /**
+ * Create action atom and push it on the stack
+ */
+ ActionAtom *pushAction(ActionType action, const TQString &arg1, const TQString &arg2);
+
+ /**
+ * Pop @p atom from the stack.
+ * @p atom must be last item on the stack
+ */
+ void popAction(ActionAtom *atom);
+
+ /**
+ * Perform the specified action
+ */
+ void performAction(const ActionAtom *);
+
+ /**
+ * Perform all actions currently on the stack, remove them from the stack and
+ * save result
+ * @return whether save was successful
+ */
+ bool performAllActions();
+
+ /**
+ * Returns whether the stack contains any actions
+ */
+ bool dirty();
+
+ void addEntry(const TQString &menuName, const TQString &menuId);
+ void removeEntry(const TQString &menuName, const TQString &menuId);
+
+ void addMenu(const TQString &menuName, const TQString &menuFile);
+ void moveMenu(const TQString &oldMenu, const TQString &newMenu);
+ void removeMenu(const TQString &menuName);
+
+ void setLayout(const TQString &menuName, const TQStringList &layout);
+
+ /**
+ * Returns a unique menu-name for a new menu under @p menuName
+ * inspired by @p newMenu and not part of @p excludeList
+ */
+ TQString uniqueMenuName(const TQString &menuName, const TQString &newMenu, const TQStringList &excludeList);
+
+protected:
+ /**
+ * Finds menu @p menuName in @p elem.
+ * If @p create is true, the menu is created if it doesn't exist yet.
+ * @return The menu dom-node of @p menuName
+ */
+ TQDomElement findMenu(TQDomElement elem, const TQString &menuName, bool create);
+
+private:
+ TQString m_error;
+ TQString m_fileName;
+
+ TQDomDocument m_doc;
+ bool m_bDirty;
+
+ TQPtrList<ActionAtom> m_actionList;
+ TQStringList m_removedEntries;
+};
+
+
+#endif
diff --git a/kmenuedit/menuinfo.cpp b/kmenuedit/menuinfo.cpp
new file mode 100644
index 000000000..03c3a2e57
--- /dev/null
+++ b/kmenuedit/menuinfo.cpp
@@ -0,0 +1,502 @@
+/*
+ * Copyright (C) 2003 Waldo Bastian <[email protected]>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ */
+
+#include "menuinfo.h"
+#include "menufile.h"
+
+#include <tqregexp.h>
+
+#include <kdesktopfile.h>
+#include <khotkeys.h>
+#include <kstandarddirs.h>
+
+//
+// MenuFolderInfo
+//
+
+static TQStringList *s_allShortcuts = 0;
+static TQStringList *s_newShortcuts = 0;
+static TQStringList *s_freeShortcuts = 0;
+static TQStringList *s_deletedApps = 0;
+
+// Add separator
+void MenuFolderInfo::add(MenuSeparatorInfo *info, bool initial)
+{
+ if (initial)
+ initialLayout.append(info);
+}
+
+// Add sub menu
+void MenuFolderInfo::add(MenuFolderInfo *info, bool initial)
+{
+ subFolders.append(info);
+ if (initial)
+ initialLayout.append(info);
+}
+
+// Remove sub menu (without deleting it)
+void MenuFolderInfo::take(MenuFolderInfo *info)
+{
+ subFolders.take(subFolders.findRef(info));
+}
+
+// Remove sub menu (without deleting it)
+bool MenuFolderInfo::takeRecursive(MenuFolderInfo *info)
+{
+ int i = subFolders.findRef(info);
+ if (i >= 0)
+ {
+ subFolders.take(i);
+ return true;
+ }
+
+ for(MenuFolderInfo *subFolderInfo = subFolders.first();
+ subFolderInfo; subFolderInfo = subFolders.next())
+ {
+ if (subFolderInfo->takeRecursive(info))
+ return true;
+ }
+ return false;
+}
+
+// Recursively update all fullIds
+void MenuFolderInfo::updateFullId(const TQString &parentId)
+{
+ fullId = parentId + id;
+
+ for(MenuFolderInfo *subFolderInfo = subFolders.first();
+ subFolderInfo; subFolderInfo = subFolders.next())
+ {
+ subFolderInfo->updateFullId(fullId);
+ }
+}
+
+// Add entry
+void MenuFolderInfo::add(MenuEntryInfo *entry, bool initial)
+{
+ entries.append(entry);
+ if (initial)
+ initialLayout.append(entry);
+}
+
+// Remove entry
+void MenuFolderInfo::take(MenuEntryInfo *entry)
+{
+ entries.removeRef(entry);
+}
+
+
+// Return a unique sub-menu caption inspired by @p caption
+TQString MenuFolderInfo::uniqueMenuCaption(const TQString &caption)
+{
+ TQRegExp r("(.*)(?=-\\d+)");
+ TQString cap = (r.search(caption) > -1) ? TQString(r.cap(1)) : caption;
+
+ TQString result = caption;
+
+ for(int n = 1; ++n; )
+ {
+ bool ok = true;
+ for(MenuFolderInfo *subFolderInfo = subFolders.first();
+ subFolderInfo; subFolderInfo = subFolders.next())
+ {
+ if (subFolderInfo->caption == result)
+ {
+ ok = false;
+ break;
+ }
+ }
+ if (ok)
+ return result;
+
+ result = cap + TQString("-%1").arg(n);
+ }
+ return TQString::null; // Never reached
+}
+
+// Return a unique item caption inspired by @p caption
+TQString MenuFolderInfo::uniqueItemCaption(const TQString &caption, const TQString &exclude)
+{
+ TQRegExp r("(.*)(?=-\\d+)");
+ TQString cap = (r.search(caption) > -1) ? TQString(r.cap(1)) : caption;
+
+ TQString result = caption;
+
+ for(int n = 1; ++n; )
+ {
+ bool ok = true;
+ if (result == exclude)
+ ok = false;
+ MenuEntryInfo *entryInfo;
+ for(TQPtrListIterator<MenuEntryInfo> it(entries);
+ ok && (entryInfo = it.current()); ++it)
+ {
+ if (entryInfo->caption == result)
+ ok = false;
+ }
+ if (ok)
+ return result;
+
+ result = cap + TQString("-%1").arg(n);
+ }
+ return TQString::null; // Never reached
+}
+
+// Return a list of existing submenu ids
+TQStringList MenuFolderInfo::existingMenuIds()
+{
+ TQStringList result;
+ for(MenuFolderInfo *subFolderInfo = subFolders.first();
+ subFolderInfo; subFolderInfo = subFolders.next())
+ {
+ result.append(subFolderInfo->id);
+ }
+ return result;
+}
+
+void MenuFolderInfo::setDirty()
+{
+ dirty = true;
+}
+
+void MenuFolderInfo::save(MenuFile *menuFile)
+{
+ if (s_deletedApps)
+ {
+ // Remove hotkeys for applications that have been deleted
+ for(TQStringList::ConstIterator it = s_deletedApps->begin();
+ it != s_deletedApps->end(); ++it)
+ {
+ KHotKeys::menuEntryDeleted(*it);
+ }
+ delete s_deletedApps;
+ s_deletedApps = 0;
+ }
+
+ if (dirty)
+ {
+ TQString local = KDesktopFile::locateLocal(directoryFile);
+
+ TDEConfig *df = 0;
+ if (directoryFile != local)
+ {
+ TDEConfig orig(directoryFile, true, false, "apps");
+ df = orig.copyTo(local);
+ }
+ else
+ {
+ df = new TDEConfig(directoryFile, false, false, "apps");
+ }
+
+ df->setDesktopGroup();
+ df->writeEntry("Name", caption);
+ df->writeEntry("GenericName", genericname);
+ df->writeEntry("Comment", comment);
+ df->writeEntry("Icon", icon);
+ df->sync();
+ delete df;
+ dirty = false;
+ }
+
+ // Save sub-menus
+ for(MenuFolderInfo *subFolderInfo = subFolders.first();
+ subFolderInfo; subFolderInfo = subFolders.next())
+ {
+ subFolderInfo->save(menuFile);
+ }
+
+ // Save entries
+ MenuEntryInfo *entryInfo;
+ for(TQPtrListIterator<MenuEntryInfo> it(entries);
+ (entryInfo = it.current()); ++it)
+ {
+ if (entryInfo->needInsertion())
+ menuFile->addEntry(fullId, entryInfo->menuId());
+ entryInfo->save();
+ }
+}
+
+bool MenuFolderInfo::hasDirt()
+{
+ if (dirty) return true;
+
+ // Check sub-menus
+ for(MenuFolderInfo *subFolderInfo = subFolders.first();
+ subFolderInfo; subFolderInfo = subFolders.next())
+ {
+ if (subFolderInfo->hasDirt()) return true;
+ }
+
+ // Check entries
+ MenuEntryInfo *entryInfo;
+ for(TQPtrListIterator<MenuEntryInfo> it(entries);
+ (entryInfo = it.current()); ++it)
+ {
+ if (entryInfo->dirty) return true;
+ if (entryInfo->shortcutDirty) return true;
+ }
+ return false;
+}
+
+KService::Ptr MenuFolderInfo::findServiceShortcut(const TDEShortcut&cut)
+{
+ KService::Ptr result;
+ // Check sub-menus
+ for(MenuFolderInfo *subFolderInfo = subFolders.first();
+ subFolderInfo; subFolderInfo = subFolders.next())
+ {
+ result = subFolderInfo->findServiceShortcut(cut);
+ if (result)
+ return result;
+ }
+
+ // Check entries
+ MenuEntryInfo *entryInfo;
+ for(TQPtrListIterator<MenuEntryInfo> it(entries);
+ (entryInfo = it.current()); ++it)
+ {
+ if (entryInfo->shortCut == cut)
+ return entryInfo->service;
+ }
+ return 0;
+}
+
+void MenuFolderInfo::setInUse(bool inUse)
+{
+ // Propagate to sub-menus
+ for(MenuFolderInfo *subFolderInfo = subFolders.first();
+ subFolderInfo; subFolderInfo = subFolders.next())
+ {
+ subFolderInfo->setInUse(inUse);
+ }
+
+ // Propagate to entries
+ MenuEntryInfo *entryInfo;
+ for(TQPtrListIterator<MenuEntryInfo> it(entries);
+ (entryInfo = it.current()); ++it)
+ {
+ entryInfo->setInUse(inUse);
+ }
+}
+
+//
+// MenuEntryInfo
+//
+
+MenuEntryInfo::~MenuEntryInfo()
+{
+ df->rollback(false);
+ delete df;
+}
+
+KDesktopFile *MenuEntryInfo::desktopFile()
+{
+ if (!df)
+ {
+ df = new KDesktopFile(service->desktopEntryPath());
+ }
+ return df;
+}
+
+void MenuEntryInfo::setDirty()
+{
+ if (dirty) return;
+
+ dirty = true;
+
+ TQString local = locateLocal("xdgdata-apps", service->menuId());
+ if (local != service->desktopEntryPath())
+ {
+ KDesktopFile *oldDf = desktopFile();
+ df = oldDf->copyTo(local);
+ df->setDesktopGroup();
+ delete oldDf;
+ }
+}
+
+bool MenuEntryInfo::needInsertion()
+{
+ // If entry is dirty and previously stored under applnk, then we need to be added explicity
+ return dirty && !service->desktopEntryPath().startsWith("/");
+}
+
+void MenuEntryInfo::save()
+{
+ if (dirty)
+ {
+ df->sync();
+ dirty = false;
+ }
+
+ if (shortcutDirty)
+ {
+ if( KHotKeys::present())
+ {
+ KHotKeys::changeMenuEntryShortcut( service->storageId(), shortCut.toStringInternal() );
+ }
+ shortcutDirty = false;
+ }
+}
+
+void MenuEntryInfo::setCaption(const TQString &_caption)
+{
+ if (caption == _caption)
+ return;
+ caption = _caption;
+ setDirty();
+ desktopFile()->writeEntry("Name", caption);
+}
+
+void MenuEntryInfo::setDescription(const TQString &_description)
+{
+ if (description == _description)
+ return;
+ description = _description;
+ setDirty();
+ desktopFile()->writeEntry("GenericName", description);
+}
+
+void MenuEntryInfo::setIcon(const TQString &_icon)
+{
+ if (icon == _icon)
+ return;
+
+ icon = _icon;
+ setDirty();
+ desktopFile()->writeEntry("Icon", icon);
+}
+
+TDEShortcut MenuEntryInfo::shortcut()
+{
+ if (!shortcutLoaded)
+ {
+ shortcutLoaded = true;
+ if( KHotKeys::present())
+ {
+ shortCut = KHotKeys::getMenuEntryShortcut( service->storageId() );
+ }
+ }
+ return shortCut;
+}
+
+static bool isEmpty(const TDEShortcut &shortCut)
+{
+ for(int i = shortCut.count(); i--;)
+ {
+ if (!shortCut.seq(i).isNull())
+ return false;
+ }
+ return true;
+}
+
+static void freeShortcut(const TDEShortcut &shortCut)
+{
+ if (!isEmpty(shortCut))
+ {
+ TQString shortcutKey = shortCut.toString();
+ if (s_newShortcuts)
+ s_newShortcuts->remove(shortcutKey);
+
+ if (!s_freeShortcuts)
+ s_freeShortcuts = new TQStringList;
+
+ s_freeShortcuts->append(shortcutKey);
+ }
+}
+
+static void allocateShortcut(const TDEShortcut &shortCut)
+{
+ if (!isEmpty(shortCut))
+ {
+ TQString shortcutKey = shortCut.toString();
+ if (s_freeShortcuts)
+ s_freeShortcuts->remove(shortcutKey);
+
+ if (!s_newShortcuts)
+ s_newShortcuts = new TQStringList;
+
+ s_newShortcuts->append(shortcutKey);
+ }
+}
+
+void MenuEntryInfo::setShortcut(const TDEShortcut &_shortcut)
+{
+ if (shortCut == _shortcut)
+ return;
+
+ freeShortcut(shortCut);
+ allocateShortcut(_shortcut);
+
+ shortCut = _shortcut;
+ if (isEmpty(shortCut))
+ shortCut = TDEShortcut(); // Normalize
+
+ shortcutLoaded = true;
+ shortcutDirty = true;
+}
+
+void MenuEntryInfo::setInUse(bool inUse)
+{
+ if (inUse)
+ {
+ TDEShortcut temp = shortcut();
+ shortCut = TDEShortcut();
+ if (isShortcutAvailable(temp))
+ shortCut = temp;
+ else
+ shortcutDirty = true;
+ allocateShortcut(shortCut);
+
+ if (s_deletedApps)
+ s_deletedApps->remove(service->storageId());
+ }
+ else
+ {
+ freeShortcut(shortcut());
+
+ // Add to list of deleted apps
+ if (!s_deletedApps)
+ s_deletedApps = new TQStringList;
+
+ s_deletedApps->append(service->storageId());
+ }
+}
+
+bool MenuEntryInfo::isShortcutAvailable(const TDEShortcut &_shortcut)
+{
+ if (shortCut == _shortcut)
+ return true;
+
+ TQString shortcutKey = _shortcut.toString();
+ bool available = true;
+ if (!s_allShortcuts)
+ {
+ s_allShortcuts = new TQStringList(KHotKeys::allShortCuts());
+ }
+ available = !s_allShortcuts->contains(shortcutKey);
+ if (available && s_newShortcuts)
+ {
+ available = !s_newShortcuts->contains(shortcutKey);
+ }
+ if (!available && s_freeShortcuts)
+ {
+ available = s_freeShortcuts->contains(shortcutKey);
+ }
+ return available;
+}
diff --git a/kmenuedit/menuinfo.h b/kmenuedit/menuinfo.h
new file mode 100644
index 000000000..9497022d0
--- /dev/null
+++ b/kmenuedit/menuinfo.h
@@ -0,0 +1,192 @@
+/*
+ * Copyright (C) 2003 Waldo Bastian <[email protected]>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ */
+
+#ifndef __menuinfo_h__
+#define __menuinfo_h__
+
+#include <tqstring.h>
+
+#include <tdeshortcut.h>
+#include <kservice.h>
+
+class MenuFile;
+class MenuEntryInfo;
+
+class MenuInfo
+{
+public:
+ MenuInfo() {}
+ virtual ~MenuInfo() {}
+};
+
+class MenuSeparatorInfo : public MenuInfo
+{
+public:
+ MenuSeparatorInfo() {}
+};
+
+class MenuFolderInfo : public MenuInfo
+{
+public:
+ MenuFolderInfo() : dirty(false), hidden(false) { subFolders.setAutoDelete(true); }
+
+ // Add separator
+ void add(MenuSeparatorInfo *, bool initial=false);
+
+ // Add sub menu
+ void add(MenuFolderInfo *, bool initial=false);
+
+ // Remove sub menu (without deleting it)
+ void take(MenuFolderInfo *);
+
+ // Remove sub menu (without deleting it)
+ // @return true if found
+ bool takeRecursive(MenuFolderInfo *info);
+
+ // Add entry
+ void add(MenuEntryInfo *, bool initial = false);
+
+ // Remove entry (without deleteing it)
+ void take(MenuEntryInfo *);
+
+ // Return a unique sub-menu caption inspired by @p caption
+ TQString uniqueMenuCaption(const TQString &caption);
+
+ // Return a unique item caption inspired by @p caption but different
+ // from @p exclude
+ TQString uniqueItemCaption(const TQString &caption, const TQString &exclude = TQString::null);
+
+ // Update full id's for this item and all submenus
+ void updateFullId(const TQString &parentId);
+
+ // Return a list of existing submenu ids
+ TQStringList existingMenuIds();
+
+ void setCaption(const TQString &_caption)
+ {
+ if (_caption == caption) return;
+ caption = _caption;
+ setDirty();
+ }
+
+ void setIcon(const TQString &_icon)
+ {
+ if (_icon == icon) return;
+ icon = _icon;
+ setDirty();
+ }
+
+ void setGenericName(const TQString &_description)
+ {
+ if (_description == genericname) return;
+ genericname = _description;
+ setDirty();
+ }
+
+ void setComment(const TQString &_comment)
+ {
+ if (_comment == comment) return;
+ comment = _comment;
+ setDirty();
+ }
+
+ // Mark menu as dirty
+ void setDirty();
+
+ // Return whether this menu or any entry or submenu contained in it is dirty.
+ bool hasDirt();
+
+ // Return whether this menu should be explicitly added to its parent menu
+ bool needInsertion();
+
+ // Save menu and all its entries and submenus
+ void save(MenuFile *);
+
+ // Search service by shortcut
+ KService::Ptr findServiceShortcut(const TDEShortcut&);
+
+ // Set whether the entry is in active use (as opposed to in the clipboard/deleted)
+ void setInUse(bool inUse);
+
+public:
+ TQString id; // Relative to parent
+ TQString fullId; // Name in tree
+ TQString caption; // Visible name
+ TQString genericname; // Generic description
+ TQString comment; // Comment
+ TQString directoryFile; // File describing this folder.
+ TQString icon; // Icon
+ TQPtrList<MenuFolderInfo> subFolders; // Sub menus in this folder
+ TQPtrList<MenuEntryInfo> entries; // Menu entries in this folder
+ TQPtrList<MenuInfo> initialLayout; // Layout of menu entries according to sycoca
+ bool dirty;
+ bool hidden;
+};
+
+class MenuEntryInfo : public MenuInfo
+{
+public:
+ MenuEntryInfo(const KService::Ptr &_service, KDesktopFile *_df = 0)
+ : service(_service), df(_df),
+ shortcutLoaded(false), shortcutDirty(false), dirty(_df != 0), hidden(false)
+ {
+ caption = service->name();
+ description = service->genericName();
+ icon = service->icon();
+ }
+ ~MenuEntryInfo();
+
+ void setCaption(const TQString &_caption);
+ void setDescription(const TQString &_description);
+ void setIcon(const TQString &_icon);
+
+ TQString menuId() const { return service->menuId(); }
+
+ TQString file() const { return service->desktopEntryPath(); }
+
+ TDEShortcut shortcut();
+ void setShortcut(const TDEShortcut &_shortcut);
+ bool isShortcutAvailable(const TDEShortcut &_shortcut);
+
+ void setDirty();
+
+ // Set whether the entry is in active use (as opposed to in the clipboard/deleted)
+ void setInUse(bool inUse);
+
+ // Return whether this menu should be explicitly added to its parent menu
+ bool needInsertion();
+
+ void save();
+
+ KDesktopFile *desktopFile();
+
+public:
+ TQString caption;
+ TQString description;
+ TQString icon;
+ KService::Ptr service;
+ KDesktopFile *df;
+ TDEShortcut shortCut;
+ bool shortcutLoaded;
+ bool shortcutDirty;
+ bool dirty;
+ bool hidden;
+};
+
+#endif
diff --git a/kmenuedit/pixmaps/CMakeLists.txt b/kmenuedit/pixmaps/CMakeLists.txt
new file mode 100644
index 000000000..a58fe9b20
--- /dev/null
+++ b/kmenuedit/pixmaps/CMakeLists.txt
@@ -0,0 +1,13 @@
+#################################################
+#
+# (C) 2010-2011 Serghei Amelian
+# serghei (DOT) amelian (AT) gmail.com
+#
+# Improvements and feedback are welcome
+#
+# This file is released under GPL >= 2
+#
+#################################################
+
+tde_install_icons( DESTINATION ${DATA_INSTALL_DIR}/kmenuedit/icons )
+tde_install_icons( DESTINATION ${DATA_INSTALL_DIR}/kcontroledit/icons )
diff --git a/kmenuedit/pixmaps/Makefile.am b/kmenuedit/pixmaps/Makefile.am
new file mode 100644
index 000000000..1e138f9db
--- /dev/null
+++ b/kmenuedit/pixmaps/Makefile.am
@@ -0,0 +1,5 @@
+kmenuediticondir = $(kde_datadir)/kmenuedit/icons
+kmenuediticon_ICON = AUTO
+
+kcontrolediticondir = $(kde_datadir)/kcontroledit/icons
+kcontrolediticon_ICON = AUTO
diff --git a/kmenuedit/pixmaps/cr22-action-filesave_and_close.png b/kmenuedit/pixmaps/cr22-action-filesave_and_close.png
new file mode 100644
index 000000000..7d4b6f365
--- /dev/null
+++ b/kmenuedit/pixmaps/cr22-action-filesave_and_close.png
Binary files differ
diff --git a/kmenuedit/pixmaps/cr22-action-menu_new.png b/kmenuedit/pixmaps/cr22-action-menu_new.png
new file mode 100644
index 000000000..56613f41c
--- /dev/null
+++ b/kmenuedit/pixmaps/cr22-action-menu_new.png
Binary files differ
diff --git a/kmenuedit/pixmaps/cr22-action-menu_new_sep.png b/kmenuedit/pixmaps/cr22-action-menu_new_sep.png
new file mode 100644
index 000000000..bff0ce6d0
--- /dev/null
+++ b/kmenuedit/pixmaps/cr22-action-menu_new_sep.png
Binary files differ
diff --git a/kmenuedit/pixmaps/cr32-action-menu_new.png b/kmenuedit/pixmaps/cr32-action-menu_new.png
new file mode 100644
index 000000000..d1e532be1
--- /dev/null
+++ b/kmenuedit/pixmaps/cr32-action-menu_new.png
Binary files differ
diff --git a/kmenuedit/pixmaps/cr32-action-menu_new_sep.png b/kmenuedit/pixmaps/cr32-action-menu_new_sep.png
new file mode 100644
index 000000000..ff02e1d2b
--- /dev/null
+++ b/kmenuedit/pixmaps/cr32-action-menu_new_sep.png
Binary files differ
diff --git a/kmenuedit/pixmaps/lo16-action-menu_new.png b/kmenuedit/pixmaps/lo16-action-menu_new.png
new file mode 100644
index 000000000..b121605f7
--- /dev/null
+++ b/kmenuedit/pixmaps/lo16-action-menu_new.png
Binary files differ
diff --git a/kmenuedit/treeview.cpp b/kmenuedit/treeview.cpp
new file mode 100644
index 000000000..7f7e31dce
--- /dev/null
+++ b/kmenuedit/treeview.cpp
@@ -0,0 +1,1581 @@
+/*
+ * Copyright (C) 2000 Matthias Elter <[email protected]>
+ * Copyright (C) 2001-2002 Raffaele Sandrini <[email protected])
+ * Copyright (C) 2003 Waldo Bastian <[email protected]>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ */
+
+#include <unistd.h>
+
+#include <tqcstring.h>
+#include <tqcursor.h>
+#include <tqdatastream.h>
+#include <tqdir.h>
+#include <tqdragobject.h>
+#include <tqfileinfo.h>
+#include <tqheader.h>
+#include <tqpainter.h>
+#include <tqpopupmenu.h>
+#include <tqregexp.h>
+#include <tqstringlist.h>
+
+#include <tdeglobal.h>
+#include <kstandarddirs.h>
+#include <kinputdialog.h>
+#include <tdelocale.h>
+#include <ksimpleconfig.h>
+#include <kdebug.h>
+#include <kiconloader.h>
+#include <kdesktopfile.h>
+#include <tdeaction.h>
+#include <tdemessagebox.h>
+#include <tdeapplication.h>
+#include <kservice.h>
+#include <kservicegroup.h>
+#include <tdemultipledrag.h>
+#include <kurldrag.h>
+
+#include "treeview.h"
+#include "treeview.moc"
+#include "khotkeys.h"
+#include "menufile.h"
+#include "menuinfo.h"
+
+#define MOVE_FOLDER 'M'
+#define COPY_FOLDER 'C'
+#define MOVE_FILE 'm'
+#define COPY_FILE 'c'
+#define COPY_SEPARATOR 'S'
+
+TreeItem::TreeItem(TQListViewItem *parent, TQListViewItem *after, const TQString& menuId, bool __init)
+ :TQListViewItem(parent, after), _hidden(false), _init(__init), _layoutDirty(false), _menuId(menuId),
+ m_folderInfo(0), m_entryInfo(0) {}
+
+TreeItem::TreeItem(TQListView *parent, TQListViewItem *after, const TQString& menuId, bool __init)
+ : TQListViewItem(parent, after), _hidden(false), _init(__init), _layoutDirty(false), _menuId(menuId),
+ m_folderInfo(0), m_entryInfo(0) {}
+
+void TreeItem::setName(const TQString &name)
+{
+ _name = name;
+ update();
+}
+
+void TreeItem::setHidden(bool b)
+{
+ if (_hidden == b) return;
+ _hidden = b;
+ update();
+}
+
+void TreeItem::update()
+{
+ TQString s = _name;
+ if (_hidden)
+ s += i18n(" [Hidden]");
+ setText(0, s);
+}
+
+void TreeItem::setOpen(bool o)
+{
+ if (o)
+ load();
+
+ TQListViewItem::setOpen(o);
+}
+
+void TreeItem::load()
+{
+ if (m_folderInfo && !_init)
+ {
+ _init = true;
+ TreeView *tv = static_cast<TreeView *>(listView());
+ tv->fillBranch(m_folderInfo, this);
+ }
+}
+
+void TreeItem::paintCell ( TQPainter * p, const TQColorGroup & cg, int column, int width, int align )
+{
+ TQListViewItem::paintCell(p, cg, column, width, align);
+
+ if (!m_folderInfo && !m_entryInfo)
+ {
+ // Draw Separator
+ int h = (height() / 2) -1;
+ if (isSelected())
+ p->setPen( cg.highlightedText() );
+ else
+ p->setPen( cg.text() );
+ p->drawLine(0, h,
+ width, h);
+ }
+}
+
+void TreeItem::setup()
+{
+ TQListViewItem::setup();
+ if (!m_folderInfo && !m_entryInfo)
+ setHeight(8);
+}
+
+static TQPixmap appIcon(const TQString &iconName)
+{
+ TQPixmap normal = TDEGlobal::iconLoader()->loadIcon(iconName, TDEIcon::Small, 0, TDEIcon::DefaultState, 0L, true);
+ // make sure they are not larger than 20x20
+ if (normal.width() > 20 || normal.height() > 20)
+ {
+ TQImage tmp = normal.convertToImage();
+ tmp = tmp.smoothScale(20, 20);
+ normal.convertFromImage(tmp);
+ }
+ return normal;
+}
+
+
+TreeView::TreeView( bool controlCenter, TDEActionCollection *ac, TQWidget *parent, const char *name )
+ : TDEListView(parent, name), m_ac(ac), m_rmb(0), m_clipboard(0),
+ m_clipboardFolderInfo(0), m_clipboardEntryInfo(0),
+ m_controlCenter(controlCenter), m_layoutDirty(false)
+{
+ setFrameStyle(TQFrame::WinPanel | TQFrame::Sunken);
+ setAllColumnsShowFocus(true);
+ setRootIsDecorated(true);
+ setSorting(-1);
+ setAcceptDrops(true);
+ setDropVisualizer(true);
+ setDragEnabled(true);
+ setMinimumWidth(240);
+
+ addColumn("");
+ header()->hide();
+
+ connect(this, TQT_SIGNAL(dropped(TQDropEvent*, TQListViewItem*, TQListViewItem*)),
+ TQT_SLOT(slotDropped(TQDropEvent*, TQListViewItem*, TQListViewItem*)));
+
+ connect(this, TQT_SIGNAL(clicked( TQListViewItem* )),
+ TQT_SLOT(itemSelected( TQListViewItem* )));
+
+ connect(this,TQT_SIGNAL(selectionChanged ( TQListViewItem * )),
+ TQT_SLOT(itemSelected( TQListViewItem* )));
+
+ connect(this, TQT_SIGNAL(rightButtonPressed(TQListViewItem*, const TQPoint&, int)),
+ TQT_SLOT(slotRMBPressed(TQListViewItem*, const TQPoint&)));
+
+ // connect actions
+ connect(m_ac->action("newitem"), TQT_SIGNAL(activated()), TQT_SLOT(newitem()));
+ connect(m_ac->action("newsubmenu"), TQT_SIGNAL(activated()), TQT_SLOT(newsubmenu()));
+ if (m_ac->action("newsep"))
+ connect(m_ac->action("newsep"), TQT_SIGNAL(activated()), TQT_SLOT(newsep()));
+
+ m_menuFile = new MenuFile( locateLocal("xdgconf-menu", "applications-tdemenuedit.menu"));
+ m_rootFolder = new MenuFolderInfo;
+ m_separator = new MenuSeparatorInfo;
+ m_drag = 0;
+
+ // Read menu format configuration information
+ TDESharedConfig::Ptr pConfig = TDESharedConfig::openConfig("kickerrc");
+
+ pConfig->setGroup("menus");
+ m_detailedMenuEntries = pConfig->readBoolEntry("DetailedMenuEntries",true);
+ if (m_detailedMenuEntries)
+ {
+ m_detailedEntriesNamesFirst = pConfig->readBoolEntry("DetailedEntriesNamesFirst",false);
+ }
+}
+
+TreeView::~TreeView() {
+ cleanupClipboard();
+ delete m_rootFolder;
+ delete m_separator;
+}
+
+void TreeView::setViewMode(bool showHidden)
+{
+ delete m_rmb;
+
+ // setup rmb menu
+ m_rmb = new TQPopupMenu(this);
+ TDEAction *action;
+
+ action = m_ac->action("edit_cut");
+ if(action) {
+ action->plug(m_rmb);
+ action->setEnabled(false);
+ connect(action, TQT_SIGNAL(activated()), TQT_SLOT(cut()));
+ }
+
+ action = m_ac->action("edit_copy");
+ if(action) {
+ action->plug(m_rmb);
+ action->setEnabled(false);
+ connect(action, TQT_SIGNAL(activated()), TQT_SLOT(copy()));
+ }
+
+ action = m_ac->action("edit_paste");
+ if(action) {
+ action->plug(m_rmb);
+ action->setEnabled(false);
+ connect(action, TQT_SIGNAL(activated()), TQT_SLOT(paste()));
+ }
+
+ m_rmb->insertSeparator();
+
+ action = m_ac->action("delete");
+ if(action) {
+ action->plug(m_rmb);
+ action->setEnabled(false);
+ connect(action, TQT_SIGNAL(activated()), TQT_SLOT(del()));
+ }
+
+ m_rmb->insertSeparator();
+
+ if(m_ac->action("newitem"))
+ m_ac->action("newitem")->plug(m_rmb);
+ if(m_ac->action("newsubmenu"))
+ m_ac->action("newsubmenu")->plug(m_rmb);
+ if(m_ac->action("newsep"))
+ m_ac->action("newsep")->plug(m_rmb);
+
+ m_showHidden = showHidden;
+ readMenuFolderInfo();
+ fill();
+}
+
+void TreeView::readMenuFolderInfo(MenuFolderInfo *folderInfo, KServiceGroup::Ptr folder, const TQString &prefix)
+{
+ if (!folderInfo)
+ {
+ folderInfo = m_rootFolder;
+ if (m_controlCenter)
+ folder = KServiceGroup::baseGroup("settings");
+ else
+ folder = KServiceGroup::root();
+ }
+
+ if (!folder || !folder->isValid())
+ return;
+
+ folderInfo->caption = folder->caption();
+ folderInfo->comment = folder->comment();
+
+ // Item names may contain ampersands. To avoid them being converted
+ // to accelerators, replace them with two ampersands.
+ folderInfo->hidden = folder->noDisplay();
+ folderInfo->directoryFile = folder->directoryEntryPath();
+ folderInfo->icon = folder->icon();
+ TQString id = folder->relPath();
+ int i = id.findRev('/', -2);
+ id = id.mid(i+1);
+ folderInfo->id = id;
+ folderInfo->fullId = prefix + id;
+
+ KServiceGroup::List list = folder->entries(true, !m_showHidden, true, m_detailedMenuEntries && !m_detailedEntriesNamesFirst);
+
+ for(KServiceGroup::List::ConstIterator it = list.begin();
+ it != list.end(); ++it)
+ {
+ KSycocaEntry * e = *it;
+
+ if (e->isType(KST_KServiceGroup))
+ {
+ KServiceGroup::Ptr g(static_cast<KServiceGroup *>(e));
+ MenuFolderInfo *subFolderInfo = new MenuFolderInfo();
+ readMenuFolderInfo(subFolderInfo, g, folderInfo->fullId);
+ folderInfo->add(subFolderInfo, true);
+ }
+ else if (e->isType(KST_KService))
+ {
+ folderInfo->add(new MenuEntryInfo(static_cast<KService *>(e)), true);
+ }
+ else if (e->isType(KST_KServiceSeparator))
+ {
+ folderInfo->add(m_separator, true);
+ }
+ }
+}
+
+void TreeView::fill()
+{
+ TQApplication::setOverrideCursor(Qt::WaitCursor);
+ clear();
+ fillBranch(m_rootFolder, 0);
+ TQApplication::restoreOverrideCursor();
+}
+
+TQString TreeView::findName(KDesktopFile *df, bool deleted)
+{
+ TQString name = df->readName();
+ if (deleted)
+ {
+ if (name == "empty")
+ name = TQString::null;
+ if (name.isEmpty())
+ {
+ TQString file = df->fileName();
+ TQString res = df->resource();
+
+ bool isLocal = true;
+ TQStringList files = TDEGlobal::dirs()->findAllResources(res.latin1(), file);
+ for(TQStringList::ConstIterator it = files.begin();
+ it != files.end();
+ ++it)
+ {
+ if (isLocal)
+ {
+ isLocal = false;
+ continue;
+ }
+
+ KDesktopFile df2(*it);
+ name = df2.readName();
+
+ if (!name.isEmpty() && (name != "empty"))
+ return name;
+ }
+ }
+ }
+ return name;
+}
+
+TreeItem *TreeView::createTreeItem(TreeItem *parent, TQListViewItem *after, MenuFolderInfo *folderInfo, bool _init)
+{
+ TreeItem *item;
+ if (parent == 0)
+ item = new TreeItem(this, after, TQString::null, _init);
+ else
+ item = new TreeItem(parent, after, TQString::null, _init);
+
+ item->setMenuFolderInfo(folderInfo);
+ item->setName(folderInfo->caption);
+ item->setPixmap(0, appIcon(folderInfo->icon));
+ item->setDirectoryPath(folderInfo->fullId);
+ item->setHidden(folderInfo->hidden);
+ item->setExpandable(true);
+ return item;
+}
+
+TreeItem *TreeView::createTreeItem(TreeItem *parent, TQListViewItem *after, MenuEntryInfo *entryInfo, bool _init)
+{
+ bool hidden = entryInfo->hidden;
+
+ TreeItem* item;
+ if (parent == 0)
+ item = new TreeItem(this, after, entryInfo->menuId(), _init);
+ else
+ item = new TreeItem(parent, after, entryInfo->menuId(),_init);
+
+ QString name;
+
+ if (m_detailedMenuEntries && entryInfo->description.length() != 0)
+ {
+ if (m_detailedEntriesNamesFirst)
+ {
+ name = entryInfo->caption + " (" + entryInfo->description + ")";
+ }
+ else
+ {
+ name = entryInfo->description + " (" + entryInfo->caption + ")";
+ }
+ }
+ else
+ {
+ name = entryInfo->caption;
+ }
+ item->setMenuEntryInfo(entryInfo);
+ item->setName(name);
+ item->setPixmap(0, appIcon(entryInfo->icon));
+
+ item->setHidden(hidden);
+ return item;
+}
+
+TreeItem *TreeView::createTreeItem(TreeItem *parent, TQListViewItem *after, MenuSeparatorInfo *, bool _init)
+{
+ TreeItem* item;
+ if (parent == 0)
+ item = new TreeItem(this, after, TQString::null, _init);
+ else
+ item = new TreeItem(parent, after, TQString::null,_init);
+
+ return item;
+}
+
+void TreeView::fillBranch(MenuFolderInfo *folderInfo, TreeItem *parent)
+{
+ TQString relPath = parent ? parent->directory() : TQString::null;
+ TQPtrListIterator<MenuInfo> it( folderInfo->initialLayout );
+ TreeItem *after = 0;
+ for (MenuInfo *info; (info = it.current()); ++it)
+ {
+ MenuEntryInfo *entry = dynamic_cast<MenuEntryInfo*>(info);
+ if (entry)
+ {
+ after = createTreeItem(parent, after, entry);
+ continue;
+ }
+
+ MenuFolderInfo *subFolder = dynamic_cast<MenuFolderInfo*>(info);
+ if (subFolder)
+ {
+ after = createTreeItem(parent, after, subFolder);
+ continue;
+ }
+ MenuSeparatorInfo *separator = dynamic_cast<MenuSeparatorInfo*>(info);
+ if (separator)
+ {
+ after = createTreeItem(parent, after, separator);
+ continue;
+ }
+ }
+}
+
+void TreeView::closeAllItems(TQListViewItem *item)
+{
+ if (!item) return;
+ while(item)
+ {
+ item->setOpen(false);
+ closeAllItems(item->firstChild());
+ item = item->nextSibling();
+ }
+}
+
+void TreeView::selectMenu(const TQString &menu)
+{
+ closeAllItems(firstChild());
+
+ if (menu.length() <= 1)
+ {
+ setCurrentItem(firstChild());
+ clearSelection();
+ return; // Root menu
+ }
+
+ TQString restMenu = menu.mid(1);
+ if (!restMenu.endsWith("/"))
+ restMenu += "/";
+
+ TreeItem *item = 0;
+ do
+ {
+ int i = restMenu.find("/");
+ TQString subMenu = restMenu.left(i+1);
+ restMenu = restMenu.mid(i+1);
+
+ item = (TreeItem*)(item ? item->firstChild() : firstChild());
+ while(item)
+ {
+ MenuFolderInfo *folderInfo = item->folderInfo();
+ if (folderInfo && (folderInfo->id == subMenu))
+ {
+ item->setOpen(true);
+ break;
+ }
+ item = (TreeItem*) item->nextSibling();
+ }
+ }
+ while( item && !restMenu.isEmpty());
+
+ if (item)
+ {
+ setCurrentItem(item);
+ ensureItemVisible(item);
+ }
+}
+
+void TreeView::selectMenuEntry(const TQString &menuEntry)
+{
+ TreeItem *item = (TreeItem *) selectedItem();
+ if (!item)
+ {
+ item = (TreeItem *) currentItem();
+ while (item && item->isDirectory())
+ item = (TreeItem*) item->nextSibling();
+ }
+ else
+ item = (TreeItem *) item->firstChild();
+
+ while(item)
+ {
+ MenuEntryInfo *entry = item->entryInfo();
+ if (entry && (entry->menuId() == menuEntry))
+ {
+ setCurrentItem(item);
+ ensureItemVisible(item);
+ return;
+ }
+ item = (TreeItem*) item->nextSibling();
+ }
+}
+
+void TreeView::itemSelected(TQListViewItem *item)
+{
+ TreeItem *_item = (TreeItem*)item;
+ bool selected = false;
+ bool dselected = false;
+ if (_item) {
+ selected = true;
+ dselected = _item->isHidden();
+ }
+
+ m_ac->action("edit_cut")->setEnabled(selected);
+ m_ac->action("edit_copy")->setEnabled(selected);
+
+ if (m_ac->action("delete"))
+ m_ac->action("delete")->setEnabled(selected && !dselected);
+
+ if(!item)
+ {
+ emit disableAction();
+ return;
+ }
+
+ if (_item->isDirectory())
+ emit entrySelected(_item->folderInfo());
+ else
+ emit entrySelected(_item->entryInfo());
+}
+
+void TreeView::currentChanged(MenuFolderInfo *folderInfo)
+{
+ TreeItem *item = (TreeItem*)selectedItem();
+ if (item == 0) return;
+ if (folderInfo == 0) return;
+
+ item->setName(folderInfo->caption);
+ item->setPixmap(0, appIcon(folderInfo->icon));
+}
+
+void TreeView::currentChanged(MenuEntryInfo *entryInfo)
+{
+ TreeItem *item = (TreeItem*)selectedItem();
+ if (item == 0) return;
+ if (entryInfo == 0) return;
+
+ QString name;
+
+ if (m_detailedMenuEntries && entryInfo->description.length() != 0)
+ {
+ if (m_detailedEntriesNamesFirst)
+ {
+ name = entryInfo->caption + " (" + entryInfo->description + ")";
+ }
+ else
+ {
+ name = entryInfo->description + " (" + entryInfo->caption + ")";
+ }
+ }
+ else
+ {
+ name = entryInfo->caption;
+ }
+ item->setName(name);
+ item->setPixmap(0, appIcon(entryInfo->icon));
+}
+
+TQStringList TreeView::fileList(const TQString& rPath)
+{
+ TQString relativePath = rPath;
+
+ // truncate "/.directory"
+ int pos = relativePath.findRev("/.directory");
+ if (pos > 0) relativePath.truncate(pos);
+
+ TQStringList filelist;
+
+ // loop through all resource dirs and build a file list
+ TQStringList resdirlist = TDEGlobal::dirs()->resourceDirs("apps");
+ for (TQStringList::ConstIterator it = resdirlist.begin(); it != resdirlist.end(); ++it)
+ {
+ TQDir dir((*it) + "/" + relativePath);
+ if(!dir.exists()) continue;
+
+ dir.setFilter(TQDir::Files);
+ dir.setNameFilter("*.desktop;*.kdelnk");
+
+ // build a list of files
+ TQStringList files = dir.entryList();
+ for (TQStringList::ConstIterator it = files.begin(); it != files.end(); ++it) {
+ // does not work?!
+ //if (filelist.contains(*it)) continue;
+
+ if (relativePath.isEmpty()) {
+ filelist.remove(*it); // hack
+ filelist.append(*it);
+ }
+ else {
+ filelist.remove(relativePath + "/" + *it); //hack
+ filelist.append(relativePath + "/" + *it);
+ }
+ }
+ }
+ return filelist;
+}
+
+TQStringList TreeView::dirList(const TQString& rPath)
+{
+ TQString relativePath = rPath;
+
+ // truncate "/.directory"
+ int pos = relativePath.findRev("/.directory");
+ if (pos > 0) relativePath.truncate(pos);
+
+ TQStringList dirlist;
+
+ // loop through all resource dirs and build a subdir list
+ TQStringList resdirlist = TDEGlobal::dirs()->resourceDirs("apps");
+ for (TQStringList::ConstIterator it = resdirlist.begin(); it != resdirlist.end(); ++it)
+ {
+ TQDir dir((*it) + "/" + relativePath);
+ if(!dir.exists()) continue;
+ dir.setFilter(TQDir::Dirs);
+
+ // build a list of subdirs
+ TQStringList subdirs = dir.entryList();
+ for (TQStringList::ConstIterator it = subdirs.begin(); it != subdirs.end(); ++it) {
+ if ((*it) == "." || (*it) == "..") continue;
+ // does not work?!
+ // if (dirlist.contains(*it)) continue;
+
+ if (relativePath.isEmpty()) {
+ dirlist.remove(*it); //hack
+ dirlist.append(*it);
+ }
+ else {
+ dirlist.remove(relativePath + "/" + *it); //hack
+ dirlist.append(relativePath + "/" + *it);
+ }
+ }
+ }
+ return dirlist;
+}
+
+bool TreeView::acceptDrag(TQDropEvent* e) const
+{
+ if (e->provides("application/x-kmenuedit-internal") &&
+ (e->source() == const_cast<TreeView *>(this)))
+ return true;
+ KURL::List urls;
+ if (KURLDrag::decode(e, urls) && (urls.count() == 1) &&
+ urls[0].isLocalFile() && urls[0].path().endsWith(".desktop"))
+ return true;
+ return false;
+}
+
+
+static TQString createDesktopFile(const TQString &file, TQString *menuId, TQStringList *excludeList)
+{
+ TQString base = file.mid(file.findRev('/')+1);
+ base = base.left(base.findRev('.'));
+
+ TQRegExp r("(.*)(?=-\\d+)");
+ base = (r.search(base) > -1) ? r.cap(1) : base;
+
+ TQString result = KService::newServicePath(true, base, menuId, excludeList);
+ excludeList->append(*menuId);
+ // Todo for Undo-support: Undo menuId allocation:
+
+ return result;
+}
+
+static KDesktopFile *copyDesktopFile(MenuEntryInfo *entryInfo, TQString *menuId, TQStringList *excludeList)
+{
+ TQString result = createDesktopFile(entryInfo->file(), menuId, excludeList);
+ KDesktopFile *df = entryInfo->desktopFile()->copyTo(result);
+ df->deleteEntry("Categories"); // Don't set any categories!
+
+ return df;
+}
+
+static TQString createDirectoryFile(const TQString &file, TQStringList *excludeList)
+{
+ TQString base = file.mid(file.findRev('/')+1);
+ base = base.left(base.findRev('.'));
+
+ TQString result;
+ int i = 1;
+ while(true)
+ {
+ if (i == 1)
+ result = base + ".directory";
+ else
+ result = base + TQString("-%1.directory").arg(i);
+
+ if (!excludeList->contains(result))
+ {
+ if (locate("xdgdata-dirs", result).isEmpty())
+ break;
+ }
+ i++;
+ }
+ excludeList->append(result);
+ result = locateLocal("xdgdata-dirs", result);
+ return result;
+}
+
+
+void TreeView::slotDropped (TQDropEvent * e, TQListViewItem *parent, TQListViewItem*after)
+{
+ if(!e) return;
+
+ // get destination folder
+ TreeItem *parentItem = static_cast<TreeItem*>(parent);
+ TQString folder = parentItem ? parentItem->directory() : TQString::null;
+ MenuFolderInfo *parentFolderInfo = parentItem ? parentItem->folderInfo() : m_rootFolder;
+
+ if (e->source() != this)
+ {
+ // External drop
+ KURL::List urls;
+ if (!KURLDrag::decode(e, urls) || (urls.count() != 1) || !urls[0].isLocalFile())
+ return;
+ TQString path = urls[0].path();
+ if (!path.endsWith(".desktop"))
+ return;
+
+ TQString menuId;
+ TQString result = createDesktopFile(path, &menuId, &m_newMenuIds);
+ KDesktopFile orig_df(path);
+ KDesktopFile *df = orig_df.copyTo(result);
+ df->deleteEntry("Categories"); // Don't set any categories!
+
+ KService *s = new KService(df);
+ s->setMenuId(menuId);
+
+ MenuEntryInfo *entryInfo = new MenuEntryInfo(s, df);
+
+ TQString oldCaption = entryInfo->caption;
+ TQString newCaption = parentFolderInfo->uniqueItemCaption(oldCaption, oldCaption);
+ entryInfo->setCaption(newCaption);
+
+ // Add file to menu
+ // m_menuFile->addEntry(folder, menuId);
+ m_menuFile->pushAction(MenuFile::ADD_ENTRY, folder, menuId);
+
+ // create the TreeItem
+ if(parentItem)
+ parentItem->setOpen(true);
+
+ // update fileInfo data
+ parentFolderInfo->add(entryInfo);
+
+ TreeItem *newItem = createTreeItem(parentItem, after, entryInfo, true);
+
+ setSelected ( newItem, true);
+ itemSelected( newItem);
+
+ m_drag = 0;
+ setLayoutDirty(parentItem);
+ return;
+ }
+
+ // is there content in the clipboard?
+ if (!m_drag) return;
+
+ if (m_dragItem == after) return; // Nothing to do
+
+ int command = m_drag;
+ if (command == MOVE_FOLDER)
+ {
+ MenuFolderInfo *folderInfo = m_dragInfo;
+ if (e->action() == TQDropEvent::Copy)
+ {
+ // Ugh.. this is hard :)
+ // * Create new .directory file
+ // Add
+ }
+ else
+ {
+ TreeItem *tmpItem = static_cast<TreeItem*>(parentItem);
+ while ( tmpItem )
+ {
+ if ( tmpItem == m_dragItem )
+ {
+ m_drag = 0;
+ return;
+ }
+ tmpItem = static_cast<TreeItem*>(tmpItem->parent() );
+ }
+
+ // Remove MenuFolderInfo
+ TreeItem *oldParentItem = static_cast<TreeItem*>(m_dragItem->parent());
+ MenuFolderInfo *oldParentFolderInfo = oldParentItem ? oldParentItem->folderInfo() : m_rootFolder;
+ oldParentFolderInfo->take(folderInfo);
+
+ // Move menu
+ TQString oldFolder = folderInfo->fullId;
+ TQString folderName = folderInfo->id;
+ TQString newFolder = m_menuFile->uniqueMenuName(folder, folderName, parentFolderInfo->existingMenuIds());
+ folderInfo->id = newFolder;
+
+ // Add file to menu
+ //m_menuFile->moveMenu(oldFolder, folder + newFolder);
+ m_menuFile->pushAction(MenuFile::MOVE_MENU, oldFolder, folder + newFolder);
+
+ // Make sure caption is unique
+ TQString newCaption = parentFolderInfo->uniqueMenuCaption(folderInfo->caption);
+ if (newCaption != folderInfo->caption)
+ {
+ folderInfo->setCaption(newCaption);
+ }
+
+ // create the TreeItem
+ if(parentItem)
+ parentItem->setOpen(true);
+
+ // update fileInfo data
+ folderInfo->updateFullId(parentFolderInfo->fullId);
+ folderInfo->setInUse(true);
+ parentFolderInfo->add(folderInfo);
+
+ if ((parentItem != oldParentItem) || !after)
+ {
+ if (oldParentItem)
+ oldParentItem->takeItem(m_dragItem);
+ else
+ takeItem(m_dragItem);
+ if (parentItem)
+ parentItem->insertItem(m_dragItem);
+ else
+ insertItem(m_dragItem);
+ }
+ m_dragItem->moveItem(after);
+ m_dragItem->setName(folderInfo->caption);
+ m_dragItem->setDirectoryPath(folderInfo->fullId);
+ setSelected(m_dragItem, true);
+ itemSelected(m_dragItem);
+ }
+ }
+ else if (command == MOVE_FILE)
+ {
+ MenuEntryInfo *entryInfo = m_dragItem->entryInfo();
+ TQString menuId = entryInfo->menuId();
+
+ if (e->action() == TQDropEvent::Copy)
+ {
+
+ // Need to copy file and then add it
+ KDesktopFile *df = copyDesktopFile(entryInfo, &menuId, &m_newMenuIds); // Duplicate
+//UNDO-ACTION: NEW_MENU_ID (menuId)
+
+ KService *s = new KService(df);
+ s->setMenuId(menuId);
+
+ entryInfo = new MenuEntryInfo(s, df);
+
+ TQString oldCaption = entryInfo->caption;
+ TQString newCaption = parentFolderInfo->uniqueItemCaption(oldCaption, oldCaption);
+ entryInfo->setCaption(newCaption);
+ }
+ else
+ {
+ del(m_dragItem, false);
+ TQString oldCaption = entryInfo->caption;
+ TQString newCaption = parentFolderInfo->uniqueItemCaption(oldCaption);
+ entryInfo->setCaption(newCaption);
+ entryInfo->setInUse(true);
+ }
+ // Add file to menu
+ // m_menuFile->addEntry(folder, menuId);
+ m_menuFile->pushAction(MenuFile::ADD_ENTRY, folder, menuId);
+
+ // create the TreeItem
+ if(parentItem)
+ parentItem->setOpen(true);
+
+ // update fileInfo data
+ parentFolderInfo->add(entryInfo);
+
+ TreeItem *newItem = createTreeItem(parentItem, after, entryInfo, true);
+
+ setSelected ( newItem, true);
+ itemSelected( newItem);
+ }
+ else if (command == COPY_SEPARATOR)
+ {
+ if (e->action() != TQDropEvent::Copy)
+ del(m_dragItem, false);
+
+ TreeItem *newItem = createTreeItem(parentItem, after, m_separator, true);
+
+ setSelected ( newItem, true);
+ itemSelected( newItem);
+ }
+ else
+ {
+ // Error
+ }
+ m_drag = 0;
+ setLayoutDirty(parentItem);
+}
+
+
+void TreeView::startDrag()
+{
+ TQDragObject *drag = dragObject();
+
+ if (!drag)
+ return;
+
+ drag->dragMove();
+}
+
+TQDragObject *TreeView::dragObject()
+{
+ m_dragPath = TQString::null;
+ TreeItem *item = (TreeItem*)selectedItem();
+ if(item == 0) return 0;
+
+ KMultipleDrag *drag = new KMultipleDrag( this );
+
+ if (item->isDirectory())
+ {
+ m_drag = MOVE_FOLDER;
+ m_dragInfo = item->folderInfo();
+ m_dragItem = item;
+ }
+ else if (item->isEntry())
+ {
+ m_drag = MOVE_FILE;
+ m_dragInfo = 0;
+ m_dragItem = item;
+ TQString menuId = item->menuId();
+ m_dragPath = item->entryInfo()->service->desktopEntryPath();
+ if (!m_dragPath.isEmpty())
+ m_dragPath = locate("apps", m_dragPath);
+ if (!m_dragPath.isEmpty())
+ {
+ KURL url;
+ url.setPath(m_dragPath);
+ drag->addDragObject( new KURLDrag(url, 0));
+ }
+ }
+ else
+ {
+ m_drag = COPY_SEPARATOR;
+ m_dragInfo = 0;
+ m_dragItem = item;
+ }
+
+ drag->addDragObject( new TQStoredDrag("application/x-kmenuedit-internal", 0));
+ if ( item->pixmap(0) )
+ drag->setPixmap(*item->pixmap(0));
+ return drag;
+}
+
+void TreeView::slotRMBPressed(TQListViewItem*, const TQPoint& p)
+{
+ TreeItem *item = (TreeItem*)selectedItem();
+ if(item == 0) return;
+
+ if(m_rmb) m_rmb->exec(p);
+}
+
+void TreeView::newsubmenu()
+{
+ TreeItem *parentItem = 0;
+ TreeItem *item = (TreeItem*)selectedItem();
+
+ bool ok;
+ TQString caption = KInputDialog::getText( i18n( "New Submenu" ),
+ i18n( "Submenu name:" ), TQString::null, &ok, this );
+
+ if (!ok) return;
+
+ TQString file = caption;
+ file.replace('/', '-');
+
+ file = createDirectoryFile(file, &m_newDirectoryList); // Create
+
+ // get destination folder
+ TQString folder;
+
+ if(!item)
+ {
+ parentItem = 0;
+ folder = TQString::null;
+ }
+ else if(item->isDirectory())
+ {
+ parentItem = item;
+ item = 0;
+ folder = parentItem->directory();
+ }
+ else
+ {
+ parentItem = static_cast<TreeItem*>(item->parent());
+ folder = parentItem ? parentItem->directory() : TQString::null;
+ }
+
+ MenuFolderInfo *parentFolderInfo = parentItem ? parentItem->folderInfo() : m_rootFolder;
+ MenuFolderInfo *folderInfo = new MenuFolderInfo();
+ folderInfo->caption = parentFolderInfo->uniqueMenuCaption(caption);
+ folderInfo->id = m_menuFile->uniqueMenuName(folder, caption, parentFolderInfo->existingMenuIds());
+ folderInfo->directoryFile = file;
+ folderInfo->icon = "package";
+ folderInfo->hidden = false;
+ folderInfo->setDirty();
+
+ KDesktopFile *df = new KDesktopFile(file);
+ df->writeEntry("Name", folderInfo->caption);
+ df->writeEntry("Icon", folderInfo->icon);
+ df->sync();
+ delete df;
+ // Add file to menu
+ // m_menuFile->addMenu(folder + folderInfo->id, file);
+ m_menuFile->pushAction(MenuFile::ADD_MENU, folder + folderInfo->id, file);
+
+ folderInfo->fullId = parentFolderInfo->fullId + folderInfo->id;
+
+ // create the TreeItem
+ if(parentItem)
+ parentItem->setOpen(true);
+
+ // update fileInfo data
+ parentFolderInfo->add(folderInfo);
+
+ TreeItem *newItem = createTreeItem(parentItem, item, folderInfo, true);
+
+ setSelected ( newItem, true);
+ itemSelected( newItem);
+
+ setLayoutDirty(parentItem);
+}
+
+void TreeView::newitem()
+{
+ TreeItem *parentItem = 0;
+ TreeItem *item = (TreeItem*)selectedItem();
+
+ bool ok;
+ TQString caption = KInputDialog::getText( i18n( "New Item" ),
+ i18n( "Item name:" ), TQString::null, &ok, this );
+
+ if (!ok) return;
+
+ TQString menuId;
+ TQString file = caption;
+ file.replace('/', '-');
+
+ file = createDesktopFile(file, &menuId, &m_newMenuIds); // Create
+
+ KDesktopFile *df = new KDesktopFile(file);
+ df->writeEntry("Name", caption);
+ df->writeEntry("Type", "Application");
+
+ // get destination folder
+ TQString folder;
+
+ if(!item)
+ {
+ parentItem = 0;
+ folder = TQString::null;
+ }
+ else if(item->isDirectory())
+ {
+ parentItem = item;
+ item = 0;
+ folder = parentItem->directory();
+ }
+ else
+ {
+ parentItem = static_cast<TreeItem*>(item->parent());
+ folder = parentItem ? parentItem->directory() : TQString::null;
+ }
+
+ MenuFolderInfo *parentFolderInfo = parentItem ? parentItem->folderInfo() : m_rootFolder;
+
+ // Add file to menu
+ // m_menuFile->addEntry(folder, menuId);
+ m_menuFile->pushAction(MenuFile::ADD_ENTRY, folder, menuId);
+
+ KService *s = new KService(df);
+ s->setMenuId(menuId);
+
+ MenuEntryInfo *entryInfo = new MenuEntryInfo(s, df);
+
+ // create the TreeItem
+ if(parentItem)
+ parentItem->setOpen(true);
+
+ // update fileInfo data
+ parentFolderInfo->add(entryInfo);
+
+ TreeItem *newItem = createTreeItem(parentItem, item, entryInfo, true);
+
+ setSelected ( newItem, true);
+ itemSelected( newItem);
+
+ setLayoutDirty(parentItem);
+}
+
+void TreeView::newsep()
+{
+ TreeItem *parentItem = 0;
+ TreeItem *item = (TreeItem*)selectedItem();
+
+ if(!item)
+ {
+ parentItem = 0;
+ }
+ else if(item->isDirectory())
+ {
+ parentItem = item;
+ item = 0;
+ }
+ else
+ {
+ parentItem = static_cast<TreeItem*>(item->parent());
+ }
+
+ // create the TreeItem
+ if(parentItem)
+ parentItem->setOpen(true);
+
+ TreeItem *newItem = createTreeItem(parentItem, item, m_separator, true);
+
+ setSelected ( newItem, true);
+ itemSelected( newItem);
+
+ setLayoutDirty(parentItem);
+}
+
+void TreeView::cut()
+{
+ copy( true );
+
+ m_ac->action("edit_cut")->setEnabled(false);
+ m_ac->action("edit_copy")->setEnabled(false);
+ m_ac->action("delete")->setEnabled(false);
+
+ // Select new current item
+ setSelected( currentItem(), true );
+ // Switch the UI to show that item
+ itemSelected( selectedItem() );
+}
+
+void TreeView::copy()
+{
+ copy( false );
+}
+
+void TreeView::copy( bool cutting )
+{
+ TreeItem *item = (TreeItem*)selectedItem();
+
+ // nil selected? -> nil to copy
+ if (item == 0) return;
+
+ if (cutting)
+ setLayoutDirty((TreeItem*)item->parent());
+
+ // clean up old stuff
+ cleanupClipboard();
+
+ // is item a folder or a file?
+ if(item->isDirectory())
+ {
+ TQString folder = item->directory();
+ if (cutting)
+ {
+ // Place in clipboard
+ m_clipboard = MOVE_FOLDER;
+ m_clipboardFolderInfo = item->folderInfo();
+
+ del(item, false);
+ }
+ else
+ {
+ // Place in clipboard
+ m_clipboard = COPY_FOLDER;
+ m_clipboardFolderInfo = item->folderInfo();
+ }
+ }
+ else if (item->isEntry())
+ {
+ if (cutting)
+ {
+ // Place in clipboard
+ m_clipboard = MOVE_FILE;
+ m_clipboardEntryInfo = item->entryInfo();
+
+ del(item, false);
+ }
+ else
+ {
+ // Place in clipboard
+ m_clipboard = COPY_FILE;
+ m_clipboardEntryInfo = item->entryInfo();
+ }
+ }
+ else
+ {
+ // Place in clipboard
+ m_clipboard = COPY_SEPARATOR;
+ if (cutting)
+ del(item, false);
+ }
+
+ m_ac->action("edit_paste")->setEnabled(true);
+}
+
+
+void TreeView::paste()
+{
+ TreeItem *parentItem = 0;
+ TreeItem *item = (TreeItem*)selectedItem();
+
+ // nil selected? -> nil to paste to
+ if (item == 0) return;
+
+ // is there content in the clipboard?
+ if (!m_clipboard) return;
+
+ // get destination folder
+ TQString folder;
+
+ if(item->isDirectory())
+ {
+ parentItem = item;
+ item = 0;
+ folder = parentItem->directory();
+ }
+ else
+ {
+ parentItem = static_cast<TreeItem*>(item->parent());
+ folder = parentItem ? parentItem->directory() : TQString::null;
+ }
+
+ MenuFolderInfo *parentFolderInfo = parentItem ? parentItem->folderInfo() : m_rootFolder;
+ int command = m_clipboard;
+ if ((command == COPY_FOLDER) || (command == MOVE_FOLDER))
+ {
+ MenuFolderInfo *folderInfo = m_clipboardFolderInfo;
+ if (command == COPY_FOLDER)
+ {
+ // Ugh.. this is hard :)
+ // * Create new .directory file
+ // Add
+ }
+ else if (command == MOVE_FOLDER)
+ {
+ // Move menu
+ TQString oldFolder = folderInfo->fullId;
+ TQString folderName = folderInfo->id;
+ TQString newFolder = m_menuFile->uniqueMenuName(folder, folderName, parentFolderInfo->existingMenuIds());
+ folderInfo->id = newFolder;
+
+ // Add file to menu
+ // m_menuFile->moveMenu(oldFolder, folder + newFolder);
+ m_menuFile->pushAction(MenuFile::MOVE_MENU, oldFolder, folder + newFolder);
+
+ // Make sure caption is unique
+ TQString newCaption = parentFolderInfo->uniqueMenuCaption(folderInfo->caption);
+ if (newCaption != folderInfo->caption)
+ {
+ folderInfo->setCaption(newCaption);
+ }
+ // create the TreeItem
+ if(parentItem)
+ parentItem->setOpen(true);
+
+ // update fileInfo data
+ folderInfo->fullId = parentFolderInfo->fullId + folderInfo->id;
+ folderInfo->setInUse(true);
+ parentFolderInfo->add(folderInfo);
+
+ TreeItem *newItem = createTreeItem(parentItem, item, folderInfo);
+
+ setSelected ( newItem, true);
+ itemSelected( newItem);
+ }
+
+ m_clipboard = COPY_FOLDER; // Next one copies.
+ }
+ else if ((command == COPY_FILE) || (command == MOVE_FILE))
+ {
+ MenuEntryInfo *entryInfo = m_clipboardEntryInfo;
+ TQString menuId;
+
+ if (command == COPY_FILE)
+ {
+ // Need to copy file and then add it
+ KDesktopFile *df = copyDesktopFile(entryInfo, &menuId, &m_newMenuIds); // Duplicate
+
+ KService *s = new KService(df);
+ s->setMenuId(menuId);
+ entryInfo = new MenuEntryInfo(s, df);
+
+ TQString oldCaption = entryInfo->caption;
+ TQString newCaption = parentFolderInfo->uniqueItemCaption(oldCaption, oldCaption);
+ entryInfo->setCaption(newCaption);
+ }
+ else if (command == MOVE_FILE)
+ {
+ menuId = entryInfo->menuId();
+ m_clipboard = COPY_FILE; // Next one copies.
+
+ TQString oldCaption = entryInfo->caption;
+ TQString newCaption = parentFolderInfo->uniqueItemCaption(oldCaption);
+ entryInfo->setCaption(newCaption);
+ entryInfo->setInUse(true);
+ }
+ // Add file to menu
+ // m_menuFile->addEntry(folder, menuId);
+ m_menuFile->pushAction(MenuFile::ADD_ENTRY, folder, menuId);
+
+ // create the TreeItem
+ if(parentItem)
+ parentItem->setOpen(true);
+
+ // update fileInfo data
+ parentFolderInfo->add(entryInfo);
+
+ TreeItem *newItem = createTreeItem(parentItem, item, entryInfo, true);
+
+ setSelected ( newItem, true);
+ itemSelected( newItem);
+ }
+ else
+ {
+ // create separator
+ if(parentItem)
+ parentItem->setOpen(true);
+
+ TreeItem *newItem = createTreeItem(parentItem, item, m_separator, true);
+
+ setSelected ( newItem, true);
+ itemSelected( newItem);
+ }
+ setLayoutDirty(parentItem);
+}
+
+void TreeView::del()
+{
+ TreeItem *item = (TreeItem*)selectedItem();
+
+ // nil selected? -> nil to delete
+ if (item == 0) return;
+
+ del(item, true);
+
+ m_ac->action("edit_cut")->setEnabled(false);
+ m_ac->action("edit_copy")->setEnabled(false);
+ m_ac->action("delete")->setEnabled(false);
+ // Select new current item
+ setSelected( currentItem(), true );
+ // Switch the UI to show that item
+ itemSelected( selectedItem() );
+}
+
+void TreeView::del(TreeItem *item, bool deleteInfo)
+{
+ TreeItem *parentItem = static_cast<TreeItem*>(item->parent());
+ // is file a .directory or a .desktop file
+ if(item->isDirectory())
+ {
+ MenuFolderInfo *folderInfo = item->folderInfo();
+
+ // Remove MenuFolderInfo
+ MenuFolderInfo *parentFolderInfo = parentItem ? parentItem->folderInfo() : m_rootFolder;
+ parentFolderInfo->take(folderInfo);
+ folderInfo->setInUse(false);
+
+ if (m_clipboard == COPY_FOLDER && (m_clipboardFolderInfo == folderInfo))
+ {
+ // Copy + Del == Cut
+ m_clipboard = MOVE_FOLDER; // Clipboard now owns folderInfo
+
+ }
+ else
+ {
+ if (folderInfo->takeRecursive(m_clipboardFolderInfo))
+ m_clipboard = MOVE_FOLDER; // Clipboard now owns m_clipboardFolderInfo
+
+ if (deleteInfo)
+ delete folderInfo; // Delete folderInfo
+ }
+
+ // Remove from menu
+ // m_menuFile->removeMenu(item->directory());
+ m_menuFile->pushAction(MenuFile::REMOVE_MENU, item->directory(), TQString::null);
+
+ // Remove tree item
+ delete item;
+ }
+ else if (item->isEntry())
+ {
+ MenuEntryInfo *entryInfo = item->entryInfo();
+ TQString menuId = entryInfo->menuId();
+
+ // Remove MenuFolderInfo
+ MenuFolderInfo *parentFolderInfo = parentItem ? parentItem->folderInfo() : m_rootFolder;
+ parentFolderInfo->take(entryInfo);
+ entryInfo->setInUse(false);
+
+ if (m_clipboard == COPY_FILE && (m_clipboardEntryInfo == entryInfo))
+ {
+ // Copy + Del == Cut
+ m_clipboard = MOVE_FILE; // Clipboard now owns entryInfo
+ }
+ else
+ {
+ if (deleteInfo)
+ delete entryInfo; // Delete entryInfo
+ }
+
+ // Remove from menu
+ TQString folder = parentItem ? parentItem->directory() : TQString::null;
+ // m_menuFile->removeEntry(folder, menuId);
+ m_menuFile->pushAction(MenuFile::REMOVE_ENTRY, folder, menuId);
+
+ // Remove tree item
+ delete item;
+ }
+ else
+ {
+ // Remove separator
+ delete item;
+ }
+ setLayoutDirty(parentItem);
+}
+
+void TreeView::cleanupClipboard() {
+ if (m_clipboard == MOVE_FOLDER)
+ delete m_clipboardFolderInfo;
+ m_clipboardFolderInfo = 0;
+
+ if (m_clipboard == MOVE_FILE)
+ delete m_clipboardEntryInfo;
+ m_clipboardEntryInfo = 0;
+
+ m_clipboard = 0;
+}
+
+static TQStringList extractLayout(TreeItem *item)
+{
+ bool firstFolder = true;
+ bool firstEntry = true;
+ TQStringList layout;
+ for(;item; item = static_cast<TreeItem*>(item->nextSibling()))
+ {
+ if (item->isDirectory())
+ {
+ if (firstFolder)
+ {
+ firstFolder = false;
+ layout << ":M"; // Add new folders here...
+ }
+ layout << (item->folderInfo()->id);
+ }
+ else if (item->isEntry())
+ {
+ if (firstEntry)
+ {
+ firstEntry = false;
+ layout << ":F"; // Add new entries here...
+ }
+ layout << (item->entryInfo()->menuId());
+ }
+ else
+ {
+ layout << ":S";
+ }
+ }
+ return layout;
+}
+
+TQStringList TreeItem::layout()
+{
+ TQStringList layout = extractLayout(static_cast<TreeItem*>(firstChild()));
+ _layoutDirty = false;
+ return layout;
+}
+
+void TreeView::saveLayout()
+{
+ if (m_layoutDirty)
+ {
+ TQStringList layout = extractLayout(static_cast<TreeItem*>(firstChild()));
+ m_menuFile->setLayout(m_rootFolder->fullId, layout);
+ m_layoutDirty = false;
+ }
+
+ TQPtrList<TQListViewItem> lst;
+ TQListViewItemIterator it( this );
+ while ( it.current() ) {
+ TreeItem *item = static_cast<TreeItem*>(it.current());
+ if ( item->isLayoutDirty() )
+ {
+ m_menuFile->setLayout(item->folderInfo()->fullId, item->layout());
+ }
+ ++it;
+ }
+}
+
+bool TreeView::save()
+{
+ saveLayout();
+ m_rootFolder->save(m_menuFile);
+
+ bool success = m_menuFile->performAllActions();
+
+ m_newMenuIds.clear();
+ m_newDirectoryList.clear();
+
+ if (success)
+ {
+ KService::rebuildKSycoca(this);
+ }
+ else
+ {
+ KMessageBox::sorry(this, "<qt>"+i18n("Menu changes could not be saved because of the following problem:")+"<br><br>"+
+ m_menuFile->error()+"</qt>");
+ }
+ return success;
+}
+
+void TreeView::setLayoutDirty(TreeItem *parentItem)
+{
+ if (parentItem)
+ parentItem->setLayoutDirty();
+ else
+ m_layoutDirty = true;
+}
+
+bool TreeView::isLayoutDirty()
+{
+ TQPtrList<TQListViewItem> lst;
+ TQListViewItemIterator it( this );
+ while ( it.current() ) {
+ if ( static_cast<TreeItem*>(it.current())->isLayoutDirty() )
+ return true;
+ ++it;
+ }
+ return false;
+}
+
+bool TreeView::dirty()
+{
+ return m_layoutDirty || m_rootFolder->hasDirt() || m_menuFile->dirty() || isLayoutDirty();
+}
+
+void TreeView::findServiceShortcut(const TDEShortcut&cut, KService::Ptr &service)
+{
+ service = m_rootFolder->findServiceShortcut(cut);
+}
+
diff --git a/kmenuedit/treeview.h b/kmenuedit/treeview.h
new file mode 100644
index 000000000..3370f5bc2
--- /dev/null
+++ b/kmenuedit/treeview.h
@@ -0,0 +1,179 @@
+/*
+ * Copyright (C) 2000 Matthias Elter <[email protected]>
+ * Copyright (C) 2001-2002 Raffaele Sandrini <[email protected]>
+ * Copyright (C) 2003 Waldo Bastian <[email protected]>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ */
+
+#ifndef __treeview_h__
+#define __treeview_h__
+
+#include <tqstring.h>
+#include <tdelistview.h>
+#include <kservice.h>
+#include <kservicegroup.h>
+
+class TQPopupMenu;
+class TDEActionCollection;
+class KDesktopFile;
+class MenuFile;
+class MenuFolderInfo;
+class MenuEntryInfo;
+class MenuSeparatorInfo;
+class TDEShortcut;
+
+class TreeItem : public TQListViewItem
+{
+public:
+ TreeItem(TQListViewItem *parent, TQListViewItem *after, const TQString &menuIdn, bool __init = false);
+ TreeItem(TQListView *parent, TQListViewItem* after, const TQString &menuId, bool __init = false);
+
+ TQString menuId() const { return _menuId; }
+
+ TQString directory() const { return _directoryPath; }
+ void setDirectoryPath(const TQString& path) { _directoryPath = path; }
+
+ MenuFolderInfo *folderInfo() { return m_folderInfo; }
+ void setMenuFolderInfo(MenuFolderInfo *folderInfo) { m_folderInfo = folderInfo; }
+
+ MenuEntryInfo *entryInfo() { return m_entryInfo; }
+ void setMenuEntryInfo(MenuEntryInfo *entryInfo) { m_entryInfo = entryInfo; }
+
+ TQString name() const { return _name; }
+ void setName(const TQString &name);
+
+ bool isDirectory() const { return m_folderInfo; }
+ bool isEntry() const { return m_entryInfo; }
+
+ bool isHidden() const { return _hidden; }
+ void setHidden(bool b);
+
+ bool isLayoutDirty() { return _layoutDirty; }
+ void setLayoutDirty() { _layoutDirty = true; }
+ TQStringList layout();
+
+ virtual void setOpen(bool o);
+ void load();
+
+ virtual void paintCell(TQPainter * p, const TQColorGroup & cg, int column, int width, int align);
+ virtual void setup();
+
+private:
+ void update();
+
+ bool _hidden : 1;
+ bool _init : 1;
+ bool _layoutDirty : 1;
+ TQString _menuId;
+ TQString _name;
+ TQString _directoryPath;
+ MenuFolderInfo *m_folderInfo;
+ MenuEntryInfo *m_entryInfo;
+};
+
+class TreeView : public TDEListView
+{
+ friend class TreeItem;
+ Q_OBJECT
+public:
+ TreeView(bool controlCenter, TDEActionCollection *ac, TQWidget *parent=0, const char *name=0);
+ ~TreeView();
+
+ void readMenuFolderInfo(MenuFolderInfo *folderInfo=0, KServiceGroup::Ptr folder=0, const TQString &prefix=TQString::null);
+ void setViewMode(bool showHidden);
+ bool save();
+
+ bool dirty();
+
+ void selectMenu(const TQString &menu);
+ void selectMenuEntry(const TQString &menuEntry);
+
+public slots:
+ void currentChanged(MenuFolderInfo *folderInfo);
+ void currentChanged(MenuEntryInfo *entryInfo);
+ void findServiceShortcut(const TDEShortcut&, KService::Ptr &);
+
+signals:
+ void entrySelected(MenuFolderInfo *folderInfo);
+ void entrySelected(MenuEntryInfo *entryInfo);
+ void disableAction();
+protected slots:
+ void itemSelected(TQListViewItem *);
+ void slotDropped(TQDropEvent *, TQListViewItem *, TQListViewItem *);
+ void slotRMBPressed(TQListViewItem*, const TQPoint&);
+
+ void newsubmenu();
+ void newitem();
+ void newsep();
+
+ void cut();
+ void copy();
+ void paste();
+ void del();
+
+protected:
+ TreeItem *createTreeItem(TreeItem *parent, TQListViewItem *after, MenuFolderInfo *folderInfo, bool _init = false);
+ TreeItem *createTreeItem(TreeItem *parent, TQListViewItem *after, MenuEntryInfo *entryInfo, bool _init = false);
+ TreeItem *createTreeItem(TreeItem *parent, TQListViewItem *after, MenuSeparatorInfo *sepInfo, bool _init = false);
+
+ void del(TreeItem *, bool deleteInfo);
+ void fill();
+ void fillBranch(MenuFolderInfo *folderInfo, TreeItem *parent);
+ TQString findName(KDesktopFile *df, bool deleted);
+
+ void closeAllItems(TQListViewItem *item);
+
+ // moving = src will be removed later
+ void copy( bool moving );
+
+ void cleanupClipboard();
+
+ bool isLayoutDirty();
+ void setLayoutDirty(TreeItem *);
+ void saveLayout();
+
+ TQStringList fileList(const TQString& relativePath);
+ TQStringList dirList(const TQString& relativePath);
+
+ virtual bool acceptDrag(TQDropEvent* event) const;
+ virtual TQDragObject *dragObject();
+ virtual void startDrag();
+
+private:
+ TDEActionCollection *m_ac;
+ TQPopupMenu *m_rmb;
+ int m_clipboard;
+ MenuFolderInfo *m_clipboardFolderInfo;
+ MenuEntryInfo *m_clipboardEntryInfo;
+ int m_drag;
+ MenuFolderInfo *m_dragInfo;
+ TreeItem *m_dragItem;
+ TQString m_dragPath;
+ bool m_showHidden;
+ bool m_controlCenter;
+ MenuFile *m_menuFile;
+ MenuFolderInfo *m_rootFolder;
+ MenuSeparatorInfo *m_separator;
+ TQStringList m_newMenuIds;
+ TQStringList m_newDirectoryList;
+ bool m_detailedMenuEntries;
+ bool m_detailedEntriesNamesFirst;
+ bool m_layoutDirty;
+};
+
+
+#endif
diff --git a/kmenuedit/uninstall.desktop b/kmenuedit/uninstall.desktop
new file mode 100644
index 000000000..e1e3e1732
--- /dev/null
+++ b/kmenuedit/uninstall.desktop
@@ -0,0 +1,2 @@
+[Desktop Entry]
+Hidden=true