mirror of
https://github.com/alicevision/Meshroom.git
synced 2025-04-28 17:57:16 +02:00
378 lines
No EOL
11 KiB
QML
378 lines
No EOL
11 KiB
QML
import QtQuick
|
|
import QtQuick.Controls
|
|
import QtQuick.Dialogs
|
|
import QtQuick.Layouts
|
|
|
|
import Controls 1.0
|
|
import MaterialIcons 2.2
|
|
import Utils 1.0
|
|
|
|
import Qt.labs.platform 1.0 as Platform
|
|
|
|
import ScriptEditor 1.0
|
|
|
|
Item {
|
|
id: root
|
|
|
|
// Defines the parent or the root Application of which this script editor is a part of
|
|
property var rootApplication: undefined;
|
|
|
|
Component {
|
|
id: clearConfirmationDialog
|
|
|
|
MessageDialog {
|
|
title: "Clear history"
|
|
|
|
preset: "Warning"
|
|
text: "This will clear all history of executed scripts."
|
|
helperText: "Are you sure you would like to continue?."
|
|
|
|
standardButtons: Dialog.Ok | Dialog.Cancel
|
|
onClosed: destroy()
|
|
}
|
|
}
|
|
|
|
function replace(text, string, replacement) {
|
|
/**
|
|
* Replaces all occurences of the string in the text
|
|
* @param text - overall text
|
|
* @param string - the string to be replaced in the text
|
|
* @param replacement - the replacement of the string
|
|
*/
|
|
// Split with the string
|
|
let lines = text.split(string)
|
|
// Return the overall text joined with the replacement
|
|
return lines.join(replacement)
|
|
}
|
|
|
|
function formatInput(text) {
|
|
/**
|
|
* Formats the text to be displayed as the input script executed
|
|
*/
|
|
|
|
// Replace the text to be RichText Supportive
|
|
return "<font color=#868686>> Input:<br>" + replace(text, "\n", "<br>") + "</font><br>"
|
|
}
|
|
|
|
function formatOutput(text) {
|
|
/**
|
|
* Formats the text to be displayed as the result of the script executed
|
|
*/
|
|
|
|
// Replace the text to be RichText Supportive
|
|
return "<font color=#49a1f3>> Result:<br>" + replace(text, "\n", "<br>") + "</font><br>"
|
|
}
|
|
|
|
function clearHistory() {
|
|
/**
|
|
* Clears all of the executed history from the script editor
|
|
*/
|
|
ScriptEditorManager.clearHistory()
|
|
input.clear()
|
|
output.clear()
|
|
}
|
|
|
|
function processScript(text = "") {
|
|
// Use either the provided/selected or the entire script
|
|
text = text || input.text
|
|
|
|
// Execute the process and fetch back the return for it
|
|
var ret = ScriptEditorManager.process(text)
|
|
|
|
// Append the input script and the output result to the output console
|
|
output.append(formatInput(text) + formatOutput(ret))
|
|
|
|
// Save the entire script after executing the commands
|
|
ScriptEditorManager.saveScript(input.text)
|
|
}
|
|
|
|
function loadScript(fileUrl) {
|
|
var request = new XMLHttpRequest()
|
|
request.open("GET", fileUrl, false)
|
|
request.send(null)
|
|
return request.responseText
|
|
}
|
|
|
|
function saveScript(fileUrl, content) {
|
|
var request = new XMLHttpRequest()
|
|
request.open("PUT", fileUrl, false)
|
|
request.send(content)
|
|
return request.status
|
|
}
|
|
|
|
implicitWidth: 500
|
|
implicitHeight: 500
|
|
|
|
Platform.FileDialog {
|
|
id: loadScriptDialog
|
|
options: Platform.FileDialog.DontUseNativeDialog
|
|
title: "Load Script"
|
|
nameFilters: ["Python Script (*.py)"]
|
|
onAccepted: {
|
|
input.clear()
|
|
input.text = loadScript(currentFile)
|
|
}
|
|
}
|
|
|
|
Platform.FileDialog {
|
|
id: saveScriptDialog
|
|
options: Platform.FileDialog.DontUseNativeDialog
|
|
title: "Save script"
|
|
nameFilters: ["Python Script (*.py)"]
|
|
fileMode: Platform.FileDialog.SaveFile
|
|
|
|
signal closed(var result)
|
|
|
|
onAccepted: {
|
|
if (Filepath.extension(currentFile) != ".py")
|
|
currentFile = currentFile + ".py"
|
|
var ret = saveScript(currentFile, input.text)
|
|
if (ret)
|
|
closed(Platform.Dialog.Accepted)
|
|
else
|
|
closed(Platform.Dialog.Rejected)
|
|
}
|
|
|
|
onRejected: closed(Platform.Dialog.Rejected)
|
|
}
|
|
|
|
ColumnLayout {
|
|
anchors.fill: parent
|
|
|
|
RowLayout {
|
|
Layout.alignment: Qt.AlignVCenter | Qt.AlignHCenter
|
|
|
|
MaterialToolButton {
|
|
font.pointSize: 13
|
|
text: MaterialIcons.file_open
|
|
ToolTip.text: "Load Script"
|
|
|
|
onClicked: {
|
|
loadScriptDialog.open()
|
|
}
|
|
}
|
|
|
|
MaterialToolButton {
|
|
font.pointSize: 13
|
|
text: MaterialIcons.save
|
|
ToolTip.text: "Save Script"
|
|
|
|
onClicked: {
|
|
saveScriptDialog.open()
|
|
}
|
|
}
|
|
|
|
MaterialToolButton {
|
|
font.pointSize: 13
|
|
text: MaterialIcons.history
|
|
ToolTip.text: "Get Previous Script"
|
|
|
|
enabled: ScriptEditorManager.hasPreviousScript;
|
|
|
|
onClicked: {
|
|
var ret = ScriptEditorManager.getPreviousScript()
|
|
|
|
if (ret != "") {
|
|
input.clear()
|
|
input.text = ret
|
|
}
|
|
}
|
|
}
|
|
|
|
MaterialToolButton {
|
|
font.pointSize: 13
|
|
text: MaterialIcons.update
|
|
ToolTip.text: "Get Next Script"
|
|
|
|
enabled: ScriptEditorManager.hasNextScript;
|
|
|
|
onClicked: {
|
|
var ret = ScriptEditorManager.getNextScript()
|
|
|
|
if (ret != "") {
|
|
input.clear()
|
|
input.text = ret
|
|
}
|
|
}
|
|
}
|
|
|
|
MaterialToolButton {
|
|
font.pointSize: 13
|
|
text: MaterialIcons.delete_sweep
|
|
ToolTip.text: "Clear History"
|
|
|
|
onClicked: {
|
|
// Confirm from the user before clearing out any history
|
|
const confirmationDialog = clearConfirmationDialog.createObject(rootApplication ? rootApplication : root);
|
|
confirmationDialog.accepted.connect(clearHistory);
|
|
confirmationDialog.open();
|
|
}
|
|
}
|
|
|
|
Item {
|
|
width: executeButton.width;
|
|
}
|
|
|
|
MaterialToolButton {
|
|
id: executeButton
|
|
font.pointSize: 13
|
|
text: MaterialIcons.play_arrow
|
|
ToolTip.text: "Execute Script"
|
|
|
|
onClicked: {
|
|
root.processScript()
|
|
}
|
|
}
|
|
|
|
Item {
|
|
Layout.fillWidth: true
|
|
}
|
|
|
|
MaterialToolButton {
|
|
font.pointSize: 13
|
|
text: MaterialIcons.backspace
|
|
ToolTip.text: "Clear Output Window"
|
|
|
|
onClicked: {
|
|
output.clear()
|
|
}
|
|
}
|
|
}
|
|
|
|
MSplitView {
|
|
id: scriptSplitView;
|
|
Layout.fillHeight: true;
|
|
Layout.fillWidth: true;
|
|
orientation: Qt.Horizontal;
|
|
|
|
// Input Text Area -- Holds the input scripts to be executed
|
|
Rectangle {
|
|
id: inputArea
|
|
SplitView.preferredWidth: root.width / 2;
|
|
|
|
color: palette.base
|
|
|
|
ListView {
|
|
id: lineNumbers
|
|
property TextMetrics textMetrics: TextMetrics { text: "9999" }
|
|
model: input.text.split(/\n/g)
|
|
anchors.left: parent.left
|
|
anchors.top: parent.top
|
|
anchors.bottom: parent.bottom
|
|
width: lineNumbers.textMetrics.boundingRect.width
|
|
clip: false
|
|
|
|
delegate: Rectangle {
|
|
width: lineNumbers.width
|
|
height: lineText.height
|
|
color: palette.mid
|
|
Text {
|
|
id: lineNumber
|
|
anchors.horizontalCenter: parent.horizontalCenter
|
|
text: index + 1
|
|
font.bold: true
|
|
color: palette.text
|
|
}
|
|
|
|
Text {
|
|
id: lineText
|
|
width: flickableInput.width
|
|
text: modelData
|
|
visible: false
|
|
wrapMode: Text.WordWrap
|
|
}
|
|
}
|
|
|
|
onContentYChanged: {
|
|
if (!moving)
|
|
return
|
|
flickableInput.contentY = contentY
|
|
}
|
|
}
|
|
|
|
Flickable {
|
|
id: flickableInput
|
|
width: parent.width
|
|
height: parent.height
|
|
contentWidth: width
|
|
contentHeight: input.contentHeight;
|
|
|
|
anchors.left: lineNumbers.right
|
|
anchors.top: parent.top
|
|
anchors.right: parent.right
|
|
anchors.bottom: parent.bottom
|
|
|
|
ScrollBar.vertical: MScrollBar {}
|
|
|
|
TextArea.flickable: TextArea {
|
|
id: input
|
|
|
|
text: ScriptEditorManager.loadLastScript()
|
|
|
|
font: lineNumbers.textMetrics.font
|
|
Layout.fillHeight: true
|
|
Layout.fillWidth: true
|
|
|
|
wrapMode: Text.WordWrap
|
|
selectByMouse: true
|
|
padding: 0
|
|
|
|
onPressed: {
|
|
root.forceActiveFocus()
|
|
}
|
|
|
|
Keys.onPressed: function(event) {
|
|
if ((event.key === Qt.Key_Enter || event.key === Qt.Key_Return) && event.modifiers === Qt.ControlModifier) {
|
|
root.processScript(input.selectedText)
|
|
}
|
|
}
|
|
}
|
|
|
|
onContentYChanged: {
|
|
if (lineNumbers.moving)
|
|
return
|
|
lineNumbers.contentY = contentY
|
|
}
|
|
}
|
|
}
|
|
|
|
// Output Text Area -- Shows the output for the executed script(s)
|
|
Rectangle {
|
|
id: outputArea
|
|
Layout.fillHeight: true
|
|
Layout.fillWidth: true
|
|
|
|
color: palette.base
|
|
|
|
Flickable {
|
|
width: parent.width
|
|
height: parent.height
|
|
contentWidth: width
|
|
contentHeight: output.contentHeight;
|
|
|
|
ScrollBar.vertical: MScrollBar {}
|
|
|
|
TextArea.flickable: TextArea {
|
|
id: output
|
|
|
|
readOnly: true
|
|
selectByMouse: true
|
|
padding: 0
|
|
Layout.fillHeight: true
|
|
Layout.fillWidth: true
|
|
wrapMode: Text.WordWrap
|
|
|
|
textFormat: Text.RichText
|
|
}
|
|
}
|
|
}
|
|
|
|
// Syntax Highlights for the Input Area for Python Based Syntax
|
|
PySyntaxHighlighter {
|
|
id: syntaxHighlighter
|
|
// The document to highlight
|
|
textDocument: input.textDocument
|
|
}
|
|
}
|
|
}
|
|
} |