4.23. kink/STR¶
Companion mod for str vals.
4.23.1. type str¶
A str is a finite immutable sequence of Unicode codepoints. Codepoints are called runes in Kink, likewise in Go.
4.23.1.1. Str1.op_add(Str2)¶
`op_add` makes a str containing the runes of `Str1` on the front, and the runes of `Str2` on the back.
Precondition:
• `Str2` must be a str
Example:
:Str <- 'foo' + 'bar'
stdout.print_line(Str.repr) # => "foobar"
4.23.1.2. Str.runes¶
`runes` returns a vec of the runes of the Str.
Example:
stdout.print_line('xyz'.runes.repr) # => [120 121 122]
4.23.1.3. Str.empty?¶
`empty?` returns whether `Str` contains one or more runes.
4.23.1.4. Str.size¶
`size` returns the number of runes in `Str` as an int num.
4.23.1.5. Str.get(Rune_index)¶
`get` returns the rune specified by `Rune_index`.
The result is an int num which represents the rune.
Example:
stdout.print_line('xyz'.get(0).repr) # => 120
4.23.1.6. Str.slice(From_pos To_pos)¶
`slice` returns the specified slice of `Str`.
Preconditions:
• `From_pos` and `To_pos` must be int nums in the range [0, Str.size]
• `From_pos` must be less than or equal to `To_pos`
The result is a str whose runes are copied from the slice of `Str`.
Example:
:Str <- 'foobar'
:Slice <- Str.slice(1 4)
stdout.print_line(Slice.repr) # => "oob"
4.23.1.7. Str.take_front(N)¶
`take_front` makes a str containing the first `N` runes of `Str`.
Precondition:
• `N` must be an int num in the range [0, Str.size].
Example:
stdout.print_line('foobar'.take_front(2).repr) # => "fo"
4.23.1.8. Str.take_back(N)¶
`take_back` makes a str containing the last `N` runes of `Str`.
Precondition:
• `N` must be an int num in the range [0, Str.size].
Example:
stdout.print_line('foobar'.take_back(2).repr) # => "ar"
4.23.1.9. Str.drop_front(N)¶
`drop_front` makes a str dropping the first `N` runes of `Str`.
Precondition:
• `N` must be an int num in the range [0, Str.size].
Example:
stdout.print_line('foobar'.drop_front(2).repr) # => "obar"
4.23.1.10. Str.drop_back(N)¶
`drop_back` makes a str dropping the last `N` runes of `Str`.
Precondition:
• `N` must be an int num in the range [0, Str.size].
Example:
stdout.print_line('foobar'.drop_back(2).repr) # => "foob"
4.23.1.11. Str.search_slice(Min_ind Slice)¶
`search_slice` searches `Slice` in `Str`.
Preconditions:
• `Min_rune_ind` must be an int num in the range [0, Str.size]
• `Slice` must be a str
`search_slice` tries to find the smallest `Ind` which is >= `Min_ind`, such that `Str.slice(Ind Slice.size)` is equal to `Slice`.
If such `Ind` is found, `search_slice` returns a vec [Ind].
If not found, `search_slice` returns an empty vec [].
Example:
:attempt_search <- {(:Str :Min_ind :Slice)
Str.search_slice(Min_ind Slice).with_just_or(
{(:Ind) stdout.print_line('found at: {}'.format(Ind.repr)) }
{ stdout.print_line('not found') }
)
}
attempt_search('foofoo' 0 'oo') # => found at: 1
attempt_search('foofoo' 1 'oo') # => found at: 1
attempt_search('foofoo' 2 'oo') # => found at: 4
attempt_search('foofoo' 3 'oo') # => found at: 4
attempt_search('foofoo' 4 'oo') # => found at: 4
attempt_search('foofoo' 5 'oo') # => not found
attempt_search('foofoo' 6 'oo') # => not found
4.23.1.12. Str.have_prefix?(Prefix)¶
`have_prefix?` returns whether `Str` has the slice `Prefix` at the front.
Precondition:
• `Prefix` must be a str.
If `Str.slice(0 Prefix.size)` is equal to `Prefix`, `have_prefix?` returns true. If not, `have_prefix?` returns false.
4.23.1.13. Str.have_suffix?(Suffix)¶
`have_suffix?` returns whether `Str` has the slice `Suffix` at the back.
Precondition:
• `Suffix` must be a str.
If `Str.slice(Str.size - Suffix.size Str.size)` is equal to `Suffix`, `have_suffix?` returns true. If not, `have_suffix?` returns false.
4.23.1.14. Str.have_slice?(Slice)¶
`have_slice?` returns whether `Str` has `Slice`.
Precondition:
• `Slice` must be a str.
If there is `Ind` such that `Str.slice(Ind Ind+Slice.size)` is equal to `Slice`, `have_slice?` returns true. If not, `have_slice?` returns false.
4.23.1.15. Str.trim¶
`trim` returns a slice of `Str` trimming space-like characters from the front and the back.
Definition of space-like characters:
• Rune which is bigger than U+0021, and which is not U+007f
`trim` returns the longest slice of `Str` which does not start with a space-like character, and does not end with a space-like character.
Exaample:
stdout.print_line(" \t\r\nfoobar\n\r\t ".trim.repr) # => "foobar"
stdout.print_line(" \t\r\n\n\r\t ".trim.repr) # => ""
4.23.1.16. Str.trim_front¶
`trim_front` returns a slice of `Str` trimming space-like characters from the front.
See `trim` for the definition of space-like characters.
`trim_front` returns the longest slice of `Str` which does not start with a space-like character.
Exaample:
stdout.print_line(" \t\r\nfoobar\n\r\t ".trim_front.repr) # => "foobar\n\r\t "
stdout.print_line(" \t\r\n\n\r\t ".trim_front.repr) # => ""
4.23.1.17. Str.trim_back¶
`trim_back` returns a slice of `Str` trimming space-like characters from the back. See `trim` for the definition of space-like characters.
`trim_back` returns the longest slice of `Str` which does not end with a space-like character.
Exaample:
stdout.print_line(" \t\r\nfoobar\n\r\t ".trim_back.repr) # => " \t\r\nfoobar"
stdout.print_line(" \t\r\n\n\r\t ".trim_back.repr) # => ""
4.23.1.18. Str.ascii_upcase¶
`ascii_upcase` converts runes of ASCII lowercase letters (from U+0061='a' to U+007A='z') in `Str` to the corresponding ASCII uppercase letters (from U+0041='A' to U+005A='Z').
`ascii_upcase` does not change runes except for ASCII lowercase letters.
Example:
stdout.print_line('Azkaban'.ascii_upcase.repr) # => "AZKABAN"
stdout.print_line('+-*/%<>'.ascii_upcase.repr) # => "+-*/%<>"
stdout.print_line('Товарищ'.ascii_upcase.repr) # => "Товарищ"
4.23.1.19. Str.ascii_downcase¶
`ascii_downcase` converts runes of ASCII uppercase letters (from U+0041='A' to U+005A='Z') in `Str` to the corresponding ASCII lowercase letters (from U+0061='a' to U+007A='z').
`ascii_downcase` does not change runes except for ASCII uppercase letters.
Example:
stdout.print_line('aZKABAN'.ascii_downcase.repr) # => "azkaban"
stdout.print_line('+-*/%<>'.ascii_downcase.repr) # => "+-*/%<>"
stdout.print_line('Товарищ'.ascii_downcase.repr) # => "Товарищ"
4.23.1.20. Str.format(... Args) / Str.format($config)¶
`format` generates a str which is formatted using `Str` as a template.
`format` has two overloads.
Overload: Str.format(... Args)
When called with variadic args `Args`, Str.format uses each element of `Args` as a positional argument.
Preconditions:
• `Str` must be a str which suffices the syntax described below.
• Each element of `Args` must have `show` method.
Overload: Str.format($config)
When called with $config fun, `format` uses $config as a config fun.
Preconditions:
• `Str` must be a str which suffices the syntax described below.
• $config must be a fun which takes a format_config val.
The format of the template str
A template str can contain placeholders delimited by braces "{" and "}". Str.format invokes .show(...[$show_config]) method of the arg corresponding to each placeholder, and emplaces the result str in the placeholder.
There are following variants of placeholders.
• "{}": takes a positional argument, with an empty str "" as the spec.
• "{%xxx}": takes a positional argument, with the spec "xxx".
• "{Foo}": takes a named argument with the name "Foo", with an empty str "" as the spec.
• "{Foo%xxx}": takes a named argument with the name "Foo", with the spec "xxx".
The name str must not contain "{", "}" and "%".
The spec str is passed to Show_config.spec(Spec) method. The spec str must not contain "{" or "}".
If you want to include literal "{" and "}" in the template str, escape them as "{{" and "}}".
Examples
stdout.print_line('{} == 0x{%04x}'.format(42 42))
# => 42 == 0x002a
stdout.print_line('{}: message: {}'.format('WARN' 'something went wrong'))
# => WARN: message: something went wrong
With locale:
:LOCALE.require_from('kink/')
:Str <- 'Id: {%04d}, Name: {}, Elevation: {%,d} meters'.format{(:C)
C.locale(LOCALE.for('da-DK'))
C.arg(12)
C.arg('Mount Fuji')
C.arg(3776.24)
}
stdout.print_line(Str)
# => Id: 0012, Name: Mount Fuji, Elevation: 3.776,24 meters
Using named args:
:LOCALE.require_from('kink/')
:Str <- 'Id: {Id%04d}, Name: {Name}, Elevation: {Elevation%,d} meters'.format{(:C)
C.locale(LOCALE.for('da-DK'))
C.named_arg('Id' 12)
C.named_arg('Name' 'Mount Fuji')
C.named_arg('Elevation' 3776.24)
}
stdout.print_line(Str)
# => Id: 0012, Name: Mount Fuji, Elevation: 3.776,24 meters
4.23.1.21. Str.op_mul(N)¶
`op_mul` makes a str repeating `Str` `N` times.
Precondition:
• `N` must be a nonnegative int num.
Example:
stdout.print_line(('foo' * 0).repr) # => ""
stdout.print_line(('foo' * 1).repr) # => "foo"
stdout.print_line(('foo' * 2).repr) # => "foofoo"
stdout.print_line(('foo' * 3).repr) # => "foofoofoo"
4.23.1.22. Str1.op_eq(Str2)¶
`op_eq` returns whether two strs contains the same sequence of runes.
Precondition:
• `Str1` and `Str2` must be strs.
Example:
stdout.print_line(('foo' == 'foo').repr) # => true
stdout.print_line(('foo' != 'foo').repr) # => false
stdout.print_line(('foo' == 'FOO').repr) # => false
stdout.print_line(('foo' != 'FOO').repr) # => true
4.23.1.23. Str1.op_lt(Str2)¶
`op_lt` returns compares the runes of the strs lexicographically, then returns whether `Str1` is less than `Str2`.
Precondition:
• `Str1` and `Str2` must be strs.
Example:
:compare <- {(:Str1 :Str2)
[:Lt? :Le? :Gt? :Ge?] = [
Str1 < Str2
Str1 <= Str2
Str1 > Str2
Str1 >= Str2
]
stdout.print_line(
'Lt?={} Le?={} Gt?={} Ge?={}'.format(Lt?.repr Le?.repr Gt?.repr Ge?.repr))
}
compare('foo' 'foo') # => Lt?=false Le?=true Gt?=false Ge?=true
compare('foo' 'fooo') # => Lt?=true Le?=true Gt?=false Ge?=false
compare('foo' 'fo') # => Lt?=false Le?=false Gt?=true Ge?=true
Note that the result of comparison may differ from java.lang.String.compareTo. The reason is that the Java method compares strings by UTF-16 units.
:JAVA.require_from('kink/javahost/')
:X <- "\x{00ffff}"
:Y <- "\x{010302}"
stdout.print_line((X < Y).repr) # => true
stdout.print_line(
(JAVA.string(X).call_method('compareTo' JAVA.string(Y)).to_kink_num < 0).repr
) # => false
4.23.1.24. Str.show(...[$config={}])¶
`show` returns `Str` itself.
`show` can take an optional parameter $config, but it is never used. This parameter exists in order to meet the requirement by `format` method.
Example:
stdout.print_line('xyz'.show)
# Output:
# xyz
stdout.print_line('xyz'.show{})
# Output:
# xyz
stdout.print_line("foo\nbar".show)
# Output:
# foo
# bar
4.23.1.25. Str.repr¶
`repr` returns the str representation of the Str for debugging, such as "xyz" and "foo\nbar".
Example:
stdout.print_line('xyz'.repr) # => "xyz"
stdout.print_line("foo\nbar".repr) # => "foo\nbar"
4.23.2. type format_config¶
A format_config is a conf val of Str.format.
4.23.2.1. Format_config.arg(Pos_arg)¶
`arg` method adds a positional arg.
Precondition:
• `Pos_arg` must support .show(...[$show_config]) method
The number of invocations of `arg` method must be equal to the unnamed placeholders.
4.23.2.2. Format_config.named_arg(Name Arg)¶
`named_arg` method adds a named arg.
Preconditions:
• `Name` must be a nonempty str which does not contain "{", "}", or "%"
• `Arg` must support .show(...[$show_config]) method
• The set of Name1, Name2 ,,, must be a superset of the names of the named arguments in the Template
All the names of the named args in the template str must be specified by `named_arg` method.
It is permitted to pass `Name` which is not the name of any named arg in the template str. For example:
stdout.print_line('X={X}, Y={Y}'.format{(:C)
C.named_arg('X' 10)
C.named_arg('Y' 20)
C.named_arg('Z' 30) # Z is not in the template str, but it is OK
}
)
# => X=10, Y=20
4.23.2.3. Format_config.locale(Locale)¶
`locale` method specifies the locale.
Precondition:
• Locale must be a locale
4.23.3. type show_config¶
`show_config` is the config val type of .show(...[$show_config]) methods.
`show` method is expected to return a str which represents the receiver val. In contrast to `repr method, which is used for debugging, the result of `show` method is intended to be used in UI messages.
Usually, `show` methods are called from Str.format. See Str.format for details.
Example usage:
:LOCALE.require_from('kink/')
:Shown <- 3776.24.show{(:S)
S.locale(LOCALE.for('da-DK'))
S.spec(',d')
}
stdout.print_line(Shown.repr) # => "3.776,24"
An implementation of show_config type can also implement extra methods such as Num_show_config.radix and .pad_zero.
For example:
:Shown <- 255.show{(:S)
S.radix(16)
S.pad_zero(4)
}
stdout.print_line(Shown.repr) # => "00ff"
4.23.3.1. Show_config.locale(Locale_name)¶
`locale` method specifies the locale which is used to generate the result of `show` method.
Precondition:
• `Locale_name` must be a locale.
4.23.3.2. Show_config.spec(Spec)¶
`spec` method specifies how the val is converted to a str.
Precondition:
• `Spec` must be a str.
The syntax of `Spec` str varies among `show` method implementations.
4.23.4. STR.is?(Val)¶
`is?` returns whether `Val` is a str val.
4.23.5. STR.from_runes(Runes)¶
`from_runes` makes a str which contains the runes.
Precondition:
• `Runes` must be a sequence which support `each` method, such as vec or iter.
• Elements of `Runes` must be int nums in the range [0x000000, 0x10ffff].
Example using vec:
:STR.require_from('kink/')
:Str <- STR.from_runes([120 121 122])
stdout.print_line(Str.repr) # => "xyz"
Example using iter:
:ITER.require_from('kink/iter/')
:STR.require_from('kink/')
:Str <- STR.from_runes(ITER.of(120 121 122))
stdout.print_line(Str.repr) # => "xyz"
4.23.6. STR.default_newline¶
`default_newline` returns the line separator str default to the runtime.
The result is either "\r\n" or "\n".