More autocompletion!
This commit is contained in:
parent
5da8dcefe5
commit
77ae54fa18
7 changed files with 151 additions and 55 deletions
|
@ -0,0 +1,101 @@
|
||||||
|
/**
|
||||||
|
* LogarithmPlotter - 2D plotter software to make BODE plots, sequences and distribution functions.
|
||||||
|
* Copyright (C) 2022 Ad5001
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import QtQuick 2.12
|
||||||
|
import QtQuick.Controls 2.12
|
||||||
|
|
||||||
|
/*!
|
||||||
|
\qmltype AutocompletionCategory
|
||||||
|
\inqmlmodule eu.ad5001.LogarithmPlotter.Setting
|
||||||
|
\brief ListView representing a category of autocompletion.
|
||||||
|
|
||||||
|
\sa ExpressionEditor
|
||||||
|
*/
|
||||||
|
ListView {
|
||||||
|
id: listFiltered
|
||||||
|
/*!
|
||||||
|
\qmlproperty int AutocompletionCategory::itemStartIndex
|
||||||
|
Start index of the first element in this list compared to the global autocompletion index.
|
||||||
|
*/
|
||||||
|
property int itemStartIndex: 0
|
||||||
|
|
||||||
|
/*!
|
||||||
|
\qmlproperty string AutocompletionCategory::category
|
||||||
|
Name of the category.
|
||||||
|
*/
|
||||||
|
property string category: ""
|
||||||
|
|
||||||
|
/*!
|
||||||
|
\qmlproperty var AutocompletionCategory::categoryItems
|
||||||
|
List of items in this category. To be filtered by the autocomplete to filters.
|
||||||
|
*/
|
||||||
|
property var categoryItems: []
|
||||||
|
|
||||||
|
/*!
|
||||||
|
\qmlproperty var AutocompletionCategory::autocompleteGenerator
|
||||||
|
Javascript function taking the name of the item to create an autocompletion item (dictionary with
|
||||||
|
a 'text', 'autocomplete', and 'cursorFinalOffset' keys.
|
||||||
|
*/
|
||||||
|
property var autocompleteGenerator: (item) => {return {'text': item, 'autocomplete': item, 'cursorFinalOffset': 0}}
|
||||||
|
|
||||||
|
/*!
|
||||||
|
\qmlproperty string AutocompletionCategory::baseText
|
||||||
|
Text to autocomplete.
|
||||||
|
*/
|
||||||
|
property string baseText: ""
|
||||||
|
width: parent.width
|
||||||
|
visible: model.length > 0
|
||||||
|
implicitHeight: contentItem.childrenRect.height + headerItem.height
|
||||||
|
model: parent.visible ? categoryItems.filter((item) => item.includes(baseText)).map(autocompleteGenerator) : []
|
||||||
|
|
||||||
|
header: Column {
|
||||||
|
width: listFiltered.width
|
||||||
|
spacing: 2
|
||||||
|
topPadding: 5
|
||||||
|
bottomPadding: 5
|
||||||
|
|
||||||
|
Text {
|
||||||
|
leftPadding: 5
|
||||||
|
text: listFiltered.category
|
||||||
|
}
|
||||||
|
|
||||||
|
Rectangle {
|
||||||
|
height: 1
|
||||||
|
color: 'black'
|
||||||
|
width: parent.width
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
delegate: Rectangle {
|
||||||
|
property bool selected: index + listFiltered.itemStartIndex == acPopupContent.itemSelected
|
||||||
|
|
||||||
|
width: autocompleteText.width
|
||||||
|
height: autocompleteText.height
|
||||||
|
color: selected ? sysPalette.highlight : 'transparent'
|
||||||
|
|
||||||
|
Text {
|
||||||
|
id: autocompleteText
|
||||||
|
topPadding: 2
|
||||||
|
bottomPadding: 2
|
||||||
|
leftPadding: 15
|
||||||
|
text: listFiltered.model[index].text
|
||||||
|
width: listFiltered.width
|
||||||
|
color: parent.selected ? sysPalette.highlightedText : sysPalette.windowText
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -22,6 +22,7 @@ import QtQuick.Dialogs 1.3 as D
|
||||||
import eu.ad5001.LogarithmPlotter.Popup 1.0 as P
|
import eu.ad5001.LogarithmPlotter.Popup 1.0 as P
|
||||||
import "../js/mathlib.js" as MathLib
|
import "../js/mathlib.js" as MathLib
|
||||||
import "../js/utils.js" as Utils
|
import "../js/utils.js" as Utils
|
||||||
|
import "../js/objects.js" as Objects
|
||||||
import "../js/parsing/parsing.js" as Parsing
|
import "../js/parsing/parsing.js" as Parsing
|
||||||
|
|
||||||
|
|
||||||
|
@ -30,7 +31,7 @@ import "../js/parsing/parsing.js" as Parsing
|
||||||
\inqmlmodule eu.ad5001.LogarithmPlotter.Setting
|
\inqmlmodule eu.ad5001.LogarithmPlotter.Setting
|
||||||
\brief Setting to edit strings and numbers.
|
\brief Setting to edit strings and numbers.
|
||||||
|
|
||||||
\sa EditorDialog, Settings, Icon
|
\sa EditorDialog, AutocompletionCategory
|
||||||
*/
|
*/
|
||||||
Item {
|
Item {
|
||||||
id: control
|
id: control
|
||||||
|
@ -177,12 +178,18 @@ Item {
|
||||||
}
|
}
|
||||||
|
|
||||||
Keys.onUpPressed: function(event) {
|
Keys.onUpPressed: function(event) {
|
||||||
acPopupContent.itemSelected = Math.max(0, acPopupContent.itemSelected-1)
|
if(acPopupContent.itemSelected == 0)
|
||||||
|
acPopupContent.itemSelected = acPopupContent.itemCount-1
|
||||||
|
else
|
||||||
|
acPopupContent.itemSelected = acPopupContent.itemSelected-1
|
||||||
event.accepted = true
|
event.accepted = true
|
||||||
}
|
}
|
||||||
|
|
||||||
Keys.onDownPressed: function(event) {
|
Keys.onDownPressed: function(event) {
|
||||||
acPopupContent.itemSelected = Math.max(0,Math.min(acPopupContent.itemCount-1, acPopupContent.itemSelected+1))
|
if(acPopupContent.itemSelected == Math.min(acPopupContent.itemCount-1))
|
||||||
|
acPopupContent.itemSelected = 0
|
||||||
|
else
|
||||||
|
acPopupContent.itemSelected = acPopupContent.itemSelected+1
|
||||||
event.accepted = true
|
event.accepted = true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -243,8 +250,8 @@ Item {
|
||||||
visible: currentToken != null && identifierTokenTypes.includes(currentToken.type)
|
visible: currentToken != null && identifierTokenTypes.includes(currentToken.type)
|
||||||
|
|
||||||
// Focus handling.
|
// Focus handling.
|
||||||
readonly property var lists: [functionsList]
|
readonly property var lists: [constantsList, functionsList, executableObjectList]
|
||||||
readonly property int itemCount: functionsList.model.length
|
readonly property int itemCount: constantsList.model.length + functionsList.model.length + executableObjectList.model.length
|
||||||
property int itemSelected: 0
|
property int itemSelected: 0
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
|
@ -275,52 +282,34 @@ Item {
|
||||||
editor.cursorPosition = startPos+autotext.autocomplete.length+autotext.cursorFinalOffset
|
editor.cursorPosition = startPos+autotext.autocomplete.length+autotext.cursorFinalOffset
|
||||||
}
|
}
|
||||||
|
|
||||||
ListView {
|
AutocompletionCategory {
|
||||||
|
id: constantsList
|
||||||
|
|
||||||
|
itemStartIndex: 0
|
||||||
|
category: qsTr("Constants")
|
||||||
|
categoryItems: Parsing.CONSTANTS_LIST
|
||||||
|
autocompleteGenerator: (item) => {return {'text': item, 'autocomplete': item + " ", 'cursorFinalOffset': 0}}
|
||||||
|
baseText: parent.currentToken.value
|
||||||
|
}
|
||||||
|
|
||||||
|
AutocompletionCategory {
|
||||||
id: functionsList
|
id: functionsList
|
||||||
//anchors.fill: parent
|
|
||||||
property int itemStartIndex: 0
|
|
||||||
width: parent.width
|
|
||||||
visible: model.length > 0
|
|
||||||
implicitHeight: contentItem.childrenRect.height + headerItem.height
|
|
||||||
model: parent.visible ?
|
|
||||||
Parsing.FUNCTIONS_LIST.filter((name) => name.includes(acPopupContent.currentToken.value))
|
|
||||||
.map((name) => {return {'text': name, 'autocomplete': name+'()', 'cursorFinalOffset': -1}}) : []
|
|
||||||
|
|
||||||
header: Column {
|
itemStartIndex: constantsList.model.length
|
||||||
width: functionsList.width
|
category: qsTr("Functions")
|
||||||
spacing: 2
|
categoryItems: Parsing.FUNCTIONS_LIST
|
||||||
topPadding: 5
|
autocompleteGenerator: (item) => {return {'text': item, 'autocomplete': item+'()', 'cursorFinalOffset': -1}}
|
||||||
bottomPadding: 5
|
baseText: parent.currentToken.value
|
||||||
|
|
||||||
Text {
|
|
||||||
leftPadding: 5
|
|
||||||
text: qsTr("Functions")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Rectangle {
|
AutocompletionCategory {
|
||||||
height: 1
|
id: executableObjectList
|
||||||
color: 'black'
|
|
||||||
width: parent.width
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
delegate: Rectangle {
|
itemStartIndex: constantsList.model.length + functionsList.model.length
|
||||||
property bool selected: index + functionsList.itemStartIndex == acPopupContent.itemSelected
|
category: qsTr("Executable Objects")
|
||||||
|
categoryItems: Objects.getObjectsName("ExecutableObject")
|
||||||
width: funcText.width
|
autocompleteGenerator: (item) => {return {'text': item, 'autocomplete': item+'()', 'cursorFinalOffset': -1}}
|
||||||
height: funcText.height
|
baseText: parent.currentToken.value
|
||||||
color: selected ? sysPalette.highlight : 'transparent'
|
|
||||||
|
|
||||||
Text {
|
|
||||||
id: funcText
|
|
||||||
topPadding: 2
|
|
||||||
bottomPadding: 2
|
|
||||||
leftPadding: 15
|
|
||||||
text: functionsList.model[index].text
|
|
||||||
width: functionsList.width
|
|
||||||
color: parent.selected ? sysPalette.highlightedText : sysPalette.windowText
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,3 +5,4 @@ Icon 1.0 Icon.qml
|
||||||
ListSetting 1.0 ListSetting.qml
|
ListSetting 1.0 ListSetting.qml
|
||||||
TextSetting 1.0 TextSetting.qml
|
TextSetting 1.0 TextSetting.qml
|
||||||
ExpressionEditor 1.0 ExpressionEditor.qml
|
ExpressionEditor 1.0 ExpressionEditor.qml
|
||||||
|
AutocompletionCategory 1.0 AutocompletionCategory.qml
|
||||||
|
|
|
@ -28,6 +28,7 @@ var evalVariables = { // Variables not provided by expr-eval.js, needs to be pro
|
||||||
"pi": Math.PI,
|
"pi": Math.PI,
|
||||||
"π": Math.PI,
|
"π": Math.PI,
|
||||||
"inf": Infinity,
|
"inf": Infinity,
|
||||||
|
"infinity": Infinity,
|
||||||
"Infinity": Infinity,
|
"Infinity": Infinity,
|
||||||
"∞": Infinity,
|
"∞": Infinity,
|
||||||
"e": Math.E,
|
"e": Math.E,
|
||||||
|
|
|
@ -148,9 +148,11 @@ function expression(tokens) {
|
||||||
|
|
||||||
switch(type) {
|
switch(type) {
|
||||||
case ExprEval.INUMBER:
|
case ExprEval.INUMBER:
|
||||||
if (typeof item.value === 'number' && item.value < 0) {
|
if(item.value == Infinity) {
|
||||||
|
nstack.push("\\infty")
|
||||||
|
} else if(typeof item.value === 'number' && item.value < 0) {
|
||||||
nstack.push(par(item.value));
|
nstack.push(par(item.value));
|
||||||
} else if (Array.isArray(item.value)) {
|
} else if(Array.isArray(item.value)) {
|
||||||
nstack.push('[' + item.value.map(ExprEval.escapeValue).join(', ') + ']');
|
nstack.push('[' + item.value.map(ExprEval.escapeValue).join(', ') + ']');
|
||||||
} else {
|
} else {
|
||||||
nstack.push(ExprEval.escapeValue(item.value));
|
nstack.push(ExprEval.escapeValue(item.value));
|
||||||
|
|
|
@ -58,11 +58,12 @@ function getObjectsName(objType) {
|
||||||
* @return {array} List of names of the objects.
|
* @return {array} List of names of the objects.
|
||||||
*/
|
*/
|
||||||
if(objType == "ExecutableObject") {
|
if(objType == "ExecutableObject") {
|
||||||
var types = getExecutableTypes()
|
// NOTE: QMLJS does not support flatMap.
|
||||||
var elementNames = ['']
|
// return getExecutableTypes().flatMap(elemType => currentObjects[elemType].map(obj => obj.name))
|
||||||
types.forEach(function(elemType){
|
let types = getExecutableTypes()
|
||||||
|
let elementNames = ['']
|
||||||
|
for(let elemType of types)
|
||||||
elementNames = elementNames.concat(currentObjects[elemType].map(obj => obj.name))
|
elementNames = elementNames.concat(currentObjects[elemType].map(obj => obj.name))
|
||||||
})
|
|
||||||
return elementNames
|
return elementNames
|
||||||
}
|
}
|
||||||
if(currentObjects[objType] == undefined) return []
|
if(currentObjects[objType] == undefined) return []
|
||||||
|
|
|
@ -28,3 +28,4 @@ var Token = TK.Token
|
||||||
var Tokenizer = TK.ExpressionTokenizer
|
var Tokenizer = TK.ExpressionTokenizer
|
||||||
|
|
||||||
var FUNCTIONS_LIST = Reference.FUNCTIONS_LIST
|
var FUNCTIONS_LIST = Reference.FUNCTIONS_LIST
|
||||||
|
var CONSTANTS_LIST = Reference.CONSTANTS_LIST
|
||||||
|
|
Loading…
Reference in a new issue