More autocompletion!

This commit is contained in:
Adsooi 2022-10-20 00:37:02 +02:00
parent 5da8dcefe5
commit 77ae54fa18
Signed by: Ad5001
GPG key ID: 7251B1AF90B960F9
7 changed files with 151 additions and 55 deletions

View file

@ -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
}
}
}

View file

@ -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
}
}
} }
} }
} }

View file

@ -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

View file

@ -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,

View file

@ -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));

View file

@ -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 []

View file

@ -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