2import QGroundControl.Controls
3import QGroundControl.Logging
10 readonly property real _indicatorWidth: ScreenTools.defaultFontPixelWidth * 0.4
11 readonly property real _margin: ScreenTools.defaultFontPixelWidth
12 readonly property real _rowHeight: ScreenTools.defaultFontPixelHeight * 1.6
14 function _levelColor(lvl) {
16 case LogEntry.Debug: return qgcPal.colorGrey
17 case LogEntry.Info: return qgcPal.text
18 case LogEntry.Warning: return qgcPal.colorOrange
19 case LogEntry.Critical: return qgcPal.colorRed
20 case LogEntry.Fatal: return qgcPal.colorRed
21 default: return qgcPal.text
28 colorGroupEnabled: enabled
35 // ── Column header ────────────────────────────────────────
36 HorizontalHeaderView {
39 Layout.fillWidth: true
44 required property var display
46 color: qgcPal.windowShade
47 implicitHeight: headerLabel.contentHeight + ScreenTools.defaultFontPixelHeight * 0.4
48 implicitWidth: headerLabel.contentWidth + ScreenTools.defaultFontPixelWidth * 2
53 anchors.left: parent.left
54 anchors.leftMargin: _margin * 0.5
55 anchors.verticalCenter: parent.verticalCenter
57 font.family: ScreenTools.fixedFontFamily
58 font.pointSize: ScreenTools.defaultFontPointSize
59 text: parent.display ?? ""
63 anchors.bottom: parent.bottom
64 color: qgcPal.groupBorder
70 anchors.right: parent.right
71 color: qgcPal.groupBorder
78 // ── Log table ────────────────────────────────────────────
82 property bool _loadCompleted: false
83 property real _cachedWidth: 0
85 Layout.fillHeight: true
86 Layout.fillWidth: true
87 boundsBehavior: Flickable.StopAtBounds
90 columnWidthProvider: function (col) {
91 const explicit_ = tableView.explicitColumnWidth(col)
95 const cw = ScreenTools.defaultFontPixelWidth
96 const w = _cachedWidth
97 const compact = w < cw * 60
99 case LogEntry.TimestampColumn:
100 return compact ? cw * 9 : cw * 12;
101 case LogEntry.LevelColumn:
102 return compact ? cw * 5 : cw * 7;
103 case LogEntry.CategoryColumn:
104 return compact ? cw * 14 : cw * 22;
105 case LogEntry.SourceColumn:
106 return compact ? cw * 16 : cw * 22;
107 case LogEntry.MessageColumn: {
108 const fixedWidth = compact ? cw * 44 : cw * 63
109 return Math.max(cw * 30, w - fixedWidth)
114 model: LogManager.model
115 resizableColumns: true
120 _cachedWidth = width;
121 _layoutTimer.restart();
129 onTriggered: tableView.forceLayout()
132 delegate: Rectangle {
133 required property int column
134 required property var display
135 required property int level
136 required property int row
138 color: row % 2 === 0 ? qgcPal.window : qgcPal.windowShade
139 implicitHeight: _rowHeight
142 anchors.bottom: parent.bottom
143 anchors.left: parent.left
144 anchors.top: parent.top
145 color: root._levelColor(parent.level)
146 visible: parent.column === 0 && parent.level >= LogEntry.Warning
147 width: _indicatorWidth
151 anchors.left: parent.left
152 anchors.leftMargin: parent.column === 0 ? _indicatorWidth + _margin * 0.3 : _margin * 0.3
153 anchors.right: parent.right
154 anchors.rightMargin: _margin * 0.3
155 anchors.verticalCenter: parent.verticalCenter
156 color: root._levelColor(parent.level)
157 elide: Text.ElideRight
158 font.family: ScreenTools.fixedFontFamily
159 font.pointSize: ScreenTools.defaultFontPointSize
160 text: parent.display ?? ""
165 color: qgcPal.colorGrey
166 text: qsTr("No log entries")
167 visible: tableView.rows === 0
168 x: tableView.contentX + (tableView.width - width) / 2
169 y: tableView.contentY + (tableView.height - height) / 2
172 Component.onCompleted: {
173 _cachedWidth = width;
174 _loadCompleted = true;
176 positionViewAtRow(rows - 1, TableView.AlignBottom);
185 if (tableView.rows > 0)
186 tableView.positionViewAtRow(tableView.rows - 1, TableView.AlignBottom);
191 function onRowsInserted() {
192 if (tableView._loadCompleted && followTail.checked)
193 scrollTimer.restart();
196 target: LogManager.model
200 // ── Separator ─────────────────────────────────────────────
202 Layout.fillWidth: true
203 Layout.preferredHeight: 1
204 color: qgcPal.groupBorder
207 // ── Filter bar ────────────────────────────────────────────
209 Layout.fillWidth: true
210 Layout.preferredHeight: filterRow.implicitHeight + _margin
211 color: qgcPal.windowShade
217 anchors.leftMargin: _margin
218 anchors.rightMargin: _margin
219 spacing: _margin * 0.75
224 model: [qsTr("All Levels"), qsTr("Debug"), qsTr("Info"), qsTr("Warning"), qsTr("Critical"), qsTr("Fatal")]
227 Component.onCompleted: currentIndex = LogManager.model.filterLevel + 1
228 onActivated: index => {
229 LogManager.model.filterLevel = index - 1;
236 model: [qsTr("All Categories")].concat(LogManager.model.categoriesList)
239 onActivated: index => {
240 LogManager.model.filterCategory = index === 0 ? "" : model[index];
247 Layout.fillWidth: true
248 Layout.minimumWidth: _margin * 10
249 placeholderText: qsTr("Search…")
251 onTextChanged: LogManager.model.setFilterTextDeferred(text)
255 ToolTip.text: qsTr("Regex search")
256 ToolTip.visible: hovered
258 checked: LogManager.model.filterRegex
261 onClicked: LogManager.model.filterRegex = checked
265 color: qgcPal.colorRed
267 text: qsTr("\u26A0 Disk Error")
268 visible: LogManager.hasError
273 onClicked: LogManager.clearError()
285 if (checked && tableView._loadCompleted && tableView.rows > 0)
286 tableView.positionViewAtRow(tableView.rows - 1, TableView.AlignBottom);
291 text: qsTr("Categories")
293 onClicked: filtersDialogFactory.open()
297 text: qsTr("Settings")
299 onClicked: settingsDialogFactory.open()
305 QGCPopupDialogFactory {
306 id: filtersDialogFactory
308 dialogComponent: Component {
309 LoggingCategoriesDialog {
314 QGCPopupDialogFactory {
315 id: settingsDialogFactory
317 dialogComponent: Component {
318 LoggingSettingsDialog {