mirror of
https://git.mirrors.martin98.com/https://github.com/Ultimaker/Cura
synced 2025-04-19 20:29:40 +08:00
255 lines
8.7 KiB
QML
255 lines
8.7 KiB
QML
// Copyright (c) 2021 Ultimaker B.V.
|
|
// Cura is released under the terms of the LGPLv3 or higher.
|
|
|
|
import QtQuick 2.15
|
|
import QtQuick.Controls 2.15
|
|
|
|
import UM 1.7 as UM
|
|
|
|
SettingItem
|
|
{
|
|
id: base
|
|
property var focusItem: input
|
|
|
|
property string textBeforeEdit
|
|
property bool textHasChanged
|
|
property bool focusGainedByClick: false
|
|
|
|
readonly property UM.IntValidator intValidator: UM.IntValidator {}
|
|
readonly property UM.FloatValidator floatValidator: UM.FloatValidator {}
|
|
readonly property UM.IntListValidator intListValidator: UM.IntListValidator {}
|
|
|
|
onFocusReceived:
|
|
{
|
|
textHasChanged = false;
|
|
textBeforeEdit = focusItem.text;
|
|
|
|
if(!focusGainedByClick)
|
|
{
|
|
// select all text when tabbing through fields (but not when selecting a field with the mouse)
|
|
focusItem.selectAll();
|
|
}
|
|
}
|
|
|
|
contents: UM.UnderlineBackground
|
|
{
|
|
id: control
|
|
|
|
anchors.fill: parent
|
|
|
|
borderColor: input.activeFocus ? UM.Theme.getColor("text_field_border_active") : "transparent"
|
|
liningColor:
|
|
{
|
|
if(!enabled)
|
|
{
|
|
return UM.Theme.getColor("text_field_border_disabled");
|
|
}
|
|
switch(propertyProvider.properties.validationState)
|
|
{
|
|
case "ValidatorState.Invalid":
|
|
case "ValidatorState.Exception":
|
|
case "ValidatorState.MinimumError":
|
|
case "ValidatorState.MaximumError":
|
|
return UM.Theme.getColor("setting_validation_error");
|
|
case "ValidatorState.MinimumWarning":
|
|
case "ValidatorState.MaximumWarning":
|
|
return UM.Theme.getColor("setting_validation_warning");
|
|
}
|
|
//Validation is OK.
|
|
if(input.activeFocus)
|
|
{
|
|
return UM.Theme.getColor("text_field_border_active");
|
|
}
|
|
if(hovered)
|
|
{
|
|
return UM.Theme.getColor("text_field_border_hovered");
|
|
}
|
|
return UM.Theme.getColor("text_field_border");
|
|
}
|
|
|
|
color: {
|
|
if(!enabled)
|
|
{
|
|
return UM.Theme.getColor("setting_control_disabled")
|
|
}
|
|
switch(propertyProvider.properties.validationState)
|
|
{
|
|
case "ValidatorState.Invalid":
|
|
case "ValidatorState.Exception":
|
|
case "ValidatorState.MinimumError":
|
|
case "ValidatorState.MaximumError":
|
|
return UM.Theme.getColor("setting_validation_error_background")
|
|
case "ValidatorState.MinimumWarning":
|
|
case "ValidatorState.MaximumWarning":
|
|
return UM.Theme.getColor("setting_validation_warning_background")
|
|
case "ValidatorState.Valid":
|
|
return UM.Theme.getColor("setting_validation_ok")
|
|
|
|
default:
|
|
return UM.Theme.getColor("text_field")
|
|
}
|
|
}
|
|
|
|
UM.Label
|
|
{
|
|
id: unitLabel
|
|
|
|
anchors
|
|
{
|
|
left: parent.left
|
|
leftMargin: Math.round(UM.Theme.getSize("setting_unit_margin").width)
|
|
right: parent.right
|
|
rightMargin: Math.round(UM.Theme.getSize("setting_unit_margin").width)
|
|
verticalCenter: parent.verticalCenter
|
|
}
|
|
|
|
text: definition.unit
|
|
//However the setting value is aligned, align the unit opposite. That way it stays readable with right-to-left languages.
|
|
horizontalAlignment: (input.effectiveHorizontalAlignment == Text.AlignLeft) ? Text.AlignRight : Text.AlignLeft
|
|
textFormat: Text.PlainText
|
|
color: UM.Theme.getColor("setting_unit")
|
|
|
|
Binding
|
|
{
|
|
target: unitLabel
|
|
property: "text"
|
|
value:
|
|
{
|
|
return propertyProvider.properties.unit;
|
|
}
|
|
}
|
|
}
|
|
|
|
TextInput
|
|
{
|
|
id: input
|
|
|
|
anchors
|
|
{
|
|
left: parent.left
|
|
leftMargin: Math.round(UM.Theme.getSize("setting_unit_margin").width)
|
|
right: parent.right
|
|
rightMargin: Math.round(UM.Theme.getSize("setting_unit_margin").width)
|
|
verticalCenter: parent.verticalCenter
|
|
}
|
|
renderType: Text.NativeRendering
|
|
|
|
Keys.onTabPressed:
|
|
{
|
|
base.setActiveFocusToNextSetting(true)
|
|
}
|
|
Keys.onBacktabPressed:
|
|
{
|
|
base.setActiveFocusToNextSetting(false)
|
|
}
|
|
|
|
Keys.onReleased:
|
|
{
|
|
if (text != textBeforeEdit)
|
|
{
|
|
textHasChanged = true;
|
|
}
|
|
if (textHasChanged)
|
|
{
|
|
propertyProvider.setPropertyValue("value", text)
|
|
}
|
|
}
|
|
|
|
onActiveFocusChanged:
|
|
{
|
|
if(activeFocus)
|
|
{
|
|
base.focusReceived();
|
|
setScrollPositionChangeLoseFocus(false);
|
|
}
|
|
else
|
|
{
|
|
setScrollPositionChangeLoseFocus(true);
|
|
}
|
|
base.focusGainedByClick = false;
|
|
}
|
|
|
|
color: !enabled ? UM.Theme.getColor("setting_control_disabled_text") : UM.Theme.getColor("setting_control_text")
|
|
selectedTextColor: UM.Theme.getColor("setting_control_text")
|
|
font: UM.Theme.getFont("default")
|
|
selectionColor: UM.Theme.getColor("text_selection")
|
|
selectByMouse: true
|
|
|
|
maximumLength: (definition.type == "str" || definition.type == "[int]") ? -1 : 12
|
|
|
|
// Since [int] & str don't have a max length, they need to be clipped (since clipping is expensive, this
|
|
// should be done as little as possible)
|
|
clip: definition.type == "str" || definition.type == "[int]"
|
|
|
|
validator: RegularExpressionValidator
|
|
{
|
|
regularExpression:
|
|
{
|
|
switch (definition.type)
|
|
{
|
|
case "[int]":
|
|
return new RegExp(intListValidator.regexString)
|
|
case "int":
|
|
return new RegExp(intValidator.regexString)
|
|
case "float":
|
|
return new RegExp(floatValidator.regexString)
|
|
default:
|
|
return new RegExp("^.*$")
|
|
}
|
|
}
|
|
}
|
|
|
|
Binding
|
|
{
|
|
target: input
|
|
property: "text"
|
|
value:
|
|
{
|
|
if (input.activeFocus)
|
|
{
|
|
// In QT6 using "when: !activeFocus" causes the value to be null when activeFocus becomes True
|
|
// Since we want the value to stay the same when giving focus to the TextInput this is being used
|
|
// in place of "when: !activeFocus"
|
|
return input.text
|
|
}
|
|
// Stacklevels
|
|
// 0: user -> unsaved change
|
|
// 1: quality changes -> saved change
|
|
// 2: quality
|
|
// 3: material -> user changed material in materialspage
|
|
// 4: variant
|
|
// 5: machine_changes
|
|
// 6: machine
|
|
if ((base.resolve != "None" && base.resolve) && (stackLevel != 0) && (stackLevel != 1))
|
|
{
|
|
// We have a resolve function. Indicates that the setting is not settable per extruder and that
|
|
// we have to choose between the resolved value (default) and the global value
|
|
// (if user has explicitly set this).
|
|
return base.resolve
|
|
}
|
|
else {
|
|
return propertyProvider.properties.value
|
|
}
|
|
}
|
|
}
|
|
|
|
MouseArea
|
|
{
|
|
id: mouseArea
|
|
anchors.fill: parent
|
|
|
|
cursorShape: Qt.IBeamCursor
|
|
|
|
onPressed:(mouse)=> {
|
|
if (!input.activeFocus)
|
|
{
|
|
base.focusGainedByClick = true
|
|
input.forceActiveFocus()
|
|
}
|
|
mouse.accepted = false
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|