1
0
mirror of https://github.com/AvengeMedia/DankMaterialShell.git synced 2026-06-07 19:59:14 -04:00

feat(PluginBrowser): add sorting and filtering options for plugins

- Introduced sorting and filtering by installed, default, category, name, and author
This commit is contained in:
purian23
2026-05-31 23:58:13 -04:00
parent a44bef5796
commit 0c3659a612
3 changed files with 425 additions and 8 deletions
+16
View File
@@ -154,6 +154,8 @@ Singleton {
property var trayItemOrder: [] property var trayItemOrder: []
property var recentColors: [] property var recentColors: []
property bool showThirdPartyPlugins: false property bool showThirdPartyPlugins: false
property bool pluginBrowserInstalledFirst: false
property string pluginBrowserSortMode: "default"
property string launchPrefix: "" property string launchPrefix: ""
property string lastBrightnessDevice: "" property string lastBrightnessDevice: ""
property var brightnessExponentialDevices: ({}) property var brightnessExponentialDevices: ({})
@@ -964,6 +966,20 @@ Singleton {
saveSettings(); saveSettings();
} }
function setPluginBrowserInstalledFirst(enabled) {
pluginBrowserInstalledFirst = enabled;
saveSettings();
}
function setPluginBrowserSortMode(mode) {
if (mode === "type" || mode === "contributor")
mode = "author";
if (mode !== "default" && mode !== "name" && mode !== "author" && mode !== "category")
mode = "default";
pluginBrowserSortMode = mode;
saveSettings();
}
function setLaunchPrefix(prefix) { function setLaunchPrefix(prefix) {
launchPrefix = prefix; launchPrefix = prefix;
saveSettings(); saveSettings();
@@ -56,6 +56,8 @@ var SPEC = {
trayItemOrder: { def: [] }, trayItemOrder: { def: [] },
recentColors: { def: [] }, recentColors: { def: [] },
showThirdPartyPlugins: { def: false }, showThirdPartyPlugins: { def: false },
pluginBrowserInstalledFirst: { def: false },
pluginBrowserSortMode: { def: "default" },
launchPrefix: { def: "" }, launchPrefix: { def: "" },
lastBrightnessDevice: { def: "" }, lastBrightnessDevice: { def: "" },
+407 -8
View File
@@ -1,5 +1,6 @@
import QtQuick import QtQuick
import QtQuick.Controls import QtQuick.Controls
import QtQuick.Layouts
import Quickshell import Quickshell
import qs.Common import qs.Common
import qs.Modals.Common import qs.Modals.Common
@@ -20,9 +21,179 @@ FloatingWindow {
parentWindow: parentModal parentWindow: parentModal
property bool pendingInstallHandled: false property bool pendingInstallHandled: false
property string typeFilter: "" property string typeFilter: ""
property string categoryFilter: "all"
property var categoryFilterOptions: []
property var availableLetters: []
readonly property bool activeCategorySort: normalizedSortMode(SessionData.pluginBrowserSortMode) === "category"
readonly property bool showCategoryFilters: activeCategorySort && categoryFilterOptions.length > 1
readonly property bool showLetterIndex: {
var mode = normalizedSortMode(SessionData.pluginBrowserSortMode);
return (mode === "name" || mode === "author") && availableLetters.length > 1;
}
readonly property var sortChipOptions: [
{ id: "installed", label: I18n.tr("Installed", "plugin browser filter chip"), toggle: true },
{ id: "default", label: I18n.tr("Default", "plugin browser sort option"), toggle: false },
{ id: "name", label: I18n.tr("Name", "plugin browser sort option"), toggle: false },
{ id: "author", label: I18n.tr("Contributor", "plugin browser sort option"), toggle: false },
{ id: "category", label: I18n.tr("Category", "plugin browser sort option"), toggle: false }
]
function normalizedSortMode(mode) {
if (mode === "type" || mode === "contributor")
return "author";
if (mode === "name" || mode === "author" || mode === "category")
return mode;
return "default";
}
function isSortChipSelected(chipId, toggle) {
if (toggle)
return SessionData.pluginBrowserInstalledFirst;
return normalizedSortMode(SessionData.pluginBrowserSortMode) === chipId;
}
function comparePluginName(a, b) {
var nameA = (a.name || "").toLowerCase();
var nameB = (b.name || "").toLowerCase();
if (nameA < nameB)
return -1;
if (nameA > nameB)
return 1;
return 0;
}
function comparePluginAuthor(a, b) {
var authorA = (a.author || "").toLowerCase() || "zzz";
var authorB = (b.author || "").toLowerCase() || "zzz";
if (authorA < authorB)
return -1;
if (authorA > authorB)
return 1;
return comparePluginName(a, b);
}
function comparePluginCategory(a, b) {
var catA = (a.category || "").toLowerCase() || "zzz";
var catB = (b.category || "").toLowerCase() || "zzz";
if (catA < catB)
return -1;
if (catA > catB)
return 1;
return comparePluginName(a, b);
}
function formatCategoryLabel(categoryKey) {
if (!categoryKey || categoryKey === "_uncategorized")
return I18n.tr("Uncategorized", "plugin browser category filter");
return categoryKey.charAt(0).toUpperCase() + categoryKey.slice(1);
}
function sortKeyForPlugin(plugin, mode) {
if (mode === "author")
return (plugin.author || "").trim();
if (mode === "category")
return formatCategoryLabel((plugin.category || "").toLowerCase() || "_uncategorized");
return (plugin.name || "").trim();
}
function buildCategoryFilterOptions(plugins) {
var counts = {};
for (var i = 0; i < plugins.length; i++) {
var cat = (plugins[i].category || "").toLowerCase();
if (!cat)
cat = "_uncategorized";
counts[cat] = (counts[cat] || 0) + 1;
}
var keys = Object.keys(counts).sort();
var options = [{
key: "all",
label: I18n.tr("All", "plugin browser category filter"),
count: plugins.length
}];
for (var j = 0; j < keys.length; j++) {
var key = keys[j];
options.push({
key: key,
label: formatCategoryLabel(key),
count: counts[key]
});
}
return options;
}
function categoryFilterDisplayLabel(option) {
return option.label + " (" + option.count + ")";
}
function categoryFilterLabelForKey(key) {
for (var i = 0; i < categoryFilterOptions.length; i++) {
if (categoryFilterOptions[i].key === key)
return categoryFilterDisplayLabel(categoryFilterOptions[i]);
}
return "";
}
function categoryFilterKeyForLabel(label) {
for (var i = 0; i < categoryFilterOptions.length; i++) {
if (categoryFilterDisplayLabel(categoryFilterOptions[i]) === label)
return categoryFilterOptions[i].key;
}
return "all";
}
function categoryFilterDropdownLabels() {
var labels = [];
for (var i = 0; i < categoryFilterOptions.length; i++)
labels.push(categoryFilterDisplayLabel(categoryFilterOptions[i]));
return labels;
}
function updateAvailableLetters(plugins) {
var mode = normalizedSortMode(SessionData.pluginBrowserSortMode);
if (mode !== "name" && mode !== "author") {
availableLetters = [];
return;
}
var letters = {};
for (var i = 0; i < plugins.length; i++) {
var key = sortKeyForPlugin(plugins[i], mode);
if (!key)
continue;
var letter = key.charAt(0).toUpperCase();
if (letter >= "A" && letter <= "Z")
letters[letter] = true;
}
availableLetters = Object.keys(letters).sort();
}
function refreshListLayout() {
if (!pluginBrowserList)
return;
pluginBrowserList.savedY = 0;
pluginBrowserList.cancelFlick();
pluginBrowserList.contentY = 0;
Qt.callLater(() => {
if (pluginBrowserList)
pluginBrowserList.forceLayout();
});
}
function scrollToLetter(letter) {
var mode = normalizedSortMode(SessionData.pluginBrowserSortMode);
for (var i = 0; i < filteredPlugins.length; i++) {
var key = sortKeyForPlugin(filteredPlugins[i], mode);
if (key && key.charAt(0).toUpperCase() === letter) {
pluginBrowserList.positionViewAtIndex(i, ListView.Beginning);
pluginBrowserList.savedY = pluginBrowserList.contentY;
return;
}
}
}
function updateFilteredPlugins() { function updateFilteredPlugins() {
var filtered = []; var baseFiltered = [];
var query = searchQuery ? searchQuery.toLowerCase() : ""; var query = searchQuery ? searchQuery.toLowerCase() : "";
for (var i = 0; i < allPlugins.length; i++) { for (var i = 0; i < allPlugins.length; i++) {
@@ -38,7 +209,7 @@ FloatingWindow {
} }
if (query.length === 0) { if (query.length === 0) {
filtered.push(plugin); baseFiltered.push(plugin);
continue; continue;
} }
@@ -47,20 +218,58 @@ FloatingWindow {
var author = plugin.author ? plugin.author.toLowerCase() : ""; var author = plugin.author ? plugin.author.toLowerCase() : "";
if (name.indexOf(query) !== -1 || description.indexOf(query) !== -1 || author.indexOf(query) !== -1) if (name.indexOf(query) !== -1 || description.indexOf(query) !== -1 || author.indexOf(query) !== -1)
filtered.push(plugin); baseFiltered.push(plugin);
}
categoryFilterOptions = buildCategoryFilterOptions(baseFiltered);
if (categoryFilter !== "all") {
var filterStillValid = false;
for (var c = 0; c < categoryFilterOptions.length; c++) {
if (categoryFilterOptions[c].key === categoryFilter) {
filterStillValid = true;
break;
}
}
if (!filterStillValid)
categoryFilter = "all";
}
var filtered = baseFiltered.slice();
if (activeCategorySort && categoryFilter !== "all") {
filtered = filtered.filter(p => {
var cat = (p.category || "").toLowerCase();
if (!cat)
cat = "_uncategorized";
return cat === categoryFilter;
});
} }
filtered.sort((a, b) => { filtered.sort((a, b) => {
if (SessionData.pluginBrowserInstalledFirst) {
var instA = a.installed || false;
var instB = b.installed || false;
if (instA !== instB)
return instA ? -1 : 1;
}
var sortMode = normalizedSortMode(SessionData.pluginBrowserSortMode);
if (sortMode === "name")
return comparePluginName(a, b);
if (sortMode === "author")
return comparePluginAuthor(a, b);
if (sortMode === "category")
return comparePluginCategory(a, b);
if (a.featured !== b.featured) if (a.featured !== b.featured)
return a.featured ? -1 : 1; return a.featured ? -1 : 1;
if (a.firstParty !== b.firstParty) if (a.firstParty !== b.firstParty)
return a.firstParty ? -1 : 1; return a.firstParty ? -1 : 1;
return 0; return comparePluginName(a, b);
}); });
filteredPlugins = filtered; filteredPlugins = filtered;
updateAvailableLetters(filtered);
selectedIndex = -1; selectedIndex = -1;
keyboardNavigationActive = false; keyboardNavigationActive = false;
refreshListLayout();
} }
function selectNext() { function selectNext() {
@@ -355,11 +564,161 @@ FloatingWindow {
} }
Item { Item {
id: listArea id: sortControlsRow
anchors.left: parent.left anchors.left: parent.left
anchors.right: parent.right anchors.right: parent.right
anchors.top: browserSearchField.bottom anchors.top: browserSearchField.bottom
anchors.topMargin: Theme.spacingM anchors.topMargin: Theme.spacingM
height: sortControlsLayout.implicitHeight
RowLayout {
id: sortControlsLayout
anchors.left: parent.left
anchors.right: parent.right
spacing: Theme.spacingS
Repeater {
model: root.sortChipOptions
Rectangle {
id: sortChip
required property var modelData
required property int index
Layout.fillWidth: true
Layout.preferredHeight: 32
Layout.maximumHeight: 32
property bool selected: root.isSortChipSelected(modelData.id, modelData.toggle)
property bool hovered: chipMouseArea.containsMouse
property bool pressed: chipMouseArea.pressed
implicitWidth: chipContent.implicitWidth + Theme.spacingM * 2
radius: height / 2
color: selected ? Theme.primary : Theme.surfaceVariant
Behavior on color {
ColorAnimation {
duration: Theme.shortDuration
easing.type: Theme.standardEasing
}
}
Rectangle {
anchors.fill: parent
radius: parent.radius
color: {
if (pressed)
return sortChip.selected ? Theme.primaryPressed : Theme.surfaceTextHover;
if (hovered)
return sortChip.selected ? Theme.primaryHover : Theme.surfaceTextHover;
return "transparent";
}
Behavior on color {
ColorAnimation {
duration: Theme.shorterDuration
easing.type: Theme.standardEasing
}
}
}
DankRipple {
id: chipRipple
cornerRadius: sortChip.radius
rippleColor: sortChip.selected ? Theme.primaryText : Theme.surfaceVariantText
}
Row {
id: chipContent
anchors.centerIn: parent
spacing: Theme.spacingXS
DankIcon {
name: modelData.toggle ? "download_done" : "check"
size: 16
anchors.verticalCenter: parent.verticalCenter
color: Theme.primaryText
visible: sortChip.selected
}
StyledText {
text: modelData.label
font.pixelSize: Theme.fontSizeSmall
font.weight: sortChip.selected ? Font.Medium : Font.Normal
color: sortChip.selected ? Theme.primaryText : Theme.surfaceVariantText
anchors.verticalCenter: parent.verticalCenter
}
}
MouseArea {
id: chipMouseArea
anchors.fill: parent
hoverEnabled: true
cursorShape: Qt.PointingHandCursor
onPressed: mouse => chipRipple.trigger(mouse.x, mouse.y)
onClicked: {
if (modelData.toggle) {
SessionData.setPluginBrowserInstalledFirst(!SessionData.pluginBrowserInstalledFirst);
} else {
if (modelData.id !== "category")
root.categoryFilter = "all";
SessionData.setPluginBrowserSortMode(modelData.id);
}
root.updateFilteredPlugins();
}
}
}
}
}
}
Item {
id: categoryFiltersRow
anchors.left: parent.left
anchors.right: parent.right
anchors.top: sortControlsRow.bottom
anchors.topMargin: root.showCategoryFilters ? Theme.spacingS : 0
height: root.showCategoryFilters ? 40 : 0
visible: root.showCategoryFilters
clip: true
RowLayout {
anchors.fill: parent
spacing: Theme.spacingS
StyledText {
id: categoryFilterLabel
text: I18n.tr("Filter", "plugin browser category filter label")
font.pixelSize: Theme.fontSizeSmall
color: Theme.outline
Layout.alignment: Qt.AlignVCenter
}
DankDropdown {
id: categoryFilterDropdown
Layout.fillWidth: true
Layout.preferredHeight: 32
compactMode: true
dropdownWidth: Math.max(240, categoryFiltersRow.width - categoryFilterLabel.implicitWidth - Theme.spacingS * 3)
currentValue: root.categoryFilterLabelForKey(root.categoryFilter)
options: root.categoryFilterDropdownLabels()
onValueChanged: value => {
var nextKey = root.categoryFilterKeyForLabel(value);
if (nextKey === root.categoryFilter)
return;
root.categoryFilter = nextKey;
root.updateFilteredPlugins();
}
}
}
}
Item {
id: listArea
anchors.left: parent.left
anchors.right: parent.right
anchors.top: categoryFiltersRow.bottom
anchors.topMargin: Theme.spacingM
anchors.bottom: parent.bottom anchors.bottom: parent.bottom
anchors.bottomMargin: Theme.spacingM anchors.bottomMargin: Theme.spacingM
@@ -401,17 +760,20 @@ FloatingWindow {
anchors.fill: parent anchors.fill: parent
anchors.leftMargin: Theme.spacingM anchors.leftMargin: Theme.spacingM
anchors.rightMargin: Theme.spacingM anchors.rightMargin: root.showLetterIndex ? Theme.spacingM + 18 : Theme.spacingM
anchors.topMargin: Theme.spacingS anchors.topMargin: Theme.spacingS
anchors.bottomMargin: Theme.spacingS anchors.bottomMargin: Theme.spacingS
spacing: Theme.spacingS spacing: Theme.spacingS
model: ScriptModel { model: ScriptModel {
values: root.filteredPlugins values: root.filteredPlugins
objectProp: "id"
} }
clip: true clip: true
visible: !root.isLoading visible: !root.isLoading
add: null add: null
remove: null
displaced: null displaced: null
move: null
ScrollBar.vertical: DankScrollbar { ScrollBar.vertical: DankScrollbar {
id: browserScrollbar id: browserScrollbar
@@ -675,13 +1037,13 @@ FloatingWindow {
color: Theme.outline color: Theme.outline
width: parent.width width: parent.width
wrapMode: Text.WordWrap wrapMode: Text.WordWrap
visible: modelData.description && modelData.description.length > 0 visible: (modelData.description || "").length > 0
} }
Flow { Flow {
width: parent.width width: parent.width
spacing: Theme.spacingXS spacing: Theme.spacingXS
visible: modelData.capabilities && modelData.capabilities.length > 0 visible: (modelData.capabilities || []).length > 0
Repeater { Repeater {
model: modelData.capabilities || [] model: modelData.capabilities || []
@@ -708,6 +1070,43 @@ FloatingWindow {
} }
} }
Column {
id: letterIndex
anchors.right: parent.right
anchors.top: pluginBrowserList.top
anchors.bottom: pluginBrowserList.bottom
anchors.rightMargin: Theme.spacingXS
width: 16
visible: root.showLetterIndex && !root.isLoading
spacing: 0
Repeater {
model: root.availableLetters
Item {
required property string modelData
width: letterIndex.width
height: Math.max(12, letterIndex.height / Math.max(1, root.availableLetters.length))
StyledText {
anchors.centerIn: parent
text: modelData
font.pixelSize: 10
font.weight: Font.Medium
color: letterMouseArea.containsMouse ? Theme.primary : Theme.outline
}
MouseArea {
id: letterMouseArea
anchors.fill: parent
hoverEnabled: true
cursorShape: Qt.PointingHandCursor
onClicked: root.scrollToLetter(modelData)
}
}
}
}
StyledText { StyledText {
anchors.centerIn: listArea anchors.centerIn: listArea
text: I18n.tr("No plugins found", "empty plugin list") text: I18n.tr("No plugins found", "empty plugin list")