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:
@@ -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: "" },
|
||||||
|
|
||||||
|
|||||||
@@ -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")
|
||||||
|
|||||||
Reference in New Issue
Block a user