diff --git a/app/gui/view/graph-editor/src/builtin/visualization/java_script/table.js b/app/gui/view/graph-editor/src/builtin/visualization/java_script/table.js
index 49d5fa889792..0abc083096e1 100644
--- a/app/gui/view/graph-editor/src/builtin/visualization/java_script/table.js
+++ b/app/gui/view/graph-editor/src/builtin/visualization/java_script/table.js
@@ -28,7 +28,19 @@ class TableVisualization extends Visualization {
constructor(data) {
super(data)
- this.setPreprocessor('Standard.Visualization.Table.Visualization', 'prepare_visualization')
+ this.setRowLimitAndPage(1000, 0)
+ }
+
+ setRowLimitAndPage(row_limit, page) {
+ if (this.row_limit !== row_limit || this.page !== page) {
+ this.row_limit = row_limit
+ this.page = page
+ this.setPreprocessor(
+ 'Standard.Visualization.Table.Visualization',
+ 'prepare_visualization',
+ this.row_limit.toString()
+ )
+ }
}
onDataReceived(data) {
@@ -79,21 +91,46 @@ class TableVisualization extends Visualization {
return content
}
+ function cellRenderer(params) {
+ if (params.value === null) {
+ return 'Nothing'
+ } else if (params.value === undefined) {
+ return ''
+ } else if (params.value === '') {
+ return 'Empty'
+ }
+ return params.value.toString()
+ }
+
if (!this.tabElem) {
while (this.dom.firstChild) {
this.dom.removeChild(this.dom.lastChild)
}
const style =
- '.ag-theme-alpine { --ag-grid-size: 3px; --ag-list-item-height: 20px; display: inline; }'
+ '.ag-theme-alpine { --ag-grid-size: 3px; --ag-list-item-height: 20px; display: inline; }\n' +
+ '.vis-status-bar { height: 20x; background-color: white; font-size:14px; white-space:nowrap; padding: 0 5px; overflow:hidden; border-radius: 16px }\n' +
+ '.vis-status-bar > button { width: 12px; margin: 0 2px; display: none }\n' +
+ '.vis-tbl-grid { height: calc(100% - 20px); width: 100%; }\n'
const styleElem = document.createElement('style')
styleElem.innerHTML = style
this.dom.appendChild(styleElem)
+ const statusElem = document.createElement('div')
+ statusElem.setAttributeNS(null, 'id', 'vis-tbl-status')
+ statusElem.setAttributeNS(null, 'class', 'vis-status-bar')
+ this.dom.appendChild(statusElem)
+ this.statusElem = statusElem
+
+ const gridElem = document.createElement('div')
+ gridElem.setAttributeNS(null, 'id', 'vis-tbl-grid')
+ gridElem.className = 'vis-tbl-grid'
+ this.dom.appendChild(gridElem)
+
const tabElem = document.createElement('div')
tabElem.setAttributeNS(null, 'id', 'vis-tbl-view')
tabElem.setAttributeNS(null, 'class', 'scrollable ag-theme-alpine')
- this.dom.appendChild(tabElem)
+ gridElem.appendChild(tabElem)
this.tabElem = tabElem
this.agGridOptions = {
@@ -106,6 +143,7 @@ class TableVisualization extends Visualization {
resizable: true,
minWidth: 25,
headerValueGetter: params => params.colDef.field,
+ cellRenderer: cellRenderer,
},
onColumnResized: e => this.lockColumnSize(e),
}
@@ -198,28 +236,106 @@ class TableVisualization extends Visualization {
dataTruncated = parsedData.all_rows_count !== rowData.length
}
- // If the table contains more rows than an upper limit, the engine will send only some of all rows.
+ // Update Status Bar
+ this.updateStatusBarControls(
+ parsedData.all_rows_count === undefined ? 1 : parsedData.all_rows_count,
+ dataTruncated
+ )
+
// If data is truncated, we cannot rely on sorting/filtering so will disable.
- // A pinned row is added to tell the user the row count and that filter/sort is disabled.
- const col_span = '__COL_SPAN__'
- if (dataTruncated) {
- columnDefs[0].colSpan = p => p.data[col_span] || 1
- }
this.agGridOptions.defaultColDef.filter = !dataTruncated
this.agGridOptions.defaultColDef.sortable = !dataTruncated
this.agGridOptions.api.setColumnDefs(columnDefs)
- if (dataTruncated) {
- const field = columnDefs[0].field
- const extraRow = {
- [field]: `Showing ${rowData.length} of ${parsedData.all_rows_count} rows. Sorting and filtering disabled.`,
- [col_span]: columnDefs.length,
- }
- this.agGridOptions.api.setPinnedTopRowData([extraRow])
- }
this.agGridOptions.api.setRowData(rowData)
this.updateTableSize(this.dom.getAttributeNS(null, 'width'))
}
+ makeOption(value, label) {
+ const optionElem = document.createElement('option')
+ optionElem.value = value
+ optionElem.appendChild(document.createTextNode(label))
+ return optionElem
+ }
+
+ makeButton(label, onclick) {
+ const buttonElem = document.createElement('button')
+ buttonElem.name = label
+ buttonElem.appendChild(document.createTextNode(label))
+ buttonElem.addEventListener('click', onclick)
+ return buttonElem
+ }
+
+ // Updates the status bar to reflect the current row limit and page, shown at top of the visualization.
+ // - Creates the row dropdown and page buttons.
+ // - Updated the row counts and filter available options.
+ updateStatusBarControls(all_rows_count, dataTruncated) {
+ const pageLimit = Math.ceil(all_rows_count / this.row_limit)
+ if (this.page > pageLimit) {
+ this.page = pageLimit
+ }
+
+ if (this.statusElem.childElementCount === 0) {
+ this.statusElem.appendChild(
+ this.makeButton('«', () => this.setRowLimitAndPage(this.row_limit, 0))
+ )
+ this.statusElem.appendChild(
+ this.makeButton('‹', () => this.setRowLimitAndPage(this.row_limit, this.page - 1))
+ )
+
+ const selectElem = document.createElement('select')
+ selectElem.name = 'row-limit'
+ selectElem.addEventListener('change', e => {
+ this.setRowLimitAndPage(e.target.value, this.page)
+ })
+ this.statusElem.appendChild(selectElem)
+
+ const rowCountSpanElem = document.createElement('span')
+ this.statusElem.appendChild(rowCountSpanElem)
+
+ this.statusElem.appendChild(
+ this.makeButton('›', () => this.setRowLimitAndPage(this.row_limit, this.page + 1))
+ )
+ this.statusElem.appendChild(
+ this.makeButton('»', () => this.setRowLimitAndPage(this.row_limit, pageLimit - 1))
+ )
+ }
+
+ // Enable/Disable Page buttons
+ this.statusElem.children.namedItem('«').disabled = this.page === 0
+ this.statusElem.children.namedItem('‹').disabled = this.page === 0
+ this.statusElem.children.namedItem('›').disabled = this.page === pageLimit - 1
+ this.statusElem.children.namedItem('»').disabled = this.page === pageLimit - 1
+
+ // Update row limit dropdown and row count
+ const rowCountElem = this.statusElem.getElementsByTagName('span')[0]
+ const rowLimitElem = this.statusElem.children.namedItem('row-limit')
+ if (all_rows_count > 1000) {
+ rowLimitElem.style.display = 'inline-block'
+ const rowCounts = [1000, 2500, 5000, 10000, 25000, 50000, 100000].filter(
+ r => r <= all_rows_count
+ )
+ if (
+ all_rows_count < rowCounts[rowCounts.length - 1] &&
+ rowCounts.indexOf(all_rows_count) === -1
+ ) {
+ rowCounts.push(all_rows_count)
+ }
+ rowLimitElem.innerHTML = ''
+ rowCounts.forEach(r => {
+ const option = this.makeOption(r, r.toString())
+ rowLimitElem.appendChild(option)
+ })
+ rowLimitElem.value = this.row_limit
+
+ rowCountElem.innerHTML = dataTruncated
+ ? ` of ${all_rows_count} rows (Sorting/Filtering disabled).`
+ : ` rows.`
+ } else {
+ rowLimitElem.style.display = 'none'
+ rowCountElem.innerHTML = all_rows_count === 1 ? '1 row.' : `${all_rows_count} rows.`
+ }
+ }
+
updateTableSize(clientWidth) {
// Check the grid has been initialised and return if not.
if (!this.agGridOptions) {
diff --git a/distribution/lib/Standard/Base/0.0.0-dev/src/Data/Text/Extensions.enso b/distribution/lib/Standard/Base/0.0.0-dev/src/Data/Text/Extensions.enso
index 3b9c1a1a5def..9eec4dd13f2e 100644
--- a/distribution/lib/Standard/Base/0.0.0-dev/src/Data/Text/Extensions.enso
+++ b/distribution/lib/Standard/Base/0.0.0-dev/src/Data/Text/Extensions.enso
@@ -40,6 +40,8 @@ from project.Data.Json import Json, Invalid_JSON, JS_Object
from project.Data.Numbers import Decimal, Integer, Number, Number_Parse_Error
from project.Data.Text.Text_Sub_Range import Codepoint_Ranges, Text_Sub_Range
+from project.Metadata import make_single_choice
+
import project.Data.Index_Sub_Range as Index_Sub_Range_Module
polyglot java import com.ibm.icu.lang.UCharacter
@@ -333,7 +335,8 @@ Text.match self pattern=".*" case_sensitivity=Case_Sensitivity.Sensitive =
Split with a vector of strings.
'azbzczdzezfzg'.split ['b', 'zez'] == ['az', 'zczd', 'fzg']
-Text.split : Text | Vector Text -> Case_Sensitivity -> Boolean -> Vector Text | Illegal_Argument
+@delimiter (make_single_choice [',', ';', '|', ['{tab}', "'\t'"], ['{space}', "' '"], ['{newline}', "['\n', '\r\n', '\r']"], ['Custom', ""]])
+Text.split : Text | Vector Text -> Case_Sensitivity -> Boolean -> Vector Text | Illegal_Argument
Text.split self delimiter="," case_sensitivity=Case_Sensitivity.Sensitive use_regex=False =
delimiter_is_empty = case delimiter of
_ : Text -> delimiter.is_empty
diff --git a/distribution/lib/Standard/Base/0.0.0-dev/src/Metadata.enso b/distribution/lib/Standard/Base/0.0.0-dev/src/Metadata.enso
index 6f669108d347..c985a248e65e 100644
--- a/distribution/lib/Standard/Base/0.0.0-dev/src/Metadata.enso
+++ b/distribution/lib/Standard/Base/0.0.0-dev/src/Metadata.enso
@@ -77,3 +77,11 @@ type Widget
## Describes a file chooser.
File_Browse label:(Nothing | Text)=Nothing display:Display=Display.When_Modified action:File_Action=File_Action.Open file_types:(Vector Pair)=[Pair.new "All Files" "*.*"]
+
+## PRIVATE
+make_single_choice : Vector -> Display -> Widget
+make_single_choice values display=Display.Always =
+ make_option value = case value of
+ _ : Vector -> Choice.Option value.first value.second
+ _ : Text -> Choice.Option value value.pretty
+ Widget.Single_Choice (values.map make_option) Nothing display
diff --git a/distribution/lib/Standard/Visualization/0.0.0-dev/src/Table/Visualization.enso b/distribution/lib/Standard/Visualization/0.0.0-dev/src/Table/Visualization.enso
index 16d12e2819fa..8b4be931d5f0 100644
--- a/distribution/lib/Standard/Visualization/0.0.0-dev/src/Table/Visualization.enso
+++ b/distribution/lib/Standard/Visualization/0.0.0-dev/src/Table/Visualization.enso
@@ -39,6 +39,10 @@ prepare_visualization y max_rows=1000 =
dataframe = x.read max_rows
all_rows_count = x.row_count
make_json_for_table dataframe [] all_rows_count
+ _ : Function ->
+ pairs = [['_display_text_', '[Function '+x.to_text+']']]
+ value = JS_Object.from_pairs pairs
+ JS_Object.from_pairs [["json", value]]
_ ->
js_value = x.to_js_object
value = if js_value.is_a JS_Object . not then js_value else
@@ -96,7 +100,7 @@ make_json_for_object_matrix current vector idx=0 = if idx == vector.length then
_ ->
js_object = row.to_js_object
if js_object.is_a JS_Object . not then False else
- if js_object.field_names.sort == ["type" , "constructor"] then False else
+ if js_object.field_names.sort == ["constructor", "type"] then False else
pairs = js_object.field_names.map f-> [f, make_json_for_value (js_object.get f)]
JS_Object.from_pairs pairs
if to_append == False then Nothing else
@@ -184,4 +188,5 @@ make_json_for_value val level=0 = case val of
truncated = val.columns.take 5 . map _.name
prepared = if val.column_count > 5 then truncated + ["… " + (val.column_count - 5).to_text+ " more"] else truncated
"Table{" + val.row_count.to_text + " rows x [" + (prepared.join ", ") + "]}"
+ _ : Function -> "[Function "+val.to_text+"]"
_ -> val.to_display_text