diff --git a/modules/gui/qt/Makefile.am b/modules/gui/qt/Makefile.am index add710cd4bab95d2c53ecb385578236e7aaef5b1..5412589ca8cd0b80f6a623e2c7c8f0fac8a4dff2 100644 --- a/modules/gui/qt/Makefile.am +++ b/modules/gui/qt/Makefile.am @@ -112,6 +112,10 @@ libqt_plugin_la_SOURCES = \ gui/qt/dialogs/sout/sout.cpp gui/qt/dialogs/sout/sout.hpp \ gui/qt/dialogs/sout/sout_widgets.cpp \ gui/qt/dialogs/sout/sout_widgets.hpp \ + gui/qt/dialogs/toolbar/controlbar_profile.hpp \ + gui/qt/dialogs/toolbar/controlbar_profile.cpp \ + gui/qt/dialogs/toolbar/controlbar_profile_model.cpp \ + gui/qt/dialogs/toolbar/controlbar_profile_model.hpp \ gui/qt/dialogs/vlm/vlm.cpp gui/qt/dialogs/vlm/vlm.hpp \ gui/qt/dialogs/playlists/playlists.cpp gui/qt/dialogs/playlists/playlists.hpp \ gui/qt/maininterface/compositor.hpp \ @@ -193,7 +197,8 @@ libqt_plugin_la_SOURCES = \ gui/qt/network/servicesdiscoverymodel.hpp \ gui/qt/player/input_models.cpp gui/qt/player/input_models.hpp \ gui/qt/player/player_controller.cpp gui/qt/player/player_controller.hpp gui/qt/player/player_controller_p.hpp \ - gui/qt/player/playercontrolbarmodel.cpp gui/qt/player/playercontrolbarmodel.hpp \ + gui/qt/player/player_controlbar_model.cpp gui/qt/player/player_controlbar_model.hpp \ + gui/qt/player/control_list_model.cpp gui/qt/player/control_list_model.hpp \ gui/qt/playlist/media.hpp \ gui/qt/playlist/playlist_common.cpp \ gui/qt/playlist/playlist_common.hpp \ @@ -315,6 +320,8 @@ nodist_libqt_plugin_la_SOURCES = \ gui/qt/dialogs/sout/profile_selector.moc.cpp \ gui/qt/dialogs/sout/sout.moc.cpp \ gui/qt/dialogs/sout/sout_widgets.moc.cpp \ + gui/qt/dialogs/toolbar/controlbar_profile.moc.cpp \ + gui/qt/dialogs/toolbar/controlbar_profile_model.moc.cpp \ gui/qt/dialogs/playlists/playlists.moc.cpp \ gui/qt/maininterface/compositor_dummy.moc.cpp \ gui/qt/maininterface/interface_window_handler.moc.cpp \ @@ -351,7 +358,8 @@ nodist_libqt_plugin_la_SOURCES = \ gui/qt/network/servicesdiscoverymodel.moc.cpp \ gui/qt/player/input_models.moc.cpp \ gui/qt/player/player_controller.moc.cpp \ - gui/qt/player/playercontrolbarmodel.moc.cpp \ + gui/qt/player/player_controlbar_model.moc.cpp \ + gui/qt/player/control_list_model.moc.cpp \ gui/qt/playlist/playlist_common.moc.cpp \ gui/qt/playlist/playlist_controller.moc.cpp \ gui/qt/playlist/playlist_item.moc.cpp \ diff --git a/modules/gui/qt/dialogs/toolbar/controlbar_profile.cpp b/modules/gui/qt/dialogs/toolbar/controlbar_profile.cpp new file mode 100644 index 0000000000000000000000000000000000000000..4c3e82fb3838eec7b89e670fae1e3bf2d907da94 --- /dev/null +++ b/modules/gui/qt/dialogs/toolbar/controlbar_profile.cpp @@ -0,0 +1,232 @@ +/***************************************************************************** + * Copyright (C) 2021 VLC authors and VideoLAN + * + * 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 "controlbar_profile.hpp" + +#include "player/control_list_model.hpp" +#include "player/player_controlbar_model.hpp" + + +decltype(ControlbarProfile::m_defaults) + ControlbarProfile::m_defaults = + { + { + { + "MainPlayer" + }, + { + { + { + ControlListModel::LANG_BUTTON, + ControlListModel::MENU_BUTTON + }, + { + ControlListModel::RANDOM_BUTTON, + ControlListModel::PREVIOUS_BUTTON, + ControlListModel::PLAY_BUTTON, + ControlListModel::NEXT_BUTTON, + ControlListModel::LOOP_BUTTON + }, + { + ControlListModel::VOLUME, + ControlListModel::FULLSCREEN_BUTTON + } + } + } + }, + { + { + "MiniPlayer" + }, + { + { + { + ControlListModel::ARTWORK_INFO + }, + { + ControlListModel::RANDOM_BUTTON, + ControlListModel::PREVIOUS_BUTTON, + ControlListModel::PLAY_BUTTON, + ControlListModel::NEXT_BUTTON, + ControlListModel::LOOP_BUTTON + }, + { + ControlListModel::VOLUME, + ControlListModel::PLAYER_SWITCH_BUTTON + } + } + } + } + }; + + +ControlbarProfile::ControlbarProfile(QObject *parent) : QObject(parent) +{ + injectDefaults(); +} + +PlayerControlbarModel *ControlbarProfile::newModel(const QString &identifier) +{ + if (identifier.isEmpty()) + return nullptr; + + if (m_models.contains(identifier)) + return nullptr; // can not allow the same identifier + + const auto model = new PlayerControlbarModel(this); + + connect(model, &PlayerControlbarModel::controlListChanged, this, &ControlbarProfile::generateLinearControlList); + + connect(model, &PlayerControlbarModel::dirtyChanged, this, [this](bool dirty) { + if (dirty) + ++m_dirty; + else + --m_dirty; + + emit dirtyChanged( this->dirty() ); + }); + + m_models.insert(identifier, model); + + return model; +} + +PlayerControlbarModel *ControlbarProfile::getModel(const QString &identifier) const +{ + if (m_models.contains(identifier)) + { + return m_models[identifier]; + } + else + { + return nullptr; + } +} + +void ControlbarProfile::setModelData(const QString &identifier, const std::array<QVector<int>, 3> &data) +{ + auto ptrModel = getModel(identifier); + + if (ptrModel) + { + ptrModel->loadModels(data); + } + else + { + ptrModel = newModel(identifier); + + if (!ptrModel) + return; + + ptrModel->loadModels(data); + } + + ptrModel->setDirty(true); +} + +std::array<QVector<int>, 3> ControlbarProfile::getModelData(const QString &identifier) const +{ + const auto ptrModel = getModel(identifier); + + if (!ptrModel) + return {}; + + return ptrModel->serializeModels(); +} + +void ControlbarProfile::deleteModel(const QString &identifier) +{ + if (m_models.contains(identifier)) + { + m_models[identifier]->deleteLater(); + m_models.remove(identifier); + } +} + +void ControlbarProfile::setName(const QString &name) +{ + if (name == m_name) + return; + + m_name = name; + + emit nameChanged(m_name); +} + +bool ControlbarProfile::dirty() const +{ + return (m_dirty > 0); +} + +QString ControlbarProfile::name() const +{ + return m_name; +} + +void ControlbarProfile::injectDefaults(bool resetDirty) +{ + injectModel(m_defaults); + + if (resetDirty) + this->resetDirty(); // defaults normally should not make the profile dirty +} + +void ControlbarProfile::injectModel(const QVector<ControlbarProfile::Configuration> &modelData) +{ + m_pauseControlListGeneration = true; + + for (const auto& i : modelData) + { + setModelData(i.identifier, i.data); + } + + m_pauseControlListGeneration = false; + + generateLinearControlList(); +} + +void ControlbarProfile::generateLinearControlList() +{ + if (m_pauseControlListGeneration) + return; + + // Don't bother if there is no receiver (connection): + if (receivers(SIGNAL(controlListChanged (const QVector<int>&) )) <= 0) + return; + + QVector<int> linearControls; + + for (const auto& i : m_models) + { + linearControls.append(i->serializeModels()[0] + i->serializeModels()[1] + i->serializeModels()[2]); + } + + emit controlListChanged(linearControls); +} + +void ControlbarProfile::resetDirty() +{ + if (dirty() == false) + return; + + for (auto it = m_models.constBegin(); it != m_models.constEnd(); ++it) + { + it.value()->setDirty(false); + } + + emit dirtyChanged(dirty()); +} diff --git a/modules/gui/qt/dialogs/toolbar/controlbar_profile.hpp b/modules/gui/qt/dialogs/toolbar/controlbar_profile.hpp new file mode 100644 index 0000000000000000000000000000000000000000..e9c486dd015bc61be7e3fa8d1b23576acf2eb35c --- /dev/null +++ b/modules/gui/qt/dialogs/toolbar/controlbar_profile.hpp @@ -0,0 +1,91 @@ +/***************************************************************************** + * Copyright (C) 2021 VLC authors and VideoLAN + * + * 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 ControlbarProfile_HPP +#define ControlbarProfile_HPP + +#include <QObject> +#include <QMap> +#include <QVector> +#include <array> + +class PlayerControlbarModel; + +class ControlbarProfile : public QObject +{ + Q_OBJECT + + Q_PROPERTY(bool dirty READ dirty RESET resetDirty NOTIFY dirtyChanged) + Q_PROPERTY(QString name READ name WRITE setName NOTIFY nameChanged) + + friend class ControlbarProfileModel; + +public: + explicit ControlbarProfile(QObject *parent = nullptr); + + PlayerControlbarModel* newModel(const QString& identifier); + Q_INVOKABLE PlayerControlbarModel* getModel(const QString& identifier) const; + + void setModelData(const QString& identifier, const std::array<QVector<int>, 3>& data); + std::array<QVector<int>, 3> getModelData(const QString& identifier) const; + + void deleteModel(const QString& identifier); + + Q_INVOKABLE void injectDefaults(bool resetDirty = true); + + bool dirty() const; + QString name() const; + +public slots: + void resetDirty(); + void setName(const QString& name); + +private: + // m_dirty indicates the count of PlayerControlbarModel + // residing in m_models which has the dirty property + // set true. + int m_dirty = 0; + + QString m_name {"N/A"}; + bool m_pauseControlListGeneration = false; + + // According to benchmarks, QMap performs better than + // QHash when item count is less than 32. + // Assuming model (player) count to stay below that, + // QMap is used here. + QMap<QString, PlayerControlbarModel *> m_models; + + struct Configuration { + QString identifier; + std::array<QVector<int>, 3> data; + }; + static const QVector<Configuration> m_defaults; + +private: + void injectModel(const QVector<Configuration>& modelData); + +private slots: + void generateLinearControlList(); + +signals: + void dirtyChanged(bool dirty); + void nameChanged(QString name); + + void controlListChanged(const QVector<int>& linearControlList); +}; + +#endif // ControlbarProfile_HPP diff --git a/modules/gui/qt/dialogs/toolbar/controlbar_profile_model.cpp b/modules/gui/qt/dialogs/toolbar/controlbar_profile_model.cpp new file mode 100644 index 0000000000000000000000000000000000000000..a052048bb053959d5cc2a00851f2aa63c4cb3855 --- /dev/null +++ b/modules/gui/qt/dialogs/toolbar/controlbar_profile_model.cpp @@ -0,0 +1,692 @@ +/***************************************************************************** + * Copyright (C) 2021 VLC authors and VideoLAN + * + * 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 "controlbar_profile_model.hpp" + +#include <QSettings> + +#include "qt.hpp" +#include "controlbar_profile.hpp" +#include "player/control_list_model.hpp" + +#define SETTINGS_KEY_SELECTEDPROFILE "SelectedProfile" +#define SETTINGS_ARRAYNAME_PROFILES "Profiles" +#define SETTINGS_KEY_NAME "Name" +#define SETTINGS_KEY_MODEL "Model" + +#define SETTINGS_CONTROL_SEPARATOR "," +#define SETTINGS_CONFIGURATION_SEPARATOR "|" +#define SETTINGS_PROFILE_SEPARATOR "$" + +decltype (ControlbarProfileModel::m_defaults) + ControlbarProfileModel::m_defaults = + { + { + { + "Minimalist Style" + }, + { + { + { + "MainPlayer" + }, + { + { + { + ControlListModel::PLAY_BUTTON, + ControlListModel::WIDGET_SPACER, + ControlListModel::PREVIOUS_BUTTON, + ControlListModel::STOP_BUTTON, + ControlListModel::NEXT_BUTTON, + ControlListModel::WIDGET_SPACER, + ControlListModel::RECORD_BUTTON, + ControlListModel::WIDGET_SPACER, + ControlListModel::TELETEXT_BUTTONS, + ControlListModel::WIDGET_SPACER, + ControlListModel::PLAYLIST_BUTTON, + ControlListModel::WIDGET_SPACER, + ControlListModel::VOLUME + }, + { + + }, + { + + } + } + } + }, + { + { + "MiniPlayer" + }, + { + { + { + ControlListModel::PREVIOUS_BUTTON, + ControlListModel::PLAY_BUTTON, + ControlListModel::STOP_BUTTON, + ControlListModel::NEXT_BUTTON + }, + { + + }, + { + + } + } + } + } + } + }, + { + { + "One-liner Style" + }, + { + { + { + "MainPlayer" + }, + { + { + { + ControlListModel::PLAY_BUTTON, + ControlListModel::WIDGET_SPACER, + ControlListModel::PREVIOUS_BUTTON, + ControlListModel::STOP_BUTTON, + ControlListModel::NEXT_BUTTON, + ControlListModel::WIDGET_SPACER, + ControlListModel::FULLSCREEN_BUTTON, + ControlListModel::PLAYLIST_BUTTON, + ControlListModel::EXTENDED_BUTTON, + ControlListModel::WIDGET_SPACER, + ControlListModel::WIDGET_SPACER, + ControlListModel::RECORD_BUTTON, + ControlListModel::SNAPSHOT_BUTTON, + ControlListModel::ATOB_BUTTON, + ControlListModel::FRAME_BUTTON + }, + { + ControlListModel::VOLUME + }, + { + + } + } + } + }, + { + { + "MiniPlayer" + }, + { + { + { + ControlListModel::RANDOM_BUTTON, + ControlListModel::PREVIOUS_BUTTON, + ControlListModel::PLAY_BUTTON, + ControlListModel::STOP_BUTTON, + ControlListModel::NEXT_BUTTON, + ControlListModel::LOOP_BUTTON + }, + { + + }, + { + + } + } + } + } + } + }, + { + { + "Simplest Style" + }, + { + { + { + "MainPlayer" + }, + { + { + { + ControlListModel::VOLUME + }, + { + ControlListModel::PLAY_BUTTON, + ControlListModel::NEXT_BUTTON, + ControlListModel::STOP_BUTTON + }, + { + ControlListModel::FULLSCREEN_BUTTON + } + } + } + }, + { + { + "MiniPlayer" + }, + { + { + { + ControlListModel::PREVIOUS_BUTTON, + ControlListModel::PLAY_BUTTON, + ControlListModel::NEXT_BUTTON + }, + { + + }, + { + + } + } + } + } + } + } + }; + + +ControlbarProfileModel::ControlbarProfileModel(intf_thread_t *p_intf, QObject *parent) + : QAbstractListModel(parent), + m_intf(p_intf) +{ + assert(m_intf); + + connect(this, &QAbstractListModel::rowsInserted, this, &ControlbarProfileModel::countChanged); + connect(this, &QAbstractListModel::rowsRemoved, this, &ControlbarProfileModel::countChanged); + connect(this, &QAbstractListModel::modelReset, this, &ControlbarProfileModel::countChanged); + + // To make the QML player controlbars update when model is Reset + connect(this, &QAbstractListModel::modelReset, this, &ControlbarProfileModel::selectedProfileChanged); + + // When all profiles are removed, insert defaults: + // Maybe add a dedicate button for this purpose and don't allow removing all profiles ? + connect(this, &ControlbarProfileModel::countChanged, this, [this] () { + if (rowCount() == 0) + insertDefaults(); + }); + + if (reload() == false) + { + // If initial reload fails, load the default profiles: + insertDefaults(); + } +} + +void ControlbarProfileModel::insertDefaults() +{ + // First, add a blank new profile: + // ControlbarProfile will inject the default configurations during its construction. + newProfile(tr("Default Profile")); + + // Add default profiles: + for (const auto& i : m_defaults) + { + const auto ptrNewProfile = newProfile(i.name); + if (!ptrNewProfile) + continue; + + ptrNewProfile->injectModel(i.modelData); + ptrNewProfile->resetDirty(); // default profiles should not be dirty initially + } + + setSelectedProfile(0); +} + +QString ControlbarProfileModel::generateUniqueName(const QString &name) +{ + const auto sameNameCount = std::count_if(m_profiles.begin(), + m_profiles.end(), + [name](const ControlbarProfile* i) { + return i->name() == name; + }); + + if (sameNameCount > 0) + return QString("%1 (%2)").arg(name).arg(sameNameCount + 1); + else + return name; +} + +int ControlbarProfileModel::rowCount(const QModelIndex &parent) const +{ + if (parent.isValid()) + return 0; + + return m_profiles.size(); +} + +QVariant ControlbarProfileModel::data(const QModelIndex &index, int role) const +{ + if (!index.isValid()) + return QVariant(); + + const auto ptrProfile = m_profiles.at(index.row()); + + if (!ptrProfile) + return QVariant(); + + switch (role) + { + case Qt::DisplayRole: + return ptrProfile->name(); + case MODEL_ROLE: + return QVariant::fromValue(ptrProfile); + } + + return QVariant(); +} + +QHash<int, QByteArray> ControlbarProfileModel::roleNames() const +{ + return { + { + Qt::DisplayRole, "name" + }, + { + MODEL_ROLE, "model" + } + }; +} + +bool ControlbarProfileModel::insertRows(int row, int count, const QModelIndex &parent) +{ + if (row < 0 || row > m_profiles.size()) + return false; + + beginInsertRows(parent, row, row + count - 1); + + for (int i = 0; i < count; ++i) + { + const auto profile = new ControlbarProfile(this); + profile->setName(tr("Profile %1").arg(m_profiles.size())); + + m_profiles.insert(row, profile); + } + + endInsertRows(); + + return true; +} + +bool ControlbarProfileModel::removeRows(int row, int count, const QModelIndex &parent) +{ + if (row < 0 || count < 1 || row + count > m_profiles.size()) + return false; + + beginRemoveRows(parent, row, row + count - 1); + + auto from = m_profiles.begin() + row; + auto to = from + count - 1; + std::for_each(from, to, [](auto* item) { + assert(item); + item->deleteLater(); + }); + m_profiles.erase(from, to); + + endRemoveRows(); + + return true; +} + +bool ControlbarProfileModel::setData(const QModelIndex &index, const QVariant &value, int role) +{ + if (data(index, role) != value) + { + auto ptrProfile = m_profiles.at(index.row()); + + if (!ptrProfile) + return false; + + switch (role) + { + case Qt::DisplayRole: + if (value.canConvert(QVariant::String)) + ptrProfile->setName(value.toString()); + else + return false; + break; + case MODEL_ROLE: + if (value.canConvert<ControlbarProfile*>()) + ptrProfile = qvariant_cast<ControlbarProfile*>(value); + else + return false; + break; + default: + return false; + } + + m_profiles.replace(index.row(), ptrProfile); + + emit dataChanged(index, index, { role }); + return true; + } + return false; +} + +Qt::ItemFlags ControlbarProfileModel::flags(const QModelIndex &index) const +{ + if (!index.isValid()) + return Qt::NoItemFlags; + + return (Qt::ItemIsEditable | Qt::ItemNeverHasChildren); +} + +int ControlbarProfileModel::selectedProfile() const +{ + return m_selectedProfile; +} + +ControlbarProfile* ControlbarProfileModel::currentModel() const +{ + return getProfile(selectedProfile()); +} + +void ControlbarProfileModel::save(bool clearDirty) const +{ + assert(m_intf->p_sys); + assert(m_intf->p_sys->mainSettings); + + if (!m_intf || !m_intf->p_sys || !m_intf->p_sys->mainSettings) + return; + + const auto settings = m_intf->p_sys->mainSettings; + const auto groupName = metaObject()->className(); + + settings->beginGroup(groupName); + settings->remove(""); // clear the group before save + + settings->setValue(SETTINGS_KEY_SELECTEDPROFILE, selectedProfile()); + + settings->beginWriteArray(SETTINGS_ARRAYNAME_PROFILES); + + for (int i = 0; i < m_profiles.size(); ++i) + { + settings->setArrayIndex(i); + + const auto& ptrModelMap = m_profiles.at(i)->m_models; + + QString val; + for (auto it = ptrModelMap.constBegin(); it != ptrModelMap.end(); ++it) + { + const QString identifier = it.key(); + + const auto serializedModels = m_profiles.at(i)->getModelData(identifier); + + static const auto join = [](const QVector<int>& list) { + QString ret; + for (auto i : list) + { + ret += QString::number(i) + SETTINGS_CONTROL_SEPARATOR; + } + if (!ret.isEmpty()) + ret.chop(1); + return ret; + }; + + val += QString(SETTINGS_PROFILE_SEPARATOR + "%1" + SETTINGS_CONFIGURATION_SEPARATOR + "%2" + SETTINGS_CONFIGURATION_SEPARATOR + "%3" + SETTINGS_CONFIGURATION_SEPARATOR + "%4").arg(identifier, + join(serializedModels[0]), + join(serializedModels[1]), + join(serializedModels[2])); + } + + if (clearDirty) + m_profiles.at(i)->resetDirty(); + + settings->setValue(SETTINGS_KEY_NAME, m_profiles.at(i)->name()); + settings->setValue(SETTINGS_KEY_MODEL, val); + } + + settings->endArray(); + settings->endGroup(); +} + +bool ControlbarProfileModel::reload() +{ + assert(m_intf->p_sys); + assert(m_intf->p_sys->mainSettings); + + if (!m_intf || !m_intf->p_sys || !m_intf->p_sys->mainSettings) + return false; + + const auto settings = m_intf->p_sys->mainSettings; + const auto groupName = metaObject()->className(); + + settings->beginGroup(groupName); + + const int size = settings->beginReadArray(SETTINGS_ARRAYNAME_PROFILES); + + if (size <= 0) + { + settings->endArray(); + settings->endGroup(); + + return false; + } + + beginResetModel(); + + decltype (m_profiles) profiles; + for (int i = 0; i < size; ++i) + { + settings->setArrayIndex(i); + + const QString modelValue = settings->value(SETTINGS_KEY_MODEL).toString(); + if (modelValue.isEmpty()) + continue; + + const auto val = modelValue.splitRef(SETTINGS_PROFILE_SEPARATOR); + if (val.isEmpty()) + continue; + + const auto ptrNewProfile = new ControlbarProfile(this); + ptrNewProfile->setName(settings->value(SETTINGS_KEY_NAME).toString()); + + for (auto j : val) + { + if (j.isEmpty()) + continue; + + const auto alignments = j.split(SETTINGS_CONFIGURATION_SEPARATOR); + + if (alignments.length() != 4) + continue; + + if (alignments[0].toString().isEmpty()) + continue; + + static const auto split = [](auto ref) { + QVector<int> list; + + if (ref.isEmpty()) + return list; + + for (auto i : ref.split(SETTINGS_CONTROL_SEPARATOR)) + { + bool ok = false; + int k = i.toInt(&ok); + + if (ok) + list.append(k); + } + return list; + }; + + const std::array<QVector<int>, 3> data { split(alignments[1]), + split(alignments[2]), + split(alignments[3]) }; + + ptrNewProfile->setModelData(alignments[0].toString(), data); + ptrNewProfile->resetDirty(); // Newly loaded model can not be dirty + } + + profiles.append(ptrNewProfile); + } + + settings->endArray(); + + m_selectedProfile = -1; + std::for_each(m_profiles.begin(), m_profiles.end(), [](auto i) { delete i; }); + + m_profiles = std::move(profiles); + + endResetModel(); + + bool ok = false; + int index = settings->value(SETTINGS_KEY_SELECTEDPROFILE).toInt(&ok); + + if (ok) + setSelectedProfile(index); + else + setSelectedProfile(0); + + settings->endGroup(); + + return true; +} + +bool ControlbarProfileModel::setSelectedProfile(int selectedProfile) +{ + if (m_selectedProfile == selectedProfile) + return false; + + const auto ptrProfileNew = getProfile(selectedProfile); + const auto ptrProfileOld = getProfile(m_selectedProfile); + + assert(ptrProfileNew); + + if (!ptrProfileNew) + return false; + + connect(ptrProfileNew, &ControlbarProfile::controlListChanged, this, &ControlbarProfileModel::selectedProfileControlListChanged); + connect(this, &QAbstractListModel::modelReset, ptrProfileNew, &ControlbarProfile::generateLinearControlList); + connect(this, &ControlbarProfileModel::selectedProfileChanged, ptrProfileNew, &ControlbarProfile::generateLinearControlList); + + if (ptrProfileOld && (ptrProfileNew != ptrProfileOld)) + { + disconnect(ptrProfileOld, &ControlbarProfile::controlListChanged, this, &ControlbarProfileModel::selectedProfileControlListChanged); + disconnect(this, &QAbstractListModel::modelReset, ptrProfileOld, &ControlbarProfile::generateLinearControlList); + disconnect(this, &ControlbarProfileModel::selectedProfileChanged, ptrProfileOld, &ControlbarProfile::generateLinearControlList); + } + + m_selectedProfile = selectedProfile; + + emit selectedProfileChanged(); + + return true; +} + +ControlbarProfile *ControlbarProfileModel::getProfile(int index) const +{ + if (index < 0 || index >= m_profiles.size()) + return nullptr; + + return m_profiles.at(index); +} + +ControlbarProfile *ControlbarProfileModel::newProfile(const QString &name) +{ + if (name.isEmpty()) + return nullptr; + + const auto ptrProfile = newProfile(); + + ptrProfile->setName(generateUniqueName(name)); + + return ptrProfile; +} + +ControlbarProfile *ControlbarProfileModel::newProfile() +{ + const auto ptrNewProfile = new ControlbarProfile(this); + + beginInsertRows(QModelIndex(), m_profiles.size(), m_profiles.size()); + + m_profiles.append(ptrNewProfile); + + endInsertRows(); + + return ptrNewProfile; +} + +ControlbarProfile *ControlbarProfileModel::cloneProfile(const ControlbarProfile *profile) +{ + const auto ptrNewProfile = newProfile(profile->name()); + + if (!ptrNewProfile) + return nullptr; + + for (auto it = profile->m_models.constBegin(); it != profile->m_models.constEnd(); ++it) + { + ptrNewProfile->setModelData(it.key(), profile->getModelData(it.key())); + ptrNewProfile->resetDirty(); + } + + return ptrNewProfile; +} + +void ControlbarProfileModel::cloneSelectedProfile(const QString &newProfileName) +{ + const auto ptrModel = currentModel(); + + assert(ptrModel); + if (!ptrModel) + return; + + const auto ptrNewModel = cloneProfile(ptrModel); + + assert(ptrNewModel); + if (!ptrNewModel) + return; + + ptrNewModel->setName(generateUniqueName(newProfileName)); +} + +void ControlbarProfileModel::deleteSelectedProfile() +{ + const auto ptrSelectedProfile = getProfile(m_selectedProfile); + + if (!ptrSelectedProfile) + return; + + const auto _selectedProfile = m_selectedProfile; + + beginRemoveRows(QModelIndex(), _selectedProfile, _selectedProfile); + + m_selectedProfile = -1; + + delete ptrSelectedProfile; + m_profiles.removeAt(_selectedProfile); + + endRemoveRows(); + + if (getProfile(_selectedProfile - 1)) + setSelectedProfile(_selectedProfile - 1); + else + setSelectedProfile(_selectedProfile); +} diff --git a/modules/gui/qt/dialogs/toolbar/controlbar_profile_model.hpp b/modules/gui/qt/dialogs/toolbar/controlbar_profile_model.hpp new file mode 100644 index 0000000000000000000000000000000000000000..108229bbe7086b5652eae737b171d20482377050 --- /dev/null +++ b/modules/gui/qt/dialogs/toolbar/controlbar_profile_model.hpp @@ -0,0 +1,110 @@ +/***************************************************************************** + * Copyright (C) 2021 VLC authors and VideoLAN + * + * 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 CONTROLBARPROFILEMODEL_H +#define CONTROLBARPROFILEMODEL_H + +#include <QAbstractListModel> +#include <array> + +#include "controlbar_profile.hpp" + +struct intf_thread_t; + +class ControlbarProfileModel : public QAbstractListModel +{ + Q_OBJECT + + Q_PROPERTY(int selectedProfile READ selectedProfile WRITE setSelectedProfile NOTIFY selectedProfileChanged) + Q_PROPERTY(ControlbarProfile* currentModel READ currentModel NOTIFY selectedProfileChanged) + + Q_PROPERTY(int count READ rowCount NOTIFY countChanged) + +public: + explicit ControlbarProfileModel(intf_thread_t *p_intf, QObject *parent = nullptr); + + // Basic functionality: + int rowCount(const QModelIndex &parent = QModelIndex()) const override; + + QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override; + + QHash<int, QByteArray> roleNames() const override; + + // Editable: + Q_INVOKABLE bool setData(const QModelIndex &index, const QVariant &value, + int role = Qt::DisplayRole) override; + + Qt::ItemFlags flags(const QModelIndex& index) const override; + + // Add data: + Q_INVOKABLE bool insertRows(int row, int count, const QModelIndex &parent = QModelIndex()) override; + + // Remove data: + bool removeRows(int row, int count, const QModelIndex &parent = QModelIndex()) override; + +public: + enum Roles { + MODEL_ROLE = Qt::UserRole, + }; + + int selectedProfile() const; + ControlbarProfile* currentModel() const; + + ControlbarProfile* cloneProfile(const ControlbarProfile* profile); + Q_INVOKABLE void cloneSelectedProfile(const QString& newProfileName); + + Q_INVOKABLE ControlbarProfile* getProfile(int index) const; + + Q_INVOKABLE ControlbarProfile* newProfile(const QString& name); + ControlbarProfile* newProfile(); + + Q_INVOKABLE void deleteSelectedProfile(); + +public slots: + void save(bool clearDirty = true) const; + bool reload(); + + bool setSelectedProfile(int selectedProfile); + +signals: + void countChanged(); + void selectedProfileChanged(); + + void selectedProfileControlListChanged(const QVector<int>& linearControlList); + +private: + QVector<ControlbarProfile *> m_profiles; + + int m_selectedProfile = -1; + + struct Profile { + QString name; + QVector<ControlbarProfile::Configuration> modelData; + }; + + static const QVector<Profile> m_defaults; + +private: + void insertDefaults(); + + QString generateUniqueName(const QString& name); + +protected: + intf_thread_t *m_intf = nullptr; +}; + +#endif // CONTROLBARPROFILEMODEL_H diff --git a/modules/gui/qt/maininterface/main_interface.cpp b/modules/gui/qt/maininterface/main_interface.cpp index e0bcdcb0a402cabe8d3f123032f395960c982f9d..206cedd25d46d76465bc466c5bd6e18c44fae99b 100644 --- a/modules/gui/qt/maininterface/main_interface.cpp +++ b/modules/gui/qt/maininterface/main_interface.cpp @@ -50,6 +50,8 @@ #include "vlc_media_library.h" +#include "dialogs/toolbar/controlbar_profile_model.hpp" + #include <QCloseEvent> #include <QKeyEvent> @@ -155,6 +157,9 @@ MainInterface::MainInterface(intf_thread_t *_p_intf , QWidget* parent, Qt::Windo m_colorScheme = new ColorSchemeModel(this); m_colorScheme->setCurrent(currentColorScheme); + /* Controlbar Profile Model Creation */ + m_controlbarProfileModel = new ControlbarProfileModel(p_intf, this); + /* Should the UI stays on top of other windows */ b_interfaceOnTop = var_InheritBool( p_intf, "video-on-top" ); diff --git a/modules/gui/qt/maininterface/main_interface.hpp b/modules/gui/qt/maininterface/main_interface.hpp index d8deb10c47b69a6d47987100eef1e303289c9169..1af9804d11b97d706dc37faa3a12a46d975d1094 100644 --- a/modules/gui/qt/maininterface/main_interface.hpp +++ b/modules/gui/qt/maininterface/main_interface.hpp @@ -59,6 +59,7 @@ class QTimer; class StandardPLPanel; struct vout_window_t; class VideoSurfaceProvider; +class ControlbarProfileModel; class WindowStateHolder : public QObject { @@ -159,6 +160,8 @@ class MainInterface : public QVLCMW Q_PROPERTY(bool hasToolbarMenu READ hasToolbarMenu NOTIFY hasToolbarMenuChanged) Q_PROPERTY(bool canShowVideoPIP READ canShowVideoPIP CONSTANT) Q_PROPERTY(bool pinVideoControls READ pinVideoControls WRITE setPinVideoControls NOTIFY pinVideoControlsChanged) + Q_PROPERTY(ControlbarProfileModel* controlbarProfileModel READ controlbarProfileModel CONSTANT) + public: /* tors */ @@ -207,6 +210,7 @@ public: inline bool canShowVideoPIP() const { return m_canShowVideoPIP; } inline void setCanShowVideoPIP(bool canShowVideoPIP) { m_canShowVideoPIP = canShowVideoPIP; } inline bool pinVideoControls() const { return m_pinVideoControls; } + inline ControlbarProfileModel* controlbarProfileModel() const { return m_controlbarProfileModel; } bool hasEmbededVideo() const; VideoSurfaceProvider* getVideoSurfaceProvider() const; @@ -290,6 +294,8 @@ protected: VLCVarChoiceModel* m_extraInterfaces; + ControlbarProfileModel* m_controlbarProfileModel; + public slots: void toggleUpdateSystrayMenu(); void showUpdateSystrayMenu(); diff --git a/modules/gui/qt/maininterface/mainui.cpp b/modules/gui/qt/maininterface/mainui.cpp index 678b2f87e0c2278581abd784df099251851da5c5..0c518b259949f4b8a176333d1a750a771ea2eb8a 100644 --- a/modules/gui/qt/maininterface/mainui.cpp +++ b/modules/gui/qt/maininterface/mainui.cpp @@ -18,7 +18,11 @@ #include "medialibrary/mlplaylist.hpp" #include "player/player_controller.hpp" -#include "player/playercontrolbarmodel.hpp" +#include "player/player_controlbar_model.hpp" +#include "player/control_list_model.hpp" + +#include "dialogs/toolbar/controlbar_profile_model.hpp" +#include "dialogs/toolbar/controlbar_profile.hpp" #include "playlist/playlist_model.hpp" #include "playlist/playlist_controller.hpp" @@ -235,7 +239,11 @@ void MainUI::registerQMLTypes() qmlRegisterType<QmlEventFilter>( "org.videolan.vlc", 0, 1, "EventFilter" ); - qmlRegisterType<PlayerControlBarModel>( "org.videolan.vlc", 0, 1, "PlayerControlBarModel"); + qRegisterMetaType<ControlbarProfile*>(); + qRegisterMetaType<ControlbarProfileModel*>(); + qmlRegisterUncreatableType<ControlbarProfile>("org.videolan.vlc", 0, 1, "ControlbarProfile", ""); + qmlRegisterUncreatableType<PlayerControlbarModel>("org.videolan.vlc", 0, 1, "PlayerControlbarModel", ""); + qmlRegisterUncreatableType<ControlListModel>( "org.videolan.vlc", 0, 1, "ControlListModel", "" ); qRegisterMetaType<QmlMainContext*>(); qmlRegisterType<QmlGlobalMenu>( "org.videolan.vlc", 0, 1, "QmlGlobalMenu" ); diff --git a/modules/gui/qt/player/control_list_model.cpp b/modules/gui/qt/player/control_list_model.cpp new file mode 100644 index 0000000000000000000000000000000000000000..4c9d8e5eb5421a60cc372019a65b7166da862c62 --- /dev/null +++ b/modules/gui/qt/player/control_list_model.cpp @@ -0,0 +1,155 @@ +/***************************************************************************** + * Copyright (C) 2021 VLC authors and VideoLAN + * + * 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 "control_list_model.hpp" + +ControlListModel::ControlListModel(QObject *parent) : QAbstractListModel(parent) +{ + connect(this, &QAbstractListModel::rowsInserted, this, &ControlListModel::countChanged); + connect(this, &QAbstractListModel::rowsRemoved, this, &ControlListModel::countChanged); + connect(this, &QAbstractListModel::modelReset, this, &ControlListModel::countChanged); +} + +int ControlListModel::rowCount(const QModelIndex &parent) const +{ + if (parent.isValid()) + return 0; + + return m_controls.size(); +} + +QVariant ControlListModel::data(const QModelIndex &index, int role) const +{ + if (!index.isValid()) + return QVariant(); + + const ControlType control = m_controls.at(index.row()); + + switch (role) { + case ID_ROLE: + return QVariant(control); + } + return QVariant(); +} + +bool ControlListModel::setData(const QModelIndex &index, const QVariant &value, int role) +{ + ControlType control = m_controls.at(index.row()); + + switch (role) { + case ID_ROLE: + if (value.canConvert(QVariant::Int)) + control = static_cast<ControlType>(value.toInt()); + else + return false; + break; + } + + if (setButtonAt(index.row(), control)) { + emit dataChanged(index, index, { role }); + return true; + } + return false; +} + +Qt::ItemFlags ControlListModel::flags(const QModelIndex &index) const +{ + if (!index.isValid()) + return Qt::NoItemFlags; + + return (Qt::ItemIsEditable | Qt::ItemNeverHasChildren); +} + +QHash<int, QByteArray> ControlListModel::roleNames() const +{ + return { + { + ID_ROLE, "id" + } + }; +} + +QVector<int> ControlListModel::getControls() const +{ + QVector<int> list; + + for (auto i : m_controls) + { + list.append(static_cast<int>(i)); + } + + return list; +} + +void ControlListModel::setControls(const QVector<int> &list) +{ + beginResetModel(); + + m_controls.resize(list.size()); + + for (int i = 0; i < list.size(); ++i) + { + m_controls[i] = static_cast<ControlType>(list.at(i)); + } + + endResetModel(); +} + +bool ControlListModel::setButtonAt(int index, const ControlType &button) +{ + if(index < 0 || index >= m_controls.size()) + return false; + + const ControlType oldControl = m_controls.at(index); + + if (button == oldControl) + return false; + + m_controls[index] = button; + return true; +} + +void ControlListModel::insert(int index, QVariantMap bdata) +{ + beginInsertRows(QModelIndex(), index, index); + m_controls.insert(index, static_cast<ControlType>(bdata.value("id").toInt())); + endInsertRows(); +} +void ControlListModel::move(int src, int dest) +{ + if(src == dest) + return; + + beginMoveRows(QModelIndex(), src, src, QModelIndex(), dest + (src < dest ? 1 : 0)); + m_controls.move(src, dest); + endMoveRows(); +} + +void ControlListModel::remove(int index) +{ + beginRemoveRows(QModelIndex(), index, index); + m_controls.remove(index); + endRemoveRows(); +} + +void ControlListModel::clear() +{ + beginResetModel(); + m_controls.clear(); + endResetModel(); +} diff --git a/modules/gui/qt/player/playercontrolbarmodel.hpp b/modules/gui/qt/player/control_list_model.hpp similarity index 63% rename from modules/gui/qt/player/playercontrolbarmodel.hpp rename to modules/gui/qt/player/control_list_model.hpp index d439bbbdb0574dd8ba7884454a3383f307921973..34d2f1aebe38ba59ea68a104ba561f96c2906c59 100644 --- a/modules/gui/qt/player/playercontrolbarmodel.hpp +++ b/modules/gui/qt/player/control_list_model.hpp @@ -1,5 +1,5 @@ /***************************************************************************** - * Copyright (C) 2019 VLC authors and VideoLAN + * Copyright (C) 2021 VLC authors and VideoLAN * * 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 @@ -16,32 +16,25 @@ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA. *****************************************************************************/ -#ifndef CONTROLLERMODEL_H -#define CONTROLLERMODEL_H +#ifndef CONTROLLISTMODEL_HPP +#define CONTROLLISTMODEL_HPP #include <QAbstractListModel> #include <QVector> -#include "util/qml_main_context.hpp" - -class PlayerControlBarModel : public QAbstractListModel +class ControlListModel : public QAbstractListModel { Q_OBJECT - Q_PROPERTY(QmlMainContext* mainCtx READ getMainCtx WRITE setMainCtx NOTIFY ctxChanged) - Q_PROPERTY(QString configName READ getConfigName WRITE setConfigName NOTIFY configNameChanged) Q_PROPERTY(int count READ rowCount NOTIFY countChanged) - public: - explicit PlayerControlBarModel(QObject *_parent = nullptr); - struct IconToolButton - { - int id; - }; - enum{ - ID_ROLE + explicit ControlListModel(QObject *parent = nullptr); + + enum Roles { + ID_ROLE = Qt::UserRole }; - enum ButtonType_e + + enum ControlType { PLAY_BUTTON, STOP_BUTTON, @@ -82,7 +75,7 @@ public: WIDGET_SPACER_EXTEND, WIDGET_MAX }; - Q_ENUM(ButtonType_e) + Q_ENUM(ControlType) // Basic functionality: int rowCount(const QModelIndex &parent = QModelIndex()) const override; @@ -97,43 +90,21 @@ public: virtual QHash<int, QByteArray> roleNames() const override; - inline QmlMainContext* getMainCtx() const { return m_mainCtx; } - void setMainCtx(QmlMainContext*); - - inline QString getConfigName() { return configName; } - void setConfigName(QString name); - - static QString getSerializedDefaultStyle(); + QVector<int> getControls() const; + void setControls(const QVector<int>& list); signals: - void ctxChanged(QmlMainContext*); - void configNameChanged(QString); void countChanged(); -protected: - intf_thread_t *p_intf = nullptr; - private: - QVector<IconToolButton> mButtons; - QString configName; - - void parseAndAdd(const QString& config); - void parseDefault(const QVector<IconToolButton>& config); - - bool setButtonAt(int index, const IconToolButton &button); - void addProfiles(); - void loadConfig(); - - QmlMainContext* m_mainCtx = nullptr; + QVector<ControlType> m_controls; + bool setButtonAt(int index, const ControlType &button); public slots: Q_INVOKABLE void insert(int index, QVariantMap bdata); Q_INVOKABLE void move(int src,int dest); Q_INVOKABLE void remove(int index); - Q_INVOKABLE void reloadConfig(QString config); - Q_INVOKABLE void saveConfig(); - Q_INVOKABLE QString getConfig(); - Q_INVOKABLE void reloadModel(); + Q_INVOKABLE void clear(); }; -#endif // CONTROLLERMODEL_H +#endif // CONTROLLISTMODEL_HPP diff --git a/modules/gui/qt/player/player_controlbar_model.cpp b/modules/gui/qt/player/player_controlbar_model.cpp new file mode 100644 index 0000000000000000000000000000000000000000..9496e8f7ba5334a0a671030eabfd89313da38dcc --- /dev/null +++ b/modules/gui/qt/player/player_controlbar_model.cpp @@ -0,0 +1,91 @@ +/***************************************************************************** + * Copyright (C) 2021 VLC authors and VideoLAN + * + * 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 "player_controlbar_model.hpp" + +#include "control_list_model.hpp" + +PlayerControlbarModel::PlayerControlbarModel(QObject *parent) : QObject(parent) +{ + m_left = new ControlListModel(this); + m_center = new ControlListModel(this); + m_right = new ControlListModel(this); + + connect(m_left, &ControlListModel::countChanged, this, &PlayerControlbarModel::contentChanged); + connect(m_center, &ControlListModel::countChanged, this, &PlayerControlbarModel::contentChanged); + connect(m_right, &ControlListModel::countChanged, this, &PlayerControlbarModel::contentChanged); + + connect(m_left, &QAbstractListModel::dataChanged, this, &PlayerControlbarModel::contentChanged); + connect(m_center, &QAbstractListModel::dataChanged, this, &PlayerControlbarModel::contentChanged); + connect(m_right, &QAbstractListModel::dataChanged, this, &PlayerControlbarModel::contentChanged); +} + +PlayerControlbarModel::~PlayerControlbarModel() +{ + setDirty(false); +} + +bool PlayerControlbarModel::dirty() const +{ + return m_dirty; +} + +std::array<QVector<int>, 3> PlayerControlbarModel::serializeModels() const +{ + return { left()->getControls(), + center()->getControls(), + right()->getControls() }; +} + +void PlayerControlbarModel::loadModels(const std::array<QVector<int>, 3> &array) +{ + left()->setControls(array.at(0)); + center()->setControls(array.at(1)); + right()->setControls(array.at(2)); +} + +ControlListModel *PlayerControlbarModel::left() const +{ + return m_left; +} + +ControlListModel *PlayerControlbarModel::center() const +{ + return m_center; +} + +ControlListModel *PlayerControlbarModel::right() const +{ + return m_right; +} + +void PlayerControlbarModel::setDirty(bool dirty) +{ + if (m_dirty == dirty) + return; + + m_dirty = dirty; + emit dirtyChanged(m_dirty); +} + +void PlayerControlbarModel::contentChanged() +{ + setDirty(true); + + emit controlListChanged(); +} diff --git a/modules/gui/qt/player/player_controlbar_model.hpp b/modules/gui/qt/player/player_controlbar_model.hpp new file mode 100644 index 0000000000000000000000000000000000000000..a6b71319cf90f578c885e18ff5d750d710116ff1 --- /dev/null +++ b/modules/gui/qt/player/player_controlbar_model.hpp @@ -0,0 +1,68 @@ +/***************************************************************************** + * Copyright (C) 2021 VLC authors and VideoLAN + * + * 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 PLAYERCONTROLBARMODEL_HPP +#define PLAYERCONTROLBARMODEL_HPP + +#include <QObject> +#include <array> + +class ControlListModel; + +class PlayerControlbarModel : public QObject +{ + Q_OBJECT + + Q_PROPERTY(bool dirty READ dirty WRITE setDirty NOTIFY dirtyChanged) + + Q_PROPERTY(ControlListModel* left READ left CONSTANT) + Q_PROPERTY(ControlListModel* center READ center CONSTANT) + Q_PROPERTY(ControlListModel* right READ right CONSTANT) + +public: + explicit PlayerControlbarModel(QObject *parent = nullptr); + ~PlayerControlbarModel(); + + bool dirty() const; + + std::array<QVector<int>, 3> serializeModels() const; + void loadModels(const std::array<QVector<int>, 3>& array); + + ControlListModel* left() const; + ControlListModel* center() const; + ControlListModel* right() const; + +public slots: + void setDirty(bool dirty); + +signals: + void dirtyChanged(bool dirty); + void controlListChanged(); + +private: + bool m_dirty = false; + + ControlListModel* m_left = nullptr; + ControlListModel* m_center = nullptr; + ControlListModel* m_right = nullptr; + +private slots: + void contentChanged(); +}; + +#endif diff --git a/modules/gui/qt/player/playercontrolbarmodel.cpp b/modules/gui/qt/player/playercontrolbarmodel.cpp deleted file mode 100644 index 6aa486cfa5debd514ac03c28cfc7aac0707b0414..0000000000000000000000000000000000000000 --- a/modules/gui/qt/player/playercontrolbarmodel.cpp +++ /dev/null @@ -1,326 +0,0 @@ -/***************************************************************************** - * Copyright (C) 2019 VLC authors and VideoLAN - * - * 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 <QSettings> - -#include "qt.hpp" -#include "playercontrolbarmodel.hpp" - -enum default_align { - ALIGN_LEFT = 0, - ALIGN_CENTER, - ALIGN_RIGHT, - ALIGN_SIZE -}; - -static const QVector<PlayerControlBarModel::IconToolButton> MAIN_TB_DEFAULT[default_align::ALIGN_SIZE] = { - { - // left - {PlayerControlBarModel::LANG_BUTTON}, - {PlayerControlBarModel::MENU_BUTTON} - }, - { - // center - {PlayerControlBarModel::RANDOM_BUTTON}, - {PlayerControlBarModel::PREVIOUS_BUTTON}, - {PlayerControlBarModel::PLAY_BUTTON}, - {PlayerControlBarModel::NEXT_BUTTON}, - {PlayerControlBarModel::LOOP_BUTTON} - }, - { - // right - {PlayerControlBarModel::VOLUME}, - {PlayerControlBarModel::FULLSCREEN_BUTTON} - } -}; - -static const QVector<PlayerControlBarModel::IconToolButton> MINI_TB_DEFAULT[default_align::ALIGN_SIZE] = { - { - // left - {PlayerControlBarModel::ARTWORK_INFO} - }, - { - // center - {PlayerControlBarModel::RANDOM_BUTTON}, - {PlayerControlBarModel::PREVIOUS_BUTTON}, - {PlayerControlBarModel::PLAY_BUTTON}, - {PlayerControlBarModel::NEXT_BUTTON}, - {PlayerControlBarModel::LOOP_BUTTON} - }, - { - // right - {PlayerControlBarModel::VOLUME}, - {PlayerControlBarModel::PLAYER_SWITCH_BUTTON} - } -}; - - -PlayerControlBarModel::PlayerControlBarModel(QObject *_parent) : QAbstractListModel(_parent) -{ - configName = "MainPlayerToolbar"; - - connect(this, &QAbstractListModel::rowsInserted, this, &PlayerControlBarModel::countChanged); - connect(this, &QAbstractListModel::rowsRemoved, this, &PlayerControlBarModel::countChanged); - connect(this, &QAbstractListModel::modelReset, this, &PlayerControlBarModel::countChanged); -} - -void PlayerControlBarModel::saveConfig() -{ - getSettings()->setValue(configName,getConfig()); -} - -QString PlayerControlBarModel::getConfig() -{ - QString config=""; - for (IconToolButton it: mButtons) { - config += QString::number(it.id); - config += ";"; - } - return config; -} - -void PlayerControlBarModel::reloadConfig(QString config) -{ - beginResetModel(); - mButtons.clear(); - if (!config.isEmpty()) - parseAndAdd(config); - endResetModel(); -} - -void PlayerControlBarModel::reloadModel() -{ - beginResetModel(); - mButtons.clear(); - - QVariant config = getSettings() ->value(configName); - - if (!config.isNull() && config.canConvert<QString>()) - parseAndAdd(config.toString()); - else - { - const auto configAndAlignment = configName.split("-"); - if (configAndAlignment.size() == 2) - { - const auto alignment = configAndAlignment[1]; - if (configAndAlignment[0] == QLatin1String("MainPlayerToolbar")) - { - if (alignment == "left") - parseDefault(MAIN_TB_DEFAULT[default_align::ALIGN_LEFT]); - else if (alignment == "center") - parseDefault(MAIN_TB_DEFAULT[default_align::ALIGN_CENTER]); - else if (alignment == "right") - parseDefault(MAIN_TB_DEFAULT[default_align::ALIGN_RIGHT]); - } - else - { - if (alignment == "left") - parseDefault(MINI_TB_DEFAULT[default_align::ALIGN_LEFT]); - else if (alignment == "center") - parseDefault(MINI_TB_DEFAULT[default_align::ALIGN_CENTER]); - else if (alignment == "right") - parseDefault(MINI_TB_DEFAULT[default_align::ALIGN_RIGHT]); - - } - } - } - endResetModel(); -} - -void PlayerControlBarModel::parseDefault(const QVector<PlayerControlBarModel::IconToolButton>& config) -{ - beginInsertRows(QModelIndex(),rowCount(),rowCount() + config.size()); - for (const auto& i : config) - mButtons.append(i); - endInsertRows(); -} - -void PlayerControlBarModel::parseAndAdd(const QString &config) -{ - beginInsertRows(QModelIndex(),rowCount(),rowCount()+config.split(";", - #if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0) - Qt::SkipEmptyParts - #else - QString::SkipEmptyParts - #endif - ).length() - 1); - - for (const QString& iconPropertyTxt : config.split( ";", - #if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0) - Qt::SkipEmptyParts - #else - QString::SkipEmptyParts - #endif - ) ) - { - QStringList list2 = iconPropertyTxt.trimmed().split( "-" ); - - if( list2.count() < 1 ) - { - msg_Warn( p_intf, "Parsing error 1. Please, report this." ); - continue; - } - bool ok; - ButtonType_e i_type = static_cast<ButtonType_e>(list2.at( 0 ).toInt( &ok )); - if( !ok ) - { - msg_Warn( p_intf, "Parsing error 2. Please, report this." ); - continue; - } - - IconToolButton itButton = {i_type}; - mButtons.append(itButton); - } - - endInsertRows(); -} - -int PlayerControlBarModel::rowCount(const QModelIndex &parent) const -{ - if (parent.isValid() ) - return 0; - - return mButtons.size(); -} - -QVariant PlayerControlBarModel::data(const QModelIndex &index, int role) const -{ - if (!index.isValid() ) - return QVariant(); - - const IconToolButton button = mButtons.at(index.row()); - - switch (role) { - case ID_ROLE: - return QVariant(button.id); - } - return QVariant(); -} - -bool PlayerControlBarModel::setData(const QModelIndex &index, const QVariant &value, int role) -{ - IconToolButton button = mButtons.at(index.row()); - switch (role) { - case ID_ROLE: - button.id = value.toInt(); - break; - } - - if (setButtonAt(index.row(),button)) { - emit dataChanged(index, index, QVector<int>() << role); - return true; - } - return false; -} - -Qt::ItemFlags PlayerControlBarModel::flags(const QModelIndex &index) const -{ - if (!index.isValid()) - return Qt::NoItemFlags; - - return Qt::ItemIsEditable; -} - -QHash<int, QByteArray> PlayerControlBarModel::roleNames() const -{ - QHash<int, QByteArray> names; - - names[ID_ROLE] = "id"; - - return names; -} -bool PlayerControlBarModel::setButtonAt(int index, const IconToolButton &button) -{ - if(index < 0 || index >= mButtons.size()) - return false; - const IconToolButton &oldButton = mButtons.at(index); - - if (button.id == oldButton.id) - return false; - - mButtons[index] = button; - return true; -} - -void PlayerControlBarModel::setMainCtx(QmlMainContext* ctx) -{ - if(ctx == nullptr && m_mainCtx == ctx) - return; - m_mainCtx = ctx; - p_intf = m_mainCtx->getIntf(); - assert(p_intf != nullptr); - reloadModel(); - emit ctxChanged(ctx); -} - -void PlayerControlBarModel::setConfigName(QString name) -{ - if(configName == name) - return; - configName = name; - if (m_mainCtx) - reloadModel(); - emit configNameChanged(name); -} - -QString PlayerControlBarModel::getSerializedDefaultStyle() -{ - QString out; - - auto serialize = [](auto style) - { - QString _out; - for (size_t i = 0; i < default_align::ALIGN_SIZE; i++) - { - for (const auto& it : style[i]) - { - _out += QString::number(it.id) + ";"; - } - _out.chop(1); - _out += "#"; - } - _out.chop(1); - return _out; - }; - - out += serialize(MAIN_TB_DEFAULT); - out += " | "; - out += serialize(MINI_TB_DEFAULT); - - return out; -} - -void PlayerControlBarModel::insert(int index, QVariantMap bdata) -{ - beginInsertRows(QModelIndex(),index,index); - mButtons.insert(index, { bdata.value("id").toInt() }); - endInsertRows(); -} -void PlayerControlBarModel::move(int src, int dest) -{ - if(src == dest) return; - beginMoveRows(QModelIndex(),src,src,QModelIndex(),dest + (src < dest ? 1:0)); - mButtons.move(src,dest); - endMoveRows(); -} - -void PlayerControlBarModel::remove(int index) -{ - beginRemoveRows(QModelIndex(),index,index); - mButtons.remove(index); - endRemoveRows(); -} diff --git a/modules/gui/qt/player/qml/ControlBar.qml b/modules/gui/qt/player/qml/ControlBar.qml index a867bca770569adbea1f7b5896a4f15c9cde37e0..17655e357850618052225e53abd937f6bd2ffc27 100644 --- a/modules/gui/qt/player/qml/ControlBar.qml +++ b/modules/gui/qt/player/qml/ControlBar.qml @@ -40,7 +40,7 @@ Widgets.NavigableFocusScope { readonly property alias sliderY: row2.y property int textPosition: ControlBar.TimeTextPosition.AboveSlider property VLCColors colors: VLCStyle.nightColors - property var configs: ["MainPlayerToolbar-left", "MainPlayerToolbar-center", "MainPlayerToolbar-right"] + property alias identifier: playerButtonsLayout.identifier property alias sliderHeight: trackPositionSlider.barHeight property alias sliderBackgroundColor: trackPositionSlider.backgroundColor property alias sliderProgressColor: trackPositionSlider.progressBarColor @@ -145,8 +145,6 @@ Widgets.NavigableFocusScope { bottomMargin: VLCStyle.applicationVerticalMargin } - models: [playerControlBarModel_left, playerControlBarModel_center, playerControlBarModel_right] - navigationUpItem: trackPositionSlider.enabled ? trackPositionSlider : root.navigationUpItem colors: root.colors @@ -194,23 +192,4 @@ Widgets.NavigableFocusScope { Keys.onDownPressed: playerButtonsLayout.focus = true } - - - PlayerControlBarModel{ - id:playerControlBarModel_left - mainCtx: mainctx - configName: root.configs[0] - } - - PlayerControlBarModel{ - id:playerControlBarModel_center - mainCtx: mainctx - configName: root.configs[1] - } - - PlayerControlBarModel{ - id:playerControlBarModel_right - mainCtx: mainctx - configName: root.configs[2] - } } diff --git a/modules/gui/qt/player/qml/ControlButtons.qml b/modules/gui/qt/player/qml/ControlButtons.qml index e04c00d45c61d99225afb6f2444d8505bef66246..c3f3dac1427bcaa58ae998dc1a19b59e082ed783 100644 --- a/modules/gui/qt/player/qml/ControlButtons.qml +++ b/modules/gui/qt/player/qml/ControlButtons.qml @@ -35,74 +35,74 @@ Item{ signal requestLockUnlockAutoHide(bool lock, var source) property var buttonL: [ - { id: PlayerControlBarModel.PLAY_BUTTON, label: VLCIcons.play, text: i18n.qtr("Play")}, - { id: PlayerControlBarModel.STOP_BUTTON, label: VLCIcons.stop, text: i18n.qtr("Stop")}, - { id: PlayerControlBarModel.OPEN_BUTTON, label: VLCIcons.eject, text: i18n.qtr("Open")}, - { id: PlayerControlBarModel.PREVIOUS_BUTTON, label: VLCIcons.previous, text: i18n.qtr("Previous")}, - { id: PlayerControlBarModel.NEXT_BUTTON, label: VLCIcons.next, text: i18n.qtr("Next")}, - { id: PlayerControlBarModel.SLOWER_BUTTON, label: VLCIcons.slower, text: i18n.qtr("Slower")}, - { id: PlayerControlBarModel.FASTER_BUTTON, label: VLCIcons.faster, text: i18n.qtr("Faster")}, - { id: PlayerControlBarModel.FULLSCREEN_BUTTON, label: VLCIcons.fullscreen, text: i18n.qtr("Fullscreen")}, - { id: PlayerControlBarModel.EXTENDED_BUTTON, label: VLCIcons.extended, text: i18n.qtr("Extended panel")}, - { id: PlayerControlBarModel.PLAYLIST_BUTTON, label: VLCIcons.playlist, text: i18n.qtr("Playlist")}, - { id: PlayerControlBarModel.SNAPSHOT_BUTTON, label: VLCIcons.snapshot, text: i18n.qtr("Snapshot")}, - { id: PlayerControlBarModel.RECORD_BUTTON, label: VLCIcons.record, text: i18n.qtr("Record")}, - { id: PlayerControlBarModel.ATOB_BUTTON, label: VLCIcons.atob, text: i18n.qtr("A-B Loop")}, - { id: PlayerControlBarModel.FRAME_BUTTON, label: VLCIcons.frame_by_frame, text: i18n.qtr("Frame By Frame")}, - { id: PlayerControlBarModel.SKIP_BACK_BUTTON, label: VLCIcons.skip_back, text: i18n.qtr("Step backward")}, - { id: PlayerControlBarModel.SKIP_FW_BUTTON, label: VLCIcons.skip_for, text: i18n.qtr("Step forward")}, - { id: PlayerControlBarModel.QUIT_BUTTON, label: VLCIcons.clear, text: i18n.qtr("Quit")}, - { id: PlayerControlBarModel.RANDOM_BUTTON, label: VLCIcons.shuffle_on, text: i18n.qtr("Random")}, - { id: PlayerControlBarModel.LOOP_BUTTON, label: VLCIcons.repeat_all, text: i18n.qtr("Loop")}, - { id: PlayerControlBarModel.INFO_BUTTON, label: VLCIcons.info, text: i18n.qtr("Information")}, - { id: PlayerControlBarModel.LANG_BUTTON, label: VLCIcons.audiosub, text: i18n.qtr("Open subtitles")}, - { id: PlayerControlBarModel.MENU_BUTTON, label: VLCIcons.menu, text: i18n.qtr("Menu Button")}, - { id: PlayerControlBarModel.BACK_BUTTON, label: VLCIcons.exit, text: i18n.qtr("Back Button")}, - { id: PlayerControlBarModel.CHAPTER_PREVIOUS_BUTTON, label: VLCIcons.dvd_prev, text: i18n.qtr("Previous chapter")}, - { id: PlayerControlBarModel.CHAPTER_NEXT_BUTTON, label: VLCIcons.dvd_next, text: i18n.qtr("Next chapter")}, - { id: PlayerControlBarModel.VOLUME, label: VLCIcons.volume_high, text: i18n.qtr("Volume Widget")}, - { id: PlayerControlBarModel.TELETEXT_BUTTONS, label: VLCIcons.tvtelx, text: i18n.qtr("Teletext")}, - { id: PlayerControlBarModel.ASPECT_RATIO_COMBOBOX, label: VLCIcons.aspect_ratio, text: i18n.qtr("Aspect Ratio")}, - { id: PlayerControlBarModel.WIDGET_SPACER, label: VLCIcons.space, text: i18n.qtr("Spacer")}, - { id: PlayerControlBarModel.WIDGET_SPACER_EXTEND, label: VLCIcons.space, text: i18n.qtr("Expanding Spacer")}, - { id: PlayerControlBarModel.PLAYER_SWITCH_BUTTON, label: VLCIcons.fullscreen, text: i18n.qtr("Switch Player")}, - { id: PlayerControlBarModel.ARTWORK_INFO, label: VLCIcons.info, text: i18n.qtr("Artwork Info")} + { id: ControlListModel.PLAY_BUTTON, label: VLCIcons.play, text: i18n.qtr("Play")}, + { id: ControlListModel.STOP_BUTTON, label: VLCIcons.stop, text: i18n.qtr("Stop")}, + { id: ControlListModel.OPEN_BUTTON, label: VLCIcons.eject, text: i18n.qtr("Open")}, + { id: ControlListModel.PREVIOUS_BUTTON, label: VLCIcons.previous, text: i18n.qtr("Previous")}, + { id: ControlListModel.NEXT_BUTTON, label: VLCIcons.next, text: i18n.qtr("Next")}, + { id: ControlListModel.SLOWER_BUTTON, label: VLCIcons.slower, text: i18n.qtr("Slower")}, + { id: ControlListModel.FASTER_BUTTON, label: VLCIcons.faster, text: i18n.qtr("Faster")}, + { id: ControlListModel.FULLSCREEN_BUTTON, label: VLCIcons.fullscreen, text: i18n.qtr("Fullscreen")}, + { id: ControlListModel.EXTENDED_BUTTON, label: VLCIcons.extended, text: i18n.qtr("Extended panel")}, + { id: ControlListModel.PLAYLIST_BUTTON, label: VLCIcons.playlist, text: i18n.qtr("Playlist")}, + { id: ControlListModel.SNAPSHOT_BUTTON, label: VLCIcons.snapshot, text: i18n.qtr("Snapshot")}, + { id: ControlListModel.RECORD_BUTTON, label: VLCIcons.record, text: i18n.qtr("Record")}, + { id: ControlListModel.ATOB_BUTTON, label: VLCIcons.atob, text: i18n.qtr("A-B Loop")}, + { id: ControlListModel.FRAME_BUTTON, label: VLCIcons.frame_by_frame, text: i18n.qtr("Frame By Frame")}, + { id: ControlListModel.SKIP_BACK_BUTTON, label: VLCIcons.skip_back, text: i18n.qtr("Step backward")}, + { id: ControlListModel.SKIP_FW_BUTTON, label: VLCIcons.skip_for, text: i18n.qtr("Step forward")}, + { id: ControlListModel.QUIT_BUTTON, label: VLCIcons.clear, text: i18n.qtr("Quit")}, + { id: ControlListModel.RANDOM_BUTTON, label: VLCIcons.shuffle_on, text: i18n.qtr("Random")}, + { id: ControlListModel.LOOP_BUTTON, label: VLCIcons.repeat_all, text: i18n.qtr("Loop")}, + { id: ControlListModel.INFO_BUTTON, label: VLCIcons.info, text: i18n.qtr("Information")}, + { id: ControlListModel.LANG_BUTTON, label: VLCIcons.audiosub, text: i18n.qtr("Open subtitles")}, + { id: ControlListModel.MENU_BUTTON, label: VLCIcons.menu, text: i18n.qtr("Menu Button")}, + { id: ControlListModel.BACK_BUTTON, label: VLCIcons.exit, text: i18n.qtr("Back Button")}, + { id: ControlListModel.CHAPTER_PREVIOUS_BUTTON, label: VLCIcons.dvd_prev, text: i18n.qtr("Previous chapter")}, + { id: ControlListModel.CHAPTER_NEXT_BUTTON, label: VLCIcons.dvd_next, text: i18n.qtr("Next chapter")}, + { id: ControlListModel.VOLUME, label: VLCIcons.volume_high, text: i18n.qtr("Volume Widget")}, + { id: ControlListModel.TELETEXT_BUTTONS, label: VLCIcons.tvtelx, text: i18n.qtr("Teletext")}, + { id: ControlListModel.ASPECT_RATIO_COMBOBOX, label: VLCIcons.aspect_ratio, text: i18n.qtr("Aspect Ratio")}, + { id: ControlListModel.WIDGET_SPACER, label: VLCIcons.space, text: i18n.qtr("Spacer")}, + { id: ControlListModel.WIDGET_SPACER_EXTEND, label: VLCIcons.space, text: i18n.qtr("Expanding Spacer")}, + { id: ControlListModel.PLAYER_SWITCH_BUTTON, label: VLCIcons.fullscreen, text: i18n.qtr("Switch Player")}, + { id: ControlListModel.ARTWORK_INFO, label: VLCIcons.info, text: i18n.qtr("Artwork Info")} ] function returnbuttondelegate(inpID){ switch (inpID){ - case PlayerControlBarModel.RANDOM_BUTTON: return randomBtnDelegate - case PlayerControlBarModel.PREVIOUS_BUTTON: return prevBtnDelegate - case PlayerControlBarModel.PLAY_BUTTON: return playBtnDelegate - case PlayerControlBarModel.NEXT_BUTTON: return nextBtnDelegate - case PlayerControlBarModel.LOOP_BUTTON: return repeatBtnDelegate - case PlayerControlBarModel.LANG_BUTTON: return langBtnDelegate - case PlayerControlBarModel.PLAYLIST_BUTTON:return playlistBtnDelegate - case PlayerControlBarModel.MENU_BUTTON:return menuBtnDelegate - case PlayerControlBarModel.CHAPTER_PREVIOUS_BUTTON:return chapterPreviousBtnDelegate - case PlayerControlBarModel.CHAPTER_NEXT_BUTTON:return chapterNextBtnDelegate - case PlayerControlBarModel.BACK_BUTTON:return backBtnDelegate - case PlayerControlBarModel.WIDGET_SPACER:return spacerDelegate - case PlayerControlBarModel.WIDGET_SPACER_EXTEND:return extendiblespacerDelegate - case PlayerControlBarModel.RECORD_BUTTON: return recordBtnDelegate - case PlayerControlBarModel.FULLSCREEN_BUTTON: return fullScreenBtnDelegate - case PlayerControlBarModel.ATOB_BUTTON: return toggleABloopstateDelegate - case PlayerControlBarModel.SNAPSHOT_BUTTON: return snapshotBtnDelegate - case PlayerControlBarModel.STOP_BUTTON: return stopBtndelgate - case PlayerControlBarModel.INFO_BUTTON: return mediainfoBtnDelegate - case PlayerControlBarModel.FRAME_BUTTON: return framebyframeDelegate - case PlayerControlBarModel.FASTER_BUTTON: return fasterBtnDelegate - case PlayerControlBarModel.SLOWER_BUTTON: return slowerBtnDelegate - case PlayerControlBarModel.OPEN_BUTTON: return openmediaBtnDelegate - case PlayerControlBarModel.EXTENDED_BUTTON: return extdSettingsBtnDelegate - case PlayerControlBarModel.SKIP_FW_BUTTON: return stepFwdBtnDelegate - case PlayerControlBarModel.SKIP_BACK_BUTTON: return stepBackBtnDelegate - case PlayerControlBarModel.QUIT_BUTTON: return quitBtnDelegate - case PlayerControlBarModel.VOLUME: return volumeBtnDelegate - case PlayerControlBarModel.ASPECT_RATIO_COMBOBOX: return aspectRatioDelegate - case PlayerControlBarModel.TELETEXT_BUTTONS: return teletextdelegate - case PlayerControlBarModel.PLAYER_SWITCH_BUTTON: return playerSwitchBtnDelegate - case PlayerControlBarModel.ARTWORK_INFO: return artworkInfoDelegate + case ControlListModel.RANDOM_BUTTON: return randomBtnDelegate + case ControlListModel.PREVIOUS_BUTTON: return prevBtnDelegate + case ControlListModel.PLAY_BUTTON: return playBtnDelegate + case ControlListModel.NEXT_BUTTON: return nextBtnDelegate + case ControlListModel.LOOP_BUTTON: return repeatBtnDelegate + case ControlListModel.LANG_BUTTON: return langBtnDelegate + case ControlListModel.PLAYLIST_BUTTON:return playlistBtnDelegate + case ControlListModel.MENU_BUTTON:return menuBtnDelegate + case ControlListModel.CHAPTER_PREVIOUS_BUTTON:return chapterPreviousBtnDelegate + case ControlListModel.CHAPTER_NEXT_BUTTON:return chapterNextBtnDelegate + case ControlListModel.BACK_BUTTON:return backBtnDelegate + case ControlListModel.WIDGET_SPACER:return spacerDelegate + case ControlListModel.WIDGET_SPACER_EXTEND:return extendiblespacerDelegate + case ControlListModel.RECORD_BUTTON: return recordBtnDelegate + case ControlListModel.FULLSCREEN_BUTTON: return fullScreenBtnDelegate + case ControlListModel.ATOB_BUTTON: return toggleABloopstateDelegate + case ControlListModel.SNAPSHOT_BUTTON: return snapshotBtnDelegate + case ControlListModel.STOP_BUTTON: return stopBtndelgate + case ControlListModel.INFO_BUTTON: return mediainfoBtnDelegate + case ControlListModel.FRAME_BUTTON: return framebyframeDelegate + case ControlListModel.FASTER_BUTTON: return fasterBtnDelegate + case ControlListModel.SLOWER_BUTTON: return slowerBtnDelegate + case ControlListModel.OPEN_BUTTON: return openmediaBtnDelegate + case ControlListModel.EXTENDED_BUTTON: return extdSettingsBtnDelegate + case ControlListModel.SKIP_FW_BUTTON: return stepFwdBtnDelegate + case ControlListModel.SKIP_BACK_BUTTON: return stepBackBtnDelegate + case ControlListModel.QUIT_BUTTON: return quitBtnDelegate + case ControlListModel.VOLUME: return volumeBtnDelegate + case ControlListModel.ASPECT_RATIO_COMBOBOX: return aspectRatioDelegate + case ControlListModel.TELETEXT_BUTTONS: return teletextdelegate + case ControlListModel.PLAYER_SWITCH_BUTTON: return playerSwitchBtnDelegate + case ControlListModel.ARTWORK_INFO: return artworkInfoDelegate } console.log("button delegate id " + inpID + " doesn't exists") return spacerDelegate diff --git a/modules/gui/qt/player/qml/MiniPlayer.qml b/modules/gui/qt/player/qml/MiniPlayer.qml index 748751da31a6f975501cde1aab4e87c39a797784..9cc983dc05218168c4bc48019338ca226c4e0628 100644 --- a/modules/gui/qt/player/qml/MiniPlayer.qml +++ b/modules/gui/qt/player/qml/MiniPlayer.qml @@ -83,7 +83,7 @@ Widgets.NavigableFocusScope { sliderHeight: VLCStyle.dp(3, VLCStyle.scale) sliderBackgroundColor: colors.sliderBarMiniplayerBgColor sliderProgressColor: colors.accent - configs: ["MiniPlayerToolbar-left", "MiniPlayerToolbar-center", "MiniPlayerToolbar-right"] + identifier: "MiniPlayer" navigationParent: root Keys.onPressed: { diff --git a/modules/gui/qt/player/qml/Player.qml b/modules/gui/qt/player/qml/Player.qml index e7190fcd346495700c3ff033fd5388f76e6bcdf3..ebe95f57920e6ae649f0c80c2aa97a7061efbb2e 100644 --- a/modules/gui/qt/player/qml/Player.qml +++ b/modules/gui/qt/player/qml/Player.qml @@ -499,6 +499,8 @@ Widgets.NavigableFocusScope { navigationUpItem: playlistpopup.showPlaylist ? playlistpopup : (audioControls.visible ? audioControls : topcontrolView) onRequestLockUnlockAutoHide: rootPlayer.lockUnlockAutoHide(lock, source) + + identifier: "MainPlayer" } } } diff --git a/modules/gui/qt/player/qml/PlayerButtonsLayout.qml b/modules/gui/qt/player/qml/PlayerButtonsLayout.qml index b6016988cd3352f510729371590e356eaee7cf34..15f11ae2938af6842b9e9ebc6781da1de5ec3060 100644 --- a/modules/gui/qt/player/qml/PlayerButtonsLayout.qml +++ b/modules/gui/qt/player/qml/PlayerButtonsLayout.qml @@ -42,24 +42,20 @@ Widgets.NavigableFocusScope { property real spacing: VLCStyle.margin_normal // spacing between controls property real layoutSpacing: VLCStyle.margin_xlarge // spacing between layouts (left, center, and right) - signal requestLockUnlockAutoHide(bool lock, var source) - - enum Alignment { - Left = 0, - Center = 1, - Right = 2 + property string identifier + readonly property var model: { + if (!!mainInterface.controlbarProfileModel.currentModel) + mainInterface.controlbarProfileModel.currentModel.getModel(identifier) + else + undefined } - property var models: [] + signal requestLockUnlockAutoHide(bool lock, var source) - Connections { - target: mainInterface - onToolBarConfUpdated: { - models[PlayerButtonsLayout.Alignment.Left].reloadModel() - models[PlayerButtonsLayout.Alignment.Center].reloadModel() - models[PlayerButtonsLayout.Alignment.Right].reloadModel() - } + Component.onCompleted: { + console.assert(!!identifier) + console.assert(identifier.length > 0) } ControlButtons { @@ -70,13 +66,9 @@ Widgets.NavigableFocusScope { onRequestLockUnlockAutoHide: playerButtonsLayout.requestLockUnlockAutoHide(lock, source) } - ButtonsLayout { + Loader { id: buttonrow_left - model: models[PlayerButtonsLayout.Alignment.Left] - - extraWidth: (buttonrow_center.x - buttonrow_left.x - minimumWidth - layoutSpacing) - anchors { left: parent.left verticalCenter: parent.verticalCenter @@ -86,20 +78,26 @@ Widgets.NavigableFocusScope { bottomMargin: marginBottom rightMargin: layoutSpacing } - - visible: extraWidth < 0 ? false : true // extraWidth < 0 means there is not even available space for minimumSize - navigationParent: playerButtonsLayout - navigationRightItem: buttonrow_center + active: !!playerButtonsLayout.model && !!playerButtonsLayout.model.left - focus: true + sourceComponent: ButtonsLayout { + model: playerButtonsLayout.model.left + + extraWidth: (buttonrow_center.x - buttonrow_left.x - minimumWidth - layoutSpacing) + + visible: extraWidth < 0 ? false : true // extraWidth < 0 means there is not even available space for minimumSize + + navigationParent: playerButtonsLayout + navigationRightItem: buttonrow_center + + focus: true + } } - ButtonsLayout { + Loader { id: buttonrow_center - model: models[PlayerButtonsLayout.Alignment.Center] - anchors { centerIn: parent @@ -107,17 +105,19 @@ Widgets.NavigableFocusScope { bottomMargin: playerButtonsLayout.marginBottom } - navigationParent: playerButtonsLayout - navigationLeftItem: buttonrow_left - navigationRightItem: buttonrow_right - } + active: !!playerButtonsLayout.model && !!playerButtonsLayout.model.center - ButtonsLayout { - id: buttonrow_right + sourceComponent: ButtonsLayout { + model: playerButtonsLayout.model.center - model: models[PlayerButtonsLayout.Alignment.Right] + navigationParent: playerButtonsLayout + navigationLeftItem: buttonrow_left + navigationRightItem: buttonrow_right + } + } - extraWidth: (playerButtonsLayout.width - (buttonrow_center.x + buttonrow_center.width) - minimumWidth - (2 * layoutSpacing)) + Loader { + id: buttonrow_right anchors { right: parent.right @@ -129,9 +129,19 @@ Widgets.NavigableFocusScope { leftMargin: layoutSpacing } - visible: extraWidth < 0 ? false : true // extraWidth < 0 means there is not even available space for minimumSize + active: !!playerButtonsLayout.model && !!playerButtonsLayout.model.right + + sourceComponent: ButtonsLayout { - navigationParent: playerButtonsLayout - navigationLeftItem: buttonrow_center + + model: playerButtonsLayout.model.right + + extraWidth: (playerButtonsLayout.width - (buttonrow_center.x + buttonrow_center.width) - minimumWidth - (2 * layoutSpacing)) + + visible: extraWidth < 0 ? false : true // extraWidth < 0 means there is not even available space for minimumSize + + navigationParent: playerButtonsLayout + navigationLeftItem: buttonrow_center + } } } diff --git a/modules/gui/qt/player/qml/TopBar.qml b/modules/gui/qt/player/qml/TopBar.qml index 4aa53d15a8e89a17cdb6ae6163ee39e8bd1e7c3f..c621c63f1f80a8671e465f7b73bc27fa49ac346d 100644 --- a/modules/gui/qt/player/qml/TopBar.qml +++ b/modules/gui/qt/player/qml/TopBar.qml @@ -269,7 +269,7 @@ Widgets.NavigableFocusScope{ Widgets.IconToolButton { id: playlistButton - objectName: PlayerControlBarModel.PLAYLIST_BUTTON + objectName: ControlListModel.PLAYLIST_BUTTON size: VLCStyle.banner_icon_size iconText: VLCIcons.playlist text: i18n.qtr("Playlist")