From 2a05d04e353991a0fbdbd9fdfc6176550f89ebf3 Mon Sep 17 00:00:00 2001 From: Ronan Amicel Date: Tue, 9 Aug 2022 15:43:44 +0200 Subject: [PATCH 1/5] Ignore diacritics in autocomplete Works for: - Choice - Choice List - Reference - Reference List Co-Authored-By: Louis Delbosc --- app/client/lib/ACIndex.ts | 10 ++++++++-- app/client/models/ColumnACIndexes.ts | 4 ++-- app/client/widgets/ChoiceListEditor.ts | 4 ++-- app/client/widgets/ReferenceEditor.ts | 4 ++-- app/client/widgets/ReferenceListEditor.ts | 6 +++--- app/common/gutil.ts | 5 +++++ 6 files changed, 22 insertions(+), 11 deletions(-) diff --git a/app/client/lib/ACIndex.ts b/app/client/lib/ACIndex.ts index c3f638b4..9a103d98 100644 --- a/app/client/lib/ACIndex.ts +++ b/app/client/lib/ACIndex.ts @@ -7,7 +7,7 @@ * "lush" would only match the "L" in "Lavender". */ -import {localeCompare, nativeCompare, sortedIndex} from 'app/common/gutil'; +import {localeCompare, nativeCompare, removeDiacritics, sortedIndex} from 'app/common/gutil'; import {DomContents} from 'grainjs'; import escapeRegExp = require("lodash/escapeRegExp"); @@ -17,6 +17,12 @@ export interface ACItem { cleanText: string; } +// Returns a normalized, trimmed, lowercase version of a string, so that autocomplete is case- +// and accent-insensitive. +export function cleanText(text: string): string { + return removeDiacritics(text).trim().toLowerCase(); +} + // Regexp used to split text into words; includes nearly all punctuation. This means that // "foo-bar" may be searched by "bar", but it's impossible to search for punctuation itself (e.g. // "a-b" and "a+b" are not distinguished). (It's easy to exclude unicode punctuation too if the @@ -91,7 +97,7 @@ export class ACIndexImpl implements ACIndex { // The main search function. SearchText will be cleaned (trimmed and lowercased) at the start. // Empty search text returns the first N items in the search universe. public search(searchText: string): ACResults { - const cleanedSearchText = searchText.trim().toLowerCase(); + const cleanedSearchText = cleanText(searchText); const searchWords = cleanedSearchText.split(wordSepRegexp).filter(w => w); // Maps item index in _allItems to its score. diff --git a/app/client/models/ColumnACIndexes.ts b/app/client/models/ColumnACIndexes.ts index 4020e7db..b0bbad20 100644 --- a/app/client/models/ColumnACIndexes.ts +++ b/app/client/models/ColumnACIndexes.ts @@ -8,7 +8,7 @@ * * It is currently used for auto-complete in the ReferenceEditor and ReferenceListEditor widgets. */ -import {ACIndex, ACIndexImpl} from 'app/client/lib/ACIndex'; +import {ACIndex, ACIndexImpl, cleanText as clean} from 'app/client/lib/ACIndex'; import {ColumnCache} from 'app/client/models/ColumnCache'; import {UserError} from 'app/client/models/errors'; import {TableData} from 'app/client/models/TableData'; @@ -45,7 +45,7 @@ export class ColumnACIndexes { const items: ICellItem[] = valColumn.map((val, i) => { const rowId = rowIds[i]; const text = formatter.formatAny(val); - const cleanText = text.trim().toLowerCase(); + const cleanText = clean(text); return {rowId, text, cleanText}; }); items.sort(itemCompare); diff --git a/app/client/widgets/ChoiceListEditor.ts b/app/client/widgets/ChoiceListEditor.ts index 7a986f47..1bcc242e 100644 --- a/app/client/widgets/ChoiceListEditor.ts +++ b/app/client/widgets/ChoiceListEditor.ts @@ -1,5 +1,5 @@ import {createGroup} from 'app/client/components/commands'; -import {ACIndexImpl, ACItem, ACResults, buildHighlightedDom, HighlightFunc} from 'app/client/lib/ACIndex'; +import {ACIndexImpl, ACItem, ACResults, buildHighlightedDom, cleanText as clean, HighlightFunc} from 'app/client/lib/ACIndex'; import {IAutocompleteOptions} from 'app/client/lib/autocomplete'; import {IToken, TokenField, tokenFieldStyles} from 'app/client/lib/TokenField'; import {colors, testId} from 'app/client/ui2018/cssVars'; @@ -16,7 +16,7 @@ import {choiceToken, cssChoiceACItem, cssChoiceToken} from 'app/client/widgets/C import {icon} from 'app/client/ui2018/icons'; export class ChoiceItem implements ACItem, IToken { - public cleanText: string = this.label.toLowerCase().trim(); + public cleanText: string = clean(this.label); constructor( public label: string, public isInvalid: boolean, // If set, this token is not one of the valid choices. diff --git a/app/client/widgets/ReferenceEditor.ts b/app/client/widgets/ReferenceEditor.ts index 7625af7b..61f25ac8 100644 --- a/app/client/widgets/ReferenceEditor.ts +++ b/app/client/widgets/ReferenceEditor.ts @@ -1,4 +1,4 @@ -import { ACResults, buildHighlightedDom, HighlightFunc } from 'app/client/lib/ACIndex'; +import { ACResults, buildHighlightedDom, cleanText as clean, HighlightFunc } from 'app/client/lib/ACIndex'; import { Autocomplete } from 'app/client/lib/autocomplete'; import { ICellItem } from 'app/client/models/ColumnACIndexes'; import { reportError } from 'app/client/models/errors'; @@ -115,7 +115,7 @@ export class ReferenceEditor extends NTextEditor { this._showAddNew = false; if (!this._enableAddNew || !text) { return result; } - const cleanText = text.trim().toLowerCase(); + const cleanText = clean(text); if (result.items.find((item) => item.cleanText === cleanText)) { return result; } diff --git a/app/client/widgets/ReferenceListEditor.ts b/app/client/widgets/ReferenceListEditor.ts index 58e6375e..ccf2910e 100644 --- a/app/client/widgets/ReferenceListEditor.ts +++ b/app/client/widgets/ReferenceListEditor.ts @@ -1,5 +1,5 @@ import { createGroup } from 'app/client/components/commands'; -import { ACItem, ACResults, HighlightFunc } from 'app/client/lib/ACIndex'; +import { ACItem, ACResults, cleanText as clean, HighlightFunc } from 'app/client/lib/ACIndex'; import { IAutocompleteOptions } from 'app/client/lib/autocomplete'; import { IToken, TokenField, tokenFieldStyles } from 'app/client/lib/TokenField'; import { reportError } from 'app/client/models/errors'; @@ -26,7 +26,7 @@ class ReferenceItem implements IToken, ACItem { * similar to getItemText() from IAutocompleteOptions. */ public label: string = typeof this.rowId === 'number' ? String(this.rowId) : this.text; - public cleanText: string = this.text.trim().toLowerCase(); + public cleanText: string = clean(this.text); constructor( public text: string, @@ -264,7 +264,7 @@ export class ReferenceListEditor extends NewBaseEditor { this._showAddNew = false; if (!this._enableAddNew || !text) { return result; } - const cleanText = text.trim().toLowerCase(); + const cleanText = clean(text); if (result.items.find((item) => item.cleanText === cleanText)) { return result; } diff --git a/app/common/gutil.ts b/app/common/gutil.ts index b3c37676..998ea012 100644 --- a/app/common/gutil.ts +++ b/app/common/gutil.ts @@ -55,6 +55,11 @@ export function capitalizeFirstWord(str: string): string { return str.replace(/\b[a-z]/i, c => c.toUpperCase()); } +// Remove diacritics (accents and other signs). +export function removeDiacritics(text: string): string { + return text.normalize("NFKD").replace(/[\u0300-\u036f]/g, "") +} + // Returns whether the string n represents a valid number. // http://stackoverflow.com/questions/18082/validate-numbers-in-javascript-isnumeric export function isNumber(n: string): boolean { From 0249b3d31b2888ce29e0370cee91a3e00bdc3f9f Mon Sep 17 00:00:00 2001 From: Ronan Amicel Date: Wed, 17 Aug 2022 18:15:32 +0200 Subject: [PATCH 2/5] Add tests for autocomplete without accents --- test/nbrowser/ChoiceList.ts | 17 +++++++++++++++++ test/nbrowser/ReferenceColumns.ts | 12 ++++++++++++ test/nbrowser/ReferenceList.ts | 12 ++++++++++++ 3 files changed, 41 insertions(+) diff --git a/test/nbrowser/ChoiceList.ts b/test/nbrowser/ChoiceList.ts index 8cbb434e..d528ecb9 100644 --- a/test/nbrowser/ChoiceList.ts +++ b/test/nbrowser/ChoiceList.ts @@ -301,6 +301,23 @@ describe('ChoiceList', function() { await gu.waitForServer(); assert.equal(await driver.find('.cell_editor').isPresent(), false); assert.equal(await gu.getCell({rowNum: 1, col: 'B'}).getText(), 'Blue\nGreen\nBlack'); + + + // Starting to type names without accents should match the actual choices + await gu.addColumn("Accents"); + await api.applyUserActions(docId, [ + ['ModifyColumn', 'Table1', 'Accents', { + type: 'ChoiceList', + widgetOptions: JSON.stringify({ + choices: ['Adélaïde', 'Adèle', 'Agnès', 'Amélie'], + }) + }], + ]); + await gu.getCell({rowNum: 1, col: 'Accents'}).click(); + await driver.sendKeys('Ade', Key.ENTER); + await driver.sendKeys('Agne', Key.ENTER); + await driver.sendKeys('Ame', Key.ENTER); + assert.deepEqual(await getEditorTokens(), ['Adélaïde', 'Agnès', 'Amélie']); }); it('should be visible in formulas', async () => { diff --git a/test/nbrowser/ReferenceColumns.ts b/test/nbrowser/ReferenceColumns.ts index 5f9d09ca..b799d193 100644 --- a/test/nbrowser/ReferenceColumns.ts +++ b/test/nbrowser/ReferenceColumns.ts @@ -413,6 +413,18 @@ describe('ReferenceColumns', function() { ['Dark Slate Blue', 'Dark Slate Gray', 'Slate Blue', 'Medium Slate Blue']); await driver.sendKeys(Key.ESCAPE); + // Starting to type Añil with the accent + await driver.sendKeys('añ'); + assert.deepEqual(await getACOptions(2), + ['Añil', 'Alice Blue']); + await driver.sendKeys(Key.ESCAPE); + + // Starting to type Añil without the accent should work too + await driver.sendKeys('an'); + assert.deepEqual(await getACOptions(2), + ['Añil', 'Alice Blue']); + await driver.sendKeys(Key.ESCAPE); + await driver.sendKeys('blac'); assert.deepEqual(await getACOptions(6), ['Black', 'Blanched Almond', 'Blue', 'Blue Violet', 'Alice Blue', 'Cadet Blue']); diff --git a/test/nbrowser/ReferenceList.ts b/test/nbrowser/ReferenceList.ts index 2a43ce63..86eb7e73 100644 --- a/test/nbrowser/ReferenceList.ts +++ b/test/nbrowser/ReferenceList.ts @@ -709,6 +709,18 @@ describe('ReferenceList', function() { ['Dark Slate Blue', 'Dark Slate Gray', 'Slate Blue', 'Medium Slate Blue']); await driver.sendKeys(Key.ESCAPE); + // Starting to type Añil with the accent + await driver.sendKeys('añ'); + assert.deepEqual(await getACOptions(2), + ['Añil', 'Alice Blue']); + await driver.sendKeys(Key.ESCAPE); + + // Starting to type Añil without the accent should work too + await driver.sendKeys('an'); + assert.deepEqual(await getACOptions(2), + ['Añil', 'Alice Blue']); + await driver.sendKeys(Key.ESCAPE); + await driver.sendKeys('blac'); assert.deepEqual(await getACOptions(6), ['Black', 'Blanched Almond', 'Blue', 'Blue Violet', 'Alice Blue', 'Cadet Blue']); From 4003d2d7c630a1829846e3f2711017965ab05098 Mon Sep 17 00:00:00 2001 From: Ronan Amicel Date: Wed, 17 Aug 2022 18:35:50 +0200 Subject: [PATCH 3/5] Use lodash to remove accents and ther diacritics https://lodash.com/docs/4.17.15#deburr --- app/client/lib/ACIndex.ts | 10 ++++++---- app/common/gutil.ts | 5 ----- 2 files changed, 6 insertions(+), 9 deletions(-) diff --git a/app/client/lib/ACIndex.ts b/app/client/lib/ACIndex.ts index 9a103d98..e305fcff 100644 --- a/app/client/lib/ACIndex.ts +++ b/app/client/lib/ACIndex.ts @@ -7,9 +7,10 @@ * "lush" would only match the "L" in "Lavender". */ -import {localeCompare, nativeCompare, removeDiacritics, sortedIndex} from 'app/common/gutil'; +import {localeCompare, nativeCompare, sortedIndex} from 'app/common/gutil'; import {DomContents} from 'grainjs'; import escapeRegExp = require("lodash/escapeRegExp"); +import deburr = require("lodash/deburr"); export interface ACItem { // This should be a trimmed lowercase version of the item's text. It may be an accessor. @@ -17,10 +18,11 @@ export interface ACItem { cleanText: string; } -// Returns a normalized, trimmed, lowercase version of a string, so that autocomplete is case- -// and accent-insensitive. +// Returns a trimmed, lowercase version of a string, +// from which accents and other diacritics have been removed, +// so that autocomplete is case- and accent-insensitive. export function cleanText(text: string): string { - return removeDiacritics(text).trim().toLowerCase(); + return deburr(text).trim().toLowerCase(); } // Regexp used to split text into words; includes nearly all punctuation. This means that diff --git a/app/common/gutil.ts b/app/common/gutil.ts index 998ea012..b3c37676 100644 --- a/app/common/gutil.ts +++ b/app/common/gutil.ts @@ -55,11 +55,6 @@ export function capitalizeFirstWord(str: string): string { return str.replace(/\b[a-z]/i, c => c.toUpperCase()); } -// Remove diacritics (accents and other signs). -export function removeDiacritics(text: string): string { - return text.normalize("NFKD").replace(/[\u0300-\u036f]/g, "") -} - // Returns whether the string n represents a valid number. // http://stackoverflow.com/questions/18082/validate-numbers-in-javascript-isnumeric export function isNumber(n: string): boolean { From cd16650e0e992857ad1ecaebc763a12af6f310cd Mon Sep 17 00:00:00 2001 From: Ronan Amicel Date: Wed, 17 Aug 2022 18:40:46 +0200 Subject: [PATCH 4/5] Rename text cleaning function to "normalizeText" --- app/client/lib/ACIndex.ts | 4 ++-- app/client/models/ColumnACIndexes.ts | 4 ++-- app/client/widgets/ChoiceListEditor.ts | 4 ++-- app/client/widgets/ReferenceEditor.ts | 4 ++-- app/client/widgets/ReferenceListEditor.ts | 6 +++--- 5 files changed, 11 insertions(+), 11 deletions(-) diff --git a/app/client/lib/ACIndex.ts b/app/client/lib/ACIndex.ts index e305fcff..44eb83f1 100644 --- a/app/client/lib/ACIndex.ts +++ b/app/client/lib/ACIndex.ts @@ -21,7 +21,7 @@ export interface ACItem { // Returns a trimmed, lowercase version of a string, // from which accents and other diacritics have been removed, // so that autocomplete is case- and accent-insensitive. -export function cleanText(text: string): string { +export function normalizeText(text: string): string { return deburr(text).trim().toLowerCase(); } @@ -99,7 +99,7 @@ export class ACIndexImpl implements ACIndex { // The main search function. SearchText will be cleaned (trimmed and lowercased) at the start. // Empty search text returns the first N items in the search universe. public search(searchText: string): ACResults { - const cleanedSearchText = cleanText(searchText); + const cleanedSearchText = normalizeText(searchText); const searchWords = cleanedSearchText.split(wordSepRegexp).filter(w => w); // Maps item index in _allItems to its score. diff --git a/app/client/models/ColumnACIndexes.ts b/app/client/models/ColumnACIndexes.ts index b0bbad20..bd323a1e 100644 --- a/app/client/models/ColumnACIndexes.ts +++ b/app/client/models/ColumnACIndexes.ts @@ -8,7 +8,7 @@ * * It is currently used for auto-complete in the ReferenceEditor and ReferenceListEditor widgets. */ -import {ACIndex, ACIndexImpl, cleanText as clean} from 'app/client/lib/ACIndex'; +import {ACIndex, ACIndexImpl, normalizeText} from 'app/client/lib/ACIndex'; import {ColumnCache} from 'app/client/models/ColumnCache'; import {UserError} from 'app/client/models/errors'; import {TableData} from 'app/client/models/TableData'; @@ -45,7 +45,7 @@ export class ColumnACIndexes { const items: ICellItem[] = valColumn.map((val, i) => { const rowId = rowIds[i]; const text = formatter.formatAny(val); - const cleanText = clean(text); + const cleanText = normalizeText(text); return {rowId, text, cleanText}; }); items.sort(itemCompare); diff --git a/app/client/widgets/ChoiceListEditor.ts b/app/client/widgets/ChoiceListEditor.ts index 1bcc242e..75674031 100644 --- a/app/client/widgets/ChoiceListEditor.ts +++ b/app/client/widgets/ChoiceListEditor.ts @@ -1,5 +1,5 @@ import {createGroup} from 'app/client/components/commands'; -import {ACIndexImpl, ACItem, ACResults, buildHighlightedDom, cleanText as clean, HighlightFunc} from 'app/client/lib/ACIndex'; +import {ACIndexImpl, ACItem, ACResults, buildHighlightedDom, normalizeText, HighlightFunc} from 'app/client/lib/ACIndex'; import {IAutocompleteOptions} from 'app/client/lib/autocomplete'; import {IToken, TokenField, tokenFieldStyles} from 'app/client/lib/TokenField'; import {colors, testId} from 'app/client/ui2018/cssVars'; @@ -16,7 +16,7 @@ import {choiceToken, cssChoiceACItem, cssChoiceToken} from 'app/client/widgets/C import {icon} from 'app/client/ui2018/icons'; export class ChoiceItem implements ACItem, IToken { - public cleanText: string = clean(this.label); + public cleanText: string = normalizeText(this.label); constructor( public label: string, public isInvalid: boolean, // If set, this token is not one of the valid choices. diff --git a/app/client/widgets/ReferenceEditor.ts b/app/client/widgets/ReferenceEditor.ts index 61f25ac8..e09d9cb5 100644 --- a/app/client/widgets/ReferenceEditor.ts +++ b/app/client/widgets/ReferenceEditor.ts @@ -1,4 +1,4 @@ -import { ACResults, buildHighlightedDom, cleanText as clean, HighlightFunc } from 'app/client/lib/ACIndex'; +import { ACResults, buildHighlightedDom, normalizeText, HighlightFunc } from 'app/client/lib/ACIndex'; import { Autocomplete } from 'app/client/lib/autocomplete'; import { ICellItem } from 'app/client/models/ColumnACIndexes'; import { reportError } from 'app/client/models/errors'; @@ -115,7 +115,7 @@ export class ReferenceEditor extends NTextEditor { this._showAddNew = false; if (!this._enableAddNew || !text) { return result; } - const cleanText = clean(text); + const cleanText = normalizeText(text); if (result.items.find((item) => item.cleanText === cleanText)) { return result; } diff --git a/app/client/widgets/ReferenceListEditor.ts b/app/client/widgets/ReferenceListEditor.ts index ccf2910e..cb757c70 100644 --- a/app/client/widgets/ReferenceListEditor.ts +++ b/app/client/widgets/ReferenceListEditor.ts @@ -1,5 +1,5 @@ import { createGroup } from 'app/client/components/commands'; -import { ACItem, ACResults, cleanText as clean, HighlightFunc } from 'app/client/lib/ACIndex'; +import { ACItem, ACResults, normalizeText, HighlightFunc } from 'app/client/lib/ACIndex'; import { IAutocompleteOptions } from 'app/client/lib/autocomplete'; import { IToken, TokenField, tokenFieldStyles } from 'app/client/lib/TokenField'; import { reportError } from 'app/client/models/errors'; @@ -26,7 +26,7 @@ class ReferenceItem implements IToken, ACItem { * similar to getItemText() from IAutocompleteOptions. */ public label: string = typeof this.rowId === 'number' ? String(this.rowId) : this.text; - public cleanText: string = clean(this.text); + public cleanText: string = normalizeText(this.text); constructor( public text: string, @@ -264,7 +264,7 @@ export class ReferenceListEditor extends NewBaseEditor { this._showAddNew = false; if (!this._enableAddNew || !text) { return result; } - const cleanText = clean(text); + const cleanText = normalizeText(text); if (result.items.find((item) => item.cleanText === cleanText)) { return result; } From b286256f969d447319a4bee5ad7bde39170130c4 Mon Sep 17 00:00:00 2001 From: Paul Fitzpatrick Date: Wed, 17 Aug 2022 19:07:48 -0400 Subject: [PATCH 5/5] add link to new self-managed handbook (#246) Adds a link to a new page for documenting configuration of self-managed Grist. --- README.md | 15 +++++++++------ 1 file changed, 9 insertions(+), 6 deletions(-) diff --git a/README.md b/README.md index 565dd0f2..24bafba3 100644 --- a/README.md +++ b/README.md @@ -98,6 +98,10 @@ To enable gVisor sandboxing, set `--env GRIST_SANDBOX_FLAVOR=gvisor`. This should work with default docker settings, but may not work in all environments. +You can find a lot more about configuring Grist, setting up authentication, +and running it on a public server in our +[Self-Managed Grist](https://support.getgrist.com/self-managed/) handbook. + ## Building from source To build Grist from source, follow these steps: @@ -137,11 +141,10 @@ You can change your name in `Profile Settings` in the [User Menu](https://support.getgrist.com/glossary/#user-menu). For multi-user operation, or if you wish to access Grist across the -public internet, you'll want to connect it to your own single sign-in service. -There's a `docker-compose` template at https://community.getgrist.com/t/a-template-for-self-hosting-grist-with-traefik-and-docker-compose/856 -covering using Let's Encrypt for certificates and Google for sign-ins. -You can also use [SAML](https://github.com/gristlabs/grist-core/blob/main/app/server/lib/SamlConfig.ts). -Grist has been tested with [Authentik](https://goauthentik.io/) and [Auth0](https://auth0.com/). +public internet, you'll want to connect it to your own Single Sign-On service. +There are a lot of ways to do this, including [SAML and forward authentication](https://support.getgrist.com/self-managed/#how-do-i-set-up-authentication). +Grist has been tested with [Authentik](https://goauthentik.io/), [Auth0](https://auth0.com/), +and Google/Microsoft sign-ins via [Dex](https://dexidp.io/). ## Why free and open source software @@ -151,7 +154,7 @@ here, combined with business-specific software designed to scale it to many user etc. Grist Labs is an open-core company. We offer Grist hosting as a -service, with free and paid plans. We intend to also develop and sell +service, with free and paid plans. We also develop and sell features related to Grist using a proprietary license, targeted at the needs of enterprises with large self-managed installations. We see data portability and autonomy as a key value Grist can bring to our