From efb4ad6a1f968c7b57724477e3568dc42290de8e Mon Sep 17 00:00:00 2001
From: csmig <33138761+csmig@users.noreply.github.com>
Date: Tue, 10 Sep 2024 12:43:07 -0400
Subject: [PATCH] feat: enhanced string column filter (#1367)
---
client/src/css/stigman.css | 10 ++
client/src/img/match-case.svg | 4 +
client/src/img/match-word.svg | 3 +
client/src/js/SM/ColumnFilters.js | 213 +++++++++++++++++++++++-------
4 files changed, 183 insertions(+), 47 deletions(-)
create mode 100644 client/src/img/match-case.svg
create mode 100644 client/src/img/match-word.svg
diff --git a/client/src/css/stigman.css b/client/src/css/stigman.css
index 9a1682d1b..5851884b8 100644
--- a/client/src/css/stigman.css
+++ b/client/src/css/stigman.css
@@ -2263,4 +2263,14 @@ td.x-grid3-hd-over .x-grid3-hd-inner {
}
.x-item-disabled {
filter: saturate(0)
+}
+
+.sm-match-case-icon {
+ background-image: url(../img/match-case.svg);
+ background-size: contain;
+}
+
+.sm-match-word-icon {
+ background-image: url(../img/match-word.svg);
+ background-size: contain;
}
\ No newline at end of file
diff --git a/client/src/img/match-case.svg b/client/src/img/match-case.svg
new file mode 100644
index 000000000..ba5a84bcc
--- /dev/null
+++ b/client/src/img/match-case.svg
@@ -0,0 +1,4 @@
+
+
diff --git a/client/src/img/match-word.svg b/client/src/img/match-word.svg
new file mode 100644
index 000000000..b07a20eb0
--- /dev/null
+++ b/client/src/img/match-word.svg
@@ -0,0 +1,3 @@
+
\ No newline at end of file
diff --git a/client/src/js/SM/ColumnFilters.js b/client/src/js/SM/ColumnFilters.js
index 27fca43ca..48bbc57e1 100644
--- a/client/src/js/SM/ColumnFilters.js
+++ b/client/src/js/SM/ColumnFilters.js
@@ -78,10 +78,9 @@ SM.ColumnFilters.extend = function extend (extended = Ext.grid.GridView) {
// // iterate the menu items and set the condition(s) for each dataIndex
for (const stringItem of stringItems) {
- const dataIndex = stringItem.filter.dataIndex
const value = stringItem.getValue()
- if (value) {
- conditions[dataIndex] = value
+ if (value.value) {
+ conditions[stringItem.filter.dataIndex] = value
}
}
for (const selectItem of selectItems) {
@@ -99,24 +98,32 @@ SM.ColumnFilters.extend = function extend (extended = Ext.grid.GridView) {
for (const dataIndex of Object.keys(conditions)) {
filterFns.push({
fn: function (record) {
- const value = record.data[dataIndex]
- if (Array.isArray(value)) {
+ const cellValue = record.data[dataIndex]
+ const condition = conditions[dataIndex]
+ if (Array.isArray(cellValue)) {
// the record data is an Array of values
- if (Array.isArray(conditions[dataIndex])) {
- if (conditions[dataIndex].includes('') && value.length === 0) return true
- return value.some( v => conditions[dataIndex].includes(v))
+ if (Array.isArray(condition)) {
+ if (condition.includes('') && cellValue.length === 0) return true
+ return cellValue.some( v => condition.includes(v))
}
}
// the record data is a scalar value (we're missing object handling?)
- if (Array.isArray(conditions[dataIndex])) {
- return conditions[dataIndex].includes(value)
+ if (Array.isArray(condition)) {
+ return condition.includes(cellValue)
}
else {
- // match case-insensitive condition anywhere in value
- const a = value.toLowerCase()
- const b = conditions[dataIndex].toLowerCase()
- return a.indexOf(b) > -1
+ // string matches
+ const a = condition.matchCase ? cellValue : cellValue.toLowerCase()
+ const b = condition.matchCase ? condition.value : condition.value.toLowerCase()
+ let found
+ if (condition.matchWord) {
+ found = a.search(new RegExp(`\\b${b}\\b`))
+ }
+ else {
+ found = a.indexOf(b)
+ }
+ return condition.condition ? found > -1 : found === -1
}
}
})
@@ -126,14 +133,16 @@ SM.ColumnFilters.extend = function extend (extended = Ext.grid.GridView) {
onFilterChange: function (item, value) {
switch (item.filter.type) {
case 'string':
- item.column.filtered = !!(item.getValue())
+ item.column.filtered = !!(item.getValue()?.value)
break
case 'values':
- const hmenuItems = this.hmenu.items.items
- const hmenuPeers = hmenuItems.filter( i => i.filter?.type === 'values' && i.filter?.dataIndex === item.filter.dataIndex)
- const hmenuPeersChecked = hmenuPeers.map( i => i.checked)
- item.column.filtered = hmenuPeersChecked.includes(false)
- break
+ {
+ const hmenuItems = this.hmenu.items.items
+ const hmenuPeers = hmenuItems.filter( i => i.filter?.type === 'values' && i.filter?.dataIndex === item.filter.dataIndex)
+ const hmenuPeersChecked = hmenuPeers.map( i => i.checked)
+ item.column.filtered = hmenuPeersChecked.includes(false)
+ break
+ }
case 'selectall':
item.column.filtered = !(!!value)
break
@@ -273,33 +282,27 @@ SM.ColumnFilters.extend = function extend (extended = Ext.grid.GridView) {
for (const col of this.cm.config) {
switch (col.filter?.type) {
- case 'string':
+ case 'string': {
if (col.renderer) {
col.configRenderer = col.renderer
col.renderer = SM.ColumnFilters.Renderers.highlighterShim
}
- const stringItem = hmenu.add(new SM.ColumnFilters.SearchTextField({
- emptyText: "Contains...",
- height: 24,
+ const stringItem = hmenu.add(new SM.ColumnFilters.StringPanel({
+ hideOnClick: false,
column: col,
filter: { dataIndex: col.dataIndex, type: 'string'},
- enableKeyEvents: true,
- hideParent: true,
listeners: {
- input: function (item, e) {
- _this.onFilterChange(item, item.value)
+ filterchanged: function (panel) {
+ _this.onFilterChange(panel, panel.getValue())
},
- keyup: function (item, e) {
- const k = e.getKey()
- if (k == e.RETURN) {
- e.stopEvent();
- hmenu.hide(true)
- }
+ enterkey: function () {
+ hmenu.hide(true)
}
}
}))
hmenu.filterItems.stringItems.push(stringItem)
break
+ }
case 'values':
dynamicColumns.push(col)
break
@@ -314,28 +317,143 @@ SM.ColumnFilters.extend = function extend (extended = Ext.grid.GridView) {
SM.ColumnFilters.GridView = SM.ColumnFilters.extend(Ext.grid.GridView)
SM.ColumnFilters.GridViewBuffered = SM.ColumnFilters.extend(Ext.ux.grid.BufferView)
-SM.ColumnFilters.SearchTextField = Ext.extend(Ext.form.TextField, {
+SM.ColumnFilters.StringMatchTextField = Ext.extend(Ext.form.TextField, {
initComponent: function () {
const config = {
- autoCreate: {tag: 'input', type: 'search', size: '20', autocomplete: 'off'}
+ autoCreate: {tag: 'input', type: 'search', size: '20', autocomplete: 'off'},
+ enableKeyEvents: true
}
Ext.apply(this, Ext.apply(this.initialConfig, config))
- SM.ColumnFilters.SearchTextField.superclass.initComponent.call(this)
+ this.superclass().initComponent.call(this)
this.addEvents( 'input' )
},
initEvents: function () {
- SM.ColumnFilters.SearchTextField.superclass.initEvents.call(this)
+ this.superclass().initEvents.call(this)
this.mon(this.el, {
scope: this,
input: this.onInput
})
},
onInput: function (e) {
- this.column.filter.value = this.getValue()
this.fireEvent('input', this, e);
}
})
+SM.ColumnFilters.StringMatchConditionComboBox = Ext.extend(Ext.form.ComboBox, {
+ initComponent: function () {
+ const store = new Ext.data.ArrayStore({
+ fields: ['display', 'value'],
+ data: [['Includes', true], ['Excludes', false]]
+ })
+ const config = {
+ listClass: 'x-menu',
+ store,
+ triggerAction: 'all',
+ mode: 'local',
+ editable: false,
+ valueField: 'value',
+ displayField: 'display'
+ }
+ Ext.apply(this, Ext.apply(this.initialConfig, config))
+ this.superclass().initComponent.call(this)
+ this.setValue(true)
+ }
+})
+
+SM.ColumnFilters.StringMatchCaseButton = Ext.extend(Ext.Button, {
+ initComponent: function () {
+ const config = {
+ enableToggle: true,
+ border: false,
+ iconCls: 'sm-match-case-icon',
+ tooltip: 'Match case'
+ }
+ Ext.apply(this, Ext.apply(this.initialConfig, config))
+ this.superclass().initComponent.call(this)
+ }
+})
+
+SM.ColumnFilters.StringMatchWordButton = Ext.extend(Ext.Button, {
+ initComponent: function () {
+ const config = {
+ enableToggle: true,
+ border: false,
+ iconCls: 'sm-match-word-icon',
+ tooltip: 'Match word'
+ }
+ Ext.apply(this, Ext.apply(this.initialConfig, config))
+ this.superclass().initComponent.call(this)
+ }
+})
+
+SM.ColumnFilters.StringPanel = Ext.extend(Ext.Panel, {
+ initComponent: function () {
+ const _this = this
+
+ const onFilterChange = function () {
+ _this.column.filter.value = getValue()
+ _this.fireEvent('filterchanged', _this)
+ }
+
+ const conditionComboBox = new SM.ColumnFilters.StringMatchConditionComboBox({
+ flex: 1,
+ listeners: {
+ select: onFilterChange
+ }
+ })
+ const matchCaseButton = new SM.ColumnFilters.StringMatchCaseButton({
+ width: 24,
+ listeners: {
+ toggle: onFilterChange
+ }
+ })
+ const matchWordButton = new SM.ColumnFilters.StringMatchWordButton({
+ width: 24,
+ listeners: {
+ toggle: onFilterChange
+ }
+ })
+ const textfield = new SM.ColumnFilters.StringMatchTextField({
+ height: 24,
+ listeners: {
+ input: onFilterChange,
+ keyup: function (item, e) {
+ const k = e.getKey()
+ if (k == e.RETURN) {
+ e.stopEvent()
+ _this.fireEvent('enterkey')
+ }
+ }
+ }
+ })
+
+ function getValue () {
+ return {
+ value: textfield.getValue() ?? '',
+ condition: conditionComboBox.getValue(),
+ matchCase: matchCaseButton.pressed,
+ matchWord: matchWordButton.pressed,
+ }
+ }
+ const config = {
+ getValue,
+ items: [
+ {
+ layout: 'hbox',
+ items: [
+ conditionComboBox,
+ matchCaseButton,
+ matchWordButton
+ ]
+ },
+ textfield
+ ]
+ }
+ Ext.apply(this, Ext.apply(this.initialConfig, config))
+ this.superclass().initComponent.call(this)
+ }
+})
+
SM.ColumnFilters.Scorers = {
severity: {
low: 2,
@@ -347,15 +465,12 @@ SM.ColumnFilters.Scorers = {
SM.ColumnFilters.CompareFns = {
severity: (a, b) => {
return SM.ColumnFilters.Scorers.severity[a] - SM.ColumnFilters.Scorers.severity[b]
- },
- labels: (a, b) => {
-
},
labelIds: (a, b, collectionId) => {
if (a === "") return -1;
if (b === "") return 1;
return SM.Cache.getCollectionLabel(collectionId, a).name.localeCompare(SM.Cache.getCollectionLabel(collectionId, b).name)
- },
+ }
}
SM.ColumnFilters.Renderers = {
@@ -404,11 +519,15 @@ SM.ColumnFilters.Renderers = {
}
},
highlighterShim: function (v, m, r, ri, ci, s) {
- if (this.filter?.type === 'string' && this.filter?.value) {
- const re = new RegExp(SM.he(this.filter.value),'gi')
- v = v.replace(re,'$&')
+ if (this.filter?.type === 'string' && this.filter.value?.value && this.filter.value.condition) {
+ let searchStr = SM.he(this.filter.value.value)
+ const flags = `g${this.filter.value.matchCase ? '' : 'i'}`
+ if (this.filter.value.matchWord) {
+ searchStr = `\\b${searchStr}\\b`
+ }
+ v = v.replace(new RegExp(searchStr, flags),'$&')
}
- return this.configRenderer ? this.configRenderer.call(this, v, m, r, ri, ci, s) : v
+ return this.configRenderer ? this.configRenderer(v, m, r, ri, ci, s) : v
},
labels: function (labelId, collectionId) {
if (!labelId) return '(No value)'