diff options
Diffstat (limited to 'kmenuedit')
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 <elter@kde.org> + * + * 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 <elter@kde.org> + * + * 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 Binary files differnew file mode 100644 index 000000000..b121605f7 --- /dev/null +++ b/kmenuedit/hi16-app-kmenuedit.png diff --git a/kmenuedit/hi22-app-kmenuedit.png b/kmenuedit/hi22-app-kmenuedit.png Binary files differnew file mode 100644 index 000000000..65509221a --- /dev/null +++ b/kmenuedit/hi22-app-kmenuedit.png diff --git a/kmenuedit/hi32-app-kmenuedit.png b/kmenuedit/hi32-app-kmenuedit.png Binary files differnew file mode 100644 index 000000000..d1e532be1 --- /dev/null +++ b/kmenuedit/hi32-app-kmenuedit.png diff --git a/kmenuedit/hi48-app-kmenuedit.png b/kmenuedit/hi48-app-kmenuedit.png Binary files differnew file mode 100644 index 000000000..e6b483a7f --- /dev/null +++ b/kmenuedit/hi48-app-kmenuedit.png 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 <elter@kde.org> + * Copyright (C) 2001-2002 Raffaele Sandrini <sandrini@kde.org> + * Copyright (C) 2004 Waldo Bastian <bastian@kde.org> + * + * 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"), "bastian@kde.org"); + aboutData.addAuthor("Raffaele Sandrini", I18N_NOOP("Previous Maintainer"), "sandrini@kde.org"); + aboutData.addAuthor("Matthias Elter", I18N_NOOP("Original Author"), "elter@kde.org"); + + 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>&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>&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 <elter@kde.org> + * Lubos Lunak <l.lunak@email.cz> + * + * 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 <elter@kde.org> + * Lubos Lunak <l.lunak@email.cz> + * + * 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 <elter@kde.org> + * Copyright (C) 2001-2002 Raffaele Sandrini <sandrini@kde.org) + * + * 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 <elter@kde.org> + * + * 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>&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>&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 <elter@kde.org> + * Copyright (C) 2001-2002 Raffaele Sandrini <sandrini@kde.org) + * + * 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"), "bastian@kde.org"); + aboutData.addAuthor("Raffaele Sandrini", I18N_NOOP("Previous Maintainer"), "sandrini@kde.org"); + aboutData.addAuthor("Matthias Elter", I18N_NOOP("Original Author"), "elter@kde.org"); + + 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 <bastian@kde.org> + * + * 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 <bastian@kde.org> + * + * 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 <bastian@kde.org> + * + * 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 <bastian@kde.org> + * + * 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 Binary files differnew file mode 100644 index 000000000..7d4b6f365 --- /dev/null +++ b/kmenuedit/pixmaps/cr22-action-filesave_and_close.png diff --git a/kmenuedit/pixmaps/cr22-action-menu_new.png b/kmenuedit/pixmaps/cr22-action-menu_new.png Binary files differnew file mode 100644 index 000000000..56613f41c --- /dev/null +++ b/kmenuedit/pixmaps/cr22-action-menu_new.png diff --git a/kmenuedit/pixmaps/cr22-action-menu_new_sep.png b/kmenuedit/pixmaps/cr22-action-menu_new_sep.png Binary files differnew file mode 100644 index 000000000..bff0ce6d0 --- /dev/null +++ b/kmenuedit/pixmaps/cr22-action-menu_new_sep.png diff --git a/kmenuedit/pixmaps/cr32-action-menu_new.png b/kmenuedit/pixmaps/cr32-action-menu_new.png Binary files differnew file mode 100644 index 000000000..d1e532be1 --- /dev/null +++ b/kmenuedit/pixmaps/cr32-action-menu_new.png diff --git a/kmenuedit/pixmaps/cr32-action-menu_new_sep.png b/kmenuedit/pixmaps/cr32-action-menu_new_sep.png Binary files differnew file mode 100644 index 000000000..ff02e1d2b --- /dev/null +++ b/kmenuedit/pixmaps/cr32-action-menu_new_sep.png diff --git a/kmenuedit/pixmaps/lo16-action-menu_new.png b/kmenuedit/pixmaps/lo16-action-menu_new.png Binary files differnew file mode 100644 index 000000000..b121605f7 --- /dev/null +++ b/kmenuedit/pixmaps/lo16-action-menu_new.png 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 <elter@kde.org> + * Copyright (C) 2001-2002 Raffaele Sandrini <sandrini@kde.org) + * Copyright (C) 2003 Waldo Bastian <bastian@kde.org> + * + * 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 <elter@kde.org> + * Copyright (C) 2001-2002 Raffaele Sandrini <sandrini@kde.org> + * Copyright (C) 2003 Waldo Bastian <bastian@kde.org> + * + * 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 |