648 lines
15 KiB
648 lines
15 KiB
import QtQuick 2.0
|
|
import QtQuick.Layouts 1.0
|
|
import QtQuick.Controls 1.1
|
|
import QtQuick.Controls.Styles 1.3
|
|
import org.ethereum.qml.SortFilterProxyModel 1.0
|
|
|
|
Rectangle
|
|
{
|
|
property variant statusPane
|
|
property variant currentStatus
|
|
property int contentXPos: logStyle.generic.layout.dateWidth + logStyle.generic.layout.typeWidth - 70
|
|
|
|
function clear()
|
|
{
|
|
logsModel.clear();
|
|
statusPane.clear();
|
|
currentStatus = undefined;
|
|
}
|
|
|
|
function push(_level, _type, _content)
|
|
{
|
|
_content = _content.replace(/\n/g, " ")
|
|
logsModel.insert(0, { "type": _type, "date": Qt.formatDateTime(new Date(), "hh:mm:ss"), "content": _content, "level": _level });
|
|
}
|
|
|
|
onVisibleChanged:
|
|
{
|
|
if (currentStatus && visible && (logsModel.count === 0 || logsModel.get(0).content !== currentStatus.content || logsModel.get(0).date !== currentStatus.date))
|
|
logsModel.insert(0, { "type": currentStatus.type, "date": currentStatus.date, "content": currentStatus.content, "level": currentStatus.level });
|
|
else if (!visible)
|
|
{
|
|
for (var k = 0; k < logsModel.count; k++)
|
|
{
|
|
if (logsModel.get(k).type === "Comp") //do not keep compilation logs.
|
|
logsModel.remove(k);
|
|
}
|
|
}
|
|
}
|
|
|
|
anchors.fill: parent
|
|
radius: 10
|
|
color: "transparent"
|
|
id: logsPane
|
|
Column {
|
|
z: 2
|
|
height: parent.height - rowAction.height
|
|
width: parent.width
|
|
spacing: 0
|
|
ListModel {
|
|
id: logsModel
|
|
}
|
|
|
|
ScrollView
|
|
{
|
|
id: scrollView
|
|
height: parent.height
|
|
width: parent.width
|
|
horizontalScrollBarPolicy: Qt.ScrollBarAlwaysOff
|
|
Column
|
|
{
|
|
id: logsRect
|
|
spacing: 0
|
|
Repeater {
|
|
id: logsRepeater
|
|
clip: true
|
|
property string frontColor: "transparent"
|
|
model: SortFilterProxyModel {
|
|
id: proxyModel
|
|
source: logsModel
|
|
property var roles: ["-", "javascript", "run", "state", "comp", "deployment"]
|
|
|
|
Component.onCompleted: {
|
|
filterType = regEx(proxyModel.roles);
|
|
}
|
|
|
|
function search(_value)
|
|
{
|
|
filterContent = _value;
|
|
}
|
|
|
|
function toogleFilter(_value)
|
|
{
|
|
var count = roles.length;
|
|
for (var i in roles)
|
|
{
|
|
if (roles[i] === _value)
|
|
{
|
|
roles.splice(i, 1);
|
|
break;
|
|
}
|
|
}
|
|
if (count === roles.length)
|
|
roles.push(_value);
|
|
|
|
filterType = regEx(proxyModel.roles);
|
|
}
|
|
|
|
function regEx(_value)
|
|
{
|
|
return "(?:" + roles.join('|') + ")";
|
|
}
|
|
|
|
filterType: "(?:javascript|run|state|comp|deployment)"
|
|
filterContent: ""
|
|
filterSyntax: SortFilterProxyModel.RegExp
|
|
filterCaseSensitivity: Qt.CaseInsensitive
|
|
}
|
|
|
|
Rectangle
|
|
{
|
|
width: logStyle.generic.layout.dateWidth + logStyle.generic.layout.contentWidth + logStyle.generic.layout.typeWidth
|
|
height: 30
|
|
color:
|
|
{
|
|
var cl;
|
|
if (level === "warning" || level === "error")
|
|
cl = logStyle.generic.layout.errorColor;
|
|
else
|
|
cl = index % 2 === 0 ? "transparent" : logStyle.generic.layout.logAlternateColor;
|
|
if (index === 0)
|
|
logsRepeater.frontColor = cl;
|
|
return cl;
|
|
}
|
|
|
|
Component.onCompleted:
|
|
{
|
|
logsPane.contentXPos = logContent.x
|
|
}
|
|
|
|
MouseArea
|
|
{
|
|
anchors.fill: parent
|
|
onClicked:
|
|
{
|
|
if (logContent.elide === Text.ElideNone)
|
|
{
|
|
logContent.elide = Text.ElideRight;
|
|
logContent.wrapMode = Text.NoWrap
|
|
parent.height = 30;
|
|
}
|
|
else
|
|
{
|
|
logContent.elide = Text.ElideNone;
|
|
logContent.wrapMode = Text.WordWrap;
|
|
parent.height = logContent.lineCount * 30;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
DefaultLabel {
|
|
id: dateLabel
|
|
text: date;
|
|
font.family: logStyle.generic.layout.logLabelFont
|
|
width: logStyle.generic.layout.dateWidth
|
|
font.pointSize: appStyle.absoluteSize(-1)
|
|
clip: true
|
|
elide: Text.ElideRight
|
|
anchors.left: parent.left
|
|
anchors.leftMargin: 15
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
color: {
|
|
parent.getColor(level);
|
|
}
|
|
}
|
|
|
|
DefaultLabel {
|
|
text: type;
|
|
id: typeLabel
|
|
font.family: logStyle.generic.layout.logLabelFont
|
|
width: logStyle.generic.layout.typeWidth
|
|
clip: true
|
|
elide: Text.ElideRight
|
|
font.pointSize: appStyle.absoluteSize(-1)
|
|
anchors.left: dateLabel.right
|
|
anchors.leftMargin: 2
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
color: {
|
|
parent.getColor(level);
|
|
}
|
|
}
|
|
|
|
Text {
|
|
id: logContent
|
|
text: content;
|
|
font.family: logStyle.generic.layout.logLabelFont
|
|
width: logStyle.generic.layout.contentWidth
|
|
font.pointSize: appStyle.absoluteSize(-1)
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
elide: Text.ElideRight
|
|
anchors.left: typeLabel.right
|
|
anchors.leftMargin: 2
|
|
color: {
|
|
parent.getColor(level);
|
|
}
|
|
}
|
|
|
|
function getColor()
|
|
{
|
|
if (level === "error")
|
|
return "red";
|
|
else if (level === "warning")
|
|
return "orange";
|
|
else
|
|
return "#808080";
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Component {
|
|
id: itemDelegate
|
|
DefaultLabel {
|
|
text: styleData.value;
|
|
font.family: logStyle.generic.layout.logLabelFont
|
|
font.pointSize: appStyle.absoluteSize(-1)
|
|
color: {
|
|
if (proxyModel.get(styleData.row).level === "error")
|
|
return "red";
|
|
else if (proxyModel.get(styleData.row).level === "warning")
|
|
return "orange";
|
|
else
|
|
return "#808080";
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Rectangle
|
|
{
|
|
gradient: Gradient {
|
|
GradientStop { position: 0.0; color: "#f1f1f1" }
|
|
GradientStop { position: 1.0; color: "#d9d7da" }
|
|
}
|
|
Layout.preferredHeight: logStyle.generic.layout.headerHeight
|
|
height: logStyle.generic.layout.headerHeight
|
|
width: logsPane.width
|
|
anchors.bottom: parent.bottom
|
|
Row
|
|
{
|
|
id: rowAction
|
|
anchors.leftMargin: logStyle.generic.layout.leftMargin
|
|
anchors.left: parent.left
|
|
spacing: logStyle.generic.layout.headerButtonSpacing
|
|
height: parent.height
|
|
Rectangle
|
|
{
|
|
color: "transparent"
|
|
height: parent.height
|
|
width: 40
|
|
DefaultLabel
|
|
{
|
|
width: 40
|
|
elide: Text.ElideRight
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
color: logStyle.generic.layout.logLabelColor
|
|
font.pointSize: appStyle.absoluteSize(-3)
|
|
font.family: logStyle.generic.layout.logLabelFont
|
|
text: qsTr("Show:")
|
|
}
|
|
}
|
|
|
|
Rectangle {
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
width: 1;
|
|
height: parent.height
|
|
color: logStyle.generic.layout.buttonSeparatorColor1
|
|
}
|
|
|
|
Rectangle {
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
width: 2;
|
|
height: parent.height
|
|
color: logStyle.generic.layout.buttonSeparatorColor2
|
|
}
|
|
|
|
ToolButton {
|
|
id: javascriptButton
|
|
checkable: true
|
|
height: logStyle.generic.layout.headerButtonHeight
|
|
width: 30
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
checked: true
|
|
onCheckedChanged: {
|
|
proxyModel.toogleFilter("javascript")
|
|
}
|
|
tooltip: qsTr("JavaScript")
|
|
style:
|
|
ButtonStyle {
|
|
label:
|
|
Item {
|
|
Rectangle
|
|
{
|
|
width: labelJs.width
|
|
height: labelJs.height
|
|
anchors.centerIn: parent
|
|
color: "transparent"
|
|
DefaultLabel {
|
|
id: labelJs
|
|
width: 15
|
|
elide: Text.ElideRight
|
|
anchors.horizontalCenter: parent.horizontalCenter
|
|
font.family: logStyle.generic.layout.logLabelFont
|
|
font.pointSize: appStyle.absoluteSize(-3)
|
|
color: logStyle.generic.layout.logLabelColor
|
|
text: qsTr("JS")
|
|
}
|
|
}
|
|
}
|
|
background:
|
|
Rectangle {
|
|
color: javascriptButton.checked ? logStyle.generic.layout.buttonSelected : "transparent"
|
|
}
|
|
}
|
|
}
|
|
|
|
Rectangle {
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
width: 1;
|
|
height: parent.height
|
|
color: logStyle.generic.layout.buttonSeparatorColor1
|
|
}
|
|
|
|
Rectangle {
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
width: 2;
|
|
height: parent.height
|
|
color: logStyle.generic.layout.buttonSeparatorColor2
|
|
}
|
|
|
|
ToolButton {
|
|
id: runButton
|
|
checkable: true
|
|
height: logStyle.generic.layout.headerButtonHeight
|
|
width: 40
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
checked: true
|
|
onCheckedChanged: {
|
|
proxyModel.toogleFilter("run")
|
|
}
|
|
tooltip: qsTr("Run")
|
|
style:
|
|
ButtonStyle {
|
|
label:
|
|
Item {
|
|
Rectangle
|
|
{
|
|
width: labelRun.width
|
|
height: labelRun.height
|
|
anchors.centerIn: parent
|
|
color: "transparent"
|
|
DefaultLabel {
|
|
id: labelRun
|
|
width: 25
|
|
anchors.horizontalCenter: parent.horizontalCenter
|
|
elide: Text.ElideRight
|
|
font.family: logStyle.generic.layout.logLabelFont
|
|
font.pointSize: appStyle.absoluteSize(-3)
|
|
color: logStyle.generic.layout.logLabelColor
|
|
text: qsTr("Run")
|
|
}
|
|
}
|
|
}
|
|
background:
|
|
Rectangle {
|
|
color: runButton.checked ? logStyle.generic.layout.buttonSelected : "transparent"
|
|
}
|
|
}
|
|
}
|
|
|
|
Rectangle {
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
width: 1;
|
|
height: parent.height
|
|
color: logStyle.generic.layout.buttonSeparatorColor1
|
|
}
|
|
|
|
Rectangle {
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
width: 2;
|
|
height: parent.height
|
|
color: logStyle.generic.layout.buttonSeparatorColor2
|
|
}
|
|
|
|
ToolButton {
|
|
id: stateButton
|
|
checkable: true
|
|
height: logStyle.generic.layout.headerButtonHeight
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
width: 40
|
|
checked: true
|
|
onCheckedChanged: {
|
|
proxyModel.toogleFilter("state")
|
|
}
|
|
tooltip: qsTr("State")
|
|
style:
|
|
ButtonStyle {
|
|
label:
|
|
Item {
|
|
Rectangle
|
|
{
|
|
width: labelState.width
|
|
height: labelState.height
|
|
anchors.centerIn: parent
|
|
color: "transparent"
|
|
DefaultLabel {
|
|
id: labelState
|
|
width: 30
|
|
anchors.horizontalCenter: parent.horizontalCenter
|
|
elide: Text.ElideRight
|
|
font.family: logStyle.generic.layout.logLabelFont
|
|
font.pointSize: appStyle.absoluteSize(-3)
|
|
color: logStyle.generic.layout.logLabelColor
|
|
text: qsTr("State")
|
|
}
|
|
}
|
|
}
|
|
background:
|
|
Rectangle {
|
|
color: stateButton.checked ? logStyle.generic.layout.buttonSelected : "transparent"
|
|
}
|
|
}
|
|
}
|
|
|
|
Rectangle {
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
width: 1;
|
|
height: parent.height
|
|
color: logStyle.generic.layout.buttonSeparatorColor1
|
|
}
|
|
|
|
Rectangle {
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
width: 2;
|
|
height: parent.height
|
|
color: logStyle.generic.layout.buttonSeparatorColor2
|
|
}
|
|
|
|
ToolButton {
|
|
id: deloyButton
|
|
checkable: true
|
|
height: logStyle.generic.layout.headerButtonHeight
|
|
width: 50
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
checked: true
|
|
onCheckedChanged: {
|
|
proxyModel.toogleFilter("deployment")
|
|
}
|
|
tooltip: qsTr("Deployment")
|
|
style:
|
|
ButtonStyle {
|
|
label:
|
|
Item {
|
|
Rectangle
|
|
{
|
|
width: labelDeploy.width
|
|
height: labelDeploy.height
|
|
anchors.centerIn: parent
|
|
color: "transparent"
|
|
DefaultLabel {
|
|
width: 40
|
|
id: labelDeploy
|
|
anchors.horizontalCenter: parent.horizontalCenter
|
|
elide: Text.ElideRight
|
|
font.family: logStyle.generic.layout.logLabelFont
|
|
font.pointSize: appStyle.absoluteSize(-3)
|
|
color: logStyle.generic.layout.logLabelColor
|
|
text: qsTr("Deploy.")
|
|
}
|
|
}
|
|
}
|
|
background:
|
|
Rectangle {
|
|
color: deloyButton.checked ? logStyle.generic.layout.buttonSelected : "transparent"
|
|
}
|
|
}
|
|
}
|
|
|
|
Rectangle {
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
width: 1;
|
|
height: parent.height
|
|
color: logStyle.generic.layout.buttonSeparatorColor1
|
|
}
|
|
|
|
Rectangle {
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
width: 2;
|
|
height: parent.height
|
|
color: logStyle.generic.layout.buttonSeparatorColor2
|
|
}
|
|
}
|
|
|
|
Row
|
|
{
|
|
height: parent.height
|
|
anchors.right: parent.right
|
|
anchors.rightMargin: 10
|
|
spacing: 10
|
|
Rectangle
|
|
{
|
|
height: logStyle.generic.layout.headerButtonHeight
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
color: "transparent"
|
|
width: 20
|
|
Button
|
|
{
|
|
id: clearButton
|
|
action: clearAction
|
|
anchors.fill: parent
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
height: 25
|
|
style:
|
|
ButtonStyle {
|
|
background:
|
|
Rectangle {
|
|
height: logStyle.generic.layout.headerButtonHeight
|
|
implicitHeight: logStyle.generic.layout.headerButtonHeight
|
|
color: "transparent"
|
|
}
|
|
}
|
|
}
|
|
|
|
Image {
|
|
id: clearImage
|
|
source: clearAction.enabled ? "qrc:/qml/img/cleariconactive.png" : "qrc:/qml/img/clearicon.png"
|
|
anchors.centerIn: parent
|
|
fillMode: Image.PreserveAspectFit
|
|
width: 20
|
|
height: 25
|
|
}
|
|
|
|
Action {
|
|
id: clearAction
|
|
enabled: logsModel.count > 0
|
|
tooltip: qsTr("Clear")
|
|
onTriggered: {
|
|
logsPane.clear()
|
|
}
|
|
}
|
|
}
|
|
|
|
Rectangle
|
|
{
|
|
height: logStyle.generic.layout.headerButtonHeight
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
color: "transparent"
|
|
width: 20
|
|
Button
|
|
{
|
|
id: copyButton
|
|
action: copyAction
|
|
anchors.fill: parent
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
height: 25
|
|
style:
|
|
ButtonStyle {
|
|
background:
|
|
Rectangle {
|
|
height: logStyle.generic.layout.headerButtonHeight
|
|
implicitHeight: logStyle.generic.layout.headerButtonHeight
|
|
color: "transparent"
|
|
}
|
|
}
|
|
}
|
|
|
|
Image {
|
|
id: copyImage
|
|
source: copyAction.enabled ? "qrc:/qml/img/copyiconactive.png" : "qrc:/qml/img/copyicon.png"
|
|
anchors.centerIn: parent
|
|
fillMode: Image.PreserveAspectFit
|
|
width: 20
|
|
height: 25
|
|
}
|
|
|
|
Action {
|
|
id: copyAction
|
|
enabled: logsModel.count > 0
|
|
tooltip: qsTr("Copy to Clipboard")
|
|
onTriggered: {
|
|
var content = "";
|
|
for (var k = 0; k < logsModel.count; k++)
|
|
{
|
|
var log = logsModel.get(k);
|
|
content += log.type + "\t" + log.level + "\t" + log.date + "\t" + log.content + "\n";
|
|
}
|
|
clipboard.text = content;
|
|
}
|
|
}
|
|
}
|
|
|
|
Rectangle
|
|
{
|
|
width: 120
|
|
radius: 10
|
|
height: 25
|
|
color: "white"
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
|
|
Image
|
|
{
|
|
id: searchImg
|
|
source: "qrc:/qml/img/searchicon.png"
|
|
fillMode: Image.PreserveAspectFit
|
|
width: 20
|
|
height: 25
|
|
z: 3
|
|
}
|
|
|
|
DefaultTextField
|
|
{
|
|
id: searchBox
|
|
z: 2
|
|
width: 100
|
|
anchors.left: searchImg.right
|
|
anchors.leftMargin: -7
|
|
font.family: logStyle.generic.layout.logLabelFont
|
|
font.pointSize: appStyle.absoluteSize(-3)
|
|
font.italic: true
|
|
text: qsTr(" - Search - ")
|
|
onFocusChanged:
|
|
{
|
|
if (!focus && text === "")
|
|
text = qsTr(" - Search - ");
|
|
else if (focus && text === qsTr(" - Search - "))
|
|
text = "";
|
|
}
|
|
|
|
onTextChanged: {
|
|
if (text === qsTr(" - Search - "))
|
|
proxyModel.search("");
|
|
else
|
|
proxyModel.search(text);
|
|
}
|
|
|
|
style:
|
|
TextFieldStyle {
|
|
background: Rectangle {
|
|
radius: 10
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|