% API
Builtins
Text
Text.as_c_string
Text.as_c_string : func(text: Text -> CString)
Converts a Text value to a C-style string.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be converted to a C-style string. | - |
Return: A C-style string (CString) representing the text.
Example:
assert "Hello".as_c_string() == CString("Hello")
Text.at
Text.at : func(text: Text, index: Int -> Text)
Get the graphical cluster at a given index. This is similar to str[i] with ASCII text, but has more correct behavior for unicode text.
Negative indices are counted from the back of the text, so -1 means the last cluster, -2 means the second-to-last, and so on.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text from which to get a cluster. | - |
| index | Int |
The index of the graphical cluster (1-indexed). | - |
Return: A Text with the single graphical cluster at the given index.
Example:
assert "Amélie".at(3) == "é"
Text.by_line
Text.by_line : func(text: Text -> func(->Text?))
Returns an iterator function that can be used to iterate over the lines in a text.
This function ignores a trailing newline if there is one. If you don't want this behavior, use text.by_split($/{1 nl}/) instead.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be iterated over, line by line. | - |
Return: An iterator function that returns one line at a time, until it runs out and returns none.
Example:
text := "
line one
line two
"
lines := [line for line in text.by_line()]
assert lines == ["line one", "line two"]
Text.by_split
Text.by_split : func(text: Text, delimiter: Text = "" -> func(->Text?))
Returns an iterator function that can be used to iterate over text separated by a delimiter.
To split based on a set of delimiters, use Text.by_split_any(). If an empty text is given as the delimiter, then each split will be the graphical clusters of the text.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be iterated over in delimited chunks. | - |
| delimiter | Text |
An exact delimiter to use for splitting the text. | "" |
Return: An iterator function that returns one chunk of text at a time, separated by the given delimiter, until it runs out and returns none.
Example:
text := "one,two,three"
chunks := [chunk for chunk in text.by_split(",")]
assert chunks == ["one", "two", "three"]
Text.by_split_any
Text.by_split_any : func(text: Text, delimiters: Text = " $\t\r\n" -> func(->Text?))
Returns an iterator function that can be used to iterate over text separated by one or more characters (grapheme clusters) from a given text of delimiters.
Splitting will occur on every place where one or more of the grapheme clusters in delimiters occurs.
To split based on an exact delimiter, use Text.by_split().
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be iterated over in delimited chunks. | - |
| delimiters | Text |
Grapheme clusters to use for splitting the text. | " $\t\r\n" |
Return: An iterator function that returns one chunk of text at a time, separated by the given delimiter characters, until it runs out and returns none.
Example:
text := "one,two,;,three"
chunks := [chunk for chunk in text.by_split_any(",;")]
assert chunks == ["one", "two", "three"]
Text.caseless_equals
Text.caseless_equals : func(a: Text, b: Text, language: Text = "C" -> Bool)
Checks whether two texts are equal, ignoring the casing of the letters (i.e. case-insensitive comparison).
| Argument | Type | Description | Default |
|---|---|---|---|
| a | Text |
The first text to compare case-insensitively. | - |
| b | Text |
The second text to compare case-insensitively. | - |
| language | Text |
The ISO 639 language code for which casing rules to use. | "C" |
Return: yes if a and b are equal to each other, ignoring casing, otherwise no.
Example:
assert "A".caseless_equals("a") == yes
# Turkish lowercase "I" is "ı" (dotless I), not "i"
assert "I".caseless_equals("i", language="tr_TR") == no
Text.codepoint_names
Text.codepoint_names : func(text: Text -> [Text])
Returns a list of the names of each codepoint in the text.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text from which to extract codepoint names. | - |
Return: A list of codepoint names ([Text]).
Example:
assert "Amélie".codepoint_names() == [
"LATIN CAPITAL LETTER A",
"LATIN SMALL LETTER M",
"LATIN SMALL LETTER E WITH ACUTE",
"LATIN SMALL LETTER L",
"LATIN SMALL LETTER I",
"LATIN SMALL LETTER E",
]
Text.distance
Text.distance : func(a: Text, b: Text, language: Text = "C" -> Num)
Get an approximate distance between two texts, such that when the distance is small, the texts are similar and when the distance is large, the texts are dissimilar.
The exact distance algorithm is not specified and may be subject to change over time.
| Argument | Type | Description | Default |
|---|---|---|---|
| a | Text |
The first text to compare. | - |
| b | Text |
The second text to compare. | - |
| language | Text |
The ISO 639 language code for which character width to use. | "C" |
Return: The distance between the two texts (larger means more dissimilar).
Example:
assert "hello".distance("hello") == 0
texts := &["goodbye", "hello", "hallo"]
texts.sort(func(a,b:&Text) a.distance("hello") <> b.distance("hello"))
assert texts == ["hello", "hallo", "goodbye"]
Text.ends_with
Text.ends_with : func(text: Text, suffix: Text, remainder: &Text? = none -> Bool)
Checks if the Text ends with a literal suffix text.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be searched. | - |
| suffix | Text |
The literal suffix text to check for. | - |
| remainder | &Text? |
If non-none, this value will be set to the rest of the text up to the trailing suffix. If the suffix is not found, this value will be set to the original text. | none |
Return: yes if the text has the target, no otherwise.
Example:
assert "hello world".ends_with("world") == yes
remainder : Text
assert "hello world".ends_with("world", &remainder) == yes
assert remainder == "hello "
Text.find
Text.find : func(text: Text, target: Text, start: Int = 1 -> Int)
Find a substring within a text and return its index, if found.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be searched. | - |
| target | Text |
The target text to find. | - |
| start | Int |
The index at which to begin searching. | 1 |
Return: The index where the first occurrence of target appears, or none if it is not found.
Example:
assert "one two".find("one") == 1
assert "one two".find("two") == 5
assert "one two".find("three") == none
assert "one two".find("o", start=2) == 7
Text.from
Text.from : func(text: Text, first: Int -> Text)
Get a slice of the text, starting at the given position.
A negative index counts backwards from the end of the text, so -1 refers to the last cluster, -2 the second-to-last, etc. Slice ranges will be truncated to the length of the text.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be sliced. | - |
| first | Int |
The index to begin the slice. | - |
Return: The text from the given grapheme cluster to the end of the text.
Example:
assert "hello".from(2) == "ello"
assert "hello".from(-2) == "lo"
Text.from_c_string
Text.from_c_string : func(str: CString -> Text)
Converts a C-style string to a Text value.
| Argument | Type | Description | Default |
|---|---|---|---|
| str | CString |
The C-style string to be converted. | - |
Return: A Text value representing the C-style string.
Example:
assert Text.from_c_string(CString("Hello")) == "Hello"
Text.from_codepoint_names
Text.from_codepoint_names : func(codepoint_names: [Text] -> [Text])
Returns text that has the given codepoint names (according to the Unicode specification) as its codepoints.
The text will be normalized, so the resulting text's codepoints may not exactly match the input codepoints.
| Argument | Type | Description | Default |
|---|---|---|---|
| codepoint_names | [Text] |
The names of each codepoint in the desired text (case-insentive). | - |
Return: A new text with the specified codepoints after normalization has been applied. Any invalid names are ignored.
Example:
text := Text.from_codepoint_names([
"LATIN CAPITAL LETTER A WITH RING ABOVE",
"LATIN SMALL LETTER K",
"LATIN SMALL LETTER E",
])
assert text == "Åke"
Text.from_utf16
Text.from_utf16 : func(bytes: [Int16] -> [Text])
Returns text that has been constructed from the given UTF16 sequence.
The text will be normalized, so the resulting text's UTF16 sequence may not exactly match the input.
| Argument | Type | Description | Default |
|---|---|---|---|
| bytes | [Int16] |
The UTF-16 integers of the desired text. | - |
Return: A new text based on the input UTF16 sequence after normalization has been applied.
Example:
assert Text.from_utf16([197, 107, 101]) == "Åke"
assert Text.from_utf16([12371, 12435, 12395, 12385, 12399, 19990, 30028]) == "こんにちは世界"
Text.from_utf32
Text.from_utf32 : func(codepoints: [Int32] -> [Text])
Returns text that has been constructed from the given UTF32 codepoints.
The text will be normalized, so the resulting text's codepoints may not exactly match the input codepoints.
| Argument | Type | Description | Default |
|---|---|---|---|
| codepoints | [Int32] |
The UTF32 codepoints in the desired text. | - |
Return: A new text with the specified codepoints after normalization has been applied.
Example:
assert Text.from_utf32([197, 107, 101]) == "Åke"
Text.from_utf8
Text.from_utf8 : func(bytes: [Byte] -> [Text])
Returns text that has been constructed from the given UTF8 bytes.
The text will be normalized, so the resulting text's UTF8 bytes may not exactly match the input.
| Argument | Type | Description | Default |
|---|---|---|---|
| bytes | [Byte] |
The UTF-8 bytes of the desired text. | - |
Return: A new text based on the input UTF8 bytes after normalization has been applied.
Example:
assert Text.from_utf8([195, 133, 107, 101]) == "Åke"
Text.has
Text.has : func(text: Text, target: Text -> Bool)
Checks if the Text contains some target text.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be searched. | - |
| target | Text |
The text to search for. | - |
Return: yes if the target text is found, no otherwise.
Example:
assert "hello world".has("wo") == yes
assert "hello world".has("xxx") == no
Text.join
Text.join : func(glue: Text, pieces: [Text] -> Text)
Joins a list of text pieces with a specified glue.
| Argument | Type | Description | Default |
|---|---|---|---|
| glue | Text |
The text used to join the pieces. | - |
| pieces | [Text] |
The list of text pieces to be joined. | - |
Return: A single Text value with the pieces joined by the glue.
Example:
assert ", ".join(["one", "two", "three"]) == "one, two, three"
Text.left_pad
Text.left_pad : func(text: Text, width: Int, pad: Text = " ", language: Text = "C" -> Text)
Pad some text on the left side so it reaches a target width.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to pad. | - |
| width | Int |
The target width. | - |
| pad | Text |
The padding text. | " " |
| language | Text |
The ISO 639 language code for which character width to use. | "C" |
Return: Text with length at least width, with extra padding on the left as needed. If pad has length greater than 1, it may be partially repeated to reach the exact desired length.
Example:
assert "x".left_pad(5) == " x"
assert "x".left_pad(5, "ABC") == "ABCAx"
Text.lines
Text.lines : func(text: Text -> [Text])
Splits the text into a list of lines of text, preserving blank lines, ignoring trailing newlines, and handling \r\n the same as \n.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be split into lines. | - |
Return: A list of substrings resulting from the split.
Example:
assert "one\ntwo\nthree".lines() == ["one", "two", "three"]
assert "one\ntwo\nthree\n".lines() == ["one", "two", "three"]
assert "one\ntwo\nthree\n\n".lines() == ["one", "two", "three", ""]
assert "one\r\ntwo\r\nthree\r\n".lines() == ["one", "two", "three"]
assert "".lines() == []
Text.lower
Text.lower : func(text: Text, language: Text = "C" -> Text)
Converts all characters in the text to lowercase.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be converted to lowercase. | - |
| language | Text |
The ISO 639 language code for which casing rules to use. | "C" |
Return: The lowercase version of the text.
Example:
assert "AMÉLIE".lower() == "amélie"
assert "I".lower(language="tr_TR") == "ı"
Text.matches_glob
Text.matches_glob : func(path: Text, glob: Text -> Bool)
Return whether or not the text matches the given glob.
| Argument | Type | Description | Default |
|---|---|---|---|
| path | Text |
The text to check. | - |
| glob | Text |
The glob pattern to check. | - |
Return: Whether or not the text matches the given glob.
Example:
assert "hello world".matches_glob("h* *d")
Text.middle_pad
Text.middle_pad : func(text: Text, width: Int, pad: Text = " ", language: Text = "C" -> Text)
Pad some text on the left and right side so it reaches a target width.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to pad. | - |
| width | Int |
The target width. | - |
| pad | Text |
The padding text. | " " |
| language | Text |
The ISO 639 language code for which character width to use. | "C" |
Return: Text with length at least width, with extra padding on the left and right as needed. If pad has length greater than 1, it may be partially repeated to reach the exact desired length.
Example:
assert "x".middle_pad(6) == " x "
assert "x".middle_pad(10, "ABC") == "ABCAxABCAB"
Text.quoted
Text.quoted : func(text: Text, color: Bool = no, quotation_mark: Text = `"` -> Text)
Formats the text with quotation marks and escapes.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be quoted. | - |
| color | Bool |
Whether to add color formatting. | no |
| quotation_mark | Text |
The quotation mark to use. | " |
Return: The text formatted as a quoted text.
Example:
assert "one\ntwo".quoted() == "\"one\\ntwo\""
Text.repeat
Text.repeat : func(text: Text, count: Int -> Text)
Repeat some text multiple times.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to repeat. | - |
| count | Int |
The number of times to repeat it. (Negative numbers are equivalent to zero). | - |
Return: The text repeated the given number of times.
Example:
assert "Abc".repeat(3) == "AbcAbcAbc"
Text.replace
Text.replace : func(text: Text, target: Text, replacement: Text -> Text)
Replaces occurrences of a target text with a replacement text.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text in which to perform replacements. | - |
| target | Text |
The target text to be replaced. | - |
| replacement | Text |
The text to replace the target with. | - |
Return: The text with occurrences of the target replaced.
Example:
assert "Hello world".replace("world", "there") == "Hello there"
Text.reversed
Text.reversed : func(text: Text -> Text)
Return a text that has the grapheme clusters in reverse order.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to reverse. | - |
Return: A reversed version of the text.
Example:
assert "Abc".reversed() == "cbA"
Text.right_pad
Text.right_pad : func(text: Text, width: Int, pad: Text = " ", language: Text = "C" -> Text)
Pad some text on the right side so it reaches a target width.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to pad. | - |
| width | Int |
The target width. | - |
| pad | Text |
The padding text. | " " |
| language | Text |
The ISO 639 language code for which character width to use. | "C" |
Return: Text with length at least width, with extra padding on the right as needed. If pad has length greater than 1, it may be partially repeated to reach the exact desired length.
Example:
assert "x".right_pad(5) == "x "
assert "x".right_pad(5, "ABC") == "xABCA"
Text.slice
Text.slice : func(text: Text, from: Int = 1, to: Int = -1 -> Text)
Get a slice of the text.
A negative index counts backwards from the end of the text, so -1 refers to the last cluster, -2 the second-to-last, etc. Slice ranges will be truncated to the length of the text.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be sliced. | - |
| from | Int |
The index of the first grapheme cluster to include (1-indexed). | 1 |
| to | Int |
The index of the last grapheme cluster to include (1-indexed). | -1 |
Return: The text that spans the given grapheme cluster indices.
Example:
assert "hello".slice(2, 3) == "el"
assert "hello".slice(to=-2) == "hell"
assert "hello".slice(from=2) == "ello"
Text.split
Text.split : func(text: Text, delimiter: Text = "" -> [Text])
Splits the text into a list of substrings based on exact matches of a delimiter.
To split based on a set of delimiters, use Text.split_any(). If an empty text is given as the delimiter, then each split will be the graphical clusters of the text.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be split. | - |
| delimiter | Text |
The delimiter used to split the text. | "" |
Return: A list of subtexts resulting from the split.
Example:
assert "one,two,,three".split(",") == ["one", "two", "", "three"]
assert "abc".split() == ["a", "b", "c"]
Text.split_any
Text.split_any : func(text: Text, delimiters: Text = " $\t\r\n" -> [Text])
Splits the text into a list of substrings at one or more occurrences of a set of delimiter characters (grapheme clusters).
Splitting will occur on every place where one or more of the grapheme clusters in delimiters occurs.
To split based on an exact delimiter, use Text.split().
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be split. | - |
| delimiters | Text |
A text containing delimiters to use for splitting the text. | " $\t\r\n" |
Return: A list of subtexts resulting from the split.
Example:
assert "one, two,,three".split_any(", ") == ["one", "two", "three"]
Text.starts_with
Text.starts_with : func(text: Text, prefix: Text, remainder: &Text? = none -> Bool)
Checks if the Text starts with a literal prefix text.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be searched. | - |
| prefix | Text |
The literal prefix text to check for. | - |
| remainder | &Text? |
If non-none, this value will be set to the rest of the text after the prefix. If the prefix is not found, this value will be set to the original text. | none |
Return: yes if the text has the given prefix, no otherwise.
Example:
assert "hello world".starts_with("hello") == yes
remainder : Text
assert "hello world".starts_with("hello", &remainder) == yes
assert remainder == " world"
Text.title
Text.title : func(text: Text, language: Text = "C" -> Text)
Converts the text to title case (capitalizing the first letter of each word).
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be converted to title case. | - |
| language | Text |
The ISO 639 language code for which casing rules to use. | "C" |
Return: The text in title case.
Example:
assert "amélie".title() == "Amélie"
# In Turkish, uppercase "i" is "İ"
assert "i".title(language="tr_TR") == "İ"
Text.to
Text.to : func(text: Text, last: Int -> Text)
Get a slice of the text, ending at the given position.
A negative index counts backwards from the end of the text, so -1 refers to the last cluster, -2 the second-to-last, etc. Slice ranges will be truncated to the length of the text.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be sliced. | - |
| last | Int |
The index of the last grapheme cluster to include (1-indexed). | - |
Return: The text up to and including the given grapheme cluster.
Example:
assert "goodbye".to(3) == "goo"
assert "goodbye".to(-2) == "goodby"
Text.translate
Text.translate : func(text: Text, translations: {Text:Text} -> Text)
Takes a table mapping target texts to their replacements and performs all the replacements in the table on the whole text. At each position, the first matching replacement is applied and the matching moves on to after the replacement text, so replacement text is not recursively modified. See Text.replace() for more information about replacement behavior.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be translated. | - |
| translations | {Text:Text} |
A table mapping from target text to its replacement. | - |
Return: The text with all occurrences of the targets replaced with their corresponding replacement text.
Example:
text := "A <tag> & an ampersand".translate({
"&": "&",
"<": "<",
">": ">",
'"': """,
"'": "'",
})
assert text == "A <tag> & an ampersand"
Text.trim
Text.trim : func(text: Text, to_trim: Text = " $\t\r\n", left: Bool = yes, right: Bool = yes -> Text)
Trims the given characters (grapheme clusters) from the left and/or right side of the text.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be trimmed. | - |
| to_trim | Text |
The characters to remove from the left/right of the text. | " $\t\r\n" |
| left | Bool |
Whether or not to trim from the front of the text. | yes |
| right | Bool |
Whether or not to trim from the back of the text. | yes |
Return: The text without the trim characters at either end.
Example:
assert " x y z \n".trim() == "x y z"
assert "one,".trim(",") == "one"
assert " xyz ".trim(right=no) == "xyz "
Text.upper
Text.upper : func(text: Text, language: Text = "C" -> Text)
Converts all characters in the text to uppercase.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be converted to uppercase. | - |
| language | Text |
The ISO 639 language code for which casing rules to use. | "C" |
Return: The uppercase version of the text.
Example:
assert "amélie".upper() == "AMÉLIE"
# In Turkish, uppercase "i" is "İ"
assert "i".upper(language="tr_TR") == "İ"
Text.utf16
Text.utf16 : func(text: Text -> [Int16])
Returns a list of Unicode code points for UTF16 encoding of the text.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text from which to extract Unicode code points. | - |
Return: A list of 16-bit integer Unicode code points ([Int16]).
Example:
assert "Åke".utf16() == [197, 107, 101]
assert "こんにちは世界".utf16() == [12371, 12435, 12395, 12385, 12399, 19990, 30028]
Text.utf32
Text.utf32 : func(text: Text -> [Int32])
Returns a list of Unicode code points for UTF32 encoding of the text.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text from which to extract Unicode code points. | - |
Return: A list of 32-bit integer Unicode code points ([Int32]).
Example:
assert "Amélie".utf32() == [65, 109, 233, 108, 105, 101]
Text.utf8
Text.utf8 : func(text: Text -> [Byte])
Converts a Text value to a list of bytes representing a UTF8 encoding of the text.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to be converted to UTF8 bytes. | - |
Return: A list of bytes ([Byte]) representing the text in UTF8 encoding.
Example:
assert "Amélie".utf8() == [65, 109, 195, 169, 108, 105, 101]
Text.width
Text.width : func(text: Text -> Int)
Returns the display width of the text as seen in a terminal with appropriate font rendering. This is usually the same as the text's .length, but there are some characters like emojis that render wider than 1 cell.
This will not always be exactly accurate when your terminal's font rendering can't handle some unicode displaying correctly.
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text whose length you want. | - |
Return: An integer representing the display width of the text.
Example:
assert "Amélie".width() == 6
assert "🤠".width() == 2
Text.without_prefix
Text.without_prefix : func(text: Text, prefix: Text -> Text)
Returns the text with a given prefix removed (if present).
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to remove the prefix from. | - |
| prefix | Text |
The prefix to remove. | - |
Return: A text without the given prefix (if present) or the unmodified text if the prefix is not present.
Example:
assert "foo:baz".without_prefix("foo:") == "baz"
assert "qux".without_prefix("foo:") == "qux"
Text.without_suffix
Text.without_suffix : func(text: Text, suffix: Text -> Text)
Returns the text with a given suffix removed (if present).
| Argument | Type | Description | Default |
|---|---|---|---|
| text | Text |
The text to remove the suffix from. | - |
| suffix | Text |
The suffix to remove. | - |
Return: A text without the given suffix (if present) or the unmodified text if the suffix is not present.
Example:
assert "baz.foo".without_suffix(".foo") == "baz"
assert "qux".without_suffix(".foo") == "qux"
1 % API3 # Builtins5 # Text6 ## Text.as_c_string9 Text.as_c_string : func(text: Text -> CString)10 ```14 Argument | Type | Description | Default15 ---------|------|-------------|---------21 **Example:**23 assert "Hello".as_c_string() == CString("Hello")25 ```26 ## Text.at29 Text.at : func(text: Text, index: Int -> Text)30 ```32 Get the graphical cluster at a given index. This is similar to `str[i]` with ASCII text, but has more correct behavior for unicode text.34 Negative indices are counted from the back of the text, so `-1` means the last cluster, `-2` means the second-to-last, and so on.36 Argument | Type | Description | Default37 ---------|------|-------------|---------44 **Example:**46 assert "Amélie".at(3) == "é"48 ```49 ## Text.by_line52 Text.by_line : func(text: Text -> func(->Text?))53 ```55 Returns an iterator function that can be used to iterate over the lines in a text.57 This function ignores a trailing newline if there is one. If you don't want this behavior, use `text.by_split($/{1 nl}/)` instead.59 Argument | Type | Description | Default60 ---------|------|-------------|---------63 **Return:** An iterator function that returns one line at a time, until it runs out and returns `none`.66 **Example:**68 text := "69 line one70 line two71 "72 lines := [line for line in text.by_line()]73 assert lines == ["line one", "line two"]75 ```76 ## Text.by_split79 Text.by_split : func(text: Text, delimiter: Text = "" -> func(->Text?))80 ```82 Returns an iterator function that can be used to iterate over text separated by a delimiter.84 To split based on a set of delimiters, use Text.by_split_any().85 If an empty text is given as the delimiter, then each split will be the graphical clusters of the text.87 Argument | Type | Description | Default88 ---------|------|-------------|---------92 **Return:** An iterator function that returns one chunk of text at a time, separated by the given delimiter, until it runs out and returns `none`.95 **Example:**97 text := "one,two,three"98 chunks := [chunk for chunk in text.by_split(",")]99 assert chunks == ["one", "two", "three"]101 ```102 ## Text.by_split_any105 Text.by_split_any : func(text: Text, delimiters: Text = " $\t\r\n" -> func(->Text?))106 ```108 Returns an iterator function that can be used to iterate over text separated by one or more characters (grapheme clusters) from a given text of delimiters.110 Splitting will occur on every place where one or more of the grapheme clusters in `delimiters` occurs.111 To split based on an exact delimiter, use Text.by_split().113 Argument | Type | Description | Default114 ---------|------|-------------|---------118 **Return:** An iterator function that returns one chunk of text at a time, separated by the given delimiter characters, until it runs out and returns `none`.121 **Example:**123 text := "one,two,;,three"124 chunks := [chunk for chunk in text.by_split_any(",;")]125 assert chunks == ["one", "two", "three"]127 ```128 ## Text.caseless_equals131 Text.caseless_equals : func(a: Text, b: Text, language: Text = "C" -> Bool)132 ```134 Checks whether two texts are equal, ignoring the casing of the letters (i.e. case-insensitive comparison).136 Argument | Type | Description | Default137 ---------|------|-------------|---------145 **Example:**147 assert "A".caseless_equals("a") == yes149 # Turkish lowercase "I" is "ı" (dotless I), not "i"150 assert "I".caseless_equals("i", language="tr_TR") == no152 ```153 ## Text.codepoint_names156 Text.codepoint_names : func(text: Text -> [Text])157 ```159 Returns a list of the names of each codepoint in the text.161 Argument | Type | Description | Default162 ---------|------|-------------|---------168 **Example:**170 assert "Amélie".codepoint_names() == [171 "LATIN CAPITAL LETTER A",172 "LATIN SMALL LETTER M",173 "LATIN SMALL LETTER E WITH ACUTE",174 "LATIN SMALL LETTER L",175 "LATIN SMALL LETTER I",176 "LATIN SMALL LETTER E",177 ]179 ```180 ## Text.distance183 Text.distance : func(a: Text, b: Text, language: Text = "C" -> Num)184 ```186 Get an approximate distance between two texts, such that when the distance is small, the texts are similar and when the distance is large, the texts are dissimilar.188 The exact distance algorithm is not specified and may be subject to change over time.190 Argument | Type | Description | Default191 ---------|------|-------------|---------199 **Example:**201 assert "hello".distance("hello") == 0202 texts := &["goodbye", "hello", "hallo"]203 texts.sort(func(a,b:&Text) a.distance("hello") <> b.distance("hello"))204 assert texts == ["hello", "hallo", "goodbye"]206 ```207 ## Text.ends_with210 Text.ends_with : func(text: Text, suffix: Text, remainder: &Text? = none -> Bool)211 ```215 Argument | Type | Description | Default216 ---------|------|-------------|---------219 remainder | `&Text?` | If non-none, this value will be set to the rest of the text up to the trailing suffix. If the suffix is not found, this value will be set to the original text. | `none`224 **Example:**226 assert "hello world".ends_with("world") == yes227 remainder : Text228 assert "hello world".ends_with("world", &remainder) == yes229 assert remainder == "hello "231 ```232 ## Text.find235 Text.find : func(text: Text, target: Text, start: Int = 1 -> Int)236 ```238 Find a substring within a text and return its index, if found.240 Argument | Type | Description | Default241 ---------|------|-------------|---------246 **Return:** The index where the first occurrence of `target` appears, or `none` if it is not found.249 **Example:**251 assert "one two".find("one") == 1252 assert "one two".find("two") == 5253 assert "one two".find("three") == none254 assert "one two".find("o", start=2) == 7256 ```257 ## Text.from260 Text.from : func(text: Text, first: Int -> Text)261 ```263 Get a slice of the text, starting at the given position.265 A negative index counts backwards from the end of the text, so `-1` refers to the last cluster, `-2` the second-to-last, etc. Slice ranges will be truncated to the length of the text.267 Argument | Type | Description | Default268 ---------|------|-------------|---------275 **Example:**277 assert "hello".from(2) == "ello"278 assert "hello".from(-2) == "lo"280 ```281 ## Text.from_c_string284 Text.from_c_string : func(str: CString -> Text)285 ```289 Argument | Type | Description | Default290 ---------|------|-------------|---------296 **Example:**298 assert Text.from_c_string(CString("Hello")) == "Hello"300 ```301 ## Text.from_codepoint_names304 Text.from_codepoint_names : func(codepoint_names: [Text] -> [Text])305 ```307 Returns text that has the given codepoint names (according to the Unicode specification) as its codepoints.309 The text will be normalized, so the resulting text's codepoints may not exactly match the input codepoints.311 Argument | Type | Description | Default312 ---------|------|-------------|---------313 codepoint_names | `[Text]` | The names of each codepoint in the desired text (case-insentive). | -315 **Return:** A new text with the specified codepoints after normalization has been applied. Any invalid names are ignored.318 **Example:**320 text := Text.from_codepoint_names([321 "LATIN CAPITAL LETTER A WITH RING ABOVE",322 "LATIN SMALL LETTER K",323 "LATIN SMALL LETTER E",324 ])325 assert text == "Åke"327 ```328 ## Text.from_utf16331 Text.from_utf16 : func(bytes: [Int16] -> [Text])332 ```334 Returns text that has been constructed from the given UTF16 sequence.336 The text will be normalized, so the resulting text's UTF16 sequence may not exactly match the input.338 Argument | Type | Description | Default339 ---------|------|-------------|---------345 **Example:**347 assert Text.from_utf16([197, 107, 101]) == "Åke"348 assert Text.from_utf16([12371, 12435, 12395, 12385, 12399, 19990, 30028]) == "こんにちは世界"350 ```351 ## Text.from_utf32354 Text.from_utf32 : func(codepoints: [Int32] -> [Text])355 ```357 Returns text that has been constructed from the given UTF32 codepoints.359 The text will be normalized, so the resulting text's codepoints may not exactly match the input codepoints.361 Argument | Type | Description | Default362 ---------|------|-------------|---------368 **Example:**370 assert Text.from_utf32([197, 107, 101]) == "Åke"372 ```373 ## Text.from_utf8376 Text.from_utf8 : func(bytes: [Byte] -> [Text])377 ```379 Returns text that has been constructed from the given UTF8 bytes.381 The text will be normalized, so the resulting text's UTF8 bytes may not exactly match the input.383 Argument | Type | Description | Default384 ---------|------|-------------|---------390 **Example:**392 assert Text.from_utf8([195, 133, 107, 101]) == "Åke"394 ```395 ## Text.has398 Text.has : func(text: Text, target: Text -> Bool)399 ```403 Argument | Type | Description | Default404 ---------|------|-------------|---------411 **Example:**413 assert "hello world".has("wo") == yes414 assert "hello world".has("xxx") == no416 ```417 ## Text.join420 Text.join : func(glue: Text, pieces: [Text] -> Text)421 ```423 Joins a list of text pieces with a specified glue.425 Argument | Type | Description | Default426 ---------|------|-------------|---------433 **Example:**435 assert ", ".join(["one", "two", "three"]) == "one, two, three"437 ```438 ## Text.left_pad441 Text.left_pad : func(text: Text, width: Int, pad: Text = " ", language: Text = "C" -> Text)442 ```444 Pad some text on the left side so it reaches a target width.446 Argument | Type | Description | Default447 ---------|------|-------------|---------453 **Return:** Text with length at least `width`, with extra padding on the left as needed. If `pad` has length greater than 1, it may be partially repeated to reach the exact desired length.456 **Example:**458 assert "x".left_pad(5) == " x"459 assert "x".left_pad(5, "ABC") == "ABCAx"461 ```462 ## Text.lines465 Text.lines : func(text: Text -> [Text])466 ```468 Splits the text into a list of lines of text, preserving blank lines, ignoring trailing newlines, and handling `\r\n` the same as `\n`.470 Argument | Type | Description | Default471 ---------|------|-------------|---------477 **Example:**479 assert "one\ntwo\nthree".lines() == ["one", "two", "three"]480 assert "one\ntwo\nthree\n".lines() == ["one", "two", "three"]481 assert "one\ntwo\nthree\n\n".lines() == ["one", "two", "three", ""]482 assert "one\r\ntwo\r\nthree\r\n".lines() == ["one", "two", "three"]483 assert "".lines() == []485 ```486 ## Text.lower489 Text.lower : func(text: Text, language: Text = "C" -> Text)490 ```492 Converts all characters in the text to lowercase.494 Argument | Type | Description | Default495 ---------|------|-------------|---------502 **Example:**504 assert "AMÉLIE".lower() == "amélie"505 assert "I".lower(language="tr_TR") == "ı"507 ```508 ## Text.matches_glob511 Text.matches_glob : func(path: Text, glob: Text -> Bool)512 ```514 Return whether or not the text matches the given glob.516 Argument | Type | Description | Default517 ---------|------|-------------|---------524 **Example:**526 assert "hello world".matches_glob("h* *d")528 ```529 ## Text.middle_pad532 Text.middle_pad : func(text: Text, width: Int, pad: Text = " ", language: Text = "C" -> Text)533 ```535 Pad some text on the left and right side so it reaches a target width.537 Argument | Type | Description | Default538 ---------|------|-------------|---------544 **Return:** Text with length at least `width`, with extra padding on the left and right as needed. If `pad` has length greater than 1, it may be partially repeated to reach the exact desired length.547 **Example:**549 assert "x".middle_pad(6) == " x "550 assert "x".middle_pad(10, "ABC") == "ABCAxABCAB"552 ```553 ## Text.quoted556 Text.quoted : func(text: Text, color: Bool = no, quotation_mark: Text = `"` -> Text)557 ```559 Formats the text with quotation marks and escapes.561 Argument | Type | Description | Default562 ---------|------|-------------|---------570 **Example:**572 assert "one\ntwo".quoted() == "\"one\\ntwo\""574 ```575 ## Text.repeat578 Text.repeat : func(text: Text, count: Int -> Text)579 ```581 Repeat some text multiple times.583 Argument | Type | Description | Default584 ---------|------|-------------|---------586 count | `Int` | The number of times to repeat it. (Negative numbers are equivalent to zero). | -591 **Example:**593 assert "Abc".repeat(3) == "AbcAbcAbc"595 ```596 ## Text.replace599 Text.replace : func(text: Text, target: Text, replacement: Text -> Text)600 ```602 Replaces occurrences of a target text with a replacement text.604 Argument | Type | Description | Default605 ---------|------|-------------|---------613 **Example:**615 assert "Hello world".replace("world", "there") == "Hello there"617 ```618 ## Text.reversed621 Text.reversed : func(text: Text -> Text)622 ```624 Return a text that has the grapheme clusters in reverse order.626 Argument | Type | Description | Default627 ---------|------|-------------|---------633 **Example:**635 assert "Abc".reversed() == "cbA"637 ```638 ## Text.right_pad641 Text.right_pad : func(text: Text, width: Int, pad: Text = " ", language: Text = "C" -> Text)642 ```644 Pad some text on the right side so it reaches a target width.646 Argument | Type | Description | Default647 ---------|------|-------------|---------653 **Return:** Text with length at least `width`, with extra padding on the right as needed. If `pad` has length greater than 1, it may be partially repeated to reach the exact desired length.656 **Example:**658 assert "x".right_pad(5) == "x "659 assert "x".right_pad(5, "ABC") == "xABCA"661 ```662 ## Text.slice665 Text.slice : func(text: Text, from: Int = 1, to: Int = -1 -> Text)666 ```668 Get a slice of the text.670 A negative index counts backwards from the end of the text, so `-1` refers to the last cluster, `-2` the second-to-last, etc. Slice ranges will be truncated to the length of the text.672 Argument | Type | Description | Default673 ---------|------|-------------|---------681 **Example:**683 assert "hello".slice(2, 3) == "el"684 assert "hello".slice(to=-2) == "hell"685 assert "hello".slice(from=2) == "ello"687 ```688 ## Text.split691 Text.split : func(text: Text, delimiter: Text = "" -> [Text])692 ```694 Splits the text into a list of substrings based on exact matches of a delimiter.696 To split based on a set of delimiters, use Text.split_any().697 If an empty text is given as the delimiter, then each split will be the graphical clusters of the text.699 Argument | Type | Description | Default700 ---------|------|-------------|---------707 **Example:**709 assert "one,two,,three".split(",") == ["one", "two", "", "three"]710 assert "abc".split() == ["a", "b", "c"]712 ```713 ## Text.split_any716 Text.split_any : func(text: Text, delimiters: Text = " $\t\r\n" -> [Text])717 ```719 Splits the text into a list of substrings at one or more occurrences of a set of delimiter characters (grapheme clusters).721 Splitting will occur on every place where one or more of the grapheme clusters in `delimiters` occurs.722 To split based on an exact delimiter, use Text.split().724 Argument | Type | Description | Default725 ---------|------|-------------|---------727 delimiters | `Text` | A text containing delimiters to use for splitting the text. | `" $\t\r\n"`732 **Example:**734 assert "one, two,,three".split_any(", ") == ["one", "two", "three"]736 ```737 ## Text.starts_with740 Text.starts_with : func(text: Text, prefix: Text, remainder: &Text? = none -> Bool)741 ```745 Argument | Type | Description | Default746 ---------|------|-------------|---------749 remainder | `&Text?` | If non-none, this value will be set to the rest of the text after the prefix. If the prefix is not found, this value will be set to the original text. | `none`754 **Example:**756 assert "hello world".starts_with("hello") == yes757 remainder : Text758 assert "hello world".starts_with("hello", &remainder) == yes759 assert remainder == " world"761 ```762 ## Text.title765 Text.title : func(text: Text, language: Text = "C" -> Text)766 ```768 Converts the text to title case (capitalizing the first letter of each word).770 Argument | Type | Description | Default771 ---------|------|-------------|---------778 **Example:**780 assert "amélie".title() == "Amélie"782 # In Turkish, uppercase "i" is "İ"783 assert "i".title(language="tr_TR") == "İ"785 ```786 ## Text.to789 Text.to : func(text: Text, last: Int -> Text)790 ```792 Get a slice of the text, ending at the given position.794 A negative index counts backwards from the end of the text, so `-1` refers to the last cluster, `-2` the second-to-last, etc. Slice ranges will be truncated to the length of the text.796 Argument | Type | Description | Default797 ---------|------|-------------|---------804 **Example:**806 assert "goodbye".to(3) == "goo"807 assert "goodbye".to(-2) == "goodby"809 ```810 ## Text.translate813 Text.translate : func(text: Text, translations: {Text:Text} -> Text)814 ```816 Takes a table mapping target texts to their replacements and performs all the replacements in the table on the whole text. At each position, the first matching replacement is applied and the matching moves on to *after* the replacement text, so replacement text is not recursively modified. See Text.replace() for more information about replacement behavior.818 Argument | Type | Description | Default819 ---------|------|-------------|---------823 **Return:** The text with all occurrences of the targets replaced with their corresponding replacement text.826 **Example:**828 text := "A <tag> & an ampersand".translate({829 "&": "&",830 "<": "<",831 ">": ">",832 '"': """,833 "'": "'",834 })835 assert text == "A <tag> & an ampersand"837 ```838 ## Text.trim841 Text.trim : func(text: Text, to_trim: Text = " $\t\r\n", left: Bool = yes, right: Bool = yes -> Text)842 ```844 Trims the given characters (grapheme clusters) from the left and/or right side of the text.846 Argument | Type | Description | Default847 ---------|------|-------------|---------856 **Example:**858 assert " x y z \n".trim() == "x y z"859 assert "one,".trim(",") == "one"860 assert " xyz ".trim(right=no) == "xyz "862 ```863 ## Text.upper866 Text.upper : func(text: Text, language: Text = "C" -> Text)867 ```869 Converts all characters in the text to uppercase.871 Argument | Type | Description | Default872 ---------|------|-------------|---------879 **Example:**881 assert "amélie".upper() == "AMÉLIE"883 # In Turkish, uppercase "i" is "İ"884 assert "i".upper(language="tr_TR") == "İ"886 ```887 ## Text.utf16890 Text.utf16 : func(text: Text -> [Int16])891 ```893 Returns a list of Unicode code points for UTF16 encoding of the text.895 Argument | Type | Description | Default896 ---------|------|-------------|---------902 **Example:**904 assert "Åke".utf16() == [197, 107, 101]905 assert "こんにちは世界".utf16() == [12371, 12435, 12395, 12385, 12399, 19990, 30028]907 ```908 ## Text.utf32911 Text.utf32 : func(text: Text -> [Int32])912 ```914 Returns a list of Unicode code points for UTF32 encoding of the text.916 Argument | Type | Description | Default917 ---------|------|-------------|---------923 **Example:**925 assert "Amélie".utf32() == [65, 109, 233, 108, 105, 101]927 ```928 ## Text.utf8931 Text.utf8 : func(text: Text -> [Byte])932 ```936 Argument | Type | Description | Default937 ---------|------|-------------|---------943 **Example:**945 assert "Amélie".utf8() == [65, 109, 195, 169, 108, 105, 101]947 ```948 ## Text.width951 Text.width : func(text: Text -> Int)952 ```954 Returns the display width of the text as seen in a terminal with appropriate font rendering. This is usually the same as the text's `.length`, but there are some characters like emojis that render wider than 1 cell.956 This will not always be exactly accurate when your terminal's font rendering can't handle some unicode displaying correctly.958 Argument | Type | Description | Default959 ---------|------|-------------|---------965 **Example:**967 assert "Amélie".width() == 6968 assert "🤠".width() == 2970 ```971 ## Text.without_prefix974 Text.without_prefix : func(text: Text, prefix: Text -> Text)975 ```977 Returns the text with a given prefix removed (if present).979 Argument | Type | Description | Default980 ---------|------|-------------|---------984 **Return:** A text without the given prefix (if present) or the unmodified text if the prefix is not present.987 **Example:**989 assert "foo:baz".without_prefix("foo:") == "baz"990 assert "qux".without_prefix("foo:") == "qux"992 ```993 ## Text.without_suffix996 Text.without_suffix : func(text: Text, suffix: Text -> Text)997 ```999 Returns the text with a given suffix removed (if present).1001 Argument | Type | Description | Default1002 ---------|------|-------------|---------1006 **Return:** A text without the given suffix (if present) or the unmodified text if the suffix is not present.1009 **Example:**1011 assert "baz.foo".without_suffix(".foo") == "baz"1012 assert "qux".without_suffix(".foo") == "qux"1014 ```