Skip to content

Commit

Permalink
Tune Text.trim, fix for Text.split (#10445)
Browse files Browse the repository at this point in the history
- Rename `Location.Start` to `Location.Left`.
- Rename `Location.End` to `Location.Right`.
- Use auto-scoping for `Location`.
- Tune widgets for `Text.trim`.
- Correct signature of `Text.split`.
- Adjist `generateLocallyUniqueIdent` to not fail on bad signature.
  • Loading branch information
jdunkerley authored Jul 4, 2024
1 parent d46e4ac commit 0661f17
Show file tree
Hide file tree
Showing 21 changed files with 91 additions and 88 deletions.
5 changes: 4 additions & 1 deletion CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,10 +8,13 @@

#### Enso Enso Standard Library

- [Reamed `Data.list_directory` to `Data.list`. Removed list support from read
- [Renamed `Data.list_directory` to `Data.list`. Removed list support from read
methods.][10434]
- [Renamed `Location.Start` to `Location.Left` and `Location.End` to
`Location.Right`.][10445]

[10434]: https://github.com/enso-org/enso/pull/10434
[10445]: https://github.com/enso-org/enso/pull/10445

# Enso 2024.2

Expand Down
3 changes: 2 additions & 1 deletion app/gui2/src/stores/graph/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -240,8 +240,9 @@ export const { injectFn: useGraphStore, provideFn: provideGraphStore } = createC
// FIXME: This implementation is not robust in the context of a synchronized document,
// as the same name can likely be assigned by multiple clients.
// Consider implementing a mechanism to repair the document in case of name clashes.
const identPrefix = prefix && isIdentifier(prefix + 1) ? prefix : FALLBACK_BINDING_PREFIX
for (let i = 1; ; i++) {
const ident = (prefix ?? FALLBACK_BINDING_PREFIX) + i
const ident = identPrefix + i
assert(isIdentifier(ident))
if (!db.identifierUsed(ident) && !ignore.has(ident)) return ident
}
Expand Down
Original file line number Diff line number Diff line change
@@ -1,11 +1,11 @@
import project.Any.Any
import project.Data.Array.Array
import project.Data.Array_Proxy.Array_Proxy
import project.Data.Decimal.Decimal
import project.Data.Json.JS_Object
import project.Data.Json.Json
import project.Data.Locale.Locale
import project.Data.Map.Map
import project.Data.Decimal.Decimal
import project.Data.Numbers.Float
import project.Data.Numbers.Integer
import project.Data.Numbers.Number
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,7 @@ from project.Data.Json import Invalid_JSON, JS_Object, Json
from project.Data.Numbers import Float, Integer, Number, Number_Parse_Error
from project.Data.Range.Extensions import all
from project.Data.Text.Text_Cleanse import Cleansable_Text, Text_Cleanse
from project.Metadata import Display, Widget
from project.Metadata import Display, make_single_choice, Widget
from project.Widget_Helpers import make_data_cleanse_vector_selector, make_date_format_selector, make_date_time_format_selector, make_delimiter_selector, make_regex_text_widget, make_time_format_selector

polyglot java import com.ibm.icu.lang.UCharacter
Expand Down Expand Up @@ -382,7 +382,7 @@ Text.to_regex self case_insensitive=False = Regex.compile self case_insensitive

'azbzczdzezfzg'.split ['b', 'zez'] == ['az', 'zczd', 'fzg']
@delimiter make_delimiter_selector
Text.split : Text | Vector Text -> Case_Sensitivity -> Boolean -> Vector Text | Illegal_Argument
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
Expand Down Expand Up @@ -1181,10 +1181,12 @@ Text.to_case self case_option:Case=..Lower locale:Locale=Locale.default = case c

"HELLO".pad 9 "AB" == "HELLOABAB"
"HELLO".pad 8 "AB" == "HELLOABA"
"HELLO".pad 8 "AB" Location.Start == "BABHELLO"

"HELLO".pad 8 "AB" ..Left == "BABHELLO"
@length (Widget.Numeric_Input display=Display.Always)
@with_pad (Widget.Text_Input display=Display.Always)
@at (make_single_choice [["Left", "..Left"], ["Right", "..Right"]])
Text.pad : Integer -> Text -> Location -> Text
Text.pad self length=0 with_pad=' ' at=Location.End = case at of
Text.pad self length:Integer=0 with_pad:Text=' ' at:Location=..Right = case at of
Location.Both -> Error.throw (Illegal_Argument.Error "`Location.Both` cannot be used with `pad`.")
_ ->
with_pad_length = with_pad.length
Expand All @@ -1194,9 +1196,9 @@ Text.pad self length=0 with_pad=' ' at=Location.End = case at of
full_repetitions = pad_size.div with_pad_length
remainder = pad_size % with_pad_length
case at of
Location.Start ->
Location.Left ->
with_pad.take (Index_Sub_Range.Last remainder) + with_pad.repeat full_repetitions + self
Location.End ->
Location.Right ->
self + with_pad.repeat full_repetitions + with_pad.take (Index_Sub_Range.First remainder)

## GROUP Text
Expand All @@ -1215,42 +1217,40 @@ Text.pad self length=0 with_pad=' ' at=Location.End = case at of
Trimming whitespace from a string.

" Hello! ".trim == "Hello!"
" Hello! ".trim Location.Start == "Hello! "
" Hello! ".trim Location.End == " Hello!"
" Hello! ".trim ..Left == "Hello! "
" Hello! ".trim ..Right == " Hello!"

> Example
Trimming a specific set of letters from a string.

"ABC123".trim Location.Start "ABC" == "123"
"ABBA123".trim Location.Start "ABC" == "123"
"ABC123".trim ..Left "ABC" == "123"
"ABBA123".trim ..Left "ABC" == "123"
@where (make_single_choice [["Both", "..Both"], ["Left", "..Left"], ["Right", "..Right"]])
@what (Widget.Text_Input display=Display.Always)
Text.trim : Location -> (Text | (Text -> Boolean)) -> Text
Text.trim self where=Location.Both what=_.is_whitespace =
Text.trim self where:Location=..Both what=_.is_whitespace =
predicate = case what of
_ : Text -> what.contains _
_ -> what
break_iterator = BreakIterator.getCharacterInstance
break_iterator.setText self
start_index = case where of
Location.End -> 0
_ ->
loop current next =
if next < 0 then current else
case predicate (Text_Utils.substring self current next) of
True ->
@Tail_Call loop next break_iterator.next
False -> current
loop 0 break_iterator.next
end_index = case where of
Location.Start -> Text_Utils.char_length self
_ ->
loop current prev =
if prev < 0 then current else
case predicate (Text_Utils.substring self prev current) of
True ->
@Tail_Call loop prev break_iterator.previous
False -> current
current = break_iterator.last
loop current break_iterator.previous
start_index = if where == Location.Right then 0 else
loop current next =
if next < 0 then current else
case predicate (Text_Utils.substring self current next) of
True ->
@Tail_Call loop next break_iterator.next
False -> current
loop 0 break_iterator.next
end_index = if where == Location.Left then Text_Utils.char_length self else
loop current prev =
if prev < 0 then current else
case predicate (Text_Utils.substring self prev current) of
True ->
@Tail_Call loop prev break_iterator.previous
False -> current
current = break_iterator.last
loop current break_iterator.previous
if start_index >= end_index then "" else
Text_Utils.substring self start_index end_index

Expand Down
Original file line number Diff line number Diff line change
@@ -1,9 +1,9 @@
type Location
## Indicates the beginning of a text.
Start
Left

## Indicates the end of a text.
End
Right

## Indicates both the beginning and end of a text.
Both
Original file line number Diff line number Diff line change
Expand Up @@ -189,7 +189,7 @@ type Regex
input = "abcdefghij"
texts = pattern.split input
texts . should_equal ["abcdefghij"]
split : Text -> Boolean -> Vector Text | Type_Error
split : Text -> Boolean -> Vector Text ! Type_Error
split self (input : Text) (only_first : Boolean = False) =
Vector.build builder->
it = Match_Iterator.new self input
Expand Down
3 changes: 1 addition & 2 deletions distribution/lib/Standard/Base/0.0.0-dev/src/Metadata.enso
Original file line number Diff line number Diff line change
Expand Up @@ -6,11 +6,10 @@ import project.Data.Text.Text
import project.Data.Vector.Vector
import project.Meta
import project.Nothing.Nothing
from project.Data.Boolean import Boolean, False, True
from project.Data.Json.Extensions import all
from project.Data.Range.Extensions import all

from project.Data.Boolean import Boolean, False, True

type Display
## Parameter is always shown on the collapsed view.
Always
Expand Down
19 changes: 9 additions & 10 deletions distribution/lib/Standard/Database/0.0.0-dev/src/DB_Column.enso
Original file line number Diff line number Diff line change
Expand Up @@ -1373,16 +1373,15 @@ type DB_Column
- what: A `Text` (or text `DB_Column`) containing characters that should be
removed. By default, spaces, tabs, returns and new lines are removed.
trim : Location -> DB_Column | Text -> DB_Column
trim self where=Location.Both what='' =
Value_Type.expect_text self <|
Value_Type.expect_text what <|
new_name = self.naming_helper.function_name "trim" [self]
operator = case where of
Location.Both -> "TRIM"
Location.Start -> "LTRIM"
Location.End -> "RTRIM"
if self.connection.dialect.is_supported operator then self.make_binary_op operator what new_name else
Error.throw (Unsupported_Database_Operation.Error ("`DB_Column.trim "+where.to_text+"` is not supported by this connection."))
trim self where:Location=..Both what='' =
Value_Type.expect_text self <| Value_Type.expect_text what <|
new_name = self.naming_helper.function_name "trim" [self]
operator = case where of
Location.Both -> "TRIM"
Location.Left -> "LTRIM"
Location.Right -> "RTRIM"
if self.connection.dialect.is_supported operator then self.make_binary_op operator what new_name else
Error.throw (Unsupported_Database_Operation.Error ("`DB_Column.trim "+where.to_text+"` is not supported by this connection."))

## GROUP Standard.Base.Text
ICON column_add
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -337,7 +337,7 @@ profile_sql_if_enabled (jdbc_connection : JDBC_Connection) (~query_text : Text)
result.if_not_error <|
padded_query = query_text.to_display_text.pad length=80
millis = duration.total_milliseconds
millis_text = (millis.to_text.pad length=5 with_pad=' ' at=Location.Start) + ' ms'
millis_text = (millis.to_text.pad length=5 with_pad=' ' at=..Left) + ' ms'

# E.g. [SQLite] SELECT * FROM "test" ... --> 12 ms
log_line = "[" + db_id + "] " + padded_query + " --> " + millis_text + '\n'
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -287,7 +287,7 @@ type Postgres_Connection
self.data_link_setup.save_as_data_link destination on_existing_file

## PRIVATE
Converts this value to a JSON serializable object.
Converts this value to a JSON serializable object.
to_js_object : JS_Object
to_js_object self =
JS_Object.from_pairs [["type", "Postgres_Connection"], ["links", self.connection.tables.at "Name" . to_vector]]
Expand Down
2 changes: 1 addition & 1 deletion distribution/lib/Standard/Table/0.0.0-dev/src/Column.enso
Original file line number Diff line number Diff line change
Expand Up @@ -1402,7 +1402,7 @@ type Column
- what: A `Text` (or text `Column`) containing characters that should be
removed. By default, all whitespace is removed.
trim : Location -> Column | Text -> Column
trim self where=Location.Both what='' = Value_Type.expect_text self <|
trim self where:Location=..Both what='' = Value_Type.expect_text self <|
new_name = naming_helper.function_name "trim" [self]

trim_get = wrap_text_argument_as_value_provider what
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -74,7 +74,7 @@ type Column_Operation

## Removes the specified characters, by default any whitespace, from the
start, the end, or both ends of the input.
Trim (input : Column_Ref|Expression|Text) (where:Location = Location.Both) (what:Text|Column_Ref = "")
Trim (input : Column_Ref|Expression|Text) (where:Location=..Both) (what:Text|Column_Ref = "")

## PRIVATE
Simple_Expression.from (that:Column_Operation) =
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -178,7 +178,7 @@ type Simple_Calculation

## Removes the specified characters, by default any whitespace, from the
start, the end, or both ends of the input.
Trim (where:Location = ..Both) (what:Column_Ref|Expression|Text = "")
Trim (where:Location=..Both) (what:Column_Ref|Expression|Text = "")

## Takes the first characters from the input column.
Text_Left (length : Column_Ref|Expression|Integer = 1)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -189,7 +189,7 @@ progress_width = 70
print_progress current_progress total_count status_text =
total_count_as_text = total_count.to_text
counter_width = total_count_as_text.length
current_progress_as_text = current_progress.to_text.pad counter_width at=Location.Start
current_progress_as_text = current_progress.to_text.pad counter_width at=..Left
line = " ("+ current_progress_as_text + " / " + total_count_as_text + ") " + status_text
truncated_line = if line.length <= progress_width then line else
line.take (progress_width - 3) + '...'
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,9 +7,8 @@ from Standard.Table import Column, Table

import project.Id.Id
import project.Table as Table_Visualization
from project.Text import get_lazy_visualization_text_window
from project.Table.Visualization import make_json_for_value

from project.Text import get_lazy_visualization_text_window

## PRIVATE
Specifies that the builtin Table visualization should be used for any type,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ error_preprocessor x =
message = err.to_display_text
stack_trace = x.get_stack_trace_text.if_nothing "" . split '\n'
truncated_message = Helpers.truncate message
full_message = truncated_message + if stack_trace.length > 0 then " (" + stack_trace.at 0 . trim +")" else ""
full_message = truncated_message + if stack_trace.length > 0 then " (" + stack_trace.at 0 . trim + ")" else ""
JS_Object.from_pairs [['kind', 'Dataflow'], ['message', full_message]] . to_json

if result.is_error then result.catch else ok
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ from Standard.Base import all
import Standard.Base.Data.Vector.Builder

import Standard.Table.Row.Row
from Standard.Table import Column, Table, Excel_Workbook
from Standard.Table import Column, Excel_Workbook, Table

import Standard.Database.DB_Column.DB_Column
import Standard.Database.DB_Table.DB_Table
Expand Down Expand Up @@ -157,6 +157,8 @@ make_json_for_table dataframe all_rows_count include_index_col =
pairs = [header, value_type, data, all_rows, has_index_col, ["type", "Table"]]
JS_Object.from_pairs pairs

## PRIVATE
Create JSON serialization of values.
make_json_for_other : Any -> JS_Object
make_json_for_other x =
js_value = x.to_js_object
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2703,7 +2703,7 @@ class RuntimeVisualizationsTest extends AnyFlatSpec with Matchers {
data
}
val stringified = new String(data)
stringified shouldEqual """{"kind":"Dataflow","message":"The List is empty. (at <enso> Main.main(Enso_Test.Test.Main:7:5-40)"}"""
stringified shouldEqual """{"kind":"Dataflow","message":"The List is empty. (at <enso> Main.main(Enso_Test.Test.Main:7:5-40))"}"""
}

it should "run visualization default preprocessor" in withContext() {
Expand Down
36 changes: 18 additions & 18 deletions test/Base_Tests/src/Data/Text_Spec.enso
Original file line number Diff line number Diff line change
Expand Up @@ -1039,7 +1039,7 @@ add_specs suite_builder =
"Hello World!".pad 15 . should_equal "Hello World! "
"HELLO".pad 9 "AB" . should_equal "HELLOABAB"
"HELLO".pad 8 "AB" . should_equal "HELLOABA"
"HELLO".pad 8 "AB" Location.Start . should_equal "BABHELLO"
"HELLO".pad 8 "AB" ..Left . should_equal "BABHELLO"
"".pad 4 . should_equal " "
"A".pad 3 "" . should_fail_with Illegal_Argument
"ABCDE".pad 3 "" . should_fail_with Illegal_Argument
Expand All @@ -1056,8 +1056,8 @@ add_specs suite_builder =
'XX'.pad 5 'y\u{301}y' . should_equal 'XXy\u{301}yy\u{301}'
'XX'.pad 4 'yy\u{301}Z' . should_equal 'XXyy\u{301}'

'🚀'.pad 3 'B' Location.End . should_equal '🚀BB'
'🚀'.pad 3 'B' Location.Start . should_equal 'BB🚀'
'🚀'.pad 3 'B' ..Right . should_equal '🚀BB'
'🚀'.pad 3 'B' ..Left . should_equal 'BB🚀'

## It is technically possible to use a combining diacritical mark as
the padding, then the actual length of the text will not increase
Expand All @@ -1067,32 +1067,32 @@ add_specs suite_builder =

group_builder.specify "should allow to trim a text" <|
" Hello! ".trim . should_equal "Hello!"
" Hello! ".trim Location.Start . should_equal "Hello! "
" Hello! ".trim Location.End . should_equal " Hello!"
"ABC123".trim Location.Start "ABC" . should_equal "123"
"ABBA123".trim Location.Start "ABC" . should_equal "123"
"ABCZ-]".trim Location.Both "[A-Z]" . should_equal "BC"
" Hello! ".trim ..Left . should_equal "Hello! "
" Hello! ".trim ..Right . should_equal " Hello!"
"ABC123".trim ..Left "ABC" . should_equal "123"
"ABBA123".trim ..Left "ABC" . should_equal "123"
"ABCZ-]".trim ..Both "[A-Z]" . should_equal "BC"

" ".trim . should_equal ""
" Hello World! ".trim . should_equal "Hello World!"
" Hello World! ".trim Location.Start . should_equal "Hello World! "
" Hello World! ".trim Location.End . should_equal " Hello World!"
"ABCD".trim Location.Start "ABCDEF" . should_equal ""
"ABCD".trim Location.End "ABCDEF" . should_equal ""
"ABCD".trim Location.Both "ABCDEF" . should_equal ""
" Hello World! ".trim ..Left . should_equal "Hello World! "
" Hello World! ".trim ..Right . should_equal " Hello World!"
"ABCD".trim ..Left "ABCDEF" . should_equal ""
"ABCD".trim ..Right "ABCDEF" . should_equal ""
"ABCD".trim ..Both "ABCDEF" . should_equal ""

"".trim . should_equal ""
"A".trim . should_equal "A"
" A ".trim . should_equal "A"
' A\u{301} \n '.trim . should_equal 'A\u{301}'
"🚧".trim . should_equal "🚧"
" 🚧 🚧 ".trim . should_equal "🚧 🚧"
" 🚧 🚧 ".trim Location.End . should_equal " 🚧 🚧"
" 🚧 🚧 ".trim ..Right . should_equal " 🚧 🚧"

"ABCD".trim Location.Start (_ -> True) . should_equal ""
"ABCD".trim Location.Both (_ -> True) . should_equal ""
"ABCD".trim Location.Both (_ -> False) . should_equal "ABCD"
"123AB98".trim Location.Both _.is_digit . should_equal "AB"
"ABCD".trim ..Left (_ -> True) . should_equal ""
"ABCD".trim ..Both (_ -> True) . should_equal ""
"ABCD".trim ..Both (_ -> False) . should_equal "ABCD"
"123AB98".trim ..Both _.is_digit . should_equal "AB"

' \t\n\r'.trim . should_equal ''
'\t\t Test\nFoo\r\n'.trim . should_equal 'Test\nFoo'
Expand Down
Loading

0 comments on commit 0661f17

Please sign in to comment.