Commit 78277a8c authored by Robert Knight's avatar Robert Knight
Browse files

* Change behavior of 'New Profile' button to copying the selected

(or default) profile rather than creating a new profile which inherits from it.
Profiles created previously should not be affected.  This should remove the
possible confusion arising from the previously used inheritance behavior
* Support editing and deleting of multiple profiles at once by selecting them in
the Manage Profiles dialog and clicking 'Edit Profile'/'Delete Profile'.  String
changes deferred to KDE 4.2
* Fix memory leak and selection loss in Manage Profiles dialog when profiles
are added/removed/changed

BUG: 162461

Squashed commit of the following:

commit c6c482f76c840dfa34c36c2eee32f5e2aed499e7
Author: Robert Knight <robertknight@gmail.com>
Date:   Thu Jun 26 00:45:20 2008 +0100

    Avoid the selection being cleared when profiles are added/removed/changed in ManageProfilesDialog.  Finish addition/update/removal of items when profileAdded()/profileRemoved()/profileChanged() signals are emitted by the SessionManager.

commit 0125653d41738768d074bcd3c6445dd98eefa619
Author: Robert Knight <robertknight@gmail.com>
Date:   Wed Jun 25 11:21:44 2008 +0100

    Set model in ManageProfilesDialog setup rather than ManageProfilesDialog::updateTableModel()

commit 32704ba0bf076ec3c305515e350c016a7bdaf2f3
Author: Robert Knight <robertknight@gmail.com>
Date:   Sun Jun 22 19:42:33 2008 +0100

    Avoid creating a new QStandardItemModel on every profile table update.  Refactor creation/update of items for profiles.

commit dcd9eb9e91d64525359561cca2f8f6f54a35a2de
Author: Robert Knight <robertknight@gmail.com>
Date:   Fri Jun 13 19:52:33 2008 +0100

    * Disable previewing of a property when editing multiple profiles
    where the profiles have conflicting original values for the property
    * Change EditProfileDialog::_profileKey to EditProfileDialog::_profile

commit 020ef2d5f090a1cc2b1814c4d429051cb3cf98bc
Author: Robert Knight <robertknight@gmail.com>
Date:   Fri Jun 13 19:28:31 2008 +0100

    Add stub functions for SessionManagerTest

commit 655b0e6fa8e8891874ca5aa7bee6f81f3d76d065
Author: Robert Knight <robertknight@gmail.com>
Date:   Fri Jun 13 19:27:03 2008 +0100

    Include KDE 3 profiles when displaying available profiles
    via "konsole --list-profiles"

    Re-use SessionManager::availableProfilePaths() to get the list
    of all profiles to load in SessionManager::loadAllProfiles()

commit 93c02e6312c8a38d89a137c7260183dcf86946fa
Author: Robert Knight <robertknight@gmail.com>
Date:   Wed Jun 11 04:02:26 2008 +0100

    Export SessionManager class for testing.

commit a35eaba16cad3a3301a4636d032d789df669df9e
Author: Robert Knight <robertknight@gmail.com>
Date:   Sat May 31 22:34:33 2008 +0100

    Add SessionManager test stub

commit 347dadcd1c50157eb264bbb17423003a658bd8f5
Author: Robert Knight <robertknight@gmail.com>
Date:   Sat May 31 22:17:43 2008 +0100

    Display profile names in 'Name' field when editing multiple profiles.

commit 4d8e895fd9c95dbd954e224edcbcd68c24429bad
Author: Robert Knight <robertknight@gmail.com>
Date:   Sat May 31 22:06:49 2008 +0100

    Add test for Profile::asGroup()

commit 527535f7786d5c7744436012dbbd889348cec6a1
Author: Robert Knight <robertknight@gmail.com>
Date:   Sat May 31 21:58:15 2008 +0100

    Limit length of dialog caption when editing many profiles.

commit eb6e199e6e130b7b33ff97f124162dbb2107bba1
Author: Robert Knight <robertknight@gmail.com>
Date:   Sat May 31 21:50:53 2008 +0100

    Re-insert accidentally removed if() condition to check that deleted profile is not the default.

commit e11d4d7dac1ae1df4c401a04e8497521f240d8f0
Author: Robert Knight <robertknight@gmail.com>
Date:   Sat May 31 20:32:01 2008 +0100

    ManageProfilesDialog formatting tidy-up.

commit 69dd69a9aa8df9dfa4f1473e53b99e17898c8549
Author: Robert Knight <robertknight@gmail.com>
Date:   Sat May 31 18:57:40 2008 +0100

    When updating a profile group via SessionManager::changeProfile(), update the group's properties before calling changeProfile() on the group's members.  Otherwise the profile group's property values will be out of sync with those of the member profiles.

commit d74de728f0c19ca9786f2429fbd819938eaabe3e
Author: Robert Knight <robertknight@gmail.com>
Date:   Sat May 31 18:56:04 2008 +0100

    Add updated Profile.h missing from previous commit.

commit 67f6ec24f6871a931d3b80cc11313a1c0c8fd240
Author: Robert Knight <robertknight@gmail.com>
Date:   Sat May 31 18:54:06 2008 +0100

    Make profile groups hidden by default.

commit f4653c6953b356107ab347f173e8d04296abcbac
Author: Robert Knight <robertknight@gmail.com>
Date:   Sat May 31 18:41:49 2008 +0100

    When modifying a profile group with SessionManager::changeProfile(), iterate through the group's profiles and call changeProfile() on each of them.  Otherwise the profiles in the group are not saved to disk when they are updated.

commit cc3c14e3c2d43f75591396b154e6b6f4b430d9a9
Author: Robert Knight <robertknight@gmail.com>
Date:   Sat May 31 18:35:24 2008 +0100

    Add Profile::asGroup() convenience methods which return the profile dynamically cast to a Profile group.

commit e2db97ee0bcb4bb3f4a8d7356846c37a04708c88
Author: Robert Knight <robertknight@gmail.com>
Date:   Sat May 31 03:43:20 2008 +0100

    Make profile group hidden, prevents creation of nameless '.profile' file on disk.

commit 872113584bce97ae739a7ba46d02ee2d63f2990d
Author: Robert Knight <robertknight@gmail.com>
Date:   Sat May 31 03:22:45 2008 +0100

    Clear corresponding shortcuts when deleting a profile.

commit 0c9353ffde887ac408d87c258587d84cfa2a4d6f
Author: Robert Knight <robertknight@gmail.com>
Date:   Sat May 31 03:08:47 2008 +0100

    KDE4ProfileReader - Check that profile exists before trying to load it with KConfig

commit af7caa25931684da0d8f0f495d5acab8e6de1b41
Author: Robert Knight <robertknight@gmail.com>
Date:   Sat May 31 02:57:50 2008 +0100

    Fix enabling/disabling of buttons in ManageProfilesDialog when number of selected rows changes.

commit d98184d6684cd9a58a702ab2b23bfa0105f25a0f
Author: Robert Knight <robertknight@gmail.com>
Date:   Sat May 31 02:44:51 2008 +0100

    Add a test for Profile::clone()

commit b536355cde13523f12ed927b9a5e64b63954f21c
Author: Robert Knight <robertknight@gmail.com>
Date:   Fri May 30 02:58:03 2008 +0100

    Implement testProfile(), testProfileGroup().  Export Profile class for use in tests.

commit 8458de3fffe3755cd5ba9ebdda5795a422cfcb5f
Author: Robert Knight <robertknight@gmail.com>
Date:   Fri May 30 02:01:42 2008 +0100

    Tidy up Profile docs.

commit 6ed40ae0adc6128057b7d1c71c4bbb1259768102
Author: Robert Knight <robertknight@gmail.com>
Date:   Fri May 30 01:53:02 2008 +0100

    Push ProfileGroup::isPropertyValueUnique() up to Profile::canInheritProperty()

commit b992e0268c1708814a5843b13e321b9b4a86c05a
Author: Robert Knight <robertknight@gmail.com>
Date:   Fri May 30 00:30:45 2008 +0100

    Add Profile test stub.

commit 7c7b63e50a2bb7bcd7a26d895e77860b6a5effde
Author: Robert Knight <robertknight@gmail.com>
Date:   Thu May 29 06:12:52 2008 +0100

    Document ProfileGroup

commit 838a628dd395bdaf6ae04fb027b1efcccbbe9f32
Author: Robert Knight <robertknight@gmail.com>
Date:   Thu May 29 01:26:48 2008 +0100

    Remove unused and incorrect Profile::terminal()

commit 87e1ad1174ffa8fd1bafe37b4d52c9782708c9ea
Author: Robert Knight <robertknight@gmail.com>
Date:   Thu May 29 01:23:52 2008 +0100

    Remove unused kDebug() statements.

commit 56c78128f083ef846308a09038251786677dd087
Author: Robert Knight <robertknight@gmail.com>
Date:   Thu May 29 01:22:41 2008 +0100

    Profile documentation corrections.

commit 965803c401692720c57a65bf1534f774e1b99365
Author: Robert Knight <robertknight@gmail.com>
Date:   Thu May 29 01:10:23 2008 +0100

    Document SessionManager::fallbackProfile()

commit a222673b15f6e8c8bb53e95046ee786369deb127
Author: Robert Knight <robertknight@gmail.com>
Date:   Mon May 26 20:49:12 2008 +0100

    Disable 'New Profile' and 'Set as Default' buttons when multiple profiles are selected.  Remove incorrect comment about QAbstractItemView signals.  selectionChanged() is a slot not a signal.

commit e5e9db39487595d92fe2e7d3fa16cca731c417ec
Author: Robert Knight <robertknight@gmail.com>
Date:   Mon May 26 20:19:18 2008 +0100

    Change the way new profiles are created.  Instead of creating an almost-empty profile which inherits from the selected profile, create a new profile which inherits from the fallback profile and then clone the properties of the currently selected profile into the new profile.

commit f56cc534b2ebd691c6b3449e982c74c7e564a35a
Author: Robert Knight <robertknight@gmail.com>
Date:   Mon May 26 20:18:14 2008 +0100

    Expose the hard-coded fallback profile in the SessionManager

commit b00327479d222be60484bd781a131598f2c2482f
Author: Robert Knight <robertknight@gmail.com>
Date:   Mon May 26 20:17:35 2008 +0100

    Add method Profile::clone() to copy an existing profile, optionally copying all properties or just those differing from the current values.

commit 951620449c3a621d38f6607c0ea44e2a1a0ac087
Author: Robert Knight <robertknight@gmail.com>
Date:   Mon May 26 19:24:49 2008 +0100

    Treat a ProfileGroup with only one profile the same as a standard Profile in EditProfileDialog.

commit ecdf120b67a6b14a871cd808654fe6653bb56aeb
Author: Robert Knight <robertknight@gmail.com>
Date:   Mon May 26 19:24:08 2008 +0100

    Make ProfileGroup instances with only one profile behave the same as a standard Profile with respect to 'unique' properties such as Name and Path.

commit f72040f310946f92eff318c52da706fce00fa17e
Author: Robert Knight <robertknight@gmail.com>
Date:   Mon May 26 19:14:27 2008 +0100

    Fix double-deletion of profile group.

commit 1523239e9cbde518eb89fc70e6711463a7f7bb2a
Author: Robert Knight <robertknight@gmail.com>
Date:   Mon May 26 18:57:17 2008 +0100

    Only look at indexes in column 0 when retrieving the selected profiles.

commit 5706a8ee16bc1b87777563023e4ddf5253815c48
Author: Robert Knight <robertknight@gmail.com>
Date:   Mon May 26 18:52:47 2008 +0100

    Move to next property when skipping a 'unique' property.

commit 803eef6767620dd7a6e9b87cc97f1c110dfa66a6
Author: Robert Knight <robertknight@gmail.com>
Date:   Mon May 26 18:49:20 2008 +0100

    Modify Edit/Delete actions in ManageProfilesDialog to allow editing/deleting of multiple profiles.

commit eec6947620d51b2a41ed74a03e1a7b695d0ee811
Author: Robert Knight <robertknight@gmail.com>
Date:   Mon May 26 18:48:36 2008 +0100

    Modify EditProfileDialog to show a different caption depending on whether the passed in profile is a single profile or a group of profiles.  Disable the profile name edit and label when editing a group of profiles.

commit 13ee83b2bb888da4009bee0cb25926fc02f34701
Author: Robert Knight <robertknight@gmail.com>
Date:   Mon May 26 18:47:22 2008 +0100

    Add ProfileGroup class which provides a composite allowing multiple profiles to be treated as one.

commit 1310a4ee66e6ba9eb91f1a0dfd79c5415c3a9f4f
Author: Robert Knight <robertknight@gmail.com>
Date:   Sun May 25 00:07:47 2008 +0100

    Undo string changes in Manage Profiles dialog so that this branch can be merged before 4.1

commit edc46b4085d61d1cc14a47c429aac5fc9f52a59f
Author: Robert Knight <robertknight@gmail.com>
Date:   Fri May 23 18:34:09 2008 +0100

    Change 'New Profile' button to 'Copy Profile' to clarify how it works.  Change 'Edit Profile' to 'Edit' and 'Delete Profile' to 'Delete' to clarify that they can work on more than one profile.

svn path=/trunk/KDE/kdebase/apps/konsole/; revision=824515
parent a9b8d756
......@@ -100,7 +100,7 @@ void EditProfileDialog::save()
if ( _tempProfile->isEmpty() )
return;
SessionManager::instance()->changeProfile(_profileKey,_tempProfile->setProperties());
SessionManager::instance()->changeProfile(_profile,_tempProfile->setProperties());
// ensure that these settings are not undone by a call
// to unpreview()
......@@ -122,18 +122,48 @@ void EditProfileDialog::accept()
unpreviewAll();
KDialog::accept();
}
void EditProfileDialog::updateCaption(const QString& profileName)
QString EditProfileDialog::groupProfileNames(const ProfileGroup::Ptr group, int maxLength)
{
setCaption( i18n("Edit Profile \"%1\"",profileName) );
QString caption;
int count = group->profiles().count();
for (int i=0;i < count;i++)
{
caption += group->profiles()[i]->name();
if (i < (count-1))
{
caption += ',';
// limit caption length to prevent very long window titles
if (maxLength > 0 && caption.length() > maxLength)
{
caption += "...";
break;
}
}
}
return caption;
}
void EditProfileDialog::updateCaption(const Profile::Ptr profile)
{
const int MAX_GROUP_CAPTION_LENGTH = 25;
ProfileGroup::Ptr group = profile->asGroup();
if (group && group->profiles().count() > 1)
{
QString caption = groupProfileNames(group,MAX_GROUP_CAPTION_LENGTH);
setCaption( i18n("Edit Profile \"%1\"",caption) );
// STRINGFREEZE - Change caption for groups after KDE 4.1 is released
// setCaption( i18n("Editing %1 profiles",group->profiles().count()) )
}
else
setCaption( i18n("Edit Profile \"%1\"",profile->name()) );
}
void EditProfileDialog::setProfile(Profile::Ptr profile)
{
_profileKey = profile;
_profile = profile;
Q_ASSERT( profile );
// update caption
updateCaption(profile->name());
updateCaption(profile);
// mark each page of the dialog as out of date
// and force an update of the currently visible page
......@@ -149,7 +179,7 @@ void EditProfileDialog::setProfile(Profile::Ptr profile)
}
const Profile::Ptr EditProfileDialog::lookupProfile() const
{
return _profileKey;
return _profile;
}
void EditProfileDialog::preparePage(int page)
{
......@@ -191,9 +221,18 @@ void EditProfileDialog::selectProfileName()
}
void EditProfileDialog::setupGeneralPage(const Profile::Ptr info)
{
// basic profile options
_ui->profileNameEdit->setText( info->name() );
{
ProfileGroup::Ptr group = info->asGroup();
if (!group || group->profiles().count() < 2)
_ui->profileNameEdit->setText( info->name() );
else
{
_ui->profileNameEdit->setText( groupProfileNames(group,-1) );
_ui->profileNameLabel->setEnabled(false);
_ui->profileNameEdit->setEnabled(false);
}
}
ShellCommand command( info->command() , info->arguments() );
_ui->commandEdit->setText( command.fullCommand() );
......@@ -352,7 +391,7 @@ void EditProfileDialog::selectIcon()
void EditProfileDialog::profileNameChanged(const QString& text)
{
_tempProfile->setProperty(Profile::Name,text);
updateCaption(_tempProfile->name());
updateCaption(_tempProfile);
}
void EditProfileDialog::startInSameDir(bool sameDir)
{
......@@ -576,7 +615,7 @@ void EditProfileDialog::unpreviewAll()
// undo any preview changes
if ( !map.isEmpty() )
SessionManager::instance()->changeProfile(_profileKey,map,false);
SessionManager::instance()->changeProfile(_profile,map,false);
}
void EditProfileDialog::unpreview(int property)
{
......@@ -587,7 +626,7 @@ void EditProfileDialog::unpreview(int property)
QHash<Profile::Property,QVariant> map;
map.insert((Profile::Property)property,_previewedProperties[property]);
SessionManager::instance()->changeProfile(_profileKey,map,false);
SessionManager::instance()->changeProfile(_profile,map,false);
_previewedProperties.remove(property);
}
......@@ -618,11 +657,22 @@ void EditProfileDialog::preview(int property , const QVariant& value)
const Profile::Ptr original = lookupProfile();
if (!_previewedProperties.contains(property))
// skip previews for profile groups if the profiles in the group
// have conflicting original values for the property
//
// TODO - Save the original values for each profile and use to unpreview properties
ProfileGroup::Ptr group = original->asGroup();
if (group && group->profiles().count() > 1 &&
original->property<QVariant>((Profile::Property)property).isNull())
return;
if (!_previewedProperties.contains(property))
{
_previewedProperties.insert(property , original->property<QVariant>((Profile::Property)property) );
}
// temporary change to color scheme
SessionManager::instance()->changeProfile( _profileKey , map , false);
SessionManager::instance()->changeProfile( _profile , map , false);
}
void EditProfileDialog::previewColorScheme(const QModelIndex& index)
{
......
......@@ -197,9 +197,11 @@ private:
void unpreviewAll();
void enableIfNonEmptySelection(QWidget* widget,QItemSelectionModel* selectionModel);
void updateCaption(const QString& profileName);
void updateCaption(const Profile::Ptr profile);
void updateTransparencyWarning();
static QString groupProfileNames(const ProfileGroup::Ptr group, int maxLength = -1);
struct RadioOption
{
QAbstractButton* button;
......@@ -219,7 +221,7 @@ private:
Ui::EditProfileDialog* _ui;
Profile::Ptr _tempProfile;
Profile::Ptr _profileKey;
Profile::Ptr _profile;
// keeps track of pages which need to be updated to match the current
// profile. all elements in this vector are set to true when the
......
......@@ -43,6 +43,7 @@ using namespace Konsole;
ManageProfilesDialog::ManageProfilesDialog(QWidget* parent)
: KDialog(parent)
, _sessionModel(new QStandardItemModel(this))
{
setCaption(i18n("Manage Profiles"));
setButtons( KDialog::Close );
......@@ -57,22 +58,22 @@ ManageProfilesDialog::ManageProfilesDialog(QWidget* parent)
_ui->sessionTable->setEditTriggers(_ui->sessionTable->editTriggers() | QAbstractItemView::SelectedClicked);
// update table and listen for changes to the session types
updateTableModel();
connect( SessionManager::instance() , SIGNAL(profileAdded(Profile::Ptr)) , this,
SLOT(updateTableModel()) );
SLOT(addItems(Profile::Ptr)) );
connect( SessionManager::instance() , SIGNAL(profileRemoved(Profile::Ptr)) , this,
SLOT(updateTableModel()) );
SLOT(removeItems(Profile::Ptr)) );
connect( SessionManager::instance() , SIGNAL(profileChanged(Profile::Ptr)) , this,
SLOT(updateTableModel()) );
SLOT(updateItems(Profile::Ptr)) );
connect( SessionManager::instance() ,
SIGNAL(favoriteStatusChanged(Profile::Ptr,bool)) , this ,
SLOT(updateFavoriteStatus(Profile::Ptr,bool)) );
populateTable();
// resize the session table to the full width of the table
_ui->sessionTable->horizontalHeader()->setHighlightSections(false);
_ui->sessionTable->resizeColumnsToContents();
// allow a larger width for the shortcut column to account for the
// allow a larger width for the shortcut column to account for the
// increased with needed by the shortcut editor compared with just
// displaying the text of the shortcut
_ui->sessionTable->setColumnWidth(ShortcutColumn,
......@@ -83,7 +84,6 @@ ManageProfilesDialog::ManageProfilesDialog(QWidget* parent)
connect( _ui->editSessionButton , SIGNAL(clicked()) , this , SLOT(editSelected()) );
connect( _ui->deleteSessionButton , SIGNAL(clicked()) , this , SLOT(deleteSelected()) );
connect( _ui->setAsDefaultButton , SIGNAL(clicked()) , this , SLOT(setSelectedAsDefault()) );
}
void ManageProfilesDialog::showEvent(QShowEvent*)
......@@ -116,86 +116,108 @@ void ManageProfilesDialog::itemDataChanged(QStandardItem* item)
if ( item->column() == ShortcutColumn )
{
QKeySequence sequence = QKeySequence::fromString(item->text());
kDebug() << "New key sequence: " << item->text();
SessionManager::instance()->setShortcut(item->data(ShortcutRole).value<Profile::Ptr>(),
sequence);
}
}
void ManageProfilesDialog::updateTableModel()
int ManageProfilesDialog::rowForProfile(const Profile::Ptr info) const
{
for (int i=0;i<_sessionModel->rowCount();i++)
{
if (_sessionModel->item(i,ProfileNameColumn)->data(ProfileKeyRole)
.value<Profile::Ptr>() == info)
{
return i;
}
}
return -1;
}
void ManageProfilesDialog::removeItems(const Profile::Ptr info)
{
int row = rowForProfile(info);
if (row < 0)
return;
_sessionModel->removeRow(row);
}
void ManageProfilesDialog::updateItems(const Profile::Ptr info)
{
int row = rowForProfile(info);
if (row < 0)
return;
QList<QStandardItem*> items;
items << _sessionModel->item(row,ProfileNameColumn);
items << _sessionModel->item(row,FavoriteStatusColumn);
items << _sessionModel->item(row,ShortcutColumn);
updateItemsForProfile(info,items);
}
void ManageProfilesDialog::updateItemsForProfile(const Profile::Ptr info, QList<QStandardItem*>& items) const
{
// Profile Name
items[ProfileNameColumn]->setText(info->name());
if ( !info->icon().isEmpty() )
items[ProfileNameColumn]->setIcon( KIcon(info->icon()) );
items[ProfileNameColumn]->setData(QVariant::fromValue(info),ProfileKeyRole);
// Favorite Status
const bool isFavorite = SessionManager::instance()->findFavorites().contains(info);
if ( isFavorite )
items[FavoriteStatusColumn]->setData(KIcon("favorites"),Qt::DecorationRole);
else
items[FavoriteStatusColumn]->setData(KIcon(),Qt::DecorationRole);
items[FavoriteStatusColumn]->setData(QVariant::fromValue(info),ProfileKeyRole);
// Shortcut
QString shortcut = SessionManager::instance()->shortcut(info).
toString();
items[ShortcutColumn]->setText(shortcut);
items[ShortcutColumn]->setData(QVariant::fromValue(info),ShortcutRole);
}
void ManageProfilesDialog::addItems(const Profile::Ptr profile)
{
if (profile->isHidden())
return;
QList<QStandardItem*> items;
for (int i=0;i<3;i++)
items << new QStandardItem;
updateItemsForProfile(profile,items);
_sessionModel->appendRow(items);
}
void ManageProfilesDialog::populateTable()
{
Q_ASSERT(!_ui->sessionTable->model());
_ui->sessionTable->setModel(_sessionModel);
// ensure profiles list is complete
// this may be EXPENSIVE, but will only be done the first time
// this may be expensive, but will only be done the first time
// that the dialog is shown.
SessionManager::instance()->loadAllProfiles();
// setup session table
_sessionModel = new QStandardItemModel(this);
_sessionModel->setHorizontalHeaderLabels( QStringList() << i18n("Name")
<< i18n("Show in Menu")
<< i18n("Shortcut") );
QListIterator<Profile::Ptr> profileIter( SessionManager::instance()->loadedProfiles() );
while ( profileIter.hasNext() )
foreach(const Profile::Ptr info,SessionManager::instance()->loadedProfiles())
{
Profile::Ptr info = profileIter.next();
if ( info->isHidden() )
continue;
QList<QStandardItem*> itemList;
QStandardItem* item = new QStandardItem( info->name() );
if ( !info->icon().isEmpty() )
item->setIcon( KIcon(info->icon()) );
item->setData(QVariant::fromValue(info),ProfileKeyRole);
const bool isFavorite = SessionManager::instance()->findFavorites().contains(info);
// favorite column
QStandardItem* favoriteItem = new QStandardItem();
if ( isFavorite )
favoriteItem->setData(KIcon("favorites"),Qt::DecorationRole);
else
favoriteItem->setData(KIcon(),Qt::DecorationRole);
favoriteItem->setData(QVariant::fromValue(info),ProfileKeyRole);
// shortcut column
QStandardItem* shortcutItem = new QStandardItem();
QString shortcut = SessionManager::instance()->shortcut(info).
toString();
shortcutItem->setText(shortcut);
shortcutItem->setData(QVariant::fromValue(info),ShortcutRole);
itemList << item << favoriteItem << shortcutItem;
_sessionModel->appendRow(itemList);
addItems(info);
}
updateDefaultItem();
connect( _sessionModel , SIGNAL(itemChanged(QStandardItem*)) , this ,
SLOT(itemDataChanged(QStandardItem*)) );
_ui->sessionTable->setModel(_sessionModel);
// listen for changes in the table selection and update the state of the form's buttons
// accordingly.
//
// it appears that the selection model is changed when the model itself is replaced,
// so the signals need to be reconnected each time the model is updated.
//
// the view ( _ui->sessionTable ) has a selectionChanged() signal which it would be
// preferable to connect to instead of the one belonging to the view's current
// selection model, but QAbstractItemView::selectionChanged() is protected
//
connect( _ui->sessionTable->selectionModel() ,
SIGNAL(selectionChanged(const QItemSelection&,const QItemSelection&)) , this ,
SLOT(tableSelectionChanged(const QItemSelection&)) );
tableSelectionChanged( _ui->sessionTable->selectionModel()->selection() );
}
void ManageProfilesDialog::updateDefaultItem()
{
......@@ -220,31 +242,29 @@ void ManageProfilesDialog::updateDefaultItem()
}
}
}
void ManageProfilesDialog::tableSelectionChanged(const QItemSelection& selection)
void ManageProfilesDialog::tableSelectionChanged(const QItemSelection&)
{
bool enable = !selection.indexes().isEmpty();
const int selectedRows = _ui->sessionTable->selectionModel()->selectedRows().count();
const SessionManager* manager = SessionManager::instance();
const bool isNotDefault = enable && selectedKey() != manager->defaultProfile();
const bool isNotDefault = (selectedRows > 0) && currentProfile() != manager->defaultProfile();
_ui->editSessionButton->setEnabled(enable);
_ui->newSessionButton->setEnabled(selectedRows < 2);
_ui->editSessionButton->setEnabled(selectedRows > 0);
// do not allow the default session type to be removed
_ui->deleteSessionButton->setEnabled(isNotDefault);
_ui->setAsDefaultButton->setEnabled(isNotDefault);
_ui->setAsDefaultButton->setEnabled(isNotDefault && (selectedRows < 2));
}
void ManageProfilesDialog::deleteSelected()
{
// TODO - Possibly add a warning here if the selected profile is the parent of
// one or more other profiles since deleting a profile will change the settings
// of the profiles which inherit from it.
Q_ASSERT( selectedKey() );
Q_ASSERT( selectedKey() != SessionManager::instance()->defaultProfile() );
SessionManager::instance()->deleteProfile(selectedKey());
foreach(Profile::Ptr profile, selectedProfiles())
{
if (profile != SessionManager::instance()->defaultProfile())
SessionManager::instance()->deleteProfile(profile);
}
}
void ManageProfilesDialog::setSelectedAsDefault()
{
SessionManager::instance()->setDefaultProfile(selectedKey());
SessionManager::instance()->setDefaultProfile(currentProfile());
// do not allow the new default session type to be removed
_ui->deleteSessionButton->setEnabled(false);
_ui->setAsDefaultButton->setEnabled(false);
......@@ -256,19 +276,20 @@ void ManageProfilesDialog::newType()
{
EditProfileDialog dialog(this);
// setup a temporary profile, inheriting from the selected profile
// setup a temporary profile which is a clone of the selected profile
// or the default if no profile is selected
Profile::Ptr parentProfile;
Profile::Ptr selectedProfile = selectedKey();
Profile::Ptr sourceProfile;
Profile::Ptr selectedProfile = currentProfile();
if ( !selectedProfile )
parentProfile = SessionManager::instance()->defaultProfile();
sourceProfile = SessionManager::instance()->defaultProfile();
else
parentProfile = selectedProfile;
sourceProfile = selectedProfile;
Q_ASSERT( parentProfile );
Q_ASSERT( sourceProfile );
Profile::Ptr newProfile = Profile::Ptr(new Profile(parentProfile));
Profile::Ptr newProfile = Profile::Ptr(new Profile(SessionManager::instance()->fallbackProfile()));
newProfile->clone(sourceProfile,true);
newProfile->setProperty(Profile::Name,i18n("New Profile"));
dialog.setProfile(newProfile);
......@@ -282,23 +303,40 @@ void ManageProfilesDialog::newType()
}
void ManageProfilesDialog::editSelected()
{
Q_ASSERT( selectedKey() );
EditProfileDialog dialog(this);
dialog.setProfile(selectedKey());
// the dialog will delete the profile group when it is destroyed
ProfileGroup* group = new ProfileGroup;
foreach(Profile::Ptr profile,selectedProfiles())
group->addProfile(profile);
group->updateValues();
dialog.setProfile(Profile::Ptr(group));
dialog.exec();
}
Profile::Ptr ManageProfilesDialog::selectedKey() const
QList<Profile::Ptr> ManageProfilesDialog::selectedProfiles() const
{
QList<Profile::Ptr> list;
QItemSelectionModel* selection = _ui->sessionTable->selectionModel();
if (!selection)
return list;
foreach(const QModelIndex& index, selection->selectedIndexes())
{
if (index.column() == ProfileNameColumn)
list << index.data(ProfileKeyRole).value<Profile::Ptr>();
}
return list;
}
Profile::Ptr ManageProfilesDialog::currentProfile() const
{
QItemSelectionModel* selection = _ui->sessionTable->selectionModel();
if ( !selection || selection->selectedRows().count() != 1 )
return Profile::Ptr();
// TODO There has to be an easier way of getting the data
// associated with the currently selected item
return selection->
selectedIndexes().first().data( Qt::UserRole + 1 ).value<Profile::Ptr>();
selectedIndexes().first().data(ProfileKeyRole).value<Profile::Ptr>();
}
void ManageProfilesDialog::updateFavoriteStatus(Profile::Ptr profile, bool favorite)
{
......@@ -416,7 +454,4 @@ void ShortcutItemDelegate::paint(QPainter* painter, const QStyleOptionViewItem&
QStyledItemDelegate::paint(painter,option,index);
}
#include "ManageProfilesDialog.moc"
......@@ -86,21 +86,29 @@ private slots:
// selection changes
void tableSelectionChanged(const QItemSelection&);
// updates the profile table to be in sync with the
// session manager
void updateTableModel();
void updateFavoriteStatus(Profile::Ptr profile, bool favorite);
void addItems(const Profile::Ptr);
void updateItems(const Profile::Ptr);
void removeItems(const Profile::Ptr);
private:
Profile::Ptr selectedKey() const; // return the profile associated with the currently selected
// item in the profile table
Profile::Ptr currentProfile() const;
QList<Profile::Ptr> selectedProfiles() const;
// updates the font of the items to match
// their default / non-default profile status
void updateDefaultItem();
void updateItemsForProfile(const Profile::Ptr profile, QList<QStandardItem*>& items) const;
// updates the profile table to be in sync with the
// session manager
void populateTable();
int rowForProfile(const Profile::Ptr info) const;
void updateDefaultItem(); // updates the font of the items to match
// their default / non-default profile status
Ui::ManageProfilesDialog* _ui;
QStandardItemModel* _sessionModel;
static const int ProfileNameColumn = 0;
static const int FavoriteStatusColumn = 1;
static const int ShortcutColumn = 2;
static const int ProfileKeyRole = Qt::UserRole + 1;
......
......@@ -38,7 +38,7 @@
<bool>false</bool>
</property>
<property name="toolTip" >
<string>Edit the selected profile</string>
<string>Edit the selected profile(s)</string>
</property>
<property name="text" >
<string>Edit Profile...</string>
......@@ -51,7 +51,7 @@
<bool>false</bool>
</property>
<property name="toolTip" >
<string>Delete the selected profile</string>
<string>Delete the selected profile(s)</string>
</property>
<property name="text" >
<string>Delete Profile</string>
......@@ -93,7 +93,7 @@
<bool>true</bool>
</property>
<property name="selectionMode" >
<enum>QAbstractItemView::SingleSelection</enum>
<enum>QAbstractItemView::ExtendedSelection</enum>
</property>
<property name="selectionBehavior" >
<enum>QAbstractItemView::SelectRows</enum>
......
......@@ -141,6 +141,9 @@ FallbackProfile::FallbackProfile()
{
// Fallback settings
setProperty(Name,i18n("Shell"));
// magic path for the fallback profile which is not a valid
// non-directory file name
setProperty(Path,"FALLBACK/");
setProperty(Command,getenv("SHELL"));
setProperty(Icon,"utilities-terminal");
setProperty(Arguments,QStringList() << getenv("SHELL"));
......@@ -180,12 +183,34 @@ FallbackProfile::FallbackProfile()
// Fallback should not be shown in menus
setHidden(true);
}
Profile::Profile(Profile::Ptr parent)
: _parent(parent)
,_hidden(false)
{
}
void Profile::clone(Profile::Ptr profile, bool differentOnly)
{
const PropertyInfo* properties = DefaultPropertyNames;
while (properties->name != 0)
{
Property current = properties->property;
QVariant otherValue = profile->property<QVariant>(current);
switch (current)
{
case Name:
case Path:
break;
default:
if (!differentOnly ||
property<QVariant>(current) !=
otherValue)
{
setProperty(current,otherValue);
}
}