123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687 |
- #
- #
- # Nim's Runtime Library
- # (c) Copyright 2016 Joey Payne
- #
- # See the file "copying.txt", included in this
- # distribution, for details about the copyright.
- #
- ## This module contains various string utility routines that are uncommonly
- ## used in comparison to the ones in `strutils <strutils.html>`_.
- import std/strutils
- func expandTabs*(s: string, tabSize: int = 8): string =
- ## Expands tab characters in `s`, replacing them by spaces.
- ##
- ## The amount of inserted spaces for each tab character is the difference
- ## between the current column number and the next tab position. Tab positions
- ## occur every `tabSize` characters.
- ## The column number starts at 0 and is increased with every single character
- ## and inserted space, except for newline, which resets the column number
- ## back to 0.
- runnableExamples:
- doAssert expandTabs("\t", 4) == " "
- doAssert expandTabs("\tfoo\t", 4) == " foo "
- doAssert expandTabs("a\tb\n\txy\t", 3) == "a b\n xy "
- result = newStringOfCap(s.len + s.len shr 2)
- var pos = 0
- template addSpaces(n) =
- for j in 0 ..< n:
- result.add(' ')
- pos += 1
- for i in 0 ..< len(s):
- let c = s[i]
- if c == '\t':
- let
- denominator = if tabSize > 0: tabSize else: 1
- numSpaces = tabSize - pos mod denominator
- addSpaces(numSpaces)
- else:
- result.add(c)
- pos += 1
- if c == '\l':
- pos = 0
- func partition*(s: string, sep: string,
- right: bool = false): (string, string, string) =
- ## Splits the string at the first (if `right` is false)
- ## or last (if `right` is true) occurrence of `sep` into a 3-tuple.
- ##
- ## Returns a 3-tuple of strings, `(beforeSep, sep, afterSep)` or
- ## `(s, "", "")` if `sep` is not found and `right` is false or
- ## `("", "", s)` if `sep` is not found and `right` is true.
- ##
- ## **See also:**
- ## * `rpartition proc <#rpartition,string,string>`_
- runnableExamples:
- doAssert partition("foo:bar:baz", ":") == ("foo", ":", "bar:baz")
- doAssert partition("foo:bar:baz", ":", right = true) == ("foo:bar", ":", "baz")
- doAssert partition("foobar", ":") == ("foobar", "", "")
- doAssert partition("foobar", ":", right = true) == ("", "", "foobar")
- let position = if right: s.rfind(sep) else: s.find(sep)
- if position != -1:
- return (s[0 ..< position], sep, s[position + sep.len ..< s.len])
- return if right: ("", "", s) else: (s, "", "")
- func rpartition*(s: string, sep: string): (string, string, string) =
- ## Splits the string at the last occurrence of `sep` into a 3-tuple.
- ##
- ## Returns a 3-tuple of strings, `(beforeSep, sep, afterSep)` or
- ## `("", "", s)` if `sep` is not found. This is the same as
- ## `partition(s, sep, right = true)`.
- ##
- ## **See also:**
- ## * `partition proc <#partition,string,string,bool>`_
- runnableExamples:
- doAssert rpartition("foo:bar:baz", ":") == ("foo:bar", ":", "baz")
- doAssert rpartition("foobar", ":") == ("", "", "foobar")
- partition(s, sep, right = true)
|