/* SPDX-FileCopyrightText: 2007 Aaron Seigo SPDX-FileCopyrightText: 2008 Ménard Alexis SPDX-FileCopyrightText: 2009 Chani Armitage SPDX-FileCopyrightText: 2012 Marco Martin SPDX-License-Identifier: LGPL-2.0-or-later */ #include "private/containment_p.h" #include #include #include "config-plasma.h" #include "pluginloader.h" #include #include "debug_p.h" #include "private/applet_p.h" namespace Plasma { const char ContainmentPrivate::defaultWallpaperPlugin[] = "org.kde.image"; ContainmentPrivate::ContainmentPrivate(Containment *c) : q(c) , formFactor(Types::Planar) , location(Types::Floating) , lastScreen(-1) , type(Plasma::Containment::Type::NoContainment) // never had a screen , uiReady(false) , appletsUiReady(false) { // if the parent is an applet (i.e we are the systray) // we want to follow screen changed signals from the parent's containment auto appletParent = qobject_cast(c->parent()); if (appletParent) { QObject::connect(appletParent->containment(), &Containment::screenChanged, c, &Containment::screenChanged); } activityInfo = new KActivities::Info(activityId, q); QObject::connect(activityInfo, &KActivities::Info::nameChanged, q, &Containment::activityNameChanged); } Plasma::ContainmentPrivate::~ContainmentPrivate() { applets.clear(); } void ContainmentPrivate::addDefaultActions(QMap &actions, Containment *c, Corona *cor) { // adjust applet actions QAction *appAction = actions.value(QStringLiteral("remove")); appAction->setShortcut(QKeySequence(Qt::ALT | Qt::Key_D, Qt::ALT | Qt::Key_R)); if (c && c->d->isPanelContainment()) { appAction->setText(i18n("Remove this Panel")); } else { appAction->setText(i18n("Remove this Activity")); } appAction = actions.value(QStringLiteral("configure")); if (appAction) { appAction->setShortcut(QKeySequence(Qt::ALT | Qt::Key_D, Qt::ALT | Qt::Key_S)); appAction->setText(i18n("Activity Settings")); } // add our own actions QAction *appletBrowserAction; if (c) { appletBrowserAction = new QAction(c); } else { appletBrowserAction = new QAction(cor); } actions[QStringLiteral("add widgets")] = appletBrowserAction; appletBrowserAction->setAutoRepeat(false); appletBrowserAction->setText(i18n("Add or Manage Widgets…")); appletBrowserAction->setIcon(QIcon::fromTheme(QStringLiteral("view-group-symbolic"))); appletBrowserAction->setShortcut(QKeySequence(Qt::ALT | Qt::Key_D, Qt::Key_A)); if (c) { static_cast(c)->d->actions.insert(actions); } } KConfigGroup ContainmentPrivate::containmentActionsConfig() const { KConfigGroup cfg = KConfigGroup(q->corona()->config(), QStringLiteral("ActionPlugins")); return KConfigGroup(&cfg, QString::number((int)type)); } void ContainmentPrivate::configChanged() { KConfigGroup group = q->config(); q->setWallpaperPlugin(group.readEntry("wallpaperplugin", defaultWallpaperPlugin)); } void ContainmentPrivate::checkStatus(Plasma::Types::ItemStatus appletStatus) { // qCDebug(LOG_PLASMA) << "================== "<< appletStatus << q->status(); if (appletStatus == q->status()) { return; } if (appletStatus < q->status() || appletStatus == Plasma::Types::HiddenStatus) { // check to see if any other applet has a higher status, and stick with that if we do // we'll treat HiddenStatus as lowest as we cannot change the enum value which is highest anymore for (Applet *applet : std::as_const(applets)) { if (applet->status() > appletStatus && applet->status() != Plasma::Types::HiddenStatus) { appletStatus = applet->status(); } } } if (appletStatus != Plasma::Types::HiddenStatus) { q->setStatus(appletStatus); } } void ContainmentPrivate::triggerShowAddWidgets() { Q_EMIT q->showAddWidgetsInterface(QPointF()); } void ContainmentPrivate::containmentConstraintsEvent(Applet::Constraints constraints) { if (!q->isContainment()) { return; } // qCDebug(LOG_PLASMA) << "got containmentConstraintsEvent" << constraints; if (constraints & Applet::ImmutableConstraint) { // update actions const bool unlocked = q->immutability() == Types::Mutable; QAction *action = q->internalAction(QStringLiteral("remove")); if (action) { action->setEnabled(unlocked); action->setVisible(unlocked); } action = q->internalAction(QStringLiteral("add widgets")); if (action) { action->setEnabled(unlocked); action->setVisible(unlocked); } // tell the applets too for (Applet *a : std::as_const(applets)) { /*Why qMin? * the applets immutability() is the maximum between internal applet immutability * and the immutability of its containment. * so not set higher immutability in the internal member of Applet * or the applet will not be able to be unlocked properly */ a->setImmutability(qMin(q->immutability(), a->d->immutability)); a->updateConstraints(Applet::ImmutableConstraint); } } // pass on the constraints that are relevant here Applet::Constraints appletConstraints = Applet::NoConstraint; if (constraints & Applet::FormFactorConstraint) { appletConstraints |= Applet::FormFactorConstraint; } if (constraints & Applet::ScreenConstraint) { appletConstraints |= Applet::ScreenConstraint; } if (appletConstraints != Applet::NoConstraint) { for (Applet *applet : std::as_const(applets)) { applet->updateConstraints(appletConstraints); } } } Applet *ContainmentPrivate::createApplet(const QString &name, const QVariantList &args, uint id, const QRectF &geometryHint) { if (!q->isContainment()) { return nullptr; } if (q->immutability() != Types::Mutable && !args.contains(QVariant::fromValue(QStringLiteral("org.kde.plasma:force-create")))) { #ifndef NDEBUG // qCDebug(LOG_PLASMA) << "addApplet for" << name << "requested, but we're currently immutable!"; #endif return nullptr; } Applet *applet = PluginLoader::self()->loadApplet(name, id, args); if (!applet) { qCWarning(LOG_PLASMA) << "Applet" << name << "could not be loaded."; applet = new Applet(nullptr, KPluginMetaData(), QVariantList{QVariant(), id}); applet->setLaunchErrorMessage(i18n("Could not find requested component: %1", name)); } q->addApplet(applet, geometryHint); // mirror behavior of resorecontents: if an applet is not valid, set it immediately to uiReady if (!applet->pluginMetaData().isValid()) { applet->updateConstraints(Applet::UiReadyConstraint); } return applet; } void ContainmentPrivate::appletDeleted(Plasma::Applet *applet) { Q_EMIT q->appletAboutToBeRemoved(applet); applets.removeAll(applet); Q_EMIT q->appletRemoved(applet); Q_EMIT q->appletsChanged(); Q_EMIT q->configNeedsSaving(); } bool ContainmentPrivate::isPanelContainment() const { return type == Plasma::Containment::Type::Panel || type == Plasma::Containment::Type::CustomPanel; } void ContainmentPrivate::setStarted() { if (!q->Applet::d->started) { q->Applet::d->started = true; if (uiReady) { Q_EMIT q->uiReadyChanged(true); } } } void ContainmentPrivate::setUiReady() { // if we are the containment and there is still some incomplete applet, we're still incomplete if (!uiReady) { uiReady = true; if (q->Applet::d->started && (appletsUiReady || applets.isEmpty()) && loadingApplets.isEmpty()) { Q_EMIT q->uiReadyChanged(true); } } } void ContainmentPrivate::appletLoaded(Applet *applet) { loadingApplets.remove(applet); if (loadingApplets.isEmpty() && !appletsUiReady) { appletsUiReady = true; if (q->Applet::d->started && uiReady) { Q_EMIT q->uiReadyChanged(true); } } } }