@ -7,81 +7,259 @@ import "."
Rectangle
{
property variant currentStatus ;
function push ( _level , _type , _content )
{
_content = _content . replace ( /\n/g , " " )
logsModel . insert ( 0 , { "type" : _type , "date" : Qt . formatDateTime ( new Date ( ) , "hh:mm:ss dd.MM.yyyy" ) , "content" : _content , "level" : _level } ) ;
logsModel . insert ( 0 , { "type" : _type , "date" : Qt . formatDateTime ( new Date ( ) , "hh:mm:ss" ) , "content" : _content , "level" : _level } ) ;
}
onVisibleChanged:
{
if ( visible && ( logsModel . count === 0 || ( logsModel . get ( 0 ) . date !== currentStatus . date && logsModel . get ( 0 ) . content !== currentStatus . content ) ) )
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" ) / / d o n o t k e e p c o m p i l a t i o n l o g s .
logsModel . remove ( k ) ;
}
}
}
anchors.fill: parent
radius: 5
color: LogsPaneStyle . generic . layout . backgroundColor
border.color: LogsPaneStyle . generic . layout . borderColor
border.width: LogsPaneStyle . generic . layout . borderWidth
ColumnLayout {
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
Row
Repeater {
id: logsRepeater
clip: true
property string frontColor: "transparent"
model: SortFilterProxyModel {
id: proxyModel
source: logsModel
property var roles: [ "-" , "javascript" , "run" , "state" , "comp" ]
Component.onCompleted: {
filterType = regEx ( proxyModel . roles ) ;
}
function search ( _value )
{
id: rowAction
Layout.preferredHeight: LogsPaneStyle . generic . layout . headerHeight
height: LogsPaneStyle . generic . layout . headerHeight
anchors.leftMargin: LogsPaneStyle . generic . layout . leftMargin
anchors.left: parent . left
spacing: LogsPaneStyle . generic . layout . headerButtonSpacing
Button
filterContent = _value ;
}
function toogleFilter ( _value )
{
height: LogsPaneStyle . generic . layout . headerButtonHeight
anchors.verticalCenter: parent . verticalCenter
action: clearAction
iconSource: "qrc:/qml/img/broom.png"
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 ) ;
Action {
id: clearAction
enabled: logsModel . count > 0
tooltip: qsTr ( "Clear" )
onTriggered: {
logsModel . clear ( )
filterType = regEx ( proxyModel . roles ) ;
}
function regEx ( _value )
{
return "(?:" + roles . join ( '|' ) + ")" ;
}
Button
filterType: "(?:javascript|run|state|comp)"
filterContent: ""
filterSyntax: SortFilterProxyModel . RegExp
filterCaseSensitivity: Qt . CaseInsensitive
}
Rectangle
{
height: LogsPaneStyle . generic . layout . headerButtonHeight
width: 750
height: 30
color:
{
var cl ;
if ( level === "warning" || level === "error" )
cl = LogsPaneStyle . generic . layout . errorColor ;
else
cl = index % 2 === 0 ? "transparent" : LogsPaneStyle . generic . layout . logAlternateColor ;
if ( index === 0 )
logsRepeater . frontColor = cl ;
return cl ;
}
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 {
text: date ;
font.family: LogsPaneStyle . generic . layout . logLabelFont
width: LogsPaneStyle . generic . layout . dateWidth
font.pointSize: Style . absoluteSize ( - 1 )
anchors.left: parent . left
anchors.leftMargin: 15
anchors.verticalCenter: parent . verticalCenter
action: copytoClipBoardAction
iconSource: "qrc:/qml/img/copy.png"
color: {
parent . getColor ( level ) ;
}
}
Action {
id: copytoClipBoardAction
enabled: logsModel . count > 0
tooltip: qsTr ( "Copy to Clipboard" )
onTriggered: {
var content = "" ;
for ( var k = 0 ; k < logsModel . count ; k ++ )
DefaultLabel {
text: type ;
font.family: LogsPaneStyle . generic . layout . logLabelFont
width: LogsPaneStyle . generic . layout . typeWidth
font.pointSize: Style . absoluteSize ( - 1 )
anchors.left: parent . left
anchors.leftMargin: 100
anchors.verticalCenter: parent . verticalCenter
color: {
parent . getColor ( level ) ;
}
}
Text {
id: logContent
text: content ;
font.family: LogsPaneStyle . generic . layout . logLabelFont
width: LogsPaneStyle . generic . layout . contentWidth
font.pointSize: Style . absoluteSize ( - 1 )
anchors.verticalCenter: parent . verticalCenter
elide: Text . ElideRight
anchors.left: parent . left
anchors.leftMargin: 190
color: {
parent . getColor ( level ) ;
}
}
function getColor ( )
{
var log = logsModel . get ( k ) ;
content += log . type + "\t" + log . level + "\t" + log . date + "\t" + log . content + "\n" ;
if ( level === "error" )
return "red" ;
else if ( level === "warning" )
return "orange" ;
else
return "#808080" ;
}
}
}
clipboard . text = content ;
}
}
Component {
id: itemDelegate
DefaultLabel {
text: styleData . value ;
font.family: LogsPaneStyle . generic . layout . logLabelFont
font.pointSize: Style . 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: LogsPaneStyle . generic . layout . headerHeight
height: LogsPaneStyle . generic . layout . headerHeight
width: logsPane . width
anchors.bottom: parent . bottom
Row
{
id: rowAction
anchors.leftMargin: LogsPaneStyle . generic . layout . leftMargin
anchors.left: parent . left
spacing: LogsPaneStyle . generic . layout . headerButtonSpacing
height: parent . height
Rectangle
{
color: "transparent"
height: parent . height
width: 40
DefaultLabel
{
anchors.verticalCenter: parent . verticalCenter
color: LogsPaneStyle . generic . layout . logLabelColor
font.pointSize: Style . absoluteSize ( - 3 )
font.family: LogsPaneStyle . generic . layout . logLabelFont
text: qsTr ( "Show:" )
}
}
Rectangle {
anchors.verticalCenter: parent . verticalCenter
width: 1 ;
height: parent . height - 10
color : "#808080"
height: parent . height
color: LogsPaneStyle . generic . layout . buttonSeparatorColor1
}
Rectangle {
anchors.verticalCenter: parent . verticalCenter
width: 2 ;
height: parent . height
color: LogsPaneStyle . generic . layout . buttonSeparatorColor2
}
ToolButton {
id: javascriptButton
checkable: true
height: LogsPaneStyle . generic . layout . headerButtonHeight
width: 20
anchors.verticalCenter: parent . verticalCenter
checked: true
onCheckedChanged: {
@ -97,16 +275,35 @@ Rectangle
font.pointSize: Style . absoluteSize ( - 3 )
color: LogsPaneStyle . generic . layout . logLabelColor
anchors.centerIn: parent
text: qsTr ( "JavaScript" )
text: qsTr ( "JS" )
}
}
background:
Rectangle {
color: javascriptButton . checked ? LogsPaneStyle.generic.layout.buttonSelected : "transparent"
}
}
}
Rectangle {
anchors.verticalCenter: parent . verticalCenter
width: 1 ;
height: parent . height
color: LogsPaneStyle . generic . layout . buttonSeparatorColor1
}
Rectangle {
anchors.verticalCenter: parent . verticalCenter
width: 2 ;
height: parent . height
color: LogsPaneStyle . generic . layout . buttonSeparatorColor2
}
ToolButton {
id: runButton
checkable: true
height: LogsPaneStyle . generic . layout . headerButtonHeight
width: 30
anchors.verticalCenter: parent . verticalCenter
checked: true
onCheckedChanged: {
@ -125,7 +322,25 @@ Rectangle
text: qsTr ( "Run" )
}
}
background:
Rectangle {
color: runButton . checked ? LogsPaneStyle.generic.layout.buttonSelected : "transparent"
}
}
}
Rectangle {
anchors.verticalCenter: parent . verticalCenter
width: 1 ;
height: parent . height
color: LogsPaneStyle . generic . layout . buttonSeparatorColor1
}
Rectangle {
anchors.verticalCenter: parent . verticalCenter
width: 2 ;
height: parent . height
color: LogsPaneStyle . generic . layout . buttonSeparatorColor2
}
ToolButton {
@ -133,6 +348,7 @@ Rectangle
checkable: true
height: LogsPaneStyle . generic . layout . headerButtonHeight
anchors.verticalCenter: parent . verticalCenter
width: 35
checked: true
onCheckedChanged: {
proxyModel . toogleFilter ( "state" )
@ -145,133 +361,206 @@ Rectangle
DefaultLabel {
font.family: LogsPaneStyle . generic . layout . logLabelFont
font.pointSize: Style . absoluteSize ( - 3 )
color: "#5391d8"
color: LogsPaneStyle . generic . layout . logLabelColor
anchors.centerIn: parent
text: qsTr ( "State" )
}
}
background:
Rectangle {
color: stateButton . checked ? LogsPaneStyle.generic.layout.buttonSelected : "transparent"
}
}
}
DefaultTextField
Rectangle {
anchors.verticalCenter: parent . verticalCenter
width: 1 ;
height: parent . height
color: LogsPaneStyle . generic . layout . buttonSeparatorColor1
}
Rectangle {
anchors.verticalCenter: parent . verticalCenter
width: 2 ;
height: parent . height
color: LogsPaneStyle . generic . layout . buttonSeparatorColor2
}
}
Row
{
height: parent . height
anchors.right: parent . right
anchors.rightMargin: 10
spacing: 10
Rectangle
{
id: searchBox
height: LogsPaneStyle . generic . layout . headerButtonHeight
anchors.verticalCenter: parent . verticalCenter
width: LogsPaneStyle . generic . layout . headerInputWidth
font.family: LogsPaneStyle . generic . layout . logLabelFont
font.pointSize: Style . absoluteSize ( - 3 )
font.italic: true
onTextChanged: {
proxyModel . search ( text ) ;
color: "transparent"
width: 20
Button
{
id: clearButton
action: clearAction
anchors.fill: parent
anchors.verticalCenter: parent . verticalCenter
height: 25
style:
ButtonStyle {
background:
Rectangle {
height: LogsPaneStyle . generic . layout . headerButtonHeight
implicitHeight: LogsPaneStyle . generic . layout . headerButtonHeight
color: "transparent"
}
}
}
ListModel {
id: logsModel
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
}
TableView {
id: logsTable
clip: true
Layout.fillWidth: true
Layout.preferredHeight: parent . height - rowAction . height
headerVisible : false
onDoubleClicked:
{
var log = logsModel . get ( logsTable . currentRow ) ;
if ( log )
clipboard . text = ( log . type + "\t" + log . level + "\t" + log . date + "\t" + log . content ) ;
Action {
id: clearAction
enabled: logsModel . count > 0
tooltip: qsTr ( "Clear" )
onTriggered: {
logsModel . clear ( ) ;
}
model: SortFilterProxyModel {
id: proxyModel
source: logsModel
property var roles: [ "-" , "javascript" , "run" , "state" ]
Component.onCompleted: {
filterType = regEx ( proxyModel . roles ) ;
}
function search ( _value )
{
filterContent = _value ;
}
function toogleFilter ( _value )
{
var count = roles . length ;
for ( var i in roles )
Rectangle
{
if ( roles [ i ] === _value )
height: LogsPaneStyle . generic . layout . headerButtonHeight
anchors.verticalCenter: parent . verticalCenter
color: "transparent"
width: 20
Button
{
roles . splice ( i , 1 ) ;
break ;
id: copyButton
action: copyAction
anchors.fill: parent
anchors.verticalCenter: parent . verticalCenter
height: 25
style:
ButtonStyle {
background:
Rectangle {
height: LogsPaneStyle . generic . layout . headerButtonHeight
implicitHeight: LogsPaneStyle . generic . layout . headerButtonHeight
color: "transparent"
}
}
}
if ( count === roles . length )
roles . push ( _value ) ;
filterType = regEx ( proxyModel . roles ) ;
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
}
function regEx ( _value )
Action {
id: copyAction
enabled: logsModel . count > 0
tooltip: qsTr ( "Copy to Clipboard" )
onTriggered: {
var content = "" ;
for ( var k = 0 ; k < logsModel . count ; k ++ )
{
return "(?:" + roles . join ( '|' ) + ")" ;
var log = logsModel . get ( k ) ;
content += log . type + "\t" + log . level + "\t" + log . date + "\t" + log . content + "\n" ;
}
filterType: "(?:javascript|run|state)"
filterContent: ""
filterSyntax: SortFilterProxyModel . RegExp
filterCaseSensitivity: Qt . CaseInsensitive
clipboard . text = content ;
}
TableViewColumn
{
role: "date"
title: qsTr ( "date" )
width: LogsPaneStyle . generic . layout . dateWidth
delegate: itemDelegate
}
TableViewColumn
{
role: "type"
title: qsTr ( "type" )
width: LogsPaneStyle . generic . layout . typeWidth
delegate: itemDelegate
}
TableViewColumn
DefaultTextField
{
role: "content"
title: qsTr ( "content" )
width: LogsPaneStyle . generic . layout . contentWidth
delegate: itemDelegate
id: searchBox
anchors.verticalCenter: parent . verticalCenter
width: LogsPaneStyle . generic . layout . headerInputWidth - 50
font.family: LogsPaneStyle . generic . layout . logLabelFont
font.pointSize: Style . 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
}
}
}
rowDelegate: Item {
Rectangle
{
height: LogsPaneStyle . generic . layout . headerButtonHeight
anchors.verticalCenter: parent . verticalCenter
color: "transparent"
width: 20
Button
{
id: hideButton
action: hideAction
anchors.fill: parent
anchors.verticalCenter: parent . verticalCenter
height: 25
style:
ButtonStyle {
background:
Rectangle {
width: logsTable . width - 4
height: 17
color: styleData . alternate ? "transparent" : LogsPaneStyle . generic . layout . logAlternateColor
height: LogsPaneStyle . generic . layout . headerButtonHeight
implicitHeight: LogsPaneStyle . generic . layout . headerButtonHeight
color: "transparent"
}
}
}
Component {
id: itemDelegate
DefaultLabel {
text: styleData . value ;
font.family: LogsPaneStyle . generic . layout . logLabelFont
font.pointSize: Style . 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" ;
Image {
id: hideImage
source: "qrc:/qml/img/exit.png"
anchors.centerIn: parent
fillMode: Image . PreserveAspectFit
width: 20
height: 25
}
Action {
id: hideAction
tooltip: qsTr ( "Exit" )
onTriggered: {
logsPane . parent . toggle ( ) ;
}
}
}
}
}
}