mirror of
https://github.com/AvengeMedia/DankMaterialShell.git
synced 2026-06-16 08:05:19 -04:00
Compare commits
3 Commits
hover
..
2026ba5bd2
| Author | SHA1 | Date | |
|---|---|---|---|
| 2026ba5bd2 | |||
| db56c8d74d | |||
| 9d1a81c93c |
@@ -3,7 +3,6 @@ pragma ComponentBehavior: Bound
|
|||||||
|
|
||||||
import Quickshell
|
import Quickshell
|
||||||
import QtQuick
|
import QtQuick
|
||||||
import qs.Common
|
|
||||||
|
|
||||||
Singleton {
|
Singleton {
|
||||||
id: root
|
id: root
|
||||||
@@ -17,76 +16,8 @@ Singleton {
|
|||||||
signal popoutOpening
|
signal popoutOpening
|
||||||
signal popoutChanged
|
signal popoutChanged
|
||||||
|
|
||||||
property real hoverCursorGlobalX: 0
|
|
||||||
property real hoverCursorGlobalY: 0
|
|
||||||
|
|
||||||
function updateHoverCursor(gx, gy) {
|
|
||||||
hoverCursorGlobalX = gx;
|
|
||||||
hoverCursorGlobalY = gy;
|
|
||||||
}
|
|
||||||
|
|
||||||
function cursorOverBar(gx, gy, padding) {
|
|
||||||
const pad = padding !== undefined ? padding : 16;
|
|
||||||
const bars = KeyboardFocus.barWindows || [];
|
|
||||||
for (let i = 0; i < bars.length; i++) {
|
|
||||||
const w = bars[i];
|
|
||||||
if (!w?.visible)
|
|
||||||
continue;
|
|
||||||
if (typeof w.containsGlobalPoint === "function") {
|
|
||||||
if (w.containsGlobalPoint(gx, gy, pad))
|
|
||||||
return true;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
const item = w.contentItem;
|
|
||||||
if (!item || typeof item.mapToItem !== "function")
|
|
||||||
continue;
|
|
||||||
const topLeft = item.mapToItem(null, 0, 0);
|
|
||||||
if (!topLeft)
|
|
||||||
continue;
|
|
||||||
if (gx >= topLeft.x - pad && gx < topLeft.x + item.width + pad && gy >= topLeft.y - pad && gy < topLeft.y + item.height + pad)
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
function _isPopoutPresented(popout) {
|
|
||||||
if (!popout)
|
|
||||||
return false;
|
|
||||||
try {
|
|
||||||
if (popout.dashVisible !== undefined)
|
|
||||||
return !!popout.dashVisible;
|
|
||||||
if (popout.notificationHistoryVisible !== undefined)
|
|
||||||
return !!popout.notificationHistoryVisible;
|
|
||||||
return !!(popout.shouldBeVisible || popout.isClosing);
|
|
||||||
} catch (e) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function _openPopout(popout) {
|
|
||||||
if (popout.dashVisible !== undefined) {
|
|
||||||
if (popout.dashVisible && !popout.shouldBeVisible && !popout.isClosing)
|
|
||||||
popout.dashVisible = false;
|
|
||||||
popout.dashVisible = true;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (popout.notificationHistoryVisible !== undefined) {
|
|
||||||
popout.notificationHistoryVisible = true;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
popout.open();
|
|
||||||
}
|
|
||||||
|
|
||||||
function _closePopout(popout) {
|
function _closePopout(popout) {
|
||||||
try {
|
try {
|
||||||
if (popout?.hoverDismissEnabled) {
|
|
||||||
if (typeof popout.closeFromHoverDismiss === "function") {
|
|
||||||
popout.closeFromHoverDismiss();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (popout.hoverDismissEnabled !== undefined)
|
|
||||||
popout.hoverDismissEnabled = false;
|
|
||||||
switch (true) {
|
switch (true) {
|
||||||
case popout.dashVisible !== undefined:
|
case popout.dashVisible !== undefined:
|
||||||
popout.dashVisible = false;
|
popout.dashVisible = false;
|
||||||
@@ -158,26 +89,7 @@ Singleton {
|
|||||||
continue;
|
continue;
|
||||||
_closePopout(popout);
|
_closePopout(popout);
|
||||||
}
|
}
|
||||||
// Keep map entries until each popout's close animation finishes (hidePopout).
|
currentPopoutsByScreen = {};
|
||||||
}
|
|
||||||
|
|
||||||
function closePopoutForScreen(screen) {
|
|
||||||
if (!screen)
|
|
||||||
return;
|
|
||||||
const screenName = screen.name;
|
|
||||||
const popout = currentPopoutsByScreen[screenName];
|
|
||||||
if (!popout || _isStale(popout)) {
|
|
||||||
currentPopoutsByScreen[screenName] = null;
|
|
||||||
currentPopoutTriggers[screenName] = null;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
_closePopout(popout);
|
|
||||||
}
|
|
||||||
|
|
||||||
function cancelHoverDismiss(screen) {
|
|
||||||
const popout = getActivePopout(screen);
|
|
||||||
if (popout?.cancelHoverDismiss)
|
|
||||||
popout.cancelHoverDismiss();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function getActivePopout(screen) {
|
function getActivePopout(screen) {
|
||||||
@@ -194,8 +106,6 @@ Singleton {
|
|||||||
function requestPopout(popout, tabIndex, triggerSource) {
|
function requestPopout(popout, tabIndex, triggerSource) {
|
||||||
if (!popout || !popout.screen)
|
if (!popout || !popout.screen)
|
||||||
return;
|
return;
|
||||||
if (popout.hoverDismissEnabled !== undefined)
|
|
||||||
popout.hoverDismissEnabled = false;
|
|
||||||
screenshotActive = false;
|
screenshotActive = false;
|
||||||
const screenName = popout.screen.name;
|
const screenName = popout.screen.name;
|
||||||
const currentPopout = currentPopoutsByScreen[screenName];
|
const currentPopout = currentPopoutsByScreen[screenName];
|
||||||
@@ -271,81 +181,16 @@ Singleton {
|
|||||||
ModalManager.closeAllModalsExcept(null);
|
ModalManager.closeAllModalsExcept(null);
|
||||||
}
|
}
|
||||||
|
|
||||||
_openPopout(popout);
|
if (movedFromOtherScreen) {
|
||||||
}
|
popout.open();
|
||||||
|
} else {
|
||||||
function requestHoverPopout(popout, tabIndex, triggerSource) {
|
if (popout.dashVisible !== undefined) {
|
||||||
if (!popout || !popout.screen)
|
popout.dashVisible = true;
|
||||||
return;
|
} else if (popout.notificationHistoryVisible !== undefined) {
|
||||||
screenshotActive = false;
|
popout.notificationHistoryVisible = true;
|
||||||
const screenName = popout.screen.name;
|
|
||||||
const currentPopout = currentPopoutsByScreen[screenName];
|
|
||||||
const triggerId = triggerSource !== undefined ? triggerSource : tabIndex;
|
|
||||||
|
|
||||||
const willOpen = !(currentPopout === popout && _isPopoutPresented(popout) && triggerId !== undefined && currentPopoutTriggers[screenName] === triggerId);
|
|
||||||
if (willOpen)
|
|
||||||
popoutOpening();
|
|
||||||
|
|
||||||
let movedFromOtherScreen = false;
|
|
||||||
for (const otherScreenName in currentPopoutsByScreen) {
|
|
||||||
if (otherScreenName === screenName)
|
|
||||||
continue;
|
|
||||||
const otherPopout = currentPopoutsByScreen[otherScreenName];
|
|
||||||
if (!otherPopout)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (_isStale(otherPopout)) {
|
|
||||||
currentPopoutsByScreen[otherScreenName] = null;
|
|
||||||
currentPopoutTriggers[otherScreenName] = null;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (otherPopout === popout) {
|
|
||||||
movedFromOtherScreen = true;
|
|
||||||
currentPopoutsByScreen[otherScreenName] = null;
|
|
||||||
currentPopoutTriggers[otherScreenName] = null;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
_closePopout(otherPopout);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (currentPopout && currentPopout !== popout) {
|
|
||||||
if (_isStale(currentPopout)) {
|
|
||||||
currentPopoutsByScreen[screenName] = null;
|
|
||||||
currentPopoutTriggers[screenName] = null;
|
|
||||||
} else {
|
} else {
|
||||||
_closePopout(currentPopout);
|
popout.open();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (currentPopout === popout && _isPopoutPresented(popout) && !movedFromOtherScreen) {
|
|
||||||
if (triggerId !== undefined && currentPopoutTriggers[screenName] === triggerId)
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (tabIndex !== undefined && popout.currentTabIndex !== undefined)
|
|
||||||
popout.currentTabIndex = tabIndex;
|
|
||||||
if (popout.updateSurfacePosition)
|
|
||||||
popout.updateSurfacePosition();
|
|
||||||
currentPopoutTriggers[screenName] = triggerId;
|
|
||||||
if (popout.hoverDismissEnabled !== undefined)
|
|
||||||
popout.hoverDismissEnabled = true;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
currentPopoutTriggers[screenName] = triggerId;
|
|
||||||
currentPopoutsByScreen[screenName] = popout;
|
|
||||||
popoutChanged();
|
|
||||||
|
|
||||||
if (tabIndex !== undefined && popout.currentTabIndex !== undefined)
|
|
||||||
popout.currentTabIndex = tabIndex;
|
|
||||||
|
|
||||||
if (currentPopout !== popout)
|
|
||||||
ModalManager.closeAllModalsExcept(null);
|
|
||||||
|
|
||||||
if (popout.hoverDismissEnabled !== undefined)
|
|
||||||
popout.hoverDismissEnabled = true;
|
|
||||||
|
|
||||||
_openPopout(popout);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -397,6 +397,7 @@ Singleton {
|
|||||||
property bool audioVisualizerEnabled: true
|
property bool audioVisualizerEnabled: true
|
||||||
property string audioScrollMode: "volume"
|
property string audioScrollMode: "volume"
|
||||||
property int audioWheelScrollAmount: 5
|
property int audioWheelScrollAmount: 5
|
||||||
|
property bool audioDeviceScrollVolumeEnabled: false
|
||||||
property bool clockCompactMode: false
|
property bool clockCompactMode: false
|
||||||
property int focusedWindowSize: 1
|
property int focusedWindowSize: 1
|
||||||
property bool focusedWindowCompactMode: false
|
property bool focusedWindowCompactMode: false
|
||||||
@@ -519,13 +520,39 @@ Singleton {
|
|||||||
property real notificationSummaryFontSize: Spec.SPEC.notificationSummaryFontSize.def
|
property real notificationSummaryFontSize: Spec.SPEC.notificationSummaryFontSize.def
|
||||||
property real notificationBodyFontSize: Spec.SPEC.notificationBodyFontSize.def
|
property real notificationBodyFontSize: Spec.SPEC.notificationBodyFontSize.def
|
||||||
property bool notepadShowLineNumbers: false
|
property bool notepadShowLineNumbers: false
|
||||||
|
property bool notepadAutoSave: false
|
||||||
|
property string notepadSlideoutSide: "right"
|
||||||
|
property string notepadDefaultMode: "slideout"
|
||||||
property real notepadTransparencyOverride: -1
|
property real notepadTransparencyOverride: -1
|
||||||
property real notepadLastCustomTransparency: 0.7
|
property real notepadLastCustomTransparency: 0.7
|
||||||
|
property bool notepadUseCompositorGap: false
|
||||||
|
property int notepadEdgeGap: 0
|
||||||
|
|
||||||
|
// Compositor layout gap when enabled and available, else the manual value.
|
||||||
|
readonly property int notepadEffectiveEdgeGap: {
|
||||||
|
if (notepadUseCompositorGap) {
|
||||||
|
var g = -1;
|
||||||
|
if (CompositorService.isNiri)
|
||||||
|
g = niriLayoutGapsOverride;
|
||||||
|
else if (CompositorService.isHyprland)
|
||||||
|
g = hyprlandLayoutGapsOverride;
|
||||||
|
else if (CompositorService.isMango)
|
||||||
|
g = mangoLayoutGapsOverride;
|
||||||
|
if (g >= 0)
|
||||||
|
return g;
|
||||||
|
}
|
||||||
|
return Math.max(0, notepadEdgeGap);
|
||||||
|
}
|
||||||
|
|
||||||
onNotepadUseMonospaceChanged: saveSettings()
|
onNotepadUseMonospaceChanged: saveSettings()
|
||||||
onNotepadFontFamilyChanged: saveSettings()
|
onNotepadFontFamilyChanged: saveSettings()
|
||||||
onNotepadFontSizeChanged: saveSettings()
|
onNotepadFontSizeChanged: saveSettings()
|
||||||
onNotepadShowLineNumbersChanged: saveSettings()
|
onNotepadShowLineNumbersChanged: saveSettings()
|
||||||
|
onNotepadAutoSaveChanged: saveSettings()
|
||||||
|
onNotepadSlideoutSideChanged: saveSettings()
|
||||||
|
onNotepadDefaultModeChanged: saveSettings()
|
||||||
|
onNotepadUseCompositorGapChanged: saveSettings()
|
||||||
|
onNotepadEdgeGapChanged: saveSettings()
|
||||||
// onCenteringModeChanged: saveSettings()
|
// onCenteringModeChanged: saveSettings()
|
||||||
onNotepadTransparencyOverrideChanged: {
|
onNotepadTransparencyOverrideChanged: {
|
||||||
if (notepadTransparencyOverride > 0) {
|
if (notepadTransparencyOverride > 0) {
|
||||||
@@ -810,8 +837,7 @@ Singleton {
|
|||||||
"shadowOpacity": 60,
|
"shadowOpacity": 60,
|
||||||
"shadowColorMode": "default",
|
"shadowColorMode": "default",
|
||||||
"shadowCustomColor": "#000000",
|
"shadowCustomColor": "#000000",
|
||||||
"clickThrough": false,
|
"clickThrough": false
|
||||||
"hoverPopouts": false
|
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|||||||
@@ -156,6 +156,7 @@ var SPEC = {
|
|||||||
audioVisualizerEnabled: { def: true },
|
audioVisualizerEnabled: { def: true },
|
||||||
audioScrollMode: { def: "volume" },
|
audioScrollMode: { def: "volume" },
|
||||||
audioWheelScrollAmount: { def: 5 },
|
audioWheelScrollAmount: { def: 5 },
|
||||||
|
audioDeviceScrollVolumeEnabled: { def: false },
|
||||||
clockCompactMode: { def: false },
|
clockCompactMode: { def: false },
|
||||||
focusedWindowCompactMode: { def: false },
|
focusedWindowCompactMode: { def: false },
|
||||||
focusedWindowSize: { def: 1 },
|
focusedWindowSize: { def: 1 },
|
||||||
@@ -263,8 +264,13 @@ var SPEC = {
|
|||||||
notificationSummaryFontSize: { def: 0 },
|
notificationSummaryFontSize: { def: 0 },
|
||||||
notificationBodyFontSize: { def: 0 },
|
notificationBodyFontSize: { def: 0 },
|
||||||
notepadShowLineNumbers: { def: false },
|
notepadShowLineNumbers: { def: false },
|
||||||
|
notepadAutoSave: { def: false },
|
||||||
|
notepadSlideoutSide: { def: "right" },
|
||||||
|
notepadDefaultMode: { def: "slideout" },
|
||||||
notepadTransparencyOverride: { def: -1 },
|
notepadTransparencyOverride: { def: -1 },
|
||||||
notepadLastCustomTransparency: { def: 0.7 },
|
notepadLastCustomTransparency: { def: 0.7 },
|
||||||
|
notepadUseCompositorGap: { def: false },
|
||||||
|
notepadEdgeGap: { def: 0 },
|
||||||
|
|
||||||
soundsEnabled: { def: true },
|
soundsEnabled: { def: true },
|
||||||
useSystemSoundTheme: { def: false },
|
useSystemSoundTheme: { def: false },
|
||||||
|
|||||||
@@ -1093,11 +1093,22 @@ Item {
|
|||||||
slideoutWidth: 480
|
slideoutWidth: 480
|
||||||
expandable: true
|
expandable: true
|
||||||
expandedWidthValue: 960
|
expandedWidthValue: 960
|
||||||
|
edgeGap: SettingsData.notepadEffectiveEdgeGap
|
||||||
|
slideEdge: SettingsData.notepadSlideoutSide
|
||||||
|
|
||||||
|
onIsVisibleChanged: {
|
||||||
|
if (isVisible)
|
||||||
|
PopoutService.notepadPopout?.hide();
|
||||||
|
}
|
||||||
|
|
||||||
content: Component {
|
content: Component {
|
||||||
Notepad {
|
Notepad {
|
||||||
slideout: notepadSlideout
|
slideout: notepadSlideout
|
||||||
onHideRequested: notepadSlideout.hide()
|
onHideRequested: notepadSlideout.hide()
|
||||||
|
onPopoutRequested: {
|
||||||
|
notepadSlideout.hide();
|
||||||
|
PopoutService.openNotepadPopout();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1114,6 +1125,24 @@ Item {
|
|||||||
Component.onCompleted: PopoutService.notepadSlideouts = instances
|
Component.onCompleted: PopoutService.notepadSlideouts = instances
|
||||||
}
|
}
|
||||||
|
|
||||||
|
LazyLoader {
|
||||||
|
id: notepadPopoutLoader
|
||||||
|
active: false
|
||||||
|
|
||||||
|
Component.onCompleted: {
|
||||||
|
PopoutService.notepadPopoutLoader = notepadPopoutLoader;
|
||||||
|
}
|
||||||
|
|
||||||
|
onActiveChanged: {
|
||||||
|
if (active && item) {
|
||||||
|
PopoutService.notepadPopout = item;
|
||||||
|
PopoutService._onNotepadPopoutLoaded();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
NotepadPopoutWindow {}
|
||||||
|
}
|
||||||
|
|
||||||
LazyLoader {
|
LazyLoader {
|
||||||
id: powerMenuModalLoader
|
id: powerMenuModalLoader
|
||||||
|
|
||||||
|
|||||||
@@ -373,6 +373,10 @@ Item {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function open(): string {
|
function open(): string {
|
||||||
|
if (SettingsData.notepadDefaultMode === "popout") {
|
||||||
|
PopoutService.openNotepadPopout();
|
||||||
|
return "NOTEPAD_OPEN_SUCCESS";
|
||||||
|
}
|
||||||
var instance = getActiveNotepadInstance();
|
var instance = getActiveNotepadInstance();
|
||||||
if (instance) {
|
if (instance) {
|
||||||
instance.show();
|
instance.show();
|
||||||
@@ -382,6 +386,10 @@ Item {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function close(): string {
|
function close(): string {
|
||||||
|
if (SettingsData.notepadDefaultMode === "popout") {
|
||||||
|
PopoutService.notepadPopout?.hide();
|
||||||
|
return "NOTEPAD_CLOSE_SUCCESS";
|
||||||
|
}
|
||||||
var instance = getActiveNotepadInstance();
|
var instance = getActiveNotepadInstance();
|
||||||
if (instance) {
|
if (instance) {
|
||||||
instance.hide();
|
instance.hide();
|
||||||
@@ -391,6 +399,10 @@ Item {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function toggle(): string {
|
function toggle(): string {
|
||||||
|
if (SettingsData.notepadDefaultMode === "popout") {
|
||||||
|
PopoutService.toggleNotepadPopout();
|
||||||
|
return "NOTEPAD_TOGGLE_SUCCESS";
|
||||||
|
}
|
||||||
var instance = getActiveNotepadInstance();
|
var instance = getActiveNotepadInstance();
|
||||||
if (instance) {
|
if (instance) {
|
||||||
instance.toggle();
|
instance.toggle();
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -709,14 +709,6 @@ PanelWindow {
|
|||||||
readonly property var _rightSection: topBarContent ? (barWindow.isVertical ? topBarContent.vRightSection : topBarContent.hRightSection) : null
|
readonly property var _rightSection: topBarContent ? (barWindow.isVertical ? topBarContent.vRightSection : topBarContent.hRightSection) : null
|
||||||
readonly property real _revealProgress: topBarSlide.x + topBarSlide.y
|
readonly property real _revealProgress: topBarSlide.x + topBarSlide.y
|
||||||
|
|
||||||
function containsGlobalPoint(gx, gy, padding) {
|
|
||||||
const pad = padding !== undefined ? padding : 16;
|
|
||||||
if (!inputMask.showing)
|
|
||||||
return false;
|
|
||||||
const topLeft = inputMask.mapToItem(null, 0, 0);
|
|
||||||
return gx >= topLeft.x - pad && gx < topLeft.x + inputMask.width + pad && gy >= topLeft.y - pad && gy < topLeft.y + inputMask.height + pad;
|
|
||||||
}
|
|
||||||
|
|
||||||
function sectionRect(section, isCenter, _dep) {
|
function sectionRect(section, isCenter, _dep) {
|
||||||
if (!section)
|
if (!section)
|
||||||
return {
|
return {
|
||||||
@@ -1018,7 +1010,7 @@ PanelWindow {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function processWheel(wheel) {
|
onWheel: wheel => {
|
||||||
if (!(barConfig?.scrollEnabled ?? true) || actionInProgress) {
|
if (!(barConfig?.scrollEnabled ?? true) || actionInProgress) {
|
||||||
wheel.accepted = false;
|
wheel.accepted = false;
|
||||||
return;
|
return;
|
||||||
@@ -1087,8 +1079,6 @@ PanelWindow {
|
|||||||
|
|
||||||
wheel.accepted = false;
|
wheel.accepted = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
onWheel: wheel => processWheel(wheel)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
DankBarContent {
|
DankBarContent {
|
||||||
@@ -1100,36 +1090,6 @@ PanelWindow {
|
|||||||
centerWidgetsModel: barWindow.centerWidgetsModel
|
centerWidgetsModel: barWindow.centerWidgetsModel
|
||||||
rightWidgetsModel: barWindow.rightWidgetsModel
|
rightWidgetsModel: barWindow.rightWidgetsModel
|
||||||
}
|
}
|
||||||
|
|
||||||
MouseArea {
|
|
||||||
id: hoverPopoutArea
|
|
||||||
anchors.fill: parent
|
|
||||||
z: 1
|
|
||||||
hoverEnabled: barConfig?.hoverPopouts ?? false
|
|
||||||
enabled: hoverPopoutArea.hoverEnabled && !barWindow.clickThroughEnabled
|
|
||||||
acceptedButtons: Qt.NoButton
|
|
||||||
propagateComposedEvents: true
|
|
||||||
|
|
||||||
property real lastGlobalX: 0
|
|
||||||
property real lastGlobalY: 0
|
|
||||||
|
|
||||||
onPositionChanged: mouse => {
|
|
||||||
const gp = mapToItem(null, mouse.x, mouse.y);
|
|
||||||
lastGlobalX = gp.x;
|
|
||||||
lastGlobalY = gp.y;
|
|
||||||
topBarContent.checkHoverPopout(gp.x, gp.y);
|
|
||||||
}
|
|
||||||
|
|
||||||
onWheel: wheel => scrollArea.processWheel(wheel)
|
|
||||||
|
|
||||||
onContainsMouseChanged: {
|
|
||||||
if (containsMouse)
|
|
||||||
return;
|
|
||||||
if (topBarContent.cursorOverHoverChain(lastGlobalX, lastGlobalY))
|
|
||||||
return;
|
|
||||||
topBarContent.closeHoverSurfaces();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -32,9 +32,20 @@ BasePill {
|
|||||||
}
|
}
|
||||||
|
|
||||||
readonly property var notepadInstance: resolveNotepadInstance()
|
readonly property var notepadInstance: resolveNotepadInstance()
|
||||||
readonly property bool isActive: notepadInstance?.isVisible ?? false
|
readonly property bool popoutDefault: SettingsData.notepadDefaultMode === "popout"
|
||||||
|
readonly property bool isActive: popoutDefault ? (PopoutService.notepadPopout?.visible ?? false) : (notepadInstance?.isVisible ?? false)
|
||||||
property bool isAutoHideBar: false
|
property bool isAutoHideBar: false
|
||||||
|
|
||||||
|
function showActiveSurface() {
|
||||||
|
if (root.popoutDefault) {
|
||||||
|
PopoutService.openNotepadPopout();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const instance = prepareNotepadInstance(root.notepadInstance);
|
||||||
|
if (instance && typeof instance.show === "function")
|
||||||
|
instance.show();
|
||||||
|
}
|
||||||
|
|
||||||
function prepareNotepadInstance(instance) {
|
function prepareNotepadInstance(instance) {
|
||||||
if (instance)
|
if (instance)
|
||||||
instance.triggerUsesOverlayLayer = root.barUsesOverlayLayer;
|
instance.triggerUsesOverlayLayer = root.barUsesOverlayLayer;
|
||||||
@@ -75,20 +86,14 @@ BasePill {
|
|||||||
function openTabByIndex(tabIndex) {
|
function openTabByIndex(tabIndex) {
|
||||||
if (tabIndex < 0)
|
if (tabIndex < 0)
|
||||||
return;
|
return;
|
||||||
const instance = prepareNotepadInstance(root.notepadInstance);
|
showActiveSurface();
|
||||||
if (instance && typeof instance.show === "function") {
|
|
||||||
instance.show();
|
|
||||||
}
|
|
||||||
Qt.callLater(() => {
|
Qt.callLater(() => {
|
||||||
NotepadStorageService.switchToTab(tabIndex);
|
NotepadStorageService.switchToTab(tabIndex);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function openNewNote() {
|
function openNewNote() {
|
||||||
const instance = prepareNotepadInstance(root.notepadInstance);
|
showActiveSurface();
|
||||||
if (instance && typeof instance.show === "function") {
|
|
||||||
instance.show();
|
|
||||||
}
|
|
||||||
Qt.callLater(() => {
|
Qt.callLater(() => {
|
||||||
NotepadStorageService.createNewTab();
|
NotepadStorageService.createNewTab();
|
||||||
});
|
});
|
||||||
@@ -147,6 +152,10 @@ BasePill {
|
|||||||
openContextMenu();
|
openContextMenu();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
if (root.popoutDefault) {
|
||||||
|
PopoutService.toggleNotepadPopout();
|
||||||
|
return;
|
||||||
|
}
|
||||||
const inst = prepareNotepadInstance(root.notepadInstance);
|
const inst = prepareNotepadInstance(root.notepadInstance);
|
||||||
if (inst) {
|
if (inst) {
|
||||||
inst.toggle();
|
inst.toggle();
|
||||||
|
|||||||
@@ -1922,53 +1922,4 @@ BasePill {
|
|||||||
return;
|
return;
|
||||||
currentTrayMenu.showForTrayItem(item, anchor, screen, atBottom, vertical ?? false, axisObj);
|
currentTrayMenu.showForTrayItem(item, anchor, screen, atBottom, vertical ?? false, axisObj);
|
||||||
}
|
}
|
||||||
|
|
||||||
function _trayLayoutRoot() {
|
|
||||||
const contentChildren = root.visualContent?.children;
|
|
||||||
if (!contentChildren || contentChildren.length === 0)
|
|
||||||
return null;
|
|
||||||
const contentRoot = contentChildren[0];
|
|
||||||
return contentRoot?.layoutLoader?.item || null;
|
|
||||||
}
|
|
||||||
|
|
||||||
function _trayHitAtGlobalPoint(gx, gy) {
|
|
||||||
if (!root.visible || root.width <= 0 || root.height <= 0)
|
|
||||||
return null;
|
|
||||||
const local = root.mapFromItem(null, gx, gy);
|
|
||||||
if (local.x < 0 || local.y < 0 || local.x > root.width || local.y > root.height)
|
|
||||||
return null;
|
|
||||||
const layout = _trayLayoutRoot();
|
|
||||||
if (!layout)
|
|
||||||
return null;
|
|
||||||
const layoutLocal = layout.mapFromItem(null, gx, gy);
|
|
||||||
const children = layout.children || [];
|
|
||||||
for (let i = 0; i < children.length; i++) {
|
|
||||||
const child = children[i];
|
|
||||||
if (!child.visible || child.width <= 0 || child.height <= 0)
|
|
||||||
continue;
|
|
||||||
if (layoutLocal.x < child.x || layoutLocal.x >= child.x + child.width)
|
|
||||||
continue;
|
|
||||||
if (layoutLocal.y < child.y || layoutLocal.y >= child.y + child.height)
|
|
||||||
continue;
|
|
||||||
if (child.trayItem)
|
|
||||||
return child;
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
function hoverTriggerAtGlobalPoint(gx, gy) {
|
|
||||||
const hit = _trayHitAtGlobalPoint(gx, gy);
|
|
||||||
if (!hit?.trayItem?.hasMenu)
|
|
||||||
return "";
|
|
||||||
return "tray-" + (hit.trayItem.id || hit.itemKey || "");
|
|
||||||
}
|
|
||||||
|
|
||||||
function openHoverAtGlobalPoint(gx, gy) {
|
|
||||||
const hit = _trayHitAtGlobalPoint(gx, gy);
|
|
||||||
if (!hit?.trayItem?.hasMenu)
|
|
||||||
return false;
|
|
||||||
const anchor = hit.children?.length > 0 ? hit.children[0] : hit;
|
|
||||||
showForTrayItem(hit.trayItem, anchor, parentScreen, isAtBottom, isVerticalOrientation, axis);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -108,9 +108,6 @@ DankPopout {
|
|||||||
MprisController.setActivePlayer(player);
|
MprisController.setActivePlayer(player);
|
||||||
root.__hideDropdowns();
|
root.__hideDropdowns();
|
||||||
}
|
}
|
||||||
onDeviceSelected: device => {
|
|
||||||
root.__hideDropdowns();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -383,7 +383,27 @@ Item {
|
|||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
hoverEnabled: true
|
hoverEnabled: true
|
||||||
cursorShape: Qt.PointingHandCursor
|
cursorShape: Qt.PointingHandCursor
|
||||||
onClicked: {
|
acceptedButtons: Qt.LeftButton | Qt.RightButton
|
||||||
|
onPressed: mouse => {
|
||||||
|
if (mouse.button === Qt.RightButton) {
|
||||||
|
mouse.accepted = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
onWheel: wheelEvent => {
|
||||||
|
if (SettingsData.audioDeviceScrollVolumeEnabled && wheelEvent.x >= deviceMouseArea.width / 2) {
|
||||||
|
AudioService.handleNodeVolumeWheel(modelData, wheelEvent);
|
||||||
|
} else {
|
||||||
|
wheelEvent.accepted = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
onClicked: mouse => {
|
||||||
|
if (mouse.button === Qt.RightButton) {
|
||||||
|
if (modelData && modelData.audio) {
|
||||||
|
SessionData.suppressOSDTemporarily();
|
||||||
|
modelData.audio.muted = !modelData.audio.muted;
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
if (modelData && modelData.name) {
|
if (modelData && modelData.name) {
|
||||||
AudioService.setDefaultSinkByName(modelData.name);
|
AudioService.setDefaultSinkByName(modelData.name);
|
||||||
root.deviceSelected(modelData);
|
root.deviceSelected(modelData);
|
||||||
|
|||||||
@@ -866,7 +866,27 @@ Item {
|
|||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
hoverEnabled: true
|
hoverEnabled: true
|
||||||
cursorShape: Qt.PointingHandCursor
|
cursorShape: Qt.PointingHandCursor
|
||||||
onClicked: {
|
acceptedButtons: Qt.LeftButton | Qt.RightButton
|
||||||
|
onPressed: mouse => {
|
||||||
|
if (mouse.button === Qt.RightButton) {
|
||||||
|
mouse.accepted = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
onWheel: wheelEvent => {
|
||||||
|
const delta = wheelEvent.angleDelta.y;
|
||||||
|
if (delta !== 0) {
|
||||||
|
AudioService.cycleAudioOutputDirection(delta < 0);
|
||||||
|
wheelEvent.accepted = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
onClicked: mouse => {
|
||||||
|
if (mouse.button === Qt.RightButton) {
|
||||||
|
if (AudioService.sink?.audio) {
|
||||||
|
SessionData.suppressOSDTemporarily();
|
||||||
|
AudioService.sink.audio.muted = !AudioService.sink.audio.muted;
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
if (devicesExpanded) {
|
if (devicesExpanded) {
|
||||||
const sinks = AudioService.getAvailableSinks();
|
const sinks = AudioService.getAvailableSinks();
|
||||||
if (sinks && sinks.length > 1) {
|
if (sinks && sinks.length > 1) {
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
pragma ComponentBehavior: Bound
|
pragma ComponentBehavior: Bound
|
||||||
import QtQuick
|
import QtQuick
|
||||||
|
import QtQuick.Layouts
|
||||||
import Quickshell
|
import Quickshell
|
||||||
import Quickshell.Io
|
import Quickshell.Io
|
||||||
import qs.Common
|
import qs.Common
|
||||||
@@ -21,11 +22,24 @@ Item {
|
|||||||
property var currentTab: NotepadStorageService.tabs.length > NotepadStorageService.currentTabIndex ? NotepadStorageService.tabs[NotepadStorageService.currentTabIndex] : null
|
property var currentTab: NotepadStorageService.tabs.length > NotepadStorageService.currentTabIndex ? NotepadStorageService.tabs[NotepadStorageService.currentTabIndex] : null
|
||||||
property bool showSettingsMenu: false
|
property bool showSettingsMenu: false
|
||||||
property string pendingSaveContent: ""
|
property string pendingSaveContent: ""
|
||||||
|
readonly property bool conflictBannerVisible: currentTab !== null && NotepadStorageService.conflictTabId === currentTab.id
|
||||||
property var slideout: null
|
property var slideout: null
|
||||||
|
property bool inPopout: false
|
||||||
|
property bool surfaceVisible: slideout ? slideout.isVisible : true
|
||||||
|
|
||||||
signal hideRequested
|
signal hideRequested
|
||||||
|
signal popoutRequested
|
||||||
|
signal dockRequested
|
||||||
signal previewRequested(string content)
|
signal previewRequested(string content)
|
||||||
|
|
||||||
|
function externalSync() {
|
||||||
|
textEditor.syncFromDisk();
|
||||||
|
}
|
||||||
|
|
||||||
|
function flushAutoSave() {
|
||||||
|
textEditor.autoSaveToSession();
|
||||||
|
}
|
||||||
|
|
||||||
Ref {
|
Ref {
|
||||||
service: NotepadStorageService
|
service: NotepadStorageService
|
||||||
}
|
}
|
||||||
@@ -36,6 +50,37 @@ Item {
|
|||||||
function onAboutToHide() {
|
function onAboutToHide() {
|
||||||
textEditor.autoSaveToSession();
|
textEditor.autoSaveToSession();
|
||||||
}
|
}
|
||||||
|
function onRevealed() {
|
||||||
|
textEditor.syncFromDisk();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function showConflictBanner(diskContent) {
|
||||||
|
if (!currentTab)
|
||||||
|
return;
|
||||||
|
NotepadStorageService.flagConflict(currentTab.id, diskContent);
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolveConflictKeepEdits() {
|
||||||
|
if (!root.conflictBannerVisible)
|
||||||
|
return;
|
||||||
|
NotepadStorageService.clearConflict();
|
||||||
|
if (currentTab && currentTab.filePath && !currentTab.isTemporary) {
|
||||||
|
root.saveToFile("file://" + currentTab.filePath);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolveConflictReload() {
|
||||||
|
if (!root.conflictBannerVisible)
|
||||||
|
return;
|
||||||
|
const diskContent = NotepadStorageService.conflictDiskContent;
|
||||||
|
NotepadStorageService.clearConflict();
|
||||||
|
textEditor.reloadFromDisk(diskContent);
|
||||||
|
}
|
||||||
|
|
||||||
|
function dismissConflictBanner() {
|
||||||
|
if (root.conflictBannerVisible)
|
||||||
|
NotepadStorageService.clearConflict();
|
||||||
}
|
}
|
||||||
|
|
||||||
function hasUnsavedChanges() {
|
function hasUnsavedChanges() {
|
||||||
@@ -51,10 +96,14 @@ Item {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function performCreateNewTab() {
|
function performCreateNewTab() {
|
||||||
|
textEditor.commitLiveBuffer();
|
||||||
NotepadStorageService.createNewTab();
|
NotepadStorageService.createNewTab();
|
||||||
|
textEditor.applyingShared = true;
|
||||||
textEditor.text = "";
|
textEditor.text = "";
|
||||||
textEditor.lastSavedContent = "";
|
textEditor.lastSavedContent = "";
|
||||||
|
textEditor.loadedTabId = -1;
|
||||||
textEditor.contentLoaded = true;
|
textEditor.contentLoaded = true;
|
||||||
|
textEditor.applyingShared = false;
|
||||||
textEditor.textArea.forceActiveFocus();
|
textEditor.textArea.forceActiveFocus();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -86,7 +135,6 @@ Item {
|
|||||||
|
|
||||||
NotepadStorageService.switchToTab(tabIndex);
|
NotepadStorageService.switchToTab(tabIndex);
|
||||||
Qt.callLater(() => {
|
Qt.callLater(() => {
|
||||||
textEditor.loadCurrentTabContent();
|
|
||||||
if (currentTab) {
|
if (currentTab) {
|
||||||
root.currentFileName = currentTab.fileName || "";
|
root.currentFileName = currentTab.fileName || "";
|
||||||
root.currentFileUrl = currentTab.fileUrl || "";
|
root.currentFileUrl = currentTab.fileUrl || "";
|
||||||
@@ -100,6 +148,7 @@ Item {
|
|||||||
var content = textEditor.text;
|
var content = textEditor.text;
|
||||||
var filePath = fileUrl.toString().replace(/^file:\/\//, '');
|
var filePath = fileUrl.toString().replace(/^file:\/\//, '');
|
||||||
|
|
||||||
|
textEditor.externalWatchPaused = true;
|
||||||
saveFileView.path = "";
|
saveFileView.path = "";
|
||||||
pendingSaveContent = content;
|
pendingSaveContent = content;
|
||||||
saveFileView.path = filePath;
|
saveFileView.path = filePath;
|
||||||
@@ -109,6 +158,53 @@ Item {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function saveExternalWithFreshnessCheck() {
|
||||||
|
if (!currentTab || currentTab.isTemporary || !currentTab.filePath)
|
||||||
|
return;
|
||||||
|
const filePath = currentTab.filePath;
|
||||||
|
loadFileView.path = "";
|
||||||
|
loadFileView.path = filePath;
|
||||||
|
|
||||||
|
if (!loadFileView.waitForJob()) {
|
||||||
|
saveToFile("file://" + filePath);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
Qt.callLater(() => {
|
||||||
|
if (!currentTab || currentTab.isTemporary || currentTab.filePath !== filePath)
|
||||||
|
return;
|
||||||
|
const diskContent = loadFileView.text();
|
||||||
|
if (diskContent !== undefined && diskContent !== null && diskContent !== textEditor.text && diskContent !== textEditor.lastSavedContent) {
|
||||||
|
root.showConflictBanner(diskContent);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
saveToFile("file://" + filePath);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function autoSaveExternal() {
|
||||||
|
if (!SettingsData.notepadAutoSave)
|
||||||
|
return;
|
||||||
|
if (!currentTab || currentTab.isTemporary || !currentTab.filePath)
|
||||||
|
return;
|
||||||
|
if (!textEditor.hasUnsavedChanges())
|
||||||
|
return;
|
||||||
|
const filePath = currentTab.filePath;
|
||||||
|
loadFileView.path = "";
|
||||||
|
loadFileView.path = filePath;
|
||||||
|
if (!loadFileView.waitForJob())
|
||||||
|
return;
|
||||||
|
Qt.callLater(() => {
|
||||||
|
if (!currentTab || currentTab.isTemporary || currentTab.filePath !== filePath)
|
||||||
|
return;
|
||||||
|
const diskContent = loadFileView.text();
|
||||||
|
if (diskContent === undefined || diskContent === null)
|
||||||
|
return;
|
||||||
|
if (diskContent !== textEditor.lastSavedContent)
|
||||||
|
return;
|
||||||
|
saveToFile("file://" + filePath);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
function loadFromFile(fileUrl) {
|
function loadFromFile(fileUrl) {
|
||||||
if (hasUnsavedTemporaryContent()) {
|
if (hasUnsavedTemporaryContent()) {
|
||||||
root.pendingFileUrl = fileUrl;
|
root.pendingFileUrl = fileUrl;
|
||||||
@@ -146,14 +242,155 @@ Item {
|
|||||||
|
|
||||||
root.currentFileName = fileName;
|
root.currentFileName = fileName;
|
||||||
root.currentFileUrl = fileUrl;
|
root.currentFileUrl = fileUrl;
|
||||||
textEditor.saveCurrentTabContent();
|
textEditor.loadedTabId = currentTab.id;
|
||||||
|
NotepadStorageService.clearSessionBuffer(currentTab.id);
|
||||||
|
if (root.conflictBannerVisible)
|
||||||
|
NotepadStorageService.clearConflict();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Item {
|
||||||
|
id: conflictBanner
|
||||||
|
anchors.top: parent.top
|
||||||
|
anchors.left: parent.left
|
||||||
|
anchors.right: parent.right
|
||||||
|
height: root.conflictBannerVisible ? bannerRect.implicitHeight : 0
|
||||||
|
visible: height > 0
|
||||||
|
clip: true
|
||||||
|
z: 5
|
||||||
|
|
||||||
|
Behavior on height {
|
||||||
|
NumberAnimation {
|
||||||
|
duration: Theme.shortDuration
|
||||||
|
easing.type: Theme.standardEasing
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
StyledRect {
|
||||||
|
id: bannerRect
|
||||||
|
anchors.left: parent.left
|
||||||
|
anchors.right: parent.right
|
||||||
|
anchors.top: parent.top
|
||||||
|
implicitHeight: bannerLayout.implicitHeight + Theme.spacingM * 2
|
||||||
|
radius: Theme.cornerRadius
|
||||||
|
color: Theme.withAlpha(Theme.warning, 0.12)
|
||||||
|
border.color: Theme.withAlpha(Theme.warning, 0.5)
|
||||||
|
border.width: 1
|
||||||
|
|
||||||
|
ColumnLayout {
|
||||||
|
id: bannerLayout
|
||||||
|
anchors.fill: parent
|
||||||
|
anchors.margins: Theme.spacingM
|
||||||
|
spacing: Theme.spacingS
|
||||||
|
|
||||||
|
RowLayout {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
spacing: Theme.spacingM
|
||||||
|
|
||||||
|
DankIcon {
|
||||||
|
Layout.alignment: Qt.AlignVCenter
|
||||||
|
name: "sync_problem"
|
||||||
|
size: Theme.iconSize - 2
|
||||||
|
color: Theme.warning
|
||||||
|
}
|
||||||
|
|
||||||
|
StyledText {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
Layout.alignment: Qt.AlignVCenter
|
||||||
|
text: I18n.tr("File changed on disk")
|
||||||
|
font.pixelSize: Theme.fontSizeMedium
|
||||||
|
font.weight: Font.Medium
|
||||||
|
color: Theme.surfaceText
|
||||||
|
wrapMode: Text.NoWrap
|
||||||
|
elide: Text.ElideRight
|
||||||
|
}
|
||||||
|
|
||||||
|
DankActionButton {
|
||||||
|
Layout.alignment: Qt.AlignVCenter
|
||||||
|
iconName: "close"
|
||||||
|
iconSize: Theme.iconSizeSmall
|
||||||
|
iconColor: Theme.surfaceText
|
||||||
|
buttonSize: 28
|
||||||
|
onClicked: root.dismissConflictBanner()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Item {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
Layout.preferredHeight: 32
|
||||||
|
|
||||||
|
Row {
|
||||||
|
id: bannerActions
|
||||||
|
anchors.right: parent.right
|
||||||
|
spacing: Theme.spacingS
|
||||||
|
|
||||||
|
readonly property real available: parent.width
|
||||||
|
|
||||||
|
StyledRect {
|
||||||
|
width: Math.min(keepText.implicitWidth + Theme.spacingM * 2, Math.max(104, (bannerActions.available - bannerActions.spacing) / 2))
|
||||||
|
height: 32
|
||||||
|
radius: Theme.cornerRadius
|
||||||
|
color: "transparent"
|
||||||
|
border.color: Theme.outlineMedium
|
||||||
|
border.width: 1
|
||||||
|
|
||||||
|
StateLayer {
|
||||||
|
anchors.fill: parent
|
||||||
|
cornerRadius: parent.radius
|
||||||
|
stateColor: Theme.surfaceText
|
||||||
|
onClicked: root.resolveConflictKeepEdits()
|
||||||
|
}
|
||||||
|
|
||||||
|
StyledText {
|
||||||
|
id: keepText
|
||||||
|
anchors.centerIn: parent
|
||||||
|
width: parent.width - Theme.spacingM
|
||||||
|
text: I18n.tr("Keep My Edits")
|
||||||
|
font.pixelSize: Theme.fontSizeSmall
|
||||||
|
color: Theme.surfaceText
|
||||||
|
horizontalAlignment: Text.AlignHCenter
|
||||||
|
elide: Text.ElideRight
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
StyledRect {
|
||||||
|
width: Math.min(reloadText.implicitWidth + Theme.spacingM * 2, Math.max(116, (bannerActions.available - bannerActions.spacing) / 2))
|
||||||
|
height: 32
|
||||||
|
radius: Theme.cornerRadius
|
||||||
|
color: Theme.primary
|
||||||
|
|
||||||
|
StateLayer {
|
||||||
|
anchors.fill: parent
|
||||||
|
cornerRadius: parent.radius
|
||||||
|
stateColor: Theme.background
|
||||||
|
onClicked: root.resolveConflictReload()
|
||||||
|
}
|
||||||
|
|
||||||
|
StyledText {
|
||||||
|
id: reloadText
|
||||||
|
anchors.centerIn: parent
|
||||||
|
width: parent.width - Theme.spacingM
|
||||||
|
text: I18n.tr("Reload From Disk")
|
||||||
|
font.pixelSize: Theme.fontSizeSmall
|
||||||
|
color: Theme.background
|
||||||
|
horizontalAlignment: Text.AlignHCenter
|
||||||
|
elide: Text.ElideRight
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Column {
|
Column {
|
||||||
anchors.fill: parent
|
anchors.top: conflictBanner.bottom
|
||||||
|
anchors.topMargin: root.conflictBannerVisible ? Theme.spacingM : 0
|
||||||
|
anchors.left: parent.left
|
||||||
|
anchors.right: parent.right
|
||||||
|
anchors.bottom: parent.bottom
|
||||||
spacing: Theme.spacingM
|
spacing: Theme.spacingM
|
||||||
|
|
||||||
NotepadTabs {
|
NotepadTabs {
|
||||||
@@ -178,11 +415,12 @@ Item {
|
|||||||
id: textEditor
|
id: textEditor
|
||||||
width: parent.width
|
width: parent.width
|
||||||
height: parent.height - tabBar.height - Theme.spacingM * 2
|
height: parent.height - tabBar.height - Theme.spacingM * 2
|
||||||
|
inPopout: root.inPopout
|
||||||
|
surfaceVisible: root.surfaceVisible
|
||||||
|
|
||||||
onSaveRequested: {
|
onSaveRequested: {
|
||||||
if (currentTab && !currentTab.isTemporary && currentTab.filePath) {
|
if (currentTab && !currentTab.isTemporary && currentTab.filePath) {
|
||||||
var fileUrl = "file://" + currentTab.filePath;
|
root.saveExternalWithFreshnessCheck();
|
||||||
saveToFile(fileUrl);
|
|
||||||
} else {
|
} else {
|
||||||
root.fileDialogOpen = true;
|
root.fileDialogOpen = true;
|
||||||
saveBrowserLoader.active = true;
|
saveBrowserLoader.active = true;
|
||||||
@@ -214,12 +452,28 @@ Item {
|
|||||||
|
|
||||||
onEscapePressed: {
|
onEscapePressed: {
|
||||||
textEditor.autoSaveToSession();
|
textEditor.autoSaveToSession();
|
||||||
root.hideRequested();
|
if (showSettingsMenu) {
|
||||||
|
showSettingsMenu = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!root.inPopout) {
|
||||||
|
root.hideRequested();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
onSettingsRequested: {
|
onSettingsRequested: {
|
||||||
showSettingsMenu = !showSettingsMenu;
|
showSettingsMenu = !showSettingsMenu;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
onPopoutRequested: root.popoutRequested()
|
||||||
|
|
||||||
|
onDockRequested: root.dockRequested()
|
||||||
|
|
||||||
|
onConflictDetected: diskContent => {
|
||||||
|
root.showConflictBanner(diskContent);
|
||||||
|
}
|
||||||
|
|
||||||
|
onAutoSaveRequested: root.autoSaveExternal()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -242,17 +496,24 @@ Item {
|
|||||||
printErrors: true
|
printErrors: true
|
||||||
|
|
||||||
onSaved: {
|
onSaved: {
|
||||||
if (currentTab && saveFileView.path && pendingSaveContent) {
|
if (currentTab && saveFileView.path) {
|
||||||
NotepadStorageService.updateTabMetadata(NotepadStorageService.currentTabIndex, {
|
NotepadStorageService.updateTabMetadata(NotepadStorageService.currentTabIndex, {
|
||||||
hasUnsavedChanges: false,
|
hasUnsavedChanges: false,
|
||||||
lastSavedContent: pendingSaveContent
|
lastSavedContent: pendingSaveContent
|
||||||
});
|
});
|
||||||
root.lastSavedFileContent = pendingSaveContent;
|
root.lastSavedFileContent = pendingSaveContent;
|
||||||
pendingSaveContent = "";
|
textEditor.lastSavedContent = pendingSaveContent;
|
||||||
|
textEditor.ignoreNextExternalChange = true;
|
||||||
|
textEditor.commitLiveBuffer();
|
||||||
|
if (root.conflictBannerVisible)
|
||||||
|
NotepadStorageService.clearConflict();
|
||||||
}
|
}
|
||||||
|
textEditor.externalWatchPaused = false;
|
||||||
|
pendingSaveContent = "";
|
||||||
}
|
}
|
||||||
|
|
||||||
onSaveFailed: error => {
|
onSaveFailed: error => {
|
||||||
|
textEditor.externalWatchPaused = false;
|
||||||
pendingSaveContent = "";
|
pendingSaveContent = "";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -298,6 +559,7 @@ Item {
|
|||||||
|
|
||||||
root.currentFileName = fileName;
|
root.currentFileName = fileName;
|
||||||
root.currentFileUrl = fileUrl;
|
root.currentFileUrl = fileUrl;
|
||||||
|
textEditor.externalWatchPaused = true;
|
||||||
|
|
||||||
if (currentTab) {
|
if (currentTab) {
|
||||||
NotepadStorageService.saveTabAs(NotepadStorageService.currentTabIndex, cleanPath);
|
NotepadStorageService.saveTabAs(NotepadStorageService.currentTabIndex, cleanPath);
|
||||||
@@ -343,7 +605,7 @@ Item {
|
|||||||
browserTitle: I18n.tr("Open Notepad File")
|
browserTitle: I18n.tr("Open Notepad File")
|
||||||
browserIcon: "folder_open"
|
browserIcon: "folder_open"
|
||||||
browserType: "notepad_load"
|
browserType: "notepad_load"
|
||||||
fileExtensions: ["*.txt", "*.md", "*.*"]
|
fileExtensions: ["*"]
|
||||||
allowStacking: true
|
allowStacking: true
|
||||||
|
|
||||||
onFileSelected: path => {
|
onFileSelected: path => {
|
||||||
@@ -376,6 +638,7 @@ Item {
|
|||||||
modalHeight: contentLoader.item ? contentLoader.item.implicitHeight + Theme.spacingM * 2 : 180
|
modalHeight: contentLoader.item ? contentLoader.item.implicitHeight + Theme.spacingM * 2 : 180
|
||||||
shouldBeVisible: false
|
shouldBeVisible: false
|
||||||
allowStacking: true
|
allowStacking: true
|
||||||
|
useOverlayLayer: true
|
||||||
|
|
||||||
onBackgroundClicked: {
|
onBackgroundClicked: {
|
||||||
close();
|
close();
|
||||||
|
|||||||
@@ -0,0 +1,137 @@
|
|||||||
|
import QtQuick
|
||||||
|
import Quickshell
|
||||||
|
import qs.Common
|
||||||
|
import qs.Services
|
||||||
|
import qs.Widgets
|
||||||
|
import qs.Modules.Notepad
|
||||||
|
|
||||||
|
FloatingWindow {
|
||||||
|
id: win
|
||||||
|
|
||||||
|
property alias shouldBeVisible: win.visible
|
||||||
|
|
||||||
|
function show() {
|
||||||
|
visible = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
function hide() {
|
||||||
|
visible = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggle() {
|
||||||
|
visible = !visible;
|
||||||
|
}
|
||||||
|
|
||||||
|
title: I18n.tr("Notepad")
|
||||||
|
minimumSize: Qt.size(360, 320)
|
||||||
|
implicitWidth: 640
|
||||||
|
implicitHeight: 760
|
||||||
|
color: Theme.surfaceContainer
|
||||||
|
visible: false
|
||||||
|
|
||||||
|
onVisibleChanged: {
|
||||||
|
if (visible) {
|
||||||
|
Qt.callLater(notepad.externalSync);
|
||||||
|
} else {
|
||||||
|
notepad.flushAutoSave();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// A compositor close (e.g. niri close-window)
|
||||||
|
onClosed: win.visible = false
|
||||||
|
|
||||||
|
Item {
|
||||||
|
anchors.fill: parent
|
||||||
|
|
||||||
|
Item {
|
||||||
|
id: titleBar
|
||||||
|
anchors.top: parent.top
|
||||||
|
anchors.left: parent.left
|
||||||
|
anchors.right: parent.right
|
||||||
|
height: 44
|
||||||
|
z: 10
|
||||||
|
|
||||||
|
MouseArea {
|
||||||
|
anchors.fill: parent
|
||||||
|
onPressed: windowControls.tryStartMove()
|
||||||
|
onDoubleClicked: windowControls.tryToggleMaximize()
|
||||||
|
}
|
||||||
|
|
||||||
|
Rectangle {
|
||||||
|
anchors.fill: parent
|
||||||
|
color: Theme.surfaceContainerHigh
|
||||||
|
opacity: 0.5
|
||||||
|
}
|
||||||
|
|
||||||
|
Row {
|
||||||
|
anchors.left: parent.left
|
||||||
|
anchors.leftMargin: Theme.spacingM
|
||||||
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
|
spacing: Theme.spacingS
|
||||||
|
|
||||||
|
DankIcon {
|
||||||
|
name: "edit_note"
|
||||||
|
size: Theme.iconSize - 2
|
||||||
|
color: Theme.primary
|
||||||
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
|
}
|
||||||
|
|
||||||
|
StyledText {
|
||||||
|
text: I18n.tr("Notepad")
|
||||||
|
font.pixelSize: Theme.fontSizeLarge
|
||||||
|
font.weight: Font.Medium
|
||||||
|
color: Theme.surfaceText
|
||||||
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Row {
|
||||||
|
anchors.right: parent.right
|
||||||
|
anchors.rightMargin: Theme.spacingS
|
||||||
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
|
spacing: Theme.spacingXS
|
||||||
|
|
||||||
|
DankActionButton {
|
||||||
|
visible: windowControls.canMaximize
|
||||||
|
circular: false
|
||||||
|
iconName: win.maximized ? "fullscreen_exit" : "fullscreen"
|
||||||
|
iconSize: Theme.iconSize - 4
|
||||||
|
iconColor: Theme.surfaceText
|
||||||
|
onClicked: windowControls.tryToggleMaximize()
|
||||||
|
}
|
||||||
|
|
||||||
|
DankActionButton {
|
||||||
|
circular: false
|
||||||
|
iconName: "close"
|
||||||
|
iconSize: Theme.iconSize - 4
|
||||||
|
iconColor: Theme.surfaceText
|
||||||
|
onClicked: win.hide()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Notepad {
|
||||||
|
id: notepad
|
||||||
|
anchors.top: titleBar.bottom
|
||||||
|
anchors.left: parent.left
|
||||||
|
anchors.right: parent.right
|
||||||
|
anchors.bottom: parent.bottom
|
||||||
|
anchors.topMargin: Theme.spacingM
|
||||||
|
anchors.leftMargin: Theme.spacingM
|
||||||
|
anchors.rightMargin: Theme.spacingM
|
||||||
|
anchors.bottomMargin: Theme.spacingM
|
||||||
|
inPopout: true
|
||||||
|
surfaceVisible: win.visible
|
||||||
|
onHideRequested: win.hide()
|
||||||
|
onDockRequested: {
|
||||||
|
win.hide();
|
||||||
|
PopoutService.openNotepadSlideout();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
FloatingWindowControls {
|
||||||
|
id: windowControls
|
||||||
|
targetWindow: win
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -10,6 +10,7 @@ Item {
|
|||||||
property var cachedFontFamilies: []
|
property var cachedFontFamilies: []
|
||||||
property var cachedMonoFamilies: []
|
property var cachedMonoFamilies: []
|
||||||
property bool fontsEnumerated: false
|
property bool fontsEnumerated: false
|
||||||
|
property bool shortcutsExpanded: false
|
||||||
|
|
||||||
signal settingsRequested
|
signal settingsRequested
|
||||||
signal findRequested
|
signal findRequested
|
||||||
@@ -62,11 +63,23 @@ Item {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
MouseArea {
|
Rectangle {
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
visible: root.isVisible
|
visible: root.isVisible
|
||||||
onClicked: root.settingsRequested()
|
|
||||||
z: 50
|
z: 50
|
||||||
|
color: Qt.rgba(Theme.surface.r, Theme.surface.g, Theme.surface.b, 0.85)
|
||||||
|
|
||||||
|
WheelHandler {
|
||||||
|
// Hold scroll so the editor beneath doesn't move while settings are open.
|
||||||
|
onWheel: event => {
|
||||||
|
event.accepted = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
MouseArea {
|
||||||
|
anchors.fill: parent
|
||||||
|
onClicked: root.settingsRequested()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Rectangle {
|
Rectangle {
|
||||||
@@ -74,8 +87,8 @@ Item {
|
|||||||
visible: root.isVisible
|
visible: root.isVisible
|
||||||
anchors.horizontalCenter: parent.horizontalCenter
|
anchors.horizontalCenter: parent.horizontalCenter
|
||||||
anchors.verticalCenter: parent.verticalCenter
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
width: 360
|
width: Math.min(360, root.width - Theme.spacingL * 2)
|
||||||
height: settingsColumn.implicitHeight + Theme.spacingXL * 2
|
height: Math.min(settingsColumn.implicitHeight + Theme.spacingXL * 2, root.height - Theme.spacingL * 2)
|
||||||
radius: Theme.cornerRadius
|
radius: Theme.cornerRadius
|
||||||
color: Qt.rgba(Theme.surfaceContainer.r, Theme.surfaceContainer.g, Theme.surfaceContainer.b, Theme.notepadTransparency)
|
color: Qt.rgba(Theme.surfaceContainer.r, Theme.surfaceContainer.g, Theme.surfaceContainer.b, Theme.notepadTransparency)
|
||||||
border.color: Qt.rgba(Theme.outline.r, Theme.outline.g, Theme.outline.b, 0.08)
|
border.color: Qt.rgba(Theme.outline.r, Theme.outline.g, Theme.outline.b, 0.08)
|
||||||
@@ -93,274 +106,458 @@ Item {
|
|||||||
z: parent.z - 1
|
z: parent.z - 1
|
||||||
}
|
}
|
||||||
|
|
||||||
Column {
|
DankFlickable {
|
||||||
id: settingsColumn
|
id: settingsFlickable
|
||||||
width: parent.width - Theme.spacingXL * 2
|
anchors.fill: parent
|
||||||
anchors.horizontalCenter: parent.horizontalCenter
|
clip: true
|
||||||
anchors.top: parent.top
|
contentWidth: width
|
||||||
anchors.topMargin: Theme.spacingXL
|
contentHeight: settingsColumn.implicitHeight + Theme.spacingXL * 2
|
||||||
spacing: Theme.spacingS
|
|
||||||
|
|
||||||
Rectangle {
|
Column {
|
||||||
width: parent.width
|
id: settingsColumn
|
||||||
height: 36
|
x: Theme.spacingXL
|
||||||
color: "transparent"
|
y: Theme.spacingXL
|
||||||
|
width: settingsFlickable.width - Theme.spacingXL * 2
|
||||||
|
spacing: Theme.spacingS
|
||||||
|
|
||||||
StyledText {
|
Rectangle {
|
||||||
anchors.left: parent.left
|
|
||||||
anchors.leftMargin: -Theme.spacingXS
|
|
||||||
anchors.verticalCenter: parent.verticalCenter
|
|
||||||
text: I18n.tr("Notepad Font Settings")
|
|
||||||
font.pixelSize: Theme.fontSizeMedium
|
|
||||||
font.weight: Font.Medium
|
|
||||||
color: Theme.surfaceText
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Rectangle {
|
|
||||||
width: parent.width
|
|
||||||
height: 1
|
|
||||||
color: Qt.rgba(Theme.outline.r, Theme.outline.g, Theme.outline.b, 0.2)
|
|
||||||
}
|
|
||||||
|
|
||||||
DankToggle {
|
|
||||||
anchors.left: parent.left
|
|
||||||
anchors.leftMargin: -Theme.spacingM
|
|
||||||
width: parent.width + Theme.spacingM
|
|
||||||
text: I18n.tr("Use Monospace Font")
|
|
||||||
description: I18n.tr("Toggle fonts")
|
|
||||||
checked: SettingsData.notepadUseMonospace
|
|
||||||
onToggled: checked => {
|
|
||||||
SettingsData.notepadUseMonospace = checked;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
DankToggle {
|
|
||||||
anchors.left: parent.left
|
|
||||||
anchors.leftMargin: -Theme.spacingM
|
|
||||||
width: parent.width + Theme.spacingM
|
|
||||||
text: I18n.tr("Show Line Numbers")
|
|
||||||
description: I18n.tr("Display line numbers in editor")
|
|
||||||
checked: SettingsData.notepadShowLineNumbers
|
|
||||||
onToggled: checked => {
|
|
||||||
SettingsData.notepadShowLineNumbers = checked;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
StyledRect {
|
|
||||||
width: parent.width
|
|
||||||
height: 60
|
|
||||||
radius: Theme.cornerRadius
|
|
||||||
color: "transparent"
|
|
||||||
|
|
||||||
StateLayer {
|
|
||||||
anchors.fill: parent
|
|
||||||
anchors.leftMargin: -Theme.spacingM
|
|
||||||
width: parent.width + Theme.spacingM
|
|
||||||
stateColor: Theme.primary
|
|
||||||
cornerRadius: parent.radius
|
|
||||||
onClicked: root.findRequested()
|
|
||||||
}
|
|
||||||
|
|
||||||
Row {
|
|
||||||
anchors.left: parent.left
|
|
||||||
anchors.leftMargin: -Theme.spacingM
|
|
||||||
anchors.right: parent.right
|
|
||||||
anchors.rightMargin: Theme.spacingM
|
|
||||||
anchors.verticalCenter: parent.verticalCenter
|
|
||||||
spacing: Theme.spacingM
|
|
||||||
|
|
||||||
DankIcon {
|
|
||||||
name: "search"
|
|
||||||
size: Theme.iconSize - 2
|
|
||||||
color: Theme.primary
|
|
||||||
anchors.verticalCenter: parent.verticalCenter
|
|
||||||
}
|
|
||||||
|
|
||||||
Column {
|
|
||||||
anchors.verticalCenter: parent.verticalCenter
|
|
||||||
spacing: Theme.spacingXS
|
|
||||||
|
|
||||||
StyledText {
|
|
||||||
text: I18n.tr("Find in Text")
|
|
||||||
font.pixelSize: Theme.fontSizeMedium
|
|
||||||
font.weight: Font.Medium
|
|
||||||
color: Theme.surfaceText
|
|
||||||
}
|
|
||||||
|
|
||||||
StyledText {
|
|
||||||
text: I18n.tr("Open search bar to find text")
|
|
||||||
font.pixelSize: Theme.fontSizeSmall
|
|
||||||
color: Theme.surfaceVariantText
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Rectangle {
|
|
||||||
width: parent.width
|
|
||||||
height: visible ? (fontDropdown.height + Theme.spacingS) : 0
|
|
||||||
color: "transparent"
|
|
||||||
visible: !SettingsData.notepadUseMonospace
|
|
||||||
|
|
||||||
DankDropdown {
|
|
||||||
id: fontDropdown
|
|
||||||
anchors.left: parent.left
|
|
||||||
anchors.leftMargin: -Theme.spacingM
|
|
||||||
width: parent.width + Theme.spacingM
|
|
||||||
text: I18n.tr("Font Family")
|
|
||||||
options: cachedFontFamilies
|
|
||||||
currentValue: {
|
|
||||||
if (!SettingsData.notepadFontFamily || SettingsData.notepadFontFamily === "")
|
|
||||||
return I18n.tr("Default (Global)");
|
|
||||||
else
|
|
||||||
return SettingsData.notepadFontFamily;
|
|
||||||
}
|
|
||||||
enableFuzzySearch: true
|
|
||||||
onValueChanged: value => {
|
|
||||||
if (value && (value.startsWith("Default") || value === "Default (Global)")) {
|
|
||||||
SettingsData.notepadFontFamily = "";
|
|
||||||
} else {
|
|
||||||
SettingsData.notepadFontFamily = value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Rectangle {
|
|
||||||
width: parent.width
|
|
||||||
height: fontSizeRow.height + Theme.spacingS
|
|
||||||
color: "transparent"
|
|
||||||
|
|
||||||
Row {
|
|
||||||
id: fontSizeRow
|
|
||||||
width: parent.width
|
width: parent.width
|
||||||
spacing: Theme.spacingS
|
height: 36
|
||||||
|
color: "transparent"
|
||||||
|
|
||||||
Column {
|
StyledText {
|
||||||
width: parent.width - fontSizeControls.width - Theme.spacingM
|
anchors.left: parent.left
|
||||||
spacing: Theme.spacingXS
|
anchors.leftMargin: -Theme.spacingXS
|
||||||
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
|
text: I18n.tr("Notepad Settings")
|
||||||
|
font.pixelSize: Theme.fontSizeMedium
|
||||||
|
font.weight: Font.Medium
|
||||||
|
color: Theme.surfaceText
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
StyledText {
|
Rectangle {
|
||||||
text: I18n.tr("Font Size")
|
width: parent.width
|
||||||
font.pixelSize: Theme.fontSizeSmall
|
height: 1
|
||||||
font.weight: Font.Medium
|
color: Qt.rgba(Theme.outline.r, Theme.outline.g, Theme.outline.b, 0.2)
|
||||||
color: Theme.surfaceText
|
}
|
||||||
}
|
|
||||||
|
|
||||||
StyledText {
|
DankToggle {
|
||||||
text: SettingsData.notepadFontSize + "px"
|
anchors.left: parent.left
|
||||||
font.pixelSize: Theme.fontSizeSmall
|
anchors.leftMargin: -Theme.spacingM
|
||||||
color: Theme.surfaceVariantText
|
width: parent.width + Theme.spacingM
|
||||||
width: parent.width
|
text: I18n.tr("Use Monospace Font")
|
||||||
}
|
description: I18n.tr("Toggle fonts")
|
||||||
|
checked: SettingsData.notepadUseMonospace
|
||||||
|
onToggled: checked => {
|
||||||
|
SettingsData.notepadUseMonospace = checked;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
DankToggle {
|
||||||
|
anchors.left: parent.left
|
||||||
|
anchors.leftMargin: -Theme.spacingM
|
||||||
|
width: parent.width + Theme.spacingM
|
||||||
|
text: I18n.tr("Show Line Numbers")
|
||||||
|
description: I18n.tr("Display line numbers in editor")
|
||||||
|
checked: SettingsData.notepadShowLineNumbers
|
||||||
|
onToggled: checked => {
|
||||||
|
SettingsData.notepadShowLineNumbers = checked;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
DankToggle {
|
||||||
|
anchors.left: parent.left
|
||||||
|
anchors.leftMargin: -Theme.spacingM
|
||||||
|
width: parent.width + Theme.spacingM
|
||||||
|
text: I18n.tr("Auto-save to disk")
|
||||||
|
description: I18n.tr("Automatically save changes to opened files as you type")
|
||||||
|
checked: SettingsData.notepadAutoSave
|
||||||
|
onToggled: checked => {
|
||||||
|
SettingsData.notepadAutoSave = checked;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
StyledRect {
|
||||||
|
width: parent.width
|
||||||
|
height: 60
|
||||||
|
radius: Theme.cornerRadius
|
||||||
|
color: "transparent"
|
||||||
|
|
||||||
|
StateLayer {
|
||||||
|
anchors.fill: parent
|
||||||
|
anchors.leftMargin: -Theme.spacingM
|
||||||
|
width: parent.width + Theme.spacingM
|
||||||
|
stateColor: Theme.primary
|
||||||
|
cornerRadius: parent.radius
|
||||||
|
onClicked: root.findRequested()
|
||||||
}
|
}
|
||||||
|
|
||||||
Row {
|
Row {
|
||||||
id: fontSizeControls
|
anchors.left: parent.left
|
||||||
spacing: Theme.spacingS
|
anchors.leftMargin: -Theme.spacingM
|
||||||
|
anchors.right: parent.right
|
||||||
|
anchors.rightMargin: Theme.spacingM
|
||||||
anchors.verticalCenter: parent.verticalCenter
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
|
spacing: Theme.spacingM
|
||||||
|
|
||||||
DankActionButton {
|
DankIcon {
|
||||||
buttonSize: 32
|
name: "search"
|
||||||
iconName: "remove"
|
size: Theme.iconSize - 2
|
||||||
iconSize: Theme.iconSizeSmall
|
color: Theme.primary
|
||||||
enabled: SettingsData.notepadFontSize > 8
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
backgroundColor: Qt.rgba(Theme.surfaceVariant.r, Theme.surfaceVariant.g, Theme.surfaceVariant.b, 0.5)
|
|
||||||
iconColor: Theme.surfaceText
|
|
||||||
onClicked: {
|
|
||||||
var newSize = Math.max(8, SettingsData.notepadFontSize - 1);
|
|
||||||
SettingsData.notepadFontSize = newSize;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Rectangle {
|
Column {
|
||||||
width: 60
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
height: 32
|
spacing: Theme.spacingXS
|
||||||
radius: Theme.cornerRadius
|
|
||||||
color: Qt.rgba(Theme.surfaceVariant.r, Theme.surfaceVariant.g, Theme.surfaceVariant.b, 0.3)
|
|
||||||
border.color: Qt.rgba(Theme.outline.r, Theme.outline.g, Theme.outline.b, 0.2)
|
|
||||||
border.width: 1
|
|
||||||
|
|
||||||
StyledText {
|
StyledText {
|
||||||
anchors.centerIn: parent
|
text: I18n.tr("Find in Text")
|
||||||
text: SettingsData.notepadFontSize + "px"
|
font.pixelSize: Theme.fontSizeMedium
|
||||||
|
font.weight: Font.Medium
|
||||||
|
color: Theme.surfaceText
|
||||||
|
}
|
||||||
|
|
||||||
|
StyledText {
|
||||||
|
text: I18n.tr("Open search bar to find text")
|
||||||
|
font.pixelSize: Theme.fontSizeSmall
|
||||||
|
color: Theme.surfaceVariantText
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Rectangle {
|
||||||
|
width: parent.width
|
||||||
|
height: visible ? (fontDropdown.height + Theme.spacingS) : 0
|
||||||
|
color: "transparent"
|
||||||
|
visible: !SettingsData.notepadUseMonospace
|
||||||
|
|
||||||
|
DankDropdown {
|
||||||
|
id: fontDropdown
|
||||||
|
anchors.left: parent.left
|
||||||
|
anchors.leftMargin: -Theme.spacingM
|
||||||
|
width: parent.width + Theme.spacingM
|
||||||
|
text: I18n.tr("Font Family")
|
||||||
|
options: cachedFontFamilies
|
||||||
|
currentValue: {
|
||||||
|
if (!SettingsData.notepadFontFamily || SettingsData.notepadFontFamily === "")
|
||||||
|
return I18n.tr("Default (Global)");
|
||||||
|
else
|
||||||
|
return SettingsData.notepadFontFamily;
|
||||||
|
}
|
||||||
|
enableFuzzySearch: true
|
||||||
|
onValueChanged: value => {
|
||||||
|
if (value && (value.startsWith("Default") || value === "Default (Global)")) {
|
||||||
|
SettingsData.notepadFontFamily = "";
|
||||||
|
} else {
|
||||||
|
SettingsData.notepadFontFamily = value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Rectangle {
|
||||||
|
width: parent.width
|
||||||
|
height: fontSizeRow.height + Theme.spacingS
|
||||||
|
color: "transparent"
|
||||||
|
|
||||||
|
Row {
|
||||||
|
id: fontSizeRow
|
||||||
|
width: parent.width
|
||||||
|
spacing: Theme.spacingS
|
||||||
|
|
||||||
|
Column {
|
||||||
|
width: parent.width - fontSizeControls.width - Theme.spacingM
|
||||||
|
spacing: Theme.spacingXS
|
||||||
|
|
||||||
|
StyledText {
|
||||||
|
text: I18n.tr("Font Size")
|
||||||
font.pixelSize: Theme.fontSizeSmall
|
font.pixelSize: Theme.fontSizeSmall
|
||||||
font.weight: Font.Medium
|
font.weight: Font.Medium
|
||||||
color: Theme.surfaceText
|
color: Theme.surfaceText
|
||||||
}
|
}
|
||||||
|
|
||||||
|
StyledText {
|
||||||
|
text: SettingsData.notepadFontSize + "px"
|
||||||
|
font.pixelSize: Theme.fontSizeSmall
|
||||||
|
color: Theme.surfaceVariantText
|
||||||
|
width: parent.width
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
DankActionButton {
|
Row {
|
||||||
buttonSize: 32
|
id: fontSizeControls
|
||||||
iconName: "add"
|
spacing: Theme.spacingS
|
||||||
iconSize: Theme.iconSizeSmall
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
enabled: SettingsData.notepadFontSize < 48
|
|
||||||
backgroundColor: Qt.rgba(Theme.surfaceVariant.r, Theme.surfaceVariant.g, Theme.surfaceVariant.b, 0.5)
|
DankActionButton {
|
||||||
iconColor: Theme.surfaceText
|
buttonSize: 32
|
||||||
onClicked: {
|
iconName: "remove"
|
||||||
var newSize = Math.min(48, SettingsData.notepadFontSize + 1);
|
iconSize: Theme.iconSizeSmall
|
||||||
SettingsData.notepadFontSize = newSize;
|
enabled: SettingsData.notepadFontSize > 8
|
||||||
|
backgroundColor: Qt.rgba(Theme.surfaceVariant.r, Theme.surfaceVariant.g, Theme.surfaceVariant.b, 0.5)
|
||||||
|
iconColor: Theme.surfaceText
|
||||||
|
onClicked: {
|
||||||
|
var newSize = Math.max(8, SettingsData.notepadFontSize - 1);
|
||||||
|
SettingsData.notepadFontSize = newSize;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Rectangle {
|
||||||
|
width: 60
|
||||||
|
height: 32
|
||||||
|
radius: Theme.cornerRadius
|
||||||
|
color: Qt.rgba(Theme.surfaceVariant.r, Theme.surfaceVariant.g, Theme.surfaceVariant.b, 0.3)
|
||||||
|
border.color: Qt.rgba(Theme.outline.r, Theme.outline.g, Theme.outline.b, 0.2)
|
||||||
|
border.width: 1
|
||||||
|
|
||||||
|
StyledText {
|
||||||
|
anchors.centerIn: parent
|
||||||
|
text: SettingsData.notepadFontSize + "px"
|
||||||
|
font.pixelSize: Theme.fontSizeSmall
|
||||||
|
font.weight: Font.Medium
|
||||||
|
color: Theme.surfaceText
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
DankActionButton {
|
||||||
|
buttonSize: 32
|
||||||
|
iconName: "add"
|
||||||
|
iconSize: Theme.iconSizeSmall
|
||||||
|
enabled: SettingsData.notepadFontSize < 48
|
||||||
|
backgroundColor: Qt.rgba(Theme.surfaceVariant.r, Theme.surfaceVariant.g, Theme.surfaceVariant.b, 0.5)
|
||||||
|
iconColor: Theme.surfaceText
|
||||||
|
onClicked: {
|
||||||
|
var newSize = Math.min(48, SettingsData.notepadFontSize + 1);
|
||||||
|
SettingsData.notepadFontSize = newSize;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
Rectangle {
|
Rectangle {
|
||||||
width: parent.width
|
|
||||||
height: transparencySliderColumn.height + Theme.spacingS
|
|
||||||
color: "transparent"
|
|
||||||
|
|
||||||
Column {
|
|
||||||
id: transparencySliderColumn
|
|
||||||
width: parent.width
|
width: parent.width
|
||||||
spacing: Theme.spacingS
|
height: transparencySliderColumn.height + Theme.spacingS
|
||||||
|
color: "transparent"
|
||||||
|
|
||||||
DankToggle {
|
Column {
|
||||||
anchors.left: parent.left
|
id: transparencySliderColumn
|
||||||
anchors.leftMargin: -Theme.spacingM
|
width: parent.width
|
||||||
width: parent.width + Theme.spacingM
|
spacing: Theme.spacingS
|
||||||
text: I18n.tr("Custom Transparency")
|
|
||||||
description: I18n.tr("Override global transparency for Notepad")
|
DankToggle {
|
||||||
checked: SettingsData.notepadTransparencyOverride >= 0
|
anchors.left: parent.left
|
||||||
onToggled: checked => {
|
anchors.leftMargin: -Theme.spacingM
|
||||||
if (checked) {
|
width: parent.width + Theme.spacingM
|
||||||
SettingsData.notepadTransparencyOverride = SettingsData.notepadLastCustomTransparency;
|
text: I18n.tr("Surface Opacity")
|
||||||
} else {
|
description: I18n.tr("Override global transparency for Notepad")
|
||||||
SettingsData.notepadTransparencyOverride = -1;
|
checked: SettingsData.notepadTransparencyOverride >= 0
|
||||||
|
onToggled: checked => {
|
||||||
|
if (checked) {
|
||||||
|
SettingsData.notepadTransparencyOverride = SettingsData.notepadLastCustomTransparency;
|
||||||
|
} else {
|
||||||
|
SettingsData.notepadTransparencyOverride = -1;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
DankSlider {
|
DankSlider {
|
||||||
anchors.left: parent.left
|
anchors.left: parent.left
|
||||||
anchors.leftMargin: -Theme.spacingM
|
anchors.leftMargin: -Theme.spacingM
|
||||||
width: parent.width + Theme.spacingM
|
width: parent.width + Theme.spacingM
|
||||||
height: 24
|
height: 24
|
||||||
visible: SettingsData.notepadTransparencyOverride >= 0
|
visible: SettingsData.notepadTransparencyOverride >= 0
|
||||||
value: Math.round((SettingsData.notepadTransparencyOverride >= 0 ? SettingsData.notepadTransparencyOverride : SettingsData.popupTransparency) * 100)
|
value: Math.round((SettingsData.notepadTransparencyOverride >= 0 ? SettingsData.notepadTransparencyOverride : SettingsData.popupTransparency) * 100)
|
||||||
minimum: 0
|
minimum: 0
|
||||||
maximum: 100
|
maximum: 100
|
||||||
unit: ""
|
unit: ""
|
||||||
showValue: true
|
showValue: true
|
||||||
wheelEnabled: false
|
wheelEnabled: false
|
||||||
onSliderValueChanged: newValue => {
|
onSliderValueChanged: newValue => {
|
||||||
if (SettingsData.notepadTransparencyOverride >= 0) {
|
if (SettingsData.notepadTransparencyOverride >= 0) {
|
||||||
SettingsData.notepadTransparencyOverride = newValue / 100;
|
SettingsData.notepadTransparencyOverride = newValue / 100;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
StyledText {
|
Rectangle {
|
||||||
width: parent.width
|
width: parent.width
|
||||||
text: SettingsData.notepadUseMonospace ? I18n.tr("Using global monospace font from Settings → Personalization") : I18n.tr("Global fonts can be configured in Settings → Personalization")
|
height: gapColumn.height + Theme.spacingS
|
||||||
font.pixelSize: Theme.fontSizeSmall
|
color: "transparent"
|
||||||
color: Theme.surfaceTextMedium
|
|
||||||
wrapMode: Text.WordWrap
|
Column {
|
||||||
opacity: 0.8
|
id: gapColumn
|
||||||
|
width: parent.width
|
||||||
|
spacing: Theme.spacingS
|
||||||
|
|
||||||
|
Column {
|
||||||
|
width: parent.width
|
||||||
|
spacing: Theme.spacingXS
|
||||||
|
|
||||||
|
StyledText {
|
||||||
|
text: I18n.tr("Default Mode")
|
||||||
|
font.pixelSize: Theme.fontSizeSmall
|
||||||
|
font.weight: Font.Medium
|
||||||
|
color: Theme.surfaceText
|
||||||
|
}
|
||||||
|
|
||||||
|
DankButtonGroup {
|
||||||
|
model: [I18n.tr("Slideout"), I18n.tr("Popout")]
|
||||||
|
size: "small"
|
||||||
|
currentIndex: SettingsData.notepadDefaultMode === "popout" ? 1 : 0
|
||||||
|
onSelectionChanged: (index, selected) => {
|
||||||
|
if (!selected)
|
||||||
|
return;
|
||||||
|
SettingsData.notepadDefaultMode = index === 1 ? "popout" : "slideout";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Column {
|
||||||
|
width: parent.width
|
||||||
|
spacing: Theme.spacingXS
|
||||||
|
visible: SettingsData.notepadDefaultMode !== "popout"
|
||||||
|
|
||||||
|
StyledText {
|
||||||
|
text: I18n.tr("Open From")
|
||||||
|
font.pixelSize: Theme.fontSizeSmall
|
||||||
|
font.weight: Font.Medium
|
||||||
|
color: Theme.surfaceText
|
||||||
|
}
|
||||||
|
|
||||||
|
DankButtonGroup {
|
||||||
|
model: [I18n.tr("Right"), I18n.tr("Left")]
|
||||||
|
size: "small"
|
||||||
|
currentIndex: SettingsData.notepadSlideoutSide === "left" ? 1 : 0
|
||||||
|
onSelectionChanged: (index, selected) => {
|
||||||
|
if (!selected)
|
||||||
|
return;
|
||||||
|
SettingsData.notepadSlideoutSide = index === 1 ? "left" : "right";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
DankToggle {
|
||||||
|
anchors.left: parent.left
|
||||||
|
anchors.leftMargin: -Theme.spacingM
|
||||||
|
width: parent.width + Theme.spacingM
|
||||||
|
text: I18n.tr("Auto Compositor Gaps")
|
||||||
|
description: I18n.tr("Inset the Notepad from screen edges using the compositor's configured gaps")
|
||||||
|
checked: SettingsData.notepadUseCompositorGap
|
||||||
|
onToggled: checked => {
|
||||||
|
SettingsData.notepadUseCompositorGap = checked;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
StyledText {
|
||||||
|
visible: !SettingsData.notepadUseCompositorGap
|
||||||
|
text: I18n.tr("Manual Gaps")
|
||||||
|
font.pixelSize: Theme.fontSizeSmall
|
||||||
|
font.weight: Font.Medium
|
||||||
|
color: Theme.surfaceText
|
||||||
|
}
|
||||||
|
|
||||||
|
DankSlider {
|
||||||
|
anchors.left: parent.left
|
||||||
|
anchors.leftMargin: Theme.spacingXS
|
||||||
|
width: parent.width - Theme.spacingXS * 2
|
||||||
|
height: 24
|
||||||
|
visible: !SettingsData.notepadUseCompositorGap
|
||||||
|
value: SettingsData.notepadEdgeGap
|
||||||
|
minimum: 0
|
||||||
|
maximum: 64
|
||||||
|
unit: "px"
|
||||||
|
showValue: true
|
||||||
|
wheelEnabled: false
|
||||||
|
onSliderValueChanged: newValue => {
|
||||||
|
SettingsData.notepadEdgeGap = newValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
StyledText {
|
||||||
|
width: parent.width
|
||||||
|
text: SettingsData.notepadUseMonospace ? I18n.tr("Using global monospace font from Settings → Personalization") : I18n.tr("Global fonts can be configured in Settings → Personalization")
|
||||||
|
font.pixelSize: Theme.fontSizeSmall
|
||||||
|
color: Theme.surfaceTextMedium
|
||||||
|
wrapMode: Text.WordWrap
|
||||||
|
opacity: 0.8
|
||||||
|
}
|
||||||
|
|
||||||
|
StyledRect {
|
||||||
|
width: parent.width
|
||||||
|
implicitHeight: shortcutsHeader.height + (root.shortcutsExpanded ? shortcutsColumn.implicitHeight + Theme.spacingM : 0)
|
||||||
|
radius: Theme.cornerRadius
|
||||||
|
color: root.shortcutsExpanded ? Qt.rgba(Theme.surfaceContainer.r, Theme.surfaceContainer.g, Theme.surfaceContainer.b, 0.95) : "transparent"
|
||||||
|
border.color: root.shortcutsExpanded ? Theme.primary : Theme.outlineMedium
|
||||||
|
border.width: root.shortcutsExpanded ? 2 : 1
|
||||||
|
|
||||||
|
StateLayer {
|
||||||
|
anchors.fill: parent
|
||||||
|
stateColor: Theme.primary
|
||||||
|
cornerRadius: parent.radius
|
||||||
|
onClicked: root.shortcutsExpanded = !root.shortcutsExpanded
|
||||||
|
}
|
||||||
|
|
||||||
|
Row {
|
||||||
|
id: shortcutsHeader
|
||||||
|
anchors.top: parent.top
|
||||||
|
anchors.left: parent.left
|
||||||
|
anchors.right: parent.right
|
||||||
|
anchors.leftMargin: Theme.spacingM
|
||||||
|
anchors.rightMargin: Theme.spacingS
|
||||||
|
height: 36
|
||||||
|
spacing: Theme.spacingS
|
||||||
|
|
||||||
|
DankIcon {
|
||||||
|
name: root.shortcutsExpanded ? "expand_less" : "expand_more"
|
||||||
|
size: Theme.iconSizeSmall
|
||||||
|
color: Theme.surfaceText
|
||||||
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
|
}
|
||||||
|
|
||||||
|
StyledText {
|
||||||
|
text: I18n.tr("Keyboard Shortcuts")
|
||||||
|
font.pixelSize: Theme.fontSizeSmall
|
||||||
|
font.weight: Font.Medium
|
||||||
|
color: Theme.surfaceText
|
||||||
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Column {
|
||||||
|
id: shortcutsColumn
|
||||||
|
visible: root.shortcutsExpanded
|
||||||
|
width: parent.width - Theme.spacingL * 2
|
||||||
|
anchors.top: shortcutsHeader.bottom
|
||||||
|
anchors.horizontalCenter: parent.horizontalCenter
|
||||||
|
spacing: 2
|
||||||
|
|
||||||
|
StyledText {
|
||||||
|
width: parent.width
|
||||||
|
text: I18n.tr("Ctrl+S: Save • Ctrl+O: Open • Ctrl+N: New • Ctrl+F: Find")
|
||||||
|
font.pixelSize: Theme.fontSizeSmall
|
||||||
|
color: Theme.surfaceText
|
||||||
|
wrapMode: Text.WordWrap
|
||||||
|
horizontalAlignment: Text.AlignHCenter
|
||||||
|
}
|
||||||
|
|
||||||
|
StyledText {
|
||||||
|
width: parent.width
|
||||||
|
text: I18n.tr("Ctrl+A: Select All • Ctrl+P: Preview • Enter/Shift+Enter: Find Next/Previous • Esc: Close")
|
||||||
|
font.pixelSize: Theme.fontSizeSmall
|
||||||
|
color: Theme.surfaceText
|
||||||
|
wrapMode: Text.WordWrap
|
||||||
|
horizontalAlignment: Text.AlignHCenter
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -32,6 +32,23 @@ Column {
|
|||||||
property string pluginHighlightedHtml: ""
|
property string pluginHighlightedHtml: ""
|
||||||
property string lastPluginContent: ""
|
property string lastPluginContent: ""
|
||||||
property int loadRequestId: 0
|
property int loadRequestId: 0
|
||||||
|
property bool ignoreNextExternalChange: false
|
||||||
|
property bool watcherReloadPending: false
|
||||||
|
property bool externalWatchPaused: false
|
||||||
|
property bool inPopout: false
|
||||||
|
property bool surfaceVisible: true
|
||||||
|
// Tab ids are Date.now() timestamps (~1.78e12) which overflow a 32-bit `int`,
|
||||||
|
// corrupting the value (e.g. -946062153) and breaking buffer keying. `var`
|
||||||
|
// holds the full JS-safe integer.
|
||||||
|
property var loadedTabId: -1
|
||||||
|
property bool applyingShared: false
|
||||||
|
property bool showPathInfo: false
|
||||||
|
|
||||||
|
function currentFilePath() {
|
||||||
|
if (!currentTab)
|
||||||
|
return "";
|
||||||
|
return currentTab.isTemporary ? (NotepadStorageService.baseDir + "/" + currentTab.filePath) : currentTab.filePath;
|
||||||
|
}
|
||||||
|
|
||||||
signal saveRequested
|
signal saveRequested
|
||||||
signal openRequested
|
signal openRequested
|
||||||
@@ -40,6 +57,10 @@ Column {
|
|||||||
signal escapePressed
|
signal escapePressed
|
||||||
signal contentChanged
|
signal contentChanged
|
||||||
signal settingsRequested
|
signal settingsRequested
|
||||||
|
signal popoutRequested
|
||||||
|
signal dockRequested
|
||||||
|
signal conflictDetected(string diskContent)
|
||||||
|
signal autoSaveRequested
|
||||||
|
|
||||||
function hasUnsavedChanges() {
|
function hasUnsavedChanges() {
|
||||||
if (!currentTab || !contentLoaded) {
|
if (!currentTab || !contentLoaded) {
|
||||||
@@ -52,6 +73,12 @@ Column {
|
|||||||
return textArea.text !== lastSavedContent;
|
return textArea.text !== lastSavedContent;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function commitLiveBuffer() {
|
||||||
|
if (loadedTabId < 0 || !contentLoaded)
|
||||||
|
return;
|
||||||
|
NotepadStorageService.setSessionBuffer(loadedTabId, textArea.text, lastSavedContent);
|
||||||
|
}
|
||||||
|
|
||||||
function loadCurrentTabContent() {
|
function loadCurrentTabContent() {
|
||||||
if (!currentTab)
|
if (!currentTab)
|
||||||
return;
|
return;
|
||||||
@@ -62,8 +89,25 @@ Column {
|
|||||||
const activeTab = NotepadStorageService.tabs.length > NotepadStorageService.currentTabIndex ? NotepadStorageService.tabs[NotepadStorageService.currentTabIndex] : null;
|
const activeTab = NotepadStorageService.tabs.length > NotepadStorageService.currentTabIndex ? NotepadStorageService.tabs[NotepadStorageService.currentTabIndex] : null;
|
||||||
if (requestId !== loadRequestId || !activeTab || activeTab.id !== requestedTabId)
|
if (requestId !== loadRequestId || !activeTab || activeTab.id !== requestedTabId)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
const buffer = NotepadStorageService.getSessionBuffer(requestedTabId);
|
||||||
|
if (buffer !== undefined) {
|
||||||
|
applyingShared = true;
|
||||||
|
lastSavedContent = buffer.baseline;
|
||||||
|
textArea.text = buffer.content;
|
||||||
|
applyingShared = false;
|
||||||
|
loadedTabId = requestedTabId;
|
||||||
|
contentLoaded = true;
|
||||||
|
syncContentToPlugin();
|
||||||
|
applyDiskContent(content);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
applyingShared = true;
|
||||||
lastSavedContent = content;
|
lastSavedContent = content;
|
||||||
textArea.text = content;
|
textArea.text = content;
|
||||||
|
applyingShared = false;
|
||||||
|
loadedTabId = requestedTabId;
|
||||||
contentLoaded = true;
|
contentLoaded = true;
|
||||||
syncContentToPlugin();
|
syncContentToPlugin();
|
||||||
});
|
});
|
||||||
@@ -72,14 +116,56 @@ Column {
|
|||||||
function saveCurrentTabContent() {
|
function saveCurrentTabContent() {
|
||||||
if (!currentTab || !contentLoaded)
|
if (!currentTab || !contentLoaded)
|
||||||
return;
|
return;
|
||||||
|
if (!currentTab.isTemporary)
|
||||||
|
return;
|
||||||
NotepadStorageService.saveTabContent(NotepadStorageService.currentTabIndex, textArea.text);
|
NotepadStorageService.saveTabContent(NotepadStorageService.currentTabIndex, textArea.text);
|
||||||
lastSavedContent = textArea.text;
|
lastSavedContent = textArea.text;
|
||||||
|
NotepadStorageService.clearSessionBuffer(loadedTabId);
|
||||||
}
|
}
|
||||||
|
|
||||||
function autoSaveToSession() {
|
function autoSaveToSession() {
|
||||||
|
commitLiveBuffer();
|
||||||
if (!currentTab || !contentLoaded)
|
if (!currentTab || !contentLoaded)
|
||||||
return;
|
return;
|
||||||
saveCurrentTabContent();
|
if (currentTab.isTemporary) {
|
||||||
|
saveCurrentTabContent();
|
||||||
|
} else if (SettingsData.notepadAutoSave) {
|
||||||
|
root.autoSaveRequested();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function syncFromDisk() {
|
||||||
|
if (!currentTab)
|
||||||
|
return;
|
||||||
|
loadCurrentTabContent();
|
||||||
|
}
|
||||||
|
|
||||||
|
function applyDiskContent(diskContent) {
|
||||||
|
if (diskContent === undefined || diskContent === null)
|
||||||
|
return;
|
||||||
|
if (diskContent === textArea.text) {
|
||||||
|
lastSavedContent = diskContent;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (diskContent === lastSavedContent) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (textArea.text === lastSavedContent) {
|
||||||
|
reloadFromDisk(diskContent);
|
||||||
|
} else if (surfaceVisible) {
|
||||||
|
conflictDetected(diskContent);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function reloadFromDisk(diskContent) {
|
||||||
|
applyingShared = true;
|
||||||
|
contentLoaded = false;
|
||||||
|
textArea.text = diskContent;
|
||||||
|
lastSavedContent = diskContent;
|
||||||
|
contentLoaded = true;
|
||||||
|
applyingShared = false;
|
||||||
|
NotepadStorageService.clearSessionBuffer(loadedTabId);
|
||||||
|
syncContentToPlugin();
|
||||||
}
|
}
|
||||||
|
|
||||||
function setTextDocumentLineHeight() {
|
function setTextDocumentLineHeight() {
|
||||||
@@ -202,7 +288,8 @@ Column {
|
|||||||
if (!currentTab)
|
if (!currentTab)
|
||||||
return;
|
return;
|
||||||
const filePath = currentTab?.filePath || "";
|
const filePath = currentTab?.filePath || "";
|
||||||
const ext = filePath.split('.').pop().toLowerCase();
|
const baseName = filePath.split('/').pop();
|
||||||
|
const ext = baseName.includes('.') ? baseName.split('.').pop().toLowerCase() : "";
|
||||||
const content = textArea.text;
|
const content = textArea.text;
|
||||||
|
|
||||||
if (content === lastPluginContent && SettingsData.getBuiltInPluginSetting("dankNotepadModule", "previewActive", false) === inlinePreviewVisible) {
|
if (content === lastPluginContent && SettingsData.getBuiltInPluginSetting("dankNotepadModule", "previewActive", false) === inlinePreviewVisible) {
|
||||||
@@ -550,6 +637,7 @@ Column {
|
|||||||
Connections {
|
Connections {
|
||||||
target: NotepadStorageService
|
target: NotepadStorageService
|
||||||
function onCurrentTabIndexChanged() {
|
function onCurrentTabIndexChanged() {
|
||||||
|
root.commitLiveBuffer();
|
||||||
loadCurrentTabContent();
|
loadCurrentTabContent();
|
||||||
Qt.callLater(() => {
|
Qt.callLater(() => {
|
||||||
textArea.forceActiveFocus();
|
textArea.forceActiveFocus();
|
||||||
@@ -570,7 +658,9 @@ Column {
|
|||||||
}
|
}
|
||||||
|
|
||||||
onTextChanged: {
|
onTextChanged: {
|
||||||
if (contentLoaded && text !== lastSavedContent) {
|
// Debounced flush to the shared buffer (+ optional disk
|
||||||
|
// autosave) for every loaded tab, not just scratch notes.
|
||||||
|
if (contentLoaded && !applyingShared) {
|
||||||
autoSaveTimer.restart();
|
autoSaveTimer.restart();
|
||||||
}
|
}
|
||||||
root.contentChanged();
|
root.contentChanged();
|
||||||
@@ -744,6 +834,7 @@ Column {
|
|||||||
spacing: Theme.spacingS
|
spacing: Theme.spacingS
|
||||||
|
|
||||||
Item {
|
Item {
|
||||||
|
id: buttonBarItem
|
||||||
width: parent.width
|
width: parent.width
|
||||||
height: 32
|
height: 32
|
||||||
|
|
||||||
@@ -820,17 +911,98 @@ Column {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
DankActionButton {
|
Row {
|
||||||
|
id: rightButtonRow
|
||||||
anchors.right: parent.right
|
anchors.right: parent.right
|
||||||
anchors.verticalCenter: parent.verticalCenter
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
iconName: "more_horiz"
|
spacing: Theme.spacingS
|
||||||
iconSize: Theme.iconSize - 2
|
|
||||||
iconColor: Theme.surfaceText
|
DankActionButton {
|
||||||
onClicked: root.settingsRequested()
|
visible: !root.inPopout
|
||||||
|
iconName: "open_in_new"
|
||||||
|
iconSize: Theme.iconSize - 2
|
||||||
|
iconColor: Theme.surfaceText
|
||||||
|
onClicked: root.popoutRequested()
|
||||||
|
}
|
||||||
|
|
||||||
|
DankActionButton {
|
||||||
|
visible: root.inPopout
|
||||||
|
iconName: "dock_to_right"
|
||||||
|
iconSize: Theme.iconSize - 2
|
||||||
|
iconColor: Theme.surfaceText
|
||||||
|
onClicked: root.dockRequested()
|
||||||
|
}
|
||||||
|
|
||||||
|
DankActionButton {
|
||||||
|
iconName: "more_horiz"
|
||||||
|
iconSize: Theme.iconSize - 2
|
||||||
|
iconColor: Theme.surfaceText
|
||||||
|
onClicked: root.settingsRequested()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
StyledRect {
|
||||||
|
id: pathInfoPopup
|
||||||
|
visible: root.showPathInfo
|
||||||
|
anchors.right: parent.right
|
||||||
|
anchors.bottom: parent.top
|
||||||
|
anchors.bottomMargin: Theme.spacingS
|
||||||
|
width: Math.min(root.width, 360)
|
||||||
|
height: pathInfoRow.implicitHeight + Theme.spacingS * 2
|
||||||
|
radius: Theme.cornerRadius
|
||||||
|
color: Theme.withAlpha(Theme.surfaceContainerHigh, Theme.popupTransparency)
|
||||||
|
border.color: Theme.outlineMedium
|
||||||
|
border.width: 1
|
||||||
|
z: 10
|
||||||
|
|
||||||
|
Row {
|
||||||
|
id: pathInfoRow
|
||||||
|
anchors.left: parent.left
|
||||||
|
anchors.right: parent.right
|
||||||
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
|
anchors.leftMargin: Theme.spacingM
|
||||||
|
anchors.rightMargin: Theme.spacingM
|
||||||
|
spacing: Theme.spacingS
|
||||||
|
|
||||||
|
DankIcon {
|
||||||
|
name: currentTab && currentTab.isTemporary ? "draft" : "description"
|
||||||
|
size: Theme.iconSize - 4
|
||||||
|
color: Theme.surfaceVariantText
|
||||||
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
|
}
|
||||||
|
|
||||||
|
StyledText {
|
||||||
|
width: pathInfoRow.width - (Theme.iconSize - 4) - copyPathButton.width - Theme.spacingS * 2
|
||||||
|
text: root.currentFilePath()
|
||||||
|
font.pixelSize: Theme.fontSizeSmall
|
||||||
|
color: Theme.surfaceText
|
||||||
|
elide: Text.ElideMiddle
|
||||||
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
|
}
|
||||||
|
|
||||||
|
DankActionButton {
|
||||||
|
id: copyPathButton
|
||||||
|
iconName: "content_copy"
|
||||||
|
iconSize: Theme.iconSize - 6
|
||||||
|
iconColor: Theme.surfaceTextMedium
|
||||||
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
|
onClicked: {
|
||||||
|
const proc = clipboardCopyProcComp.createObject(root, {
|
||||||
|
content: root.currentFilePath(),
|
||||||
|
running: true
|
||||||
|
});
|
||||||
|
proc.exited.connect(() => {
|
||||||
|
ToastService.showInfo(I18n.tr("Path copied to clipboard"));
|
||||||
|
proc.destroy();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Row {
|
Row {
|
||||||
|
id: statusRow
|
||||||
width: parent.width
|
width: parent.width
|
||||||
spacing: Theme.spacingL
|
spacing: Theme.spacingL
|
||||||
|
|
||||||
@@ -853,35 +1025,46 @@ Column {
|
|||||||
opacity: 1.0
|
opacity: 1.0
|
||||||
}
|
}
|
||||||
|
|
||||||
StyledText {
|
Row {
|
||||||
text: {
|
visible: textArea.text.length > 0
|
||||||
if (autoSaveTimer.running) {
|
spacing: Theme.spacingXS
|
||||||
return I18n.tr("Auto-saving...");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (hasUnsavedChanges()) {
|
StyledText {
|
||||||
if (currentTab && currentTab.isTemporary) {
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
return I18n.tr("Unsaved note...");
|
readonly property bool savingToDisk: autoSaveTimer.running && currentTab && (currentTab.isTemporary || SettingsData.notepadAutoSave)
|
||||||
} else {
|
text: {
|
||||||
return I18n.tr("Unsaved changes");
|
if (savingToDisk) {
|
||||||
|
return I18n.tr("Saving...");
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
return I18n.tr("Saved");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
font.pixelSize: Theme.fontSizeSmall
|
|
||||||
color: {
|
|
||||||
if (autoSaveTimer.running) {
|
|
||||||
return Theme.primary;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (hasUnsavedChanges()) {
|
if (currentTab && currentTab.isTemporary) {
|
||||||
return Theme.warning;
|
return I18n.tr("Auto saved");
|
||||||
} else {
|
}
|
||||||
return Theme.success;
|
|
||||||
|
return hasUnsavedChanges() ? I18n.tr("Unsaved changes") : I18n.tr("Saved");
|
||||||
|
}
|
||||||
|
font.pixelSize: Theme.fontSizeSmall
|
||||||
|
color: {
|
||||||
|
if (savingToDisk) {
|
||||||
|
return Theme.primary;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (currentTab && currentTab.isTemporary) {
|
||||||
|
return Theme.success;
|
||||||
|
}
|
||||||
|
|
||||||
|
return hasUnsavedChanges() ? Theme.warning : Theme.success;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
opacity: textArea.text.length > 0 ? 1.0 : 0.0
|
|
||||||
|
DankActionButton {
|
||||||
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
|
iconName: "info"
|
||||||
|
iconSize: Theme.iconSizeSmall
|
||||||
|
iconColor: root.showPathInfo ? Theme.primary : Theme.surfaceTextMedium
|
||||||
|
buttonSize: 20
|
||||||
|
onClicked: root.showPathInfo = !root.showPathInfo
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -902,6 +1085,38 @@ Column {
|
|||||||
onTriggered: syncContentToPlugin()
|
onTriggered: syncContentToPlugin()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
FileView {
|
||||||
|
id: externalWatch
|
||||||
|
path: (!root.externalWatchPaused && currentTab && !currentTab.isTemporary && currentTab.filePath) ? currentTab.filePath : ""
|
||||||
|
blockLoading: true
|
||||||
|
preload: true
|
||||||
|
watchChanges: true
|
||||||
|
|
||||||
|
onFileChanged: {
|
||||||
|
root.watcherReloadPending = true;
|
||||||
|
reload();
|
||||||
|
}
|
||||||
|
|
||||||
|
onLoaded: {
|
||||||
|
if (root.ignoreNextExternalChange) {
|
||||||
|
root.ignoreNextExternalChange = false;
|
||||||
|
root.lastSavedContent = externalWatch.text();
|
||||||
|
root.watcherReloadPending = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!root.watcherReloadPending)
|
||||||
|
return;
|
||||||
|
root.watcherReloadPending = false;
|
||||||
|
if (!root.contentLoaded || !root.currentTab || root.currentTab.isTemporary)
|
||||||
|
return;
|
||||||
|
if (!root.surfaceVisible)
|
||||||
|
return;
|
||||||
|
root.applyDiskContent(externalWatch.text());
|
||||||
|
}
|
||||||
|
|
||||||
|
onLoadFailed: error => {}
|
||||||
|
}
|
||||||
|
|
||||||
Connections {
|
Connections {
|
||||||
target: SettingsData
|
target: SettingsData
|
||||||
function onBuiltInPluginSettingsChanged() {
|
function onBuiltInPluginSettingsChanged() {
|
||||||
@@ -910,4 +1125,24 @@ Column {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Connections {
|
||||||
|
target: NotepadStorageService
|
||||||
|
function onSessionBufferRevisionChanged() {
|
||||||
|
if (applyingShared || !contentLoaded || loadedTabId < 0)
|
||||||
|
return;
|
||||||
|
if (textArea.activeFocus)
|
||||||
|
return;
|
||||||
|
var buffer = NotepadStorageService.getSessionBuffer(loadedTabId);
|
||||||
|
if (buffer === undefined || buffer.content === textArea.text)
|
||||||
|
return;
|
||||||
|
if (textArea.text === lastSavedContent) {
|
||||||
|
applyingShared = true;
|
||||||
|
lastSavedContent = buffer.baseline;
|
||||||
|
textArea.text = buffer.content;
|
||||||
|
applyingShared = false;
|
||||||
|
syncContentToPlugin();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -330,24 +330,6 @@ Item {
|
|||||||
pluginPopout.toggle();
|
pluginPopout.toggle();
|
||||||
}
|
}
|
||||||
|
|
||||||
function triggerHoverPopout(widgetHostId) {
|
|
||||||
if (pillClickAction) {
|
|
||||||
triggerPopout();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (!hasPopout)
|
|
||||||
return;
|
|
||||||
|
|
||||||
const pill = isVertical ? verticalPill : horizontalPill;
|
|
||||||
const globalPos = pill.visualContent.mapToItem(null, 0, 0);
|
|
||||||
const currentScreen = parentScreen || Screen;
|
|
||||||
const barPosition = axis?.edge === "left" ? 2 : (axis?.edge === "right" ? 3 : (axis?.edge === "top" ? 0 : 1));
|
|
||||||
const pos = SettingsData.getPopupTriggerPosition(globalPos, currentScreen, barThickness, pill.visualWidth, barSpacing, barPosition, barConfig);
|
|
||||||
|
|
||||||
pluginPopout.setTriggerPosition(pos.x, pos.y, pos.width, section, currentScreen, barPosition, barThickness, barSpacing, barConfig);
|
|
||||||
PopoutManager.requestHoverPopout(pluginPopout, undefined, widgetHostId || pluginId);
|
|
||||||
}
|
|
||||||
|
|
||||||
PluginPopout {
|
PluginPopout {
|
||||||
id: pluginPopout
|
id: pluginPopout
|
||||||
contentWidth: root.popoutWidth
|
contentWidth: root.popoutWidth
|
||||||
|
|||||||
@@ -164,7 +164,6 @@ Item {
|
|||||||
scrollEnabled: defaultBar.scrollEnabled ?? true,
|
scrollEnabled: defaultBar.scrollEnabled ?? true,
|
||||||
scrollXBehavior: defaultBar.scrollXBehavior ?? "column",
|
scrollXBehavior: defaultBar.scrollXBehavior ?? "column",
|
||||||
scrollYBehavior: defaultBar.scrollYBehavior ?? "workspace",
|
scrollYBehavior: defaultBar.scrollYBehavior ?? "workspace",
|
||||||
hoverPopouts: defaultBar.hoverPopouts ?? false,
|
|
||||||
shadowIntensity: defaultBar.shadowIntensity ?? 0,
|
shadowIntensity: defaultBar.shadowIntensity ?? 0,
|
||||||
shadowOpacity: defaultBar.shadowOpacity ?? 60,
|
shadowOpacity: defaultBar.shadowOpacity ?? 60,
|
||||||
shadowDirectionMode: defaultBar.shadowDirectionMode ?? "inherit",
|
shadowDirectionMode: defaultBar.shadowDirectionMode ?? "inherit",
|
||||||
@@ -1742,19 +1741,6 @@ Item {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
SettingsToggleCard {
|
|
||||||
iconName: "touch_app"
|
|
||||||
title: I18n.tr("Hover Popouts")
|
|
||||||
description: I18n.tr("Open widget popouts by hovering over the bar. Moving to another widget switches the popout.")
|
|
||||||
visible: !dankBarTab.appearanceOnly && selectedBarConfig?.enabled
|
|
||||||
enabled: !(selectedBarConfig?.clickThrough ?? false)
|
|
||||||
opacity: (selectedBarConfig?.clickThrough ?? false) ? 0.5 : 1.0
|
|
||||||
checked: selectedBarConfig?.hoverPopouts ?? false
|
|
||||||
onToggled: checked => SettingsData.updateBarConfig(selectedBarId, {
|
|
||||||
hoverPopouts: checked
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
SettingsToggleCard {
|
SettingsToggleCard {
|
||||||
iconName: "mouse"
|
iconName: "mouse"
|
||||||
title: I18n.tr("Scroll Wheel")
|
title: I18n.tr("Scroll Wheel")
|
||||||
|
|||||||
@@ -113,6 +113,13 @@ Item {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SettingsToggleRow {
|
||||||
|
text: I18n.tr("Device list scroll volume")
|
||||||
|
description: I18n.tr("Allow adjusting device volume by scrolling on the right half of items in the device list")
|
||||||
|
checked: SettingsData.audioDeviceScrollVolumeEnabled
|
||||||
|
onToggled: checked => SettingsData.set("audioDeviceScrollVolumeEnabled", checked)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -58,6 +58,8 @@ Singleton {
|
|||||||
return SessionData.deviceMaxVolumes[name] ?? 100;
|
return SessionData.deviceMaxVolumes[name] ?? 100;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
readonly property int wheelVolumeStep: SettingsData.audioWheelScrollAmount
|
||||||
|
|
||||||
signal micMuteChanged
|
signal micMuteChanged
|
||||||
signal audioOutputCycled(string deviceName, string deviceIcon)
|
signal audioOutputCycled(string deviceName, string deviceIcon)
|
||||||
signal deviceAliasChanged(string nodeName, string newAlias)
|
signal deviceAliasChanged(string nodeName, string newAlias)
|
||||||
@@ -156,14 +158,19 @@ Singleton {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
function cycleAudioOutput() {
|
function cycleAudioOutputDirection(forward) {
|
||||||
const sinks = getAvailableSinks();
|
const sinks = getAvailableSinks();
|
||||||
if (sinks.length < 2)
|
if (sinks.length < 2)
|
||||||
return null;
|
return null;
|
||||||
|
|
||||||
const currentName = root.sink?.name ?? "";
|
const currentName = root.sink?.name ?? "";
|
||||||
const currentIndex = sinks.findIndex(s => s.name === currentName);
|
const currentIndex = sinks.findIndex(s => s.name === currentName);
|
||||||
const nextIndex = (currentIndex + 1) % sinks.length;
|
let nextIndex;
|
||||||
|
if (forward) {
|
||||||
|
nextIndex = (currentIndex + 1) % sinks.length;
|
||||||
|
} else {
|
||||||
|
nextIndex = (currentIndex - 1 + sinks.length) % sinks.length;
|
||||||
|
}
|
||||||
const nextSink = sinks[nextIndex];
|
const nextSink = sinks[nextIndex];
|
||||||
setDefaultSinkByName(nextSink.name);
|
setDefaultSinkByName(nextSink.name);
|
||||||
const name = displayName(nextSink);
|
const name = displayName(nextSink);
|
||||||
@@ -171,6 +178,10 @@ Singleton {
|
|||||||
return name;
|
return name;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function cycleAudioOutput() {
|
||||||
|
return cycleAudioOutputDirection(true);
|
||||||
|
}
|
||||||
|
|
||||||
function getDeviceAlias(nodeName) {
|
function getDeviceAlias(nodeName) {
|
||||||
if (!nodeName)
|
if (!nodeName)
|
||||||
return null;
|
return null;
|
||||||
@@ -833,6 +844,28 @@ EOFCONFIG
|
|||||||
return root.sink.audio.muted ? "Audio muted" : "Audio unmuted";
|
return root.sink.audio.muted ? "Audio muted" : "Audio unmuted";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function handleNodeVolumeWheel(node, wheelEvent) {
|
||||||
|
if (!node?.audio)
|
||||||
|
return;
|
||||||
|
|
||||||
|
SessionData.suppressOSDTemporarily();
|
||||||
|
const delta = wheelEvent.angleDelta.y;
|
||||||
|
if (delta === 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
const current = Math.round(node.audio.volume * 100);
|
||||||
|
const maxVol = getMaxVolumePercent(node);
|
||||||
|
const newVolume = delta > 0 ? Math.min(maxVol, current + root.wheelVolumeStep) : Math.max(0, current - root.wheelVolumeStep);
|
||||||
|
|
||||||
|
node.audio.muted = false;
|
||||||
|
node.audio.volume = newVolume / 100;
|
||||||
|
|
||||||
|
if (node === sink) {
|
||||||
|
playVolumeChangeSoundIfEnabled();
|
||||||
|
}
|
||||||
|
wheelEvent.accepted = true;
|
||||||
|
}
|
||||||
|
|
||||||
function setMicVolume(percentage) {
|
function setMicVolume(percentage) {
|
||||||
if (!root.source?.audio) {
|
if (!root.source?.audio) {
|
||||||
return "No audio source available";
|
return "No audio source available";
|
||||||
|
|||||||
@@ -23,6 +23,49 @@ Singleton {
|
|||||||
property var tabsBeingCreated: ({})
|
property var tabsBeingCreated: ({})
|
||||||
property bool metadataLoaded: false
|
property bool metadataLoaded: false
|
||||||
|
|
||||||
|
// Shared live edit state across slideout and popout surfaces.
|
||||||
|
property var sessionBuffers: ({})
|
||||||
|
property int sessionBufferRevision: 0
|
||||||
|
|
||||||
|
function setSessionBuffer(tabId, content, baseline) {
|
||||||
|
if (tabId === undefined || tabId === null || tabId < 0)
|
||||||
|
return
|
||||||
|
var next = Object.assign({}, sessionBuffers)
|
||||||
|
if (content !== baseline) {
|
||||||
|
next[tabId] = { content: content, baseline: baseline }
|
||||||
|
} else {
|
||||||
|
delete next[tabId]
|
||||||
|
}
|
||||||
|
sessionBuffers = next
|
||||||
|
sessionBufferRevision++
|
||||||
|
}
|
||||||
|
|
||||||
|
function getSessionBuffer(tabId) {
|
||||||
|
return sessionBuffers[tabId]
|
||||||
|
}
|
||||||
|
|
||||||
|
function clearSessionBuffer(tabId) {
|
||||||
|
if (sessionBuffers[tabId] === undefined)
|
||||||
|
return
|
||||||
|
var next = Object.assign({}, sessionBuffers)
|
||||||
|
delete next[tabId]
|
||||||
|
sessionBuffers = next
|
||||||
|
sessionBufferRevision++
|
||||||
|
}
|
||||||
|
|
||||||
|
property var conflictTabId: -1
|
||||||
|
property string conflictDiskContent: ""
|
||||||
|
|
||||||
|
function flagConflict(tabId, diskContent) {
|
||||||
|
conflictDiskContent = diskContent
|
||||||
|
conflictTabId = tabId
|
||||||
|
}
|
||||||
|
|
||||||
|
function clearConflict() {
|
||||||
|
conflictTabId = -1
|
||||||
|
conflictDiskContent = ""
|
||||||
|
}
|
||||||
|
|
||||||
Component.onCompleted: {
|
Component.onCompleted: {
|
||||||
ensureDirectories()
|
ensureDirectories()
|
||||||
}
|
}
|
||||||
@@ -209,6 +252,10 @@ Singleton {
|
|||||||
if (tabIndex < 0 || tabIndex >= tabs.length) return
|
if (tabIndex < 0 || tabIndex >= tabs.length) return
|
||||||
|
|
||||||
var newTabs = tabs.slice()
|
var newTabs = tabs.slice()
|
||||||
|
var closedTabId = newTabs[tabIndex] ? newTabs[tabIndex].id : -1
|
||||||
|
clearSessionBuffer(closedTabId)
|
||||||
|
if (conflictTabId === closedTabId)
|
||||||
|
clearConflict()
|
||||||
|
|
||||||
if (newTabs.length <= 1) {
|
if (newTabs.length <= 1) {
|
||||||
var id = Date.now()
|
var id = Date.now()
|
||||||
|
|||||||
@@ -789,21 +789,97 @@ Singleton {
|
|||||||
networkInfoModal?.close();
|
networkInfoModal?.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
function openNotepad() {
|
function closeNotepadSlideouts() {
|
||||||
|
for (var i = 0; i < notepadSlideouts.length; i++) {
|
||||||
|
if (notepadSlideouts[i] && notepadSlideouts[i].isVisible)
|
||||||
|
notepadSlideouts[i].hide();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function openNotepadSlideout() {
|
||||||
|
notepadPopout?.hide();
|
||||||
if (notepadSlideouts.length > 0) {
|
if (notepadSlideouts.length > 0) {
|
||||||
notepadSlideouts[0]?.show();
|
notepadSlideouts[0]?.show();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Keep the notepad in a single presentation for default modes
|
||||||
|
Connections {
|
||||||
|
target: SettingsData
|
||||||
|
function onNotepadDefaultModeChanged() {
|
||||||
|
if (SettingsData.notepadDefaultMode === "popout") {
|
||||||
|
var hadSlideout = false;
|
||||||
|
for (var i = 0; i < root.notepadSlideouts.length; i++) {
|
||||||
|
if (root.notepadSlideouts[i] && root.notepadSlideouts[i].isVisible) {
|
||||||
|
hadSlideout = true;
|
||||||
|
root.notepadSlideouts[i].hide();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (hadSlideout)
|
||||||
|
root.openNotepadPopout();
|
||||||
|
} else if (root.notepadPopout && root.notepadPopout.visible) {
|
||||||
|
root.notepadPopout.hide();
|
||||||
|
root.openNotepadSlideout();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function openNotepad() {
|
||||||
|
if (SettingsData.notepadDefaultMode === "popout") {
|
||||||
|
openNotepadPopout();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
openNotepadSlideout();
|
||||||
|
}
|
||||||
|
|
||||||
function closeNotepad() {
|
function closeNotepad() {
|
||||||
|
if (SettingsData.notepadDefaultMode === "popout") {
|
||||||
|
notepadPopout?.hide();
|
||||||
|
return;
|
||||||
|
}
|
||||||
if (notepadSlideouts.length > 0) {
|
if (notepadSlideouts.length > 0) {
|
||||||
notepadSlideouts[0]?.hide();
|
notepadSlideouts[0]?.hide();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function toggleNotepad() {
|
function toggleNotepad() {
|
||||||
|
if (SettingsData.notepadDefaultMode === "popout") {
|
||||||
|
toggleNotepadPopout();
|
||||||
|
return;
|
||||||
|
}
|
||||||
if (notepadSlideouts.length > 0) {
|
if (notepadSlideouts.length > 0) {
|
||||||
notepadSlideouts[0]?.toggle();
|
notepadSlideouts[0]?.toggle();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
property var notepadPopout: null
|
||||||
|
property var notepadPopoutLoader: null
|
||||||
|
property bool _notepadPopoutWantsOpen: false
|
||||||
|
|
||||||
|
function openNotepadPopout() {
|
||||||
|
closeNotepadSlideouts();
|
||||||
|
if (notepadPopout) {
|
||||||
|
notepadPopout.show();
|
||||||
|
} else if (notepadPopoutLoader) {
|
||||||
|
_notepadPopoutWantsOpen = true;
|
||||||
|
notepadPopoutLoader.active = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function _onNotepadPopoutLoaded() {
|
||||||
|
if (_notepadPopoutWantsOpen && notepadPopout) {
|
||||||
|
_notepadPopoutWantsOpen = false;
|
||||||
|
notepadPopout.show();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggleNotepadPopout() {
|
||||||
|
if (notepadPopout) {
|
||||||
|
if (!notepadPopout.visible)
|
||||||
|
closeNotepadSlideouts();
|
||||||
|
notepadPopout.toggle();
|
||||||
|
} else {
|
||||||
|
openNotepadPopout();
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,11 +13,12 @@ Row {
|
|||||||
property var initialSelection: []
|
property var initialSelection: []
|
||||||
property var currentSelection: initialSelection
|
property var currentSelection: initialSelection
|
||||||
property bool checkEnabled: true
|
property bool checkEnabled: true
|
||||||
property int buttonHeight: 40
|
property string size: "medium"
|
||||||
property int minButtonWidth: 64
|
property int buttonHeight: size === "small" ? 32 : 40
|
||||||
property int buttonPadding: Theme.spacingL
|
property int minButtonWidth: size === "small" ? 56 : 64
|
||||||
property int checkIconSize: Theme.iconSizeSmall
|
property int buttonPadding: size === "small" ? Theme.spacingM : Theme.spacingL
|
||||||
property int textSize: Theme.fontSizeMedium
|
property int checkIconSize: size === "small" ? Theme.iconSizeSmall - 2 : Theme.iconSizeSmall
|
||||||
|
property int textSize: size === "small" ? Theme.fontSizeSmall : Theme.fontSizeMedium
|
||||||
property bool userInteracted: false
|
property bool userInteracted: false
|
||||||
|
|
||||||
signal selectionChanged(int index, bool selected)
|
signal selectionChanged(int index, bool selected)
|
||||||
|
|||||||
@@ -24,7 +24,6 @@ Item {
|
|||||||
property list<real> animationExitCurve: Theme.variantPopoutExitCurve
|
property list<real> animationExitCurve: Theme.variantPopoutExitCurve
|
||||||
property bool suspendShadowWhileResizing: false
|
property bool suspendShadowWhileResizing: false
|
||||||
property bool shouldBeVisible: false
|
property bool shouldBeVisible: false
|
||||||
property bool hoverDismissEnabled: false
|
|
||||||
property var customKeyboardFocus: null
|
property var customKeyboardFocus: null
|
||||||
property bool backgroundInteractive: true
|
property bool backgroundInteractive: true
|
||||||
property bool contentHandlesKeys: false
|
property bool contentHandlesKeys: false
|
||||||
@@ -83,8 +82,6 @@ Item {
|
|||||||
readonly property real alignedY: impl.item ? impl.item.alignedY : 0
|
readonly property real alignedY: impl.item ? impl.item.alignedY : 0
|
||||||
readonly property real alignedWidth: impl.item ? impl.item.alignedWidth : 0
|
readonly property real alignedWidth: impl.item ? impl.item.alignedWidth : 0
|
||||||
readonly property real alignedHeight: impl.item ? impl.item.alignedHeight : 0
|
readonly property real alignedHeight: impl.item ? impl.item.alignedHeight : 0
|
||||||
readonly property real renderedAlignedY: impl.item ? (impl.item.renderedAlignedY ?? impl.item.alignedY) : 0
|
|
||||||
readonly property real renderedAlignedHeight: impl.item ? (impl.item.renderedAlignedHeight ?? impl.item.alignedHeight) : 0
|
|
||||||
readonly property real maskX: impl.item ? impl.item.maskX : 0
|
readonly property real maskX: impl.item ? impl.item.maskX : 0
|
||||||
readonly property real maskY: impl.item ? impl.item.maskY : 0
|
readonly property real maskY: impl.item ? impl.item.maskY : 0
|
||||||
readonly property real maskWidth: impl.item ? impl.item.maskWidth : 0
|
readonly property real maskWidth: impl.item ? impl.item.maskWidth : 0
|
||||||
@@ -175,32 +172,6 @@ Item {
|
|||||||
impl.item.close();
|
impl.item.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
function cancelHoverDismiss() {
|
|
||||||
if (impl.item?.cancelHoverDismiss)
|
|
||||||
impl.item.cancelHoverDismiss();
|
|
||||||
}
|
|
||||||
|
|
||||||
function closeFromHoverDismiss() {
|
|
||||||
hoverDismissEnabled = false;
|
|
||||||
if (impl.item) {
|
|
||||||
impl.item.animationsEnabled = true;
|
|
||||||
impl.item.animationDuration = Math.round(Theme.expressiveDurations.expressiveDefaultSpatial);
|
|
||||||
impl.item.animationExitCurve = Theme.expressiveCurves.expressiveDefaultSpatial;
|
|
||||||
}
|
|
||||||
if (dashVisible !== undefined) {
|
|
||||||
dashVisible = false;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (notificationHistoryVisible !== undefined) {
|
|
||||||
notificationHistoryVisible = false;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (impl.item)
|
|
||||||
impl.item.close();
|
|
||||||
else
|
|
||||||
close();
|
|
||||||
}
|
|
||||||
|
|
||||||
function toggle() {
|
function toggle() {
|
||||||
(shouldBeVisible || _pendingOpen) ? close() : open();
|
(shouldBeVisible || _pendingOpen) ? close() : open();
|
||||||
}
|
}
|
||||||
@@ -239,20 +210,6 @@ Item {
|
|||||||
impl.item.updateSurfacePosition();
|
impl.item.updateSurfacePosition();
|
||||||
}
|
}
|
||||||
|
|
||||||
function containsGlobalPoint(gx, gy) {
|
|
||||||
if (!screen)
|
|
||||||
return false;
|
|
||||||
const presented = shouldBeVisible || (impl.item?.isClosing ?? false);
|
|
||||||
if (!presented)
|
|
||||||
return false;
|
|
||||||
const padding = 24;
|
|
||||||
const x = alignedX - padding;
|
|
||||||
const y = renderedAlignedY - padding;
|
|
||||||
const w = alignedWidth + padding * 2;
|
|
||||||
const h = renderedAlignedHeight + padding * 2;
|
|
||||||
return gx >= x && gx <= x + w && gy >= y && gy <= y + h;
|
|
||||||
}
|
|
||||||
|
|
||||||
Loader {
|
Loader {
|
||||||
id: impl
|
id: impl
|
||||||
active: root.screen !== null
|
active: root.screen !== null
|
||||||
@@ -304,7 +261,6 @@ Item {
|
|||||||
it.screen = Qt.binding(() => root.screen);
|
it.screen = Qt.binding(() => root.screen);
|
||||||
it.effectiveBarPosition = Qt.binding(() => root.effectiveBarPosition);
|
it.effectiveBarPosition = Qt.binding(() => root.effectiveBarPosition);
|
||||||
it.effectiveBarBottomGap = Qt.binding(() => root.effectiveBarBottomGap);
|
it.effectiveBarBottomGap = Qt.binding(() => root.effectiveBarBottomGap);
|
||||||
it.hoverDismissEnabled = Qt.binding(() => root.hoverDismissEnabled);
|
|
||||||
|
|
||||||
it.shouldBeVisible = root.shouldBeVisible;
|
it.shouldBeVisible = root.shouldBeVisible;
|
||||||
if (root._primeContent && typeof it.primeContent === "function")
|
if (root._primeContent && typeof it.primeContent === "function")
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import Quickshell
|
|||||||
import Quickshell.Wayland
|
import Quickshell.Wayland
|
||||||
import qs.Common
|
import qs.Common
|
||||||
import qs.Services
|
import qs.Services
|
||||||
import qs.Widgets
|
|
||||||
|
|
||||||
Item {
|
Item {
|
||||||
id: root
|
id: root
|
||||||
@@ -408,20 +407,6 @@ Item {
|
|||||||
onFrameOwnsConnectedChromeChanged: _syncPopoutChromeState()
|
onFrameOwnsConnectedChromeChanged: _syncPopoutChromeState()
|
||||||
|
|
||||||
property bool animationsEnabled: true
|
property bool animationsEnabled: true
|
||||||
property bool hoverDismissEnabled: false
|
|
||||||
|
|
||||||
function cancelHoverDismiss() {
|
|
||||||
hoverDismissTracker.cancelPending();
|
|
||||||
}
|
|
||||||
|
|
||||||
function closeFromHoverDismiss() {
|
|
||||||
if (isClosing || !shouldBeVisible)
|
|
||||||
return;
|
|
||||||
if (popoutHandle?.closeFromHoverDismiss)
|
|
||||||
popoutHandle.closeFromHoverDismiss();
|
|
||||||
else
|
|
||||||
close();
|
|
||||||
}
|
|
||||||
|
|
||||||
function open() {
|
function open() {
|
||||||
if (!screen)
|
if (!screen)
|
||||||
@@ -776,27 +761,6 @@ Item {
|
|||||||
visible: false
|
visible: false
|
||||||
color: "transparent"
|
color: "transparent"
|
||||||
|
|
||||||
MouseArea {
|
|
||||||
anchors.fill: parent
|
|
||||||
z: -1
|
|
||||||
acceptedButtons: Qt.NoButton
|
|
||||||
hoverEnabled: true
|
|
||||||
onPositionChanged: mouse => {
|
|
||||||
const gp = mapToItem(null, mouse.x, mouse.y);
|
|
||||||
PopoutManager.updateHoverCursor(gp.x, gp.y);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
HoverDismissTracker {
|
|
||||||
id: hoverDismissTracker
|
|
||||||
anchors.fill: parent
|
|
||||||
enabled: root.hoverDismissEnabled && root.shouldBeVisible
|
|
||||||
shouldDismiss: function () {
|
|
||||||
return !PopoutManager.cursorOverBar(PopoutManager.hoverCursorGlobalX, PopoutManager.hoverCursorGlobalY);
|
|
||||||
}
|
|
||||||
onDismissRequested: root.closeFromHoverDismiss()
|
|
||||||
}
|
|
||||||
|
|
||||||
WindowBlur {
|
WindowBlur {
|
||||||
id: popoutBlur
|
id: popoutBlur
|
||||||
targetWindow: contentWindow
|
targetWindow: contentWindow
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import Quickshell
|
|||||||
import Quickshell.Wayland
|
import Quickshell.Wayland
|
||||||
import qs.Common
|
import qs.Common
|
||||||
import qs.Services
|
import qs.Services
|
||||||
import qs.Widgets
|
|
||||||
|
|
||||||
Item {
|
Item {
|
||||||
id: root
|
id: root
|
||||||
@@ -36,21 +35,6 @@ Item {
|
|||||||
property bool shouldBeVisible: false
|
property bool shouldBeVisible: false
|
||||||
property bool isClosing: false
|
property bool isClosing: false
|
||||||
property bool animationsEnabled: true
|
property bool animationsEnabled: true
|
||||||
property bool hoverDismissEnabled: false
|
|
||||||
|
|
||||||
function cancelHoverDismiss() {
|
|
||||||
hoverDismissTracker.cancelPending();
|
|
||||||
}
|
|
||||||
|
|
||||||
function closeFromHoverDismiss() {
|
|
||||||
if (isClosing || !shouldBeVisible)
|
|
||||||
return;
|
|
||||||
if (popoutHandle?.closeFromHoverDismiss)
|
|
||||||
popoutHandle.closeFromHoverDismiss();
|
|
||||||
else
|
|
||||||
close();
|
|
||||||
}
|
|
||||||
|
|
||||||
property var customKeyboardFocus: null
|
property var customKeyboardFocus: null
|
||||||
property bool backgroundInteractive: true
|
property bool backgroundInteractive: true
|
||||||
property bool contentHandlesKeys: false
|
property bool contentHandlesKeys: false
|
||||||
@@ -601,27 +585,6 @@ Item {
|
|||||||
color: "transparent"
|
color: "transparent"
|
||||||
readonly property bool closeVisualActive: root.shouldBeVisible || root.isClosing
|
readonly property bool closeVisualActive: root.shouldBeVisible || root.isClosing
|
||||||
|
|
||||||
MouseArea {
|
|
||||||
anchors.fill: parent
|
|
||||||
z: -1
|
|
||||||
acceptedButtons: Qt.NoButton
|
|
||||||
hoverEnabled: true
|
|
||||||
onPositionChanged: mouse => {
|
|
||||||
const gp = mapToItem(null, mouse.x, mouse.y);
|
|
||||||
PopoutManager.updateHoverCursor(gp.x, gp.y);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
HoverDismissTracker {
|
|
||||||
id: hoverDismissTracker
|
|
||||||
anchors.fill: parent
|
|
||||||
enabled: root.hoverDismissEnabled && root.shouldBeVisible
|
|
||||||
shouldDismiss: function () {
|
|
||||||
return !PopoutManager.cursorOverBar(PopoutManager.hoverCursorGlobalX, PopoutManager.hoverCursorGlobalY);
|
|
||||||
}
|
|
||||||
onDismissRequested: root.closeFromHoverDismiss()
|
|
||||||
}
|
|
||||||
|
|
||||||
WindowBlur {
|
WindowBlur {
|
||||||
id: popoutBlur
|
id: popoutBlur
|
||||||
targetWindow: contentWindow
|
targetWindow: contentWindow
|
||||||
|
|||||||
@@ -13,7 +13,6 @@ PanelWindow {
|
|||||||
WlrLayershell.namespace: layerNamespace
|
WlrLayershell.namespace: layerNamespace
|
||||||
|
|
||||||
property bool isVisible: false
|
property bool isVisible: false
|
||||||
property bool hoverDismissEnabled: false
|
|
||||||
property var targetScreen: null
|
property var targetScreen: null
|
||||||
property var modelData: null
|
property var modelData: null
|
||||||
property bool triggerUsesOverlayLayer: false
|
property bool triggerUsesOverlayLayer: false
|
||||||
@@ -21,17 +20,22 @@ PanelWindow {
|
|||||||
property bool expandable: false
|
property bool expandable: false
|
||||||
property bool expandedWidth: false
|
property bool expandedWidth: false
|
||||||
property real expandedWidthValue: 960
|
property real expandedWidthValue: 960
|
||||||
|
property real edgeGap: 0
|
||||||
|
property string slideEdge: "right"
|
||||||
|
readonly property bool slideFromLeft: slideEdge === "left"
|
||||||
property Component content: null
|
property Component content: null
|
||||||
property string title: ""
|
property string title: ""
|
||||||
property alias container: contentContainer
|
property alias container: contentContainer
|
||||||
property real customTransparency: -1
|
property real customTransparency: -1
|
||||||
property bool mappedVisible: false
|
property bool mappedVisible: false
|
||||||
signal aboutToHide
|
signal aboutToHide
|
||||||
|
signal revealed
|
||||||
|
|
||||||
function show() {
|
function show() {
|
||||||
mappedVisible = true;
|
mappedVisible = true;
|
||||||
Qt.callLater(() => {
|
Qt.callLater(() => {
|
||||||
isVisible = true;
|
isVisible = true;
|
||||||
|
revealed();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -40,24 +44,6 @@ PanelWindow {
|
|||||||
isVisible = false;
|
isVisible = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
function hideFromHoverDismiss() {
|
|
||||||
hoverDismissEnabled = false;
|
|
||||||
slideAnimation.duration = Math.round(Theme.expressiveDurations.expressiveDefaultSpatial);
|
|
||||||
hide();
|
|
||||||
}
|
|
||||||
|
|
||||||
function cancelHoverDismiss() {
|
|
||||||
hoverDismissTracker.cancelPending();
|
|
||||||
}
|
|
||||||
|
|
||||||
function containsGlobalPoint(gx, gy) {
|
|
||||||
if (!isVisible || !modelData)
|
|
||||||
return false;
|
|
||||||
const padding = 24;
|
|
||||||
const topLeft = slideContainer.mapToItem(null, 0, 0);
|
|
||||||
return gx >= topLeft.x - padding && gx < topLeft.x + slideContainer.width + padding && gy >= topLeft.y - padding && gy < topLeft.y + slideContainer.height + padding;
|
|
||||||
}
|
|
||||||
|
|
||||||
function toggle() {
|
function toggle() {
|
||||||
if (isVisible) {
|
if (isVisible) {
|
||||||
hide();
|
hide();
|
||||||
@@ -71,35 +57,14 @@ PanelWindow {
|
|||||||
|
|
||||||
anchors.top: true
|
anchors.top: true
|
||||||
anchors.bottom: true
|
anchors.bottom: true
|
||||||
anchors.right: true
|
anchors.right: !root.slideFromLeft
|
||||||
|
anchors.left: root.slideFromLeft
|
||||||
|
|
||||||
// Expandable: fixed max surface width; strip width is slideContainer only (keeps blur/mask aligned).
|
|
||||||
implicitWidth: expandable ? expandedWidthValue : slideoutWidth
|
implicitWidth: expandable ? expandedWidthValue : slideoutWidth
|
||||||
implicitHeight: modelData ? modelData.height : 800
|
implicitHeight: modelData ? modelData.height : 800
|
||||||
|
|
||||||
color: "transparent"
|
color: "transparent"
|
||||||
|
|
||||||
MouseArea {
|
|
||||||
anchors.fill: parent
|
|
||||||
z: -1
|
|
||||||
acceptedButtons: Qt.NoButton
|
|
||||||
hoverEnabled: true
|
|
||||||
onPositionChanged: mouse => {
|
|
||||||
const gp = mapToItem(null, mouse.x, mouse.y);
|
|
||||||
PopoutManager.updateHoverCursor(gp.x, gp.y);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
HoverDismissTracker {
|
|
||||||
id: hoverDismissTracker
|
|
||||||
anchors.fill: parent
|
|
||||||
enabled: root.hoverDismissEnabled && root.isVisible
|
|
||||||
shouldDismiss: function () {
|
|
||||||
return !PopoutManager.cursorOverBar(PopoutManager.hoverCursorGlobalX, PopoutManager.hoverCursorGlobalY);
|
|
||||||
}
|
|
||||||
onDismissRequested: root.hideFromHoverDismiss()
|
|
||||||
}
|
|
||||||
|
|
||||||
readonly property bool slideoutBlurActive: root.visible && BlurService.enabled && Theme.connectedSurfaceBlurEnabled
|
readonly property bool slideoutBlurActive: root.visible && BlurService.enabled && Theme.connectedSurfaceBlurEnabled
|
||||||
|
|
||||||
WlrLayershell.layer: (triggerUsesOverlayLayer || CompositorService.framePeerSurfacesUseOverlayForScreen(modelData)) ? WlrLayershell.Overlay : WlrLayershell.Top
|
WlrLayershell.layer: (triggerUsesOverlayLayer || CompositorService.framePeerSurfacesUseOverlayForScreen(modelData)) ? WlrLayershell.Overlay : WlrLayershell.Top
|
||||||
@@ -109,14 +74,15 @@ PanelWindow {
|
|||||||
readonly property real dpr: CompositorService.getScreenScale(root.screen)
|
readonly property real dpr: CompositorService.getScreenScale(root.screen)
|
||||||
readonly property real alignedWidth: Theme.px(expandable && expandedWidth ? expandedWidthValue : slideoutWidth, dpr)
|
readonly property real alignedWidth: Theme.px(expandable && expandedWidth ? expandedWidthValue : slideoutWidth, dpr)
|
||||||
readonly property real alignedHeight: Theme.px(modelData ? modelData.height : 800, dpr)
|
readonly property real alignedHeight: Theme.px(modelData ? modelData.height : 800, dpr)
|
||||||
|
readonly property real alignedEdgeGap: Theme.px(edgeGap, dpr)
|
||||||
readonly property real slideoutSlideSnapX: Theme.snap(slideContainer.slideOffset, dpr)
|
readonly property real slideoutSlideSnapX: Theme.snap(slideContainer.slideOffset, dpr)
|
||||||
|
|
||||||
mask: Region {
|
mask: Region {
|
||||||
item: Rectangle {
|
item: Rectangle {
|
||||||
x: root.width - slideContainer.width
|
x: root.slideFromLeft ? root.alignedEdgeGap : (root.width - slideContainer.width - root.alignedEdgeGap)
|
||||||
y: 0
|
y: root.alignedEdgeGap
|
||||||
width: slideContainer.width
|
width: slideContainer.width
|
||||||
height: root.height
|
height: root.height - root.alignedEdgeGap * 2
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -124,16 +90,21 @@ PanelWindow {
|
|||||||
id: slideContainer
|
id: slideContainer
|
||||||
anchors.top: parent.top
|
anchors.top: parent.top
|
||||||
anchors.bottom: parent.bottom
|
anchors.bottom: parent.bottom
|
||||||
anchors.right: parent.right
|
anchors.right: root.slideFromLeft ? undefined : parent.right
|
||||||
|
anchors.left: root.slideFromLeft ? parent.left : undefined
|
||||||
|
anchors.topMargin: root.alignedEdgeGap
|
||||||
|
anchors.bottomMargin: root.alignedEdgeGap
|
||||||
|
anchors.rightMargin: root.alignedEdgeGap
|
||||||
|
anchors.leftMargin: root.alignedEdgeGap
|
||||||
width: root.alignedWidth
|
width: root.alignedWidth
|
||||||
height: root.alignedHeight
|
height: root.alignedHeight - root.alignedEdgeGap * 2
|
||||||
|
|
||||||
property real slideOffset: root.alignedWidth
|
property real slideOffset: root.slideFromLeft ? -root.alignedWidth : root.alignedWidth
|
||||||
|
|
||||||
Connections {
|
Connections {
|
||||||
target: root
|
target: root
|
||||||
function onIsVisibleChanged() {
|
function onIsVisibleChanged() {
|
||||||
slideContainer.slideOffset = root.isVisible ? 0 : slideContainer.width;
|
slideContainer.slideOffset = root.isVisible ? 0 : (root.slideFromLeft ? -slideContainer.width : slideContainer.width);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -144,16 +115,13 @@ PanelWindow {
|
|||||||
easing.type: Easing.OutCubic
|
easing.type: Easing.OutCubic
|
||||||
|
|
||||||
onRunningChanged: {
|
onRunningChanged: {
|
||||||
if (!running) {
|
if (!running && !root.isVisible) {
|
||||||
if (!root.isVisible)
|
root.mappedVisible = false;
|
||||||
root.mappedVisible = false;
|
|
||||||
slideAnimation.duration = 450;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Expandable only; mask/blur bind to slideContainer geometry so they track this animation.
|
|
||||||
Behavior on width {
|
Behavior on width {
|
||||||
enabled: root.expandable
|
enabled: root.expandable
|
||||||
NumberAnimation {
|
NumberAnimation {
|
||||||
@@ -259,7 +227,6 @@ PanelWindow {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Blur region from slideContainer (not layered contentRect); position uses x + slideoutSlideSnapX, not mapToItem(root).
|
|
||||||
WindowBlur {
|
WindowBlur {
|
||||||
targetWindow: root
|
targetWindow: root
|
||||||
blurX: root.slideoutBlurActive ? slideContainer.x + root.slideoutSlideSnapX : 0
|
blurX: root.slideoutBlurActive ? slideContainer.x + root.slideoutSlideSnapX : 0
|
||||||
|
|||||||
@@ -1,28 +0,0 @@
|
|||||||
pragma ComponentBehavior: Bound
|
|
||||||
|
|
||||||
import QtQuick
|
|
||||||
|
|
||||||
Item {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
property bool enabled: false
|
|
||||||
property var shouldDismiss: null
|
|
||||||
|
|
||||||
signal dismissRequested
|
|
||||||
|
|
||||||
anchors.fill: parent
|
|
||||||
|
|
||||||
HoverHandler {
|
|
||||||
id: hoverHandler
|
|
||||||
enabled: root.enabled
|
|
||||||
onHoveredChanged: {
|
|
||||||
if (hoverHandler.hovered || !root.enabled)
|
|
||||||
return;
|
|
||||||
if (typeof root.shouldDismiss === "function" && !root.shouldDismiss())
|
|
||||||
return;
|
|
||||||
root.dismissRequested();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function cancelPending() {}
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user