New Upstream Release - nim-unicodeplus

Ready changes

Summary

Merged new upstream version: 0.8.0 (was: 0.5.1).

Resulting package

Built on 2022-03-15T22:14 (took 2m5s)

The resulting binary packages can be installed (if you have the apt repository enabled) by running one of:

apt install -t fresh-releases nim-unicodeplus-dev

Lintian Result

Diff

diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..6a04aca
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,3 @@
+nimcache/
+src/unicodeplus
+tests/tests
diff --git a/.travis.yml b/.travis.yml
index d0e2aef..f63a71e 100644
--- a/.travis.yml
+++ b/.travis.yml
@@ -1,10 +1,11 @@
 services:
   - docker
 env:
-  - NIM=0.18.0
   - NIM=0.19.0
   - NIM=0.19.6
   - NIM=0.20.0
+  - NIM=1.0.6
+  - NIM=1.2.0
 before_install:
   - docker pull nimlang/nim:$NIM
 script:
diff --git a/CHANGELOG.md b/CHANGELOG.md
index 4866ea1..2892828 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,3 +1,19 @@
+v0.8.0
+==================
+
+* Deprecate all funcs taking `seq[Rune]`
+
+v0.7.0
+==================
+
+* Add `cmpCaseless`
+
+v0.6.0
+==================
+
+* Add `toTitle`, `toUpper`, and `toLower`
+* Drop Nim 0.18 support
+
 v0.5.1
 ==================
 
diff --git a/README.md b/README.md
index 16cdef3..1ff56e0 100644
--- a/README.md
+++ b/README.md
@@ -3,11 +3,7 @@
 [![Build Status](https://img.shields.io/travis/nitely/nim-unicodeplus.svg?style=flat-square)](https://travis-ci.org/nitely/nim-unicodeplus)
 [![licence](https://img.shields.io/github/license/nitely/nim-unicodeplus.svg?style=flat-square)](https://raw.githubusercontent.com/nitely/nim-unicodeplus/master/LICENSE)
 
-A replacement for Nim's unicode module. It closely
-follows Python's API behavior for each operation.
-
-> Note: this is a WIP! But what's
-> implemented is already ready to use.
+Common unicode operations.
 
 ## Install
 
@@ -17,49 +13,34 @@ nimble install unicodeplus
 
 ## Compatibility
 
-Nim 0.18.0, +0.19.0
+Nim 0.19, 0.20, +1.0
 
 ## Usage
 
 ```nim
 import unicodeplus
 
-assert "abc def ghi".isLower()
-assert "ABC DEF GHI".isUpper()
-assert "A Title - Yes!".isTitle()
-assert "  \L".isWhiteSpace()
-assert "abc123۲⅕".isAlnum()
-assert "abcd".isAlpha()
-assert "0123456789".isDecimal()
-assert "0123456789۲".isDigit()
-assert "0123456789۲⅕".isNumeric()
-assert "abcd".isPrintable()
+assert "abc def ghi".isLower
+assert "ABC DEF GHI".isUpper
+assert "A Title - Yes!".isTitle
+assert "  \L".isWhiteSpace
+assert "abc123۲⅕".isAlnum
+assert "abcd".isAlpha
+assert "0123456789".isDecimal
+assert "0123456789۲".isDigit
+assert "0123456789۲⅕".isNumeric
+assert "abcd".isPrintable
+assert "The quick? (“brown”) fox can’t jump 32.3 feet, right?".toTitle ==
+  "The Quick? (“Brown”) Fox Can’t Jump 32.3 Feet, Right?"
+assert "The quick? (“brown”) fox can’t jump 32.3 feet, right?".toUpper ==
+  "THE QUICK? (“BROWN”) FOX CAN’T JUMP 32.3 FEET, RIGHT?"
+assert "The quIck? (“bRown”) fox cAn’T jUMp 32.3 feet, rIGHt?".toLower ==
+  "the quick? (“brown”) fox can’t jump 32.3 feet, right?"
+assert cmpCaseless("AbCσ", "aBcΣ")
 ```
 
 [docs](https://nitely.github.io/nim-unicodeplus/)
 
-## vs Nim's stdlib
-
-Although this library will be a replacement
-for Nim's unicode module, it won't be a *drop-in replacement*.
-Some of the APIs don't show the same behavior.
-
-To illustrate:
-
-```nim
-import unicode
-
-assert(not "A Title - Maybe?".isTitle)
-assert(not "I'M UPPER?".isUpper)
-```
-
-```nim
-import unicodeplus
-
-assert "A Title - Maybe?".isTitle
-assert "I'M UPPER?".isUpper
-```
-
 ## Tests
 
 ```
diff --git a/debian/changelog b/debian/changelog
index d2648ec..b24a3aa 100644
--- a/debian/changelog
+++ b/debian/changelog
@@ -1,3 +1,9 @@
+nim-unicodeplus (0.8.0-1) UNRELEASED; urgency=low
+
+  * New upstream release.
+
+ -- Debian Janitor <janitor@jelmer.uk>  Tue, 15 Mar 2022 22:12:59 -0000
+
 nim-unicodeplus (0.5.1-2) unstable; urgency=medium
 
   * Bump debhelper from old 12 to 13.
diff --git a/docs/index.html b/docs/index.html
new file mode 100644
index 0000000..f9e6083
--- /dev/null
+++ b/docs/index.html
@@ -0,0 +1,18 @@
+<!DOCTYPE html>
+<html>
+  <head>
+    <style>
+      body {
+        color: #fff;
+        background-color: #171921;
+      }
+      a {
+        color: #8be9fd;
+      }
+    </style>
+    <meta http-equiv="Refresh" content="0; url=unicodeplus.html" />
+  </head>
+  <body>
+    <p>Please follow <a href="unicodeplus.html">this link</a>.</p>
+  </body>
+</html>
diff --git a/docs/nimdoc.out.css b/docs/nimdoc.out.css
new file mode 100644
index 0000000..4ee73ea
--- /dev/null
+++ b/docs/nimdoc.out.css
@@ -0,0 +1,891 @@
+/*
+Stylesheet for use with Docutils/rst2html.
+
+See http://docutils.sf.net/docs/howto/html-stylesheets.html for how to
+customize this style sheet.
+
+Modified from Chad Skeeters' rst2html-style
+https://bitbucket.org/cskeeters/rst2html-style/
+
+Modified by Boyd Greenfield and narimiran
+*/
+
+:root {
+  --primary-background: #fff;
+  --secondary-background: ghostwhite;
+  --third-background: #e8e8e8;
+  --border: #dde;
+  --text: #222;
+  --anchor: #07b;
+  --anchor-focus: #607c9f;
+  --input-focus: #1fa0eb;
+  --strong: #3c3c3c;
+  --hint: #9A9A9A;
+  --nim-sprite-base64: url("");
+
+  --keyword: #5e8f60;
+  --identifier: #222;
+  --comment: #484a86;
+  --operator: #155da4;
+  --punctuation: black;
+  --other: black;
+  --escapeSequence: #c4891b;
+  --number: #252dbe;
+  --literal: #a4255b;
+  --raw-data: #a4255b;
+}
+
+[data-theme="dark"] {
+  --primary-background: #171921;
+  --secondary-background: #1e202a;
+  --third-background: #2b2e3b;
+  --border: #0e1014;
+  --text: #fff;
+  --anchor: #8be9fd;
+  --anchor-focus: #8be9fd;
+  --input-focus: #8be9fd;
+  --strong: #bd93f9;
+  --hint: #7A7C85;
+  --nim-sprite-base64: url("");
+
+  --keyword: #ff79c6;
+  --identifier: #f8f8f2;
+  --comment: #6272a4;
+  --operator: #ff79c6;
+  --punctuation: #f8f8f2;
+  --other: #f8f8f2;
+  --escapeSequence: #bd93f9;
+  --number: #bd93f9;
+  --literal: #f1fa8c;
+  --raw-data: #8be9fd;
+}
+
+.theme-switch-wrapper {
+  display: flex;
+  align-items: center;
+
+  em {
+    margin-left: 10px;
+    font-size: 1rem;
+  }
+}
+.theme-switch {
+  display: inline-block;
+  height: 22px;
+  position: relative;
+  width: 50px;
+}
+
+.theme-switch input {
+  display: none;
+}
+
+.slider {
+  background-color: #ccc;
+  bottom: 0;
+  cursor: pointer;
+  left: 0;
+  position: absolute;
+  right: 0;
+  top: 0;
+  transition: .4s;
+}
+
+.slider:before {
+  background-color: #fff;
+  bottom: 4px;
+  content: "";
+  height: 13px;
+  left: 4px;
+  position: absolute;
+  transition: .4s;
+  width: 13px;
+}
+
+input:checked + .slider {
+  background-color: #66bb6a;
+}
+
+input:checked + .slider:before {
+  transform: translateX(26px);
+}
+
+.slider.round {
+  border-radius: 17px;
+}
+
+.slider.round:before {
+  border-radius: 50%;
+}
+
+html {
+  font-size: 100%;
+  -webkit-text-size-adjust: 100%;
+  -ms-text-size-adjust: 100%; }
+
+body {
+  font-family: "Lato", "Helvetica Neue", "HelveticaNeue", Helvetica, Arial, sans-serif;
+  font-weight: 400;
+  font-size: 1.125em;
+  line-height: 1.5;
+  color: var(--text);
+  background-color: var(--primary-background); }
+
+/* Skeleton grid */
+.container {
+  position: relative;
+  width: 100%;
+  max-width: 1050px;
+  margin: 0 auto;
+  padding: 0;
+  box-sizing: border-box; }
+
+.column,
+.columns {
+  width: 100%;
+  float: left;
+  box-sizing: border-box;
+  margin-left: 1%;
+}
+
+.column:first-child,
+.columns:first-child {
+  margin-left: 0; }
+
+.three.columns {
+  width: 19%; }
+
+.nine.columns {
+  width: 80.0%; }
+
+.twelve.columns {
+  width: 100%;
+  margin-left: 0; }
+
+@media screen and (max-width: 860px) {
+  .three.columns {
+    display: none;
+  }
+  .nine.columns {
+    width: 98.0%;
+  }
+  body {
+    font-size: 1em;
+    line-height: 1.35;
+  }
+}
+
+cite {
+  font-style: italic !important; }
+
+
+/* Nim search input */
+div#searchInputDiv {
+  margin-bottom: 1em;
+}
+input#searchInput {
+  width: 80%;
+}
+
+/*
+ * Some custom formatting for input forms.
+ * This also fixes input form colors on Firefox with a dark system theme on Linux.
+ */
+input {
+  -moz-appearance: none;
+  background-color: var(--secondary-background);
+  color: var(--text);
+  border: 1px solid var(--border);
+  font-family: "Lato", "Helvetica Neue", "HelveticaNeue", Helvetica, Arial, sans-serif;
+  font-size: 0.9em;
+  padding: 6px;
+}
+
+input:focus {
+  border: 1px solid var(--input-focus);
+  box-shadow: 0 0 3px var(--input-focus);
+}
+
+select {
+  -moz-appearance: none;
+  background-color: var(--secondary-background);
+  color: var(--text);
+  border: 1px solid var(--border);
+  font-family: "Lato", "Helvetica Neue", "HelveticaNeue", Helvetica, Arial, sans-serif;
+  font-size: 0.9em;
+  padding: 6px;
+}
+
+select:focus {
+  border: 1px solid var(--input-focus);
+  box-shadow: 0 0 3px var(--input-focus);
+}
+
+/* Docgen styles */
+/* Links */
+a {
+  color: var(--anchor);
+  text-decoration: none;
+}
+
+a span.Identifier {
+  text-decoration: underline;
+  text-decoration-color: #aab;
+}
+
+a.reference-toplevel {
+  font-weight: bold;
+}
+
+a.toc-backref {
+  text-decoration: none;
+  color: var(--text); }
+
+a.link-seesrc {
+  color: #607c9f;
+  font-size: 0.9em;
+  font-style: italic; }
+
+a:hover,
+a:focus {
+  color: var(--anchor-focus);
+  text-decoration: underline; }
+
+a:hover span.Identifier {
+  color: var(--anchor);
+}
+
+
+sub,
+sup {
+  position: relative;
+  font-size: 75%;
+  line-height: 0;
+  vertical-align: baseline; }
+
+sup {
+  top: -0.5em; }
+
+sub {
+  bottom: -0.25em; }
+
+img {
+  width: auto;
+  height: auto;
+  max-width: 100%;
+  vertical-align: middle;
+  border: 0;
+  -ms-interpolation-mode: bicubic; }
+
+@media print {
+  * {
+    color: black !important;
+    text-shadow: none !important;
+    background: transparent !important;
+    box-shadow: none !important; }
+
+  a,
+  a:visited {
+    text-decoration: underline; }
+
+  a[href]:after {
+    content: " (" attr(href) ")"; }
+
+  abbr[title]:after {
+    content: " (" attr(title) ")"; }
+
+  .ir a:after,
+  a[href^="javascript:"]:after,
+  a[href^="#"]:after {
+    content: ""; }
+
+  pre,
+  blockquote {
+    border: 1px solid #999;
+    page-break-inside: avoid; }
+
+  thead {
+    display: table-header-group; }
+
+  tr,
+  img {
+    page-break-inside: avoid; }
+
+  img {
+    max-width: 100% !important; }
+
+  @page {
+    margin: 0.5cm; }
+
+  h1 {
+    page-break-before: always; }
+
+  h1.title {
+    page-break-before: avoid; }
+
+  p,
+  h2,
+  h3 {
+    orphans: 3;
+    widows: 3; }
+
+  h2,
+  h3 {
+    page-break-after: avoid; }
+}
+
+
+p {
+  margin-top: 0.5em;
+  margin-bottom: 0.5em;
+}
+
+small {
+  font-size: 85%; }
+
+strong {
+  font-weight: 600;
+  font-size: 0.95em;
+  color: var(--strong);
+}
+
+em {
+  font-style: italic; }
+
+h1 {
+  font-size: 1.8em;
+  font-weight: 400;
+  padding-bottom: .25em;
+  border-bottom: 6px solid var(--third-background);
+  margin-top: 2.5em;
+  margin-bottom: 1em;
+  line-height: 1.2em; }
+
+h1.title {
+  padding-bottom: 1em;
+  border-bottom: 0px;
+  font-size: 2.5em;
+  text-align: center;
+  font-weight: 900;
+  margin-top: 0.75em;
+  margin-bottom: 0em;
+}
+
+h2 {
+  font-size: 1.3em;
+  margin-top: 2em; }
+
+h2.subtitle {
+  text-align: center; }
+
+h3 {
+  font-size: 1.125em;
+  font-style: italic;
+  margin-top: 1.5em; }
+
+h4 {
+  font-size: 1.125em;
+  margin-top: 1em; }
+
+h5 {
+  font-size: 1.125em;
+  margin-top: 0.75em; }
+
+h6 {
+  font-size: 1.1em; }
+
+
+ul,
+ol {
+  padding: 0;
+  margin-top: 0.5em;
+  margin-left: 0.75em; }
+
+ul ul,
+ul ol,
+ol ol,
+ol ul {
+  margin-bottom: 0;
+  margin-left: 1.25em; }
+
+li {
+    list-style-type: circle;
+}
+
+ul.simple-boot li {
+    list-style-type: none;
+    margin-left: 0em;
+    margin-bottom: 0.5em;
+}
+
+ol.simple > li, ul.simple > li {
+  margin-bottom: 0.25em;
+  margin-left: 0.4em }
+
+ul.simple.simple-toc > li {
+    margin-top: 1em;
+}
+
+ul.simple-toc {
+  list-style: none;
+  font-size: 0.9em;
+  margin-left: -0.3em;
+  margin-top: 1em; }
+
+ul.simple-toc > li {
+    list-style-type: none;
+}
+
+ul.simple-toc-section {
+  list-style-type: circle;
+  margin-left: 1em;
+  color: #6c9aae; }
+
+
+ol.arabic {
+  list-style: decimal; }
+
+ol.loweralpha {
+  list-style: lower-alpha; }
+
+ol.upperalpha {
+  list-style: upper-alpha; }
+
+ol.lowerroman {
+  list-style: lower-roman; }
+
+ol.upperroman {
+  list-style: upper-roman; }
+
+ul.auto-toc {
+  list-style-type: none; }
+
+
+dl {
+  margin-bottom: 1.5em; }
+
+dt {
+  margin-bottom: -0.5em;
+  margin-left: 0.0em; }
+
+dd {
+  margin-left: 2.0em;
+  margin-bottom: 3.0em;
+  margin-top: 0.5em; }
+
+
+hr {
+  margin: 2em 0;
+  border: 0;
+  border-top: 1px solid #aaa; }
+
+blockquote {
+  font-size: 0.9em;
+  font-style: italic;
+  padding-left: 0.5em;
+  margin-left: 0;
+  border-left: 5px solid #bbc;
+}
+
+.pre {
+  font-family: "Source Code Pro", Monaco, Menlo, Consolas, "Courier New", monospace;
+  font-weight: 500;
+  font-size: 0.85em;
+  color: var(--text);
+  background-color: var(--third-background);
+  padding-left: 3px;
+  padding-right: 3px;
+  border-radius: 4px;
+}
+
+pre {
+  font-family: "Source Code Pro", Monaco, Menlo, Consolas, "Courier New", monospace;
+  color: var(--text);
+  font-weight: 500;
+  display: inline-block;
+  box-sizing: border-box;
+  min-width: 100%;
+  padding: 0.5em;
+  margin-top: 0.5em;
+  margin-bottom: 0.5em;
+  font-size: 0.85em;
+  white-space: pre !important;
+  overflow-y: hidden;
+  overflow-x: visible;
+  background-color: var(--secondary-background);
+  border: 1px solid var(--border);
+  -webkit-border-radius: 6px;
+  -moz-border-radius: 6px;
+  border-radius: 6px; }
+
+.pre-scrollable {
+  max-height: 340px;
+  overflow-y: scroll; }
+
+
+/* Nim line-numbered tables */
+.line-nums-table {
+  width: 100%;
+  table-layout: fixed; }
+
+table.line-nums-table {
+  border-radius: 4px;
+  border: 1px solid #cccccc;
+  background-color: ghostwhite;
+  border-collapse: separate;
+  margin-top: 15px;
+  margin-bottom: 25px; }
+
+.line-nums-table tbody {
+  border: none; }
+
+.line-nums-table td pre {
+  border: none;
+  background-color: transparent; }
+
+.line-nums-table td.blob-line-nums {
+  width: 28px; }
+
+.line-nums-table td.blob-line-nums pre {
+  color: #b0b0b0;
+  -webkit-filter: opacity(75%);
+  text-align: right;
+  border-color: transparent;
+  background-color: transparent;
+  padding-left: 0px;
+  margin-left: 0px;
+  padding-right: 0px;
+  margin-right: 0px; }
+
+
+table {
+  max-width: 100%;
+  background-color: transparent;
+  margin-top: 0.5em;
+  margin-bottom: 1.5em;
+  border-collapse: collapse;
+  border-color: var(--third-background);
+  border-spacing: 0;
+  font-size: 0.9em;
+}
+
+table th, table td {
+  padding: 0px 0.5em 0px;
+  border-color: var(--third-background);
+}
+
+table th {
+  background-color: var(--third-background);
+  border-color: var(--third-background);
+  font-weight: bold; }
+
+table th.docinfo-name {
+    background-color: transparent;
+}
+
+table tr:hover {
+  background-color: var(--third-background); }
+
+
+/* rst2html default used to remove borders from tables and images */
+.borderless, table.borderless td, table.borderless th {
+  border: 0; }
+
+table.borderless td, table.borderless th {
+  /* Override padding for "table.docutils td" with "! important".
+     The right padding separates the table cells. */
+  padding: 0 0.5em 0 0 !important; }
+
+.first {
+  /* Override more specific margin styles with "! important". */
+  margin-top: 0 !important; }
+
+.last, .with-subtitle {
+  margin-bottom: 0 !important; }
+
+.hidden {
+  display: none; }
+
+blockquote.epigraph {
+  margin: 2em 5em; }
+
+dl.docutils dd {
+  margin-bottom: 0.5em; }
+
+object[type="image/svg+xml"], object[type="application/x-shockwave-flash"] {
+  overflow: hidden; }
+
+
+div.figure {
+  margin-left: 2em;
+  margin-right: 2em; }
+
+div.footer, div.header {
+  clear: both;
+  text-align: center;
+  color: #666;
+  font-size: smaller; }
+
+div.footer {
+    padding-top: 5em;
+}
+
+div.line-block {
+  display: block;
+  margin-top: 1em;
+  margin-bottom: 1em; }
+
+div.line-block div.line-block {
+  margin-top: 0;
+  margin-bottom: 0;
+  margin-left: 1.5em; }
+
+div.topic {
+  margin: 2em; }
+
+div.search_results {
+  background-color: antiquewhite;
+  margin: 3em;
+  padding: 1em;
+  border: 1px solid #4d4d4d;
+}
+
+div#global-links ul {
+  margin-left: 0;
+  list-style-type: none;
+}
+
+div#global-links > simple-boot {
+    margin-left: 3em;
+}
+
+hr.docutils {
+  width: 75%; }
+
+img.align-left, .figure.align-left, object.align-left {
+  clear: left;
+  float: left;
+  margin-right: 1em; }
+
+img.align-right, .figure.align-right, object.align-right {
+  clear: right;
+  float: right;
+  margin-left: 1em; }
+
+img.align-center, .figure.align-center, object.align-center {
+  display: block;
+  margin-left: auto;
+  margin-right: auto; }
+
+.align-left {
+  text-align: left; }
+
+.align-center {
+  clear: both;
+  text-align: center; }
+
+.align-right {
+  text-align: right; }
+
+/* reset inner alignment in figures */
+div.align-right {
+  text-align: inherit; }
+
+p.attribution {
+  text-align: right;
+  margin-left: 50%; }
+
+p.caption {
+  font-style: italic; }
+
+p.credits {
+  font-style: italic;
+  font-size: smaller; }
+
+p.label {
+  white-space: nowrap; }
+
+p.rubric {
+  font-weight: bold;
+  font-size: larger;
+  color: maroon;
+  text-align: center; }
+
+p.topic-title {
+  font-weight: bold; }
+
+pre.address {
+  margin-bottom: 0;
+  margin-top: 0;
+  font: inherit; }
+
+pre.literal-block, pre.doctest-block, pre.math, pre.code {
+  margin-left: 2em;
+  margin-right: 2em; }
+
+pre.code .ln {
+  color: grey; }
+
+/* line numbers */
+pre.code, code {
+  background-color: #eeeeee; }
+
+pre.code .comment, code .comment {
+  color: #5c6576; }
+
+pre.code .keyword, code .keyword {
+  color: #3B0D06;
+  font-weight: bold; }
+
+pre.code .literal.string, code .literal.string {
+  color: #0c5404; }
+
+pre.code .name.builtin, code .name.builtin {
+  color: #352b84; }
+
+pre.code .deleted, code .deleted {
+  background-color: #DEB0A1; }
+
+pre.code .inserted, code .inserted {
+  background-color: #A3D289; }
+
+span.classifier {
+  font-style: oblique; }
+
+span.classifier-delimiter {
+  font-weight: bold; }
+
+span.option {
+  white-space: nowrap; }
+
+span.problematic {
+  color: #b30000; }
+
+span.section-subtitle {
+  /* font-size relative to parent (h1..h6 element) */
+  font-size: 80%; }
+
+span.DecNumber {
+  color: var(--number); }
+
+span.BinNumber {
+  color: var(--number); }
+
+span.HexNumber {
+  color: var(--number); }
+
+span.OctNumber {
+  color: var(--number); }
+
+span.FloatNumber {
+  color: var(--number); }
+
+span.Identifier {
+  color: var(--identifier); }
+
+span.Keyword {
+  font-weight: 600;
+  color: var(--keyword); }
+
+span.StringLit {
+  color: var(--literal); }
+
+span.LongStringLit {
+  color: var(--literal); }
+
+span.CharLit {
+  color: var(--literal); }
+
+span.EscapeSequence {
+  color: var(--escapeSequence); }
+
+span.Operator {
+  color: var(--operator); }
+
+span.Punctuation {
+  color: var(--punctuation); }
+
+span.Comment, span.LongComment {
+  font-style: italic;
+  font-weight: 400;
+  color: var(--comment); }
+
+span.RegularExpression {
+  color: darkviolet; }
+
+span.TagStart {
+  color: darkviolet; }
+
+span.TagEnd {
+  color: darkviolet; }
+
+span.Key {
+  color: #252dbe; }
+
+span.Value {
+  color: #252dbe; }
+
+span.RawData {
+  color: var(--raw-data); }
+
+span.Assembler {
+  color: #252dbe; }
+
+span.Preprocessor {
+  color: #252dbe; }
+
+span.Directive {
+  color: #252dbe; }
+
+span.Command, span.Rule, span.Hyperlink, span.Label, span.Reference,
+span.Other {
+  color: var(--other); }
+
+/* Pop type, const, proc, and iterator defs in nim def blocks */
+dt pre > span.Identifier, dt pre > span.Operator {
+  color: var(--identifier);
+  font-weight: 700; }
+
+dt pre > span.Keyword ~ span.Identifier, dt pre > span.Identifier ~ span.Identifier,
+dt pre > span.Operator ~ span.Identifier, dt pre > span.Other ~ span.Identifier {
+  color: var(--identifier);
+  font-weight: inherit; }
+
+/* Nim sprite for the footer (taken from main page favicon) */
+.nim-sprite {
+  display: inline-block;
+  width: 51px;
+  height: 14px;
+  background-position: 0 0;
+  background-size: 51px 14px;
+  -webkit-filter: opacity(50%);
+  background-repeat: no-repeat;
+  background-image: var(--nim-sprite-base64);
+  margin-bottom: 5px; }
+
+span.pragmadots {
+  /* Position: relative frees us up to make the dots
+  look really nice without fucking up the layout and
+  causing bulging in the parent container */
+  position: relative;
+  /* 1px down looks slightly nicer */
+  top: 1px;
+  padding: 2px;
+  background-color: var(--third-background);
+  border-radius: 4px;
+  margin: 0 2px;
+  cursor: pointer;
+  font-size: 0.8em;
+}
+
+span.pragmadots:hover {
+  background-color: var(--hint);
+}
+span.pragmawrap {
+  display: none;
+}
+
+span.attachedType {
+  display: none;
+  visibility: hidden;
+}
\ No newline at end of file
diff --git a/docs/unicodeplus.html b/docs/unicodeplus.html
new file mode 100644
index 0000000..62d5b4e
--- /dev/null
+++ b/docs/unicodeplus.html
@@ -0,0 +1,480 @@
+<?xml version="1.0" encoding="utf-8" ?>
+<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
+  "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
+<!--  This file is generated by Nim. -->
+<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
+<head>
+<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
+
+<meta name="viewport" content="width=device-width, initial-scale=1.0">
+
+<!-- Favicon -->
+<link rel="shortcut icon" href=""/>
+<link rel="icon" type="image/png" sizes="32x32" href="">
+
+<!-- Google fonts -->
+<link href='https://fonts.googleapis.com/css?family=Lato:400,600,900' rel='stylesheet' type='text/css'/>
+<link href='https://fonts.googleapis.com/css?family=Source+Code+Pro:400,500,600' rel='stylesheet' type='text/css'/>
+
+<!-- CSS -->
+<title>unicodeplus</title>
+<link rel="stylesheet" type="text/css" href="nimdoc.out.css">
+
+<script type="text/javascript" src="dochack.js"></script>
+
+<script type="text/javascript">
+function main() {
+  var pragmaDots = document.getElementsByClassName("pragmadots");
+  for (var i = 0; i < pragmaDots.length; i++) {
+    pragmaDots[i].onclick = function(event) {
+      // Hide tease
+      event.target.parentNode.style.display = "none";
+      // Show actual
+      event.target.parentNode.nextElementSibling.style.display = "inline";
+    }
+  }
+
+  const toggleSwitch = document.querySelector('.theme-switch input[type="checkbox"]');
+  function switchTheme(e) {
+      if (e.target.checked) {
+          document.documentElement.setAttribute('data-theme', 'dark');
+          localStorage.setItem('theme', 'dark');
+      } else {
+          document.documentElement.setAttribute('data-theme', 'light');
+          localStorage.setItem('theme', 'light');
+      }
+  }
+
+  toggleSwitch.addEventListener('change', switchTheme, false);
+
+
+  if (window.matchMedia && window.matchMedia('(prefers-color-scheme: dark)').matches) {
+    document.documentElement.setAttribute('data-theme', "dark");
+    toggleSwitch.checked = true;
+  } else if (window.matchMedia && window.matchMedia('(prefers-color-scheme: light)').matches) {
+    document.documentElement.setAttribute('data-theme', "light");
+    toggleSwitch.checked = false;
+  } else {
+    const currentTheme = localStorage.getItem('theme') ? localStorage.getItem('theme') : null;
+    if (currentTheme) {
+      document.documentElement.setAttribute('data-theme', currentTheme);
+
+      if (currentTheme === 'dark') {
+        toggleSwitch.checked = true;
+      }
+    }
+  }
+}
+</script>
+
+</head>
+<body onload="main()">
+<div class="document" id="documentId">
+  <div class="container">
+    <h1 class="title">unicodeplus</h1>
+    <div class="row">
+  <div class="three columns">
+  <div class="theme-switch-wrapper">
+    <label class="theme-switch" for="checkbox">
+      <input type="checkbox" id="checkbox" />
+      <div class="slider round"></div>
+    </label>
+    &nbsp;&nbsp;&nbsp; <em>Dark Mode</em>
+  </div>
+  <div id="global-links">
+    <ul class="simple">
+    </ul>
+  </div>
+  <div id="searchInputDiv">
+    Search: <input type="text" id="searchInput"
+      onkeyup="search()" />
+  </div>
+  <div>
+    Group by:
+    <select onchange="groupBy(this.value)">
+      <option value="section">Section</option>
+      <option value="type">Type</option>
+    </select>
+  </div>
+  <ul class="simple simple-toc" id="toc-list">
+<li>
+  <a class="reference reference-toplevel" href="#13" id="63">Funcs</a>
+  <ul class="simple simple-toc-section">
+      <li><a class="reference" href="#isDecimal%2CRune"
+    title="isDecimal(c: Rune): bool"><wbr />is<wbr />Decimal<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isDecimal%2Cstring"
+    title="isDecimal(s: string): bool"><wbr />is<wbr />Decimal<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isDecimal%2Cseq%5BRune%5D"
+    title="isDecimal(s: seq[Rune]): bool"><wbr />is<wbr />Decimal<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isDigit%2CRune"
+    title="isDigit(c: Rune): bool"><wbr />is<wbr />Digit<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isDigit%2Cstring"
+    title="isDigit(s: string): bool"><wbr />is<wbr />Digit<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isDigit%2Cseq%5BRune%5D"
+    title="isDigit(s: seq[Rune]): bool"><wbr />is<wbr />Digit<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isNumeric%2CRune"
+    title="isNumeric(c: Rune): bool"><wbr />is<wbr />Numeric<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isNumeric%2Cstring"
+    title="isNumeric(s: string): bool"><wbr />is<wbr />Numeric<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isNumeric%2Cseq%5BRune%5D"
+    title="isNumeric(s: seq[Rune]): bool"><wbr />is<wbr />Numeric<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isAlpha%2CRune"
+    title="isAlpha(c: Rune): bool"><wbr />is<wbr />Alpha<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isAlpha%2Cstring"
+    title="isAlpha(s: string): bool"><wbr />is<wbr />Alpha<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isAlpha%2Cseq%5BRune%5D"
+    title="isAlpha(s: seq[Rune]): bool"><wbr />is<wbr />Alpha<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isAlnum%2CRune"
+    title="isAlnum(c: Rune): bool"><wbr />is<wbr />Alnum<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isAlnum%2Cstring"
+    title="isAlnum(s: string): bool"><wbr />is<wbr />Alnum<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isAlnum%2Cseq%5BRune%5D"
+    title="isAlnum(s: seq[Rune]): bool"><wbr />is<wbr />Alnum<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isPrintable%2CRune"
+    title="isPrintable(c: Rune): bool"><wbr />is<wbr />Printable<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isPrintable%2Cstring"
+    title="isPrintable(s: string): bool"><wbr />is<wbr />Printable<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isPrintable%2Cseq%5BRune%5D"
+    title="isPrintable(s: seq[Rune]): bool"><wbr />is<wbr />Printable<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isWhiteSpace%2CRune"
+    title="isWhiteSpace(c: Rune): bool"><wbr />is<wbr />White<wbr />Space<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isWhiteSpace%2Cstring"
+    title="isWhiteSpace(s: string): bool"><wbr />is<wbr />White<wbr />Space<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isWhiteSpace%2Cseq%5BRune%5D"
+    title="isWhiteSpace(s: seq[Rune]): bool"><wbr />is<wbr />White<wbr />Space<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isUpper%2CRune"
+    title="isUpper(c: Rune): bool"><wbr />is<wbr />Upper<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isUpper%2Cstring"
+    title="isUpper(s: string): bool"><wbr />is<wbr />Upper<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isUpper%2Cseq%5BRune%5D"
+    title="isUpper(s: seq[Rune]): bool"><wbr />is<wbr />Upper<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isLower%2CRune"
+    title="isLower(c: Rune): bool"><wbr />is<wbr />Lower<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isLower%2Cstring"
+    title="isLower(s: string): bool"><wbr />is<wbr />Lower<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isLower%2Cseq%5BRune%5D"
+    title="isLower(s: seq[Rune]): bool"><wbr />is<wbr />Lower<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isTitle%2CRune"
+    title="isTitle(c: Rune): bool"><wbr />is<wbr />Title<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isTitle%2Cstring"
+    title="isTitle(s: string): bool"><wbr />is<wbr />Title<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#isTitle%2Cseq%5BRune%5D"
+    title="isTitle(s: seq[Rune]): bool"><wbr />is<wbr />Title<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#toUpper%2Cstring"
+    title="toUpper(s: string): string"><wbr />to<wbr />Upper<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#toLower%2Cstring"
+    title="toLower(s: string): string"><wbr />to<wbr />Lower<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#toTitle%2Cstring"
+    title="toTitle(s: string): string"><wbr />to<wbr />Title<span class="attachedType"></span></a></li>
+  <li><a class="reference" href="#cmpCaseless%2Cstring%2Cstring"
+    title="cmpCaseless(a, b: string): bool"><wbr />cmp<wbr />Caseless<span class="attachedType"></span></a></li>
+
+  </ul>
+</li>
+
+</ul>
+
+  </div>
+  <div class="nine columns" id="content">
+  <div id="tocRoot"></div>
+  
+  <p class="module-desc">This module provides common unicode operations</p>
+  <div class="section" id="13">
+<h1><a class="toc-backref" href="#13">Funcs</a></h1>
+<dl class="item">
+<a id="isDecimal,Rune"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isDecimal%2CRune"><span class="Identifier">isDecimal</span></a><span class="Other">(</span><span class="Identifier">c</span><span class="Other">:</span> <span class="Identifier">Rune</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+
+
+</dd>
+<a id="isDecimal,string"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isDecimal%2Cstring"><span class="Identifier">isDecimal</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">string</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+Return <tt class="docutils literal"><span class="pre">true</span></tt> if all characters in the string are decimal characters and there is at least one character, false otherwise. Decimal characters are those that can be used to form numbers in base 10, e.g. U+0660, ARABIC-INDIC DIGIT ZERO. Formally, a decimal is a character that has the property value <tt class="docutils literal"><span class="pre">Numeric_Type=Decimal</span></tt>
+
+</dd>
+<a id="isDecimal,seq[Rune]"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isDecimal%2Cseq%5BRune%5D"><span class="Identifier">isDecimal</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">seq</span><span class="Other">[</span><span class="Identifier">Rune</span><span class="Other">]</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">deprecated</span><span class="Other">:</span> <span class="StringLit">&quot;Use isDecimal(string)&quot;</span><span class="Other">,</span> <span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span>
+                                 <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+  <div class="deprecation-message">
+    <b>Deprecated:</b> Use isDecimal(string)
+  </div>
+
+
+
+</dd>
+<a id="isDigit,Rune"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isDigit%2CRune"><span class="Identifier">isDigit</span></a><span class="Other">(</span><span class="Identifier">c</span><span class="Other">:</span> <span class="Identifier">Rune</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+
+
+</dd>
+<a id="isDigit,string"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isDigit%2Cstring"><span class="Identifier">isDigit</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">string</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+Return <tt class="docutils literal"><span class="pre">true</span></tt> if all characters in the string are digits and there is at least one character, <tt class="docutils literal"><span class="pre">false</span></tt> otherwise. Digits include decimal characters and digits that need special handling, such as the compatibility superscript digits. This covers digits which cannot be used to form numbers in base 10, like the Kharosthi numbers. Formally, a digit is a character that has the property value <tt class="docutils literal"><span class="pre">Numeric_Type=Digit</span></tt> or <tt class="docutils literal"><span class="pre">Numeric_Type=Decimal</span></tt>
+
+</dd>
+<a id="isDigit,seq[Rune]"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isDigit%2Cseq%5BRune%5D"><span class="Identifier">isDigit</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">seq</span><span class="Other">[</span><span class="Identifier">Rune</span><span class="Other">]</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">deprecated</span><span class="Other">:</span> <span class="StringLit">&quot;Use isDigit(string)&quot;</span><span class="Other">,</span> <span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span>
+                               <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+  <div class="deprecation-message">
+    <b>Deprecated:</b> Use isDigit(string)
+  </div>
+
+
+
+</dd>
+<a id="isNumeric,Rune"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isNumeric%2CRune"><span class="Identifier">isNumeric</span></a><span class="Other">(</span><span class="Identifier">c</span><span class="Other">:</span> <span class="Identifier">Rune</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+
+
+</dd>
+<a id="isNumeric,string"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isNumeric%2Cstring"><span class="Identifier">isNumeric</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">string</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+Return <tt class="docutils literal"><span class="pre">true</span></tt> if all characters in the string are numeric, and there is at least one character, <tt class="docutils literal"><span class="pre">false</span></tt> otherwise. Numeric characters include digit characters, and all characters that have the Unicode numeric value property, e.g. U+2155, VULGAR FRACTION ONE FIFTH. Formally, numeric characters are those with the property value Numeric_Type=Digit, Numeric_Type=Decimal or Numeric_Type=Numeric
+
+</dd>
+<a id="isNumeric,seq[Rune]"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isNumeric%2Cseq%5BRune%5D"><span class="Identifier">isNumeric</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">seq</span><span class="Other">[</span><span class="Identifier">Rune</span><span class="Other">]</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">deprecated</span><span class="Other">:</span> <span class="StringLit">&quot;Use isNumeric(string)&quot;</span><span class="Other">,</span> <span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span>
+                                 <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+  <div class="deprecation-message">
+    <b>Deprecated:</b> Use isNumeric(string)
+  </div>
+
+
+
+</dd>
+<a id="isAlpha,Rune"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isAlpha%2CRune"><span class="Identifier">isAlpha</span></a><span class="Other">(</span><span class="Identifier">c</span><span class="Other">:</span> <span class="Identifier">Rune</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+
+
+</dd>
+<a id="isAlpha,string"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isAlpha%2Cstring"><span class="Identifier">isAlpha</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">string</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+Return <tt class="docutils literal"><span class="pre">true</span></tt> if all characters in the string are alphabetic and there is at least one character, <tt class="docutils literal"><span class="pre">false</span></tt> otherwise. Alphabetic characters are those characters defined in the UCD as “Letter”. This is not the same as the “Alphabetic” property defined in the UCD
+
+</dd>
+<a id="isAlpha,seq[Rune]"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isAlpha%2Cseq%5BRune%5D"><span class="Identifier">isAlpha</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">seq</span><span class="Other">[</span><span class="Identifier">Rune</span><span class="Other">]</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">deprecated</span><span class="Other">:</span> <span class="StringLit">&quot;Use isAlpha(string)&quot;</span><span class="Other">,</span> <span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span>
+                               <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+  <div class="deprecation-message">
+    <b>Deprecated:</b> Use isAlpha(string)
+  </div>
+
+
+
+</dd>
+<a id="isAlnum,Rune"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isAlnum%2CRune"><span class="Identifier">isAlnum</span></a><span class="Other">(</span><span class="Identifier">c</span><span class="Other">:</span> <span class="Identifier">Rune</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+
+
+</dd>
+<a id="isAlnum,string"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isAlnum%2Cstring"><span class="Identifier">isAlnum</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">string</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+Return <tt class="docutils literal"><span class="pre">true</span></tt> if all characters in the string are alphanumeric and there is at least one character, <tt class="docutils literal"><span class="pre">false</span></tt> otherwise. A <tt class="docutils literal"><span class="pre">c</span></tt> character is alphanumeric if one of the following returns <tt class="docutils literal"><span class="pre">true</span></tt>: <tt class="docutils literal"><span class="pre">c.isAlpha()</span></tt>, <tt class="docutils literal"><span class="pre">c.isDecimal()</span></tt>, <tt class="docutils literal"><span class="pre">c.isDigit()</span></tt>, or <tt class="docutils literal"><span class="pre">c.isNumeric()</span></tt>
+
+</dd>
+<a id="isAlnum,seq[Rune]"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isAlnum%2Cseq%5BRune%5D"><span class="Identifier">isAlnum</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">seq</span><span class="Other">[</span><span class="Identifier">Rune</span><span class="Other">]</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">deprecated</span><span class="Other">:</span> <span class="StringLit">&quot;Use isAlnum(string)&quot;</span><span class="Other">,</span> <span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span>
+                               <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+  <div class="deprecation-message">
+    <b>Deprecated:</b> Use isAlnum(string)
+  </div>
+
+
+
+</dd>
+<a id="isPrintable,Rune"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isPrintable%2CRune"><span class="Identifier">isPrintable</span></a><span class="Other">(</span><span class="Identifier">c</span><span class="Other">:</span> <span class="Identifier">Rune</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+
+
+</dd>
+<a id="isPrintable,string"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isPrintable%2Cstring"><span class="Identifier">isPrintable</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">string</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+Nonprintable characters are those characters defined in the UCD as “Other” or “Separator”, except for the ASCII space (0x20). Return <tt class="docutils literal"><span class="pre">true</span></tt> if all characters meet this condition or the string is empty
+
+</dd>
+<a id="isPrintable,seq[Rune]"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isPrintable%2Cseq%5BRune%5D"><span class="Identifier">isPrintable</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">seq</span><span class="Other">[</span><span class="Identifier">Rune</span><span class="Other">]</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">deprecated</span><span class="Other">:</span> <span class="StringLit">&quot;Use isPrintable(string)&quot;</span><span class="Other">,</span>
+                                   <span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+  <div class="deprecation-message">
+    <b>Deprecated:</b> Use isPrintable(string)
+  </div>
+
+
+
+</dd>
+<a id="isWhiteSpace,Rune"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isWhiteSpace%2CRune"><span class="Identifier">isWhiteSpace</span></a><span class="Other">(</span><span class="Identifier">c</span><span class="Other">:</span> <span class="Identifier">Rune</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+
+
+</dd>
+<a id="isWhiteSpace,string"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isWhiteSpace%2Cstring"><span class="Identifier">isWhiteSpace</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">string</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+Whitespace characters are those characters defined in the Unicode character database as “Other” or “Separator” and those with bidirectional property being one of “WS”, “B”, or “S”. Return <tt class="docutils literal"><span class="pre">true</span></tt> if all characters meet this condition. Return <tt class="docutils literal"><span class="pre">false</span></tt> if the string is empty
+
+</dd>
+<a id="isWhiteSpace,seq[Rune]"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isWhiteSpace%2Cseq%5BRune%5D"><span class="Identifier">isWhiteSpace</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">seq</span><span class="Other">[</span><span class="Identifier">Rune</span><span class="Other">]</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">deprecated</span><span class="Other">:</span> <span class="StringLit">&quot;Use isWhiteSpace(string)&quot;</span><span class="Other">,</span>
+                                    <span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+  <div class="deprecation-message">
+    <b>Deprecated:</b> Use isWhiteSpace(string)
+  </div>
+
+
+
+</dd>
+<a id="isUpper,Rune"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isUpper%2CRune"><span class="Identifier">isUpper</span></a><span class="Other">(</span><span class="Identifier">c</span><span class="Other">:</span> <span class="Identifier">Rune</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+
+
+</dd>
+<a id="isUpper,string"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isUpper%2Cstring"><span class="Identifier">isUpper</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">string</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+return <tt class="docutils literal"><span class="pre">true</span></tt> if all cased runes are upper-case and there is at least one cased rune
+
+</dd>
+<a id="isUpper,seq[Rune]"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isUpper%2Cseq%5BRune%5D"><span class="Identifier">isUpper</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">seq</span><span class="Other">[</span><span class="Identifier">Rune</span><span class="Other">]</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">deprecated</span><span class="Other">:</span> <span class="StringLit">&quot;Use isUpper(string)&quot;</span><span class="Other">,</span> <span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span>
+                               <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+  <div class="deprecation-message">
+    <b>Deprecated:</b> Use isUpper(string)
+  </div>
+
+
+
+</dd>
+<a id="isLower,Rune"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isLower%2CRune"><span class="Identifier">isLower</span></a><span class="Other">(</span><span class="Identifier">c</span><span class="Other">:</span> <span class="Identifier">Rune</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+
+
+</dd>
+<a id="isLower,string"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isLower%2Cstring"><span class="Identifier">isLower</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">string</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+return <tt class="docutils literal"><span class="pre">true</span></tt> if all cased runes are lower-case and there is at least one cased rune
+
+</dd>
+<a id="isLower,seq[Rune]"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isLower%2Cseq%5BRune%5D"><span class="Identifier">isLower</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">seq</span><span class="Other">[</span><span class="Identifier">Rune</span><span class="Other">]</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">deprecated</span><span class="Other">:</span> <span class="StringLit">&quot;Use isLower(string)&quot;</span><span class="Other">,</span> <span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span>
+                               <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+  <div class="deprecation-message">
+    <b>Deprecated:</b> Use isLower(string)
+  </div>
+
+
+
+</dd>
+<a id="isTitle,Rune"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isTitle%2CRune"><span class="Identifier">isTitle</span></a><span class="Other">(</span><span class="Identifier">c</span><span class="Other">:</span> <span class="Identifier">Rune</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+Return <tt class="docutils literal"><span class="pre">true</span></tt> for ligatures containing uppercase followed by lowercase letters (e.g., Dž, Lj, Nj, and Dz). Return <tt class="docutils literal"><span class="pre">false</span></tt> otherwise
+
+</dd>
+<a id="isTitle,string"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isTitle%2Cstring"><span class="Identifier">isTitle</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">string</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+A title is a unicode sequence of uncased characters followed by an uppercase character and cased characters followed by a lowercase character. Return <tt class="docutils literal"><span class="pre">false</span></tt> if the string is empty
+
+</dd>
+<a id="isTitle,seq[Rune]"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#isTitle%2Cseq%5BRune%5D"><span class="Identifier">isTitle</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">seq</span><span class="Other">[</span><span class="Identifier">Rune</span><span class="Other">]</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">deprecated</span><span class="Other">:</span> <span class="StringLit">&quot;Use isTitle(string)&quot;</span><span class="Other">,</span> <span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span>
+                               <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+  <div class="deprecation-message">
+    <b>Deprecated:</b> Use isTitle(string)
+  </div>
+
+
+
+</dd>
+<a id="toUpper,string"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#toUpper%2Cstring"><span class="Identifier">toUpper</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">string</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">string</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">inline</span><span class="Other">,</span> <span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+Return <tt class="docutils literal"><span class="pre">s</span></tt> in upper case. Beware the result may be longer than <tt class="docutils literal"><span class="pre">s</span></tt>
+
+</dd>
+<a id="toLower,string"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#toLower%2Cstring"><span class="Identifier">toLower</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">string</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">string</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">inline</span><span class="Other">,</span> <span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+Return <tt class="docutils literal"><span class="pre">s</span></tt> in lower case. Beware the result may be longer than <tt class="docutils literal"><span class="pre">s</span></tt>
+
+</dd>
+<a id="toTitle,string"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#toTitle%2Cstring"><span class="Identifier">toTitle</span></a><span class="Other">(</span><span class="Identifier">s</span><span class="Other">:</span> <span class="Identifier">string</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">string</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">inline</span><span class="Other">,</span> <span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+Return <tt class="docutils literal"><span class="pre">s</span></tt> in title case. Beware the result may be longer than <tt class="docutils literal"><span class="pre">s</span></tt>
+
+</dd>
+<a id="cmpCaseless,string,string"></a>
+<dt><pre><span class="Keyword">func</span> <a href="#cmpCaseless%2Cstring%2Cstring"><span class="Identifier">cmpCaseless</span></a><span class="Other">(</span><span class="Identifier">a</span><span class="Other">,</span> <span class="Identifier">b</span><span class="Other">:</span> <span class="Identifier">string</span><span class="Other">)</span><span class="Other">:</span> <span class="Identifier">bool</span> <span><span class="Other">{</span><span class="Other pragmadots">...</span><span class="Other">}</span></span><span class="pragmawrap"><span class="Other">{.</span><span class="pragma"><span class="Identifier">inline</span><span class="Other">,</span> <span class="Identifier">raises</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span><span class="Other">,</span> <span class="Identifier">tags</span><span class="Other">:</span> <span class="Other">[</span><span class="Other">]</span></span><span class="Other">.}</span></span></pre></dt>
+<dd>
+
+Caseless string comparison. Beware the strings are not normalized. This is meant for arbitrary text, do not use it to compare identifiers
+
+</dd>
+
+</dl></div>
+
+  </div>
+</div>
+
+    <div class="row">
+      <div class="twelve-columns footer">
+        <span class="nim-sprite"></span>
+        <br/>
+        <small style="color: var(--hint);">Made with Nim. Generated: 2020-05-30 20:25:27 UTC</small>
+      </div>
+    </div>
+  </div>
+</div>
+
+</body>
+</html>
diff --git a/src/unicodeplus.nim b/src/unicodeplus.nim
index e760ad2..e7f063f 100644
--- a/src/unicodeplus.nim
+++ b/src/unicodeplus.nim
@@ -1,22 +1,24 @@
 ## This module provides common unicode operations
 
-from unicode import
-  Rune, runes, `==`
+from std/unicode import
+  Rune, runes, `==`, fastRuneAt, fastToUtf8Copy, toUtf8
 
-import unicodedb/properties
-import unicodedb/types
+import pkg/unicodedb/properties
+import pkg/unicodedb/types
+import pkg/unicodedb/casing
+import pkg/segmentation
 
 export Rune
 
-proc genNums(): array[128, bool] =
+func genNums(): array[128, bool] =
   for i in '0'.ord .. '9'.ord:
     result[i] = true
-proc genLetters(): array[128, bool] =
+func genLetters(): array[128, bool] =
   for i in 'a'.ord .. 'z'.ord:
     result[i] = true
   for i in 'A'.ord .. 'Z'.ord:
     result[i] = true
-proc genAlphaNums(): array[128, bool] =
+func genAlphaNums(): array[128, bool] =
   for i in '0'.ord .. '9'.ord:
     result[i] = true
   for i in 'a'.ord .. 'z'.ord:
@@ -43,13 +45,13 @@ template runeCheck(s, runeProc) =
     if not result:
       break
 
-proc isDecimal*(c: Rune): bool =
+func isDecimal*(c: Rune): bool =
   result = if c.int < 128:
     nums[c.int]
   else:
     utmDecimal in c.unicodeTypes()
 
-proc isDecimal*(s: string): bool =
+func isDecimal*(s: string): bool =
   ## Return `true` if all characters in the
   ## string are decimal characters and there
   ## is at least one character, false otherwise.
@@ -60,16 +62,16 @@ proc isDecimal*(s: string): bool =
   ## value `Numeric_Type=Decimal`
   runeCheck(s, isDecimal)
 
-proc isDecimal*(s: seq[Rune]): bool =
+func isDecimal*(s: seq[Rune]): bool {.deprecated: "Use isDecimal(string)".} =
   runeCheck(s, isDecimal)
 
-proc isDigit*(c: Rune): bool =
+func isDigit*(c: Rune): bool =
   result = if c.int < 128:
     nums[c.int]
   else:
     utmDigit+utmDecimal in c.unicodeTypes()
 
-proc isDigit*(s: string): bool =
+func isDigit*(s: string): bool =
   ## Return `true` if all characters in the
   ## string are digits and there is at least
   ## one character, `false` otherwise. Digits
@@ -83,16 +85,16 @@ proc isDigit*(s: string): bool =
   ## or `Numeric_Type=Decimal`
   runeCheck(s, isDigit)
 
-proc isDigit*(s: seq[Rune]): bool =
+func isDigit*(s: seq[Rune]): bool {.deprecated: "Use isDigit(string)".} =
   runeCheck(s, isDigit)
 
-proc isNumeric*(c: Rune): bool =
+func isNumeric*(c: Rune): bool =
   result = if c.int < 128:
     nums[c.int]
   else:
     utmDigit+utmDecimal+utmNumeric in c.unicodeTypes()
 
-proc isNumeric*(s: string): bool =
+func isNumeric*(s: string): bool =
   ## Return `true` if all characters in the
   ## string are numeric, and there
   ## is at least one character, `false` otherwise.
@@ -105,16 +107,16 @@ proc isNumeric*(s: string): bool =
   ## Numeric_Type=Decimal or Numeric_Type=Numeric
   runeCheck(s, isNumeric)
 
-proc isNumeric*(s: seq[Rune]): bool =
+func isNumeric*(s: seq[Rune]): bool {.deprecated: "Use isNumeric(string)".} =
   runeCheck(s, isNumeric)
 
-proc isAlpha*(c: Rune): bool =
+func isAlpha*(c: Rune): bool =
   result = if c.int < 128:
     letters[c.int]
   else:
     c.unicodeCategory() in ctgL
 
-proc isAlpha*(s: string): bool =
+func isAlpha*(s: string): bool =
   ## Return `true` if all characters in
   ## the string are alphabetic and there
   ## is at least one character, `false` otherwise.
@@ -124,17 +126,17 @@ proc isAlpha*(s: string): bool =
   ## property defined in the UCD
   runeCheck(s, isAlpha)
 
-proc isAlpha*(s: seq[Rune]): bool =
+func isAlpha*(s: seq[Rune]): bool {.deprecated: "Use isAlpha(string)".} =
   runeCheck(s, isAlpha)
 
-proc isAlnum*(c: Rune): bool =
+func isAlnum*(c: Rune): bool =
   result = if c.int < 128:
     alphaNums[c.int]
   else:
     (c.unicodeCategory() in ctgL or
       utmDigit+utmDecimal+utmNumeric in c.unicodeTypes())
 
-proc isAlnum*(s: string): bool =
+func isAlnum*(s: string): bool =
   ## Return `true` if all characters in
   ## the string are alphanumeric and
   ## there is at least one character,
@@ -145,15 +147,15 @@ proc isAlnum*(s: string): bool =
   ## or `c.isNumeric()`
   runeCheck(s, isAlnum)
 
-proc isAlnum*(s: seq[Rune]): bool =
+func isAlnum*(s: seq[Rune]): bool {.deprecated: "Use isAlnum(string)".} =
   runeCheck(s, isAlnum)
 
-proc isPrintable*(c: Rune): bool =
+func isPrintable*(c: Rune): bool =
   result =
     c == Rune(0x20) or
     c.unicodeCategory() notin ctgC+ctgZ
 
-proc isPrintable*(s: string): bool =
+func isPrintable*(s: string): bool =
   ## Nonprintable characters are those characters
   ## defined in the UCD as “Other” or “Separator”,
   ## except for the ASCII space (0x20). Return `true` if all
@@ -161,16 +163,16 @@ proc isPrintable*(s: string): bool =
   result = true
   runeCheck(s, isPrintable)
 
-proc isPrintable*(s: seq[Rune]): bool =
+func isPrintable*(s: seq[Rune]): bool {.deprecated: "Use isPrintable(string)".} =
   result = true
   runeCheck(s, isPrintable)
 
-proc isWhiteSpace*(c: Rune): bool =
+func isWhiteSpace*(c: Rune): bool =
   result =
     c.unicodeCategory() in ctgC+ctgZ or
     c.bidirectional() in ["WS", "B", "S"]
 
-proc isWhiteSpace*(s: string): bool =
+func isWhiteSpace*(s: string): bool =
   ## Whitespace characters are those characters
   ## defined in the Unicode character database
   ## as “Other” or “Separator” and those with
@@ -180,13 +182,13 @@ proc isWhiteSpace*(s: string): bool =
   ## `false` if the string is empty
   runeCheck(s, isWhiteSpace)
 
-proc isWhiteSpace*(s: seq[Rune]): bool =
+func isWhiteSpace*(s: seq[Rune]): bool {.deprecated: "Use isWhiteSpace(string)".} =
   runeCheck(s, isWhiteSpace)
 
-proc isUpper*(c: Rune): bool =
+func isUpper*(c: Rune): bool =
   utmUppercase in c.unicodeTypes()
 
-proc isUpper*(s: string | seq[Rune]): bool =
+func isUpper*(s: string): bool =
   ## return `true` if all cased runes are
   ## upper-case and there is at least one cased rune
   result = false
@@ -197,10 +199,19 @@ proc isUpper*(s: string | seq[Rune]): bool =
       if not result:
         break
 
-proc isLower*(c: Rune): bool =
+func isUpper*(s: seq[Rune]): bool {.deprecated: "Use isUpper(string)".} =
+  result = false
+  for r in s.runes:
+    let ut = r.unicodeTypes()
+    if utmCased in ut:
+      result = utmUppercase in ut
+      if not result:
+        break
+
+func isLower*(c: Rune): bool =
   utmLowercase in c.unicodeTypes()
 
-proc isLower*(s: string | seq[Rune]): bool =
+func isLower*(s: string): bool =
   ## return `true` if all cased runes are
   ## lower-case and there is at least one cased rune
   result = false
@@ -211,7 +222,16 @@ proc isLower*(s: string | seq[Rune]): bool =
       if not result:
         break
 
-proc isTitle*(c: Rune): bool =
+func isLower*(s: seq[Rune]): bool {.deprecated: "Use isLower(string)".} =
+  result = false
+  for r in s.runes:
+    let ut = r.unicodeTypes()
+    if utmCased in ut:
+      result = utmLowercase in ut
+      if not result:
+        break
+
+func isTitle*(c: Rune): bool =
   ## Return ``true`` for ligatures
   ## containing uppercase
   ## followed by lowercase letters
@@ -219,7 +239,7 @@ proc isTitle*(c: Rune): bool =
   ## Return ``false`` otherwise
   c.unicodeCategory() == ctgLt
 
-proc isTitle*(s: string | seq[Rune]): bool =
+func isTitle*(s: string): bool =
   ## A title is a unicode sequence of
   ## uncased characters followed by an
   ## uppercase character and cased
@@ -239,33 +259,103 @@ proc isTitle*(s: string | seq[Rune]): bool =
     isLastCased = utmCased in ut
     result = true
 
-# todo: needs toLower
-#[]
-proc cmpCi*(a, b: Rune): int =
-  ## Case insensitive comparison
-  RuneImpl(a.toLower()) - RuneImpl(b.toLower())
-
-proc cmpCi*(a, b: seq[Rune]): int =
-  ## Case insensitive comparison
-  result = a.len - b.len
-  if result != 0:
-    return
-  for i in 0 .. a.high:
-    result = a[i].toLower() - b[i].toLower()
-    if result != 0:
-      return
-
-proc cmpCi*(a, b: string): int =
-  ## Case insensitive comparison
+func isTitle*(s: seq[Rune]): bool {.deprecated: "Use isTitle(string)".} =
+  result = false
+  var isLastCased = false
+  for r in s.runes:
+    let ut = r.unicodeTypes()
+    if utmUppercase in ut and isLastCased:
+      result = false
+      break
+    if utmLowercase in ut and not isLastCased:
+      result = false
+      break
+    isLastCased = utmCased in ut
+    result = true
+
+template caseConversionImpl(
+  s: string,
+  caseProc: untyped
+): untyped {.dirty.} =
+  const buffCap = 16
+  var cap = s.len
+  result = newString(cap + buffCap)
+  result.setLen(0)
+  for r in s.runes:
+    for rr in caseProc(r):
+      if (result.len >= cap).unlikely:
+        cap = result.len
+        result.setLen(result.len * 2 + buffCap)
+        result.setLen(cap)
+        cap = result.len * 2
+        # echo "realloc", " ", result.len, " ", cap
+      let pos = result.len
+      fastToUtf8Copy(rr, result, pos, false)
+
+# This follows Unicode Chapter 3,
+# "Default Case Algorithms" section
+func toUpper*(s: string): string {.inline.} =
+  ## Return `s` in upper case.
+  ## Beware the result may be
+  ## longer than `s`
+  caseConversionImpl(s, upperCase)
+
+func toLower*(s: string): string {.inline.} =
+  ## Return `s` in lower case.
+  ## Beware the result may be
+  ## longer than `s`
+  caseConversionImpl(s, lowerCase)
+
+func toTitle*(s: string): string {.inline.} =
+  ## Return `s` in title case.
+  ## Beware the result may be
+  ## longer than `s`
+  result = newString(s.len + 16)
+  result.setLen(0)
+  var r: Rune
+  var ra: int
+  for wb in s.wordsBounds:
+    ra = wb.a
+    fastRuneAt(s, ra, r, true)
+    for rr in r.titleCase:
+      let pos = result.len
+      fastToUtf8Copy(rr, result, pos, false)
+    # XXX memCopy
+    # this will preallocate if needed
+    for i in ra .. wb.b:
+      result.add s[i]
+
+func cmpCaseless*(a, b: string): bool {.inline.} =
+  ## Caseless string comparison. Beware the strings
+  ## are not normalized. This is meant for
+  ## arbitrary text, do not use it to compare
+  ## identifiers
+  template fillBuffA: untyped {.dirty.} =
+    fastRuneAt(a, riA, rA, true)
+    for c in caseFold(rA):
+      buffA[idxA] = c
+      inc idxA
+  template fillBuffB: untyped {.dirty.} =
+    fastRuneAt(b, riB, rB, true)
+    for c in caseFold(rB):
+      buffB[idxB] = c
+      inc idxB
   var
-    i = 0
-    j = 0
-    ar, br: Rune
-  while i < a.len and j < b.len:
-    fastRuneAt(a, i, ar)
-    fastRuneAt(b, j, br)
-    result = RuneImpl(ar.toLower()) - RuneImpl(br.toLower())
-    if result != 0:
-      return
-  result = a.len - b.len
-]#
+    buffA, buffB: array[8, Rune]  # at least 2x max caseFold
+    idxA, idxB = 0
+    riA, riB = 0
+    rA, rB: Rune
+  while riA < a.len and riB < b.len:
+    fillBuffA()
+    fillBuffB()
+    while idxA < idxB and riA < a.len:
+      fillBuffA()
+    while idxB < idxA and riB < b.len:
+      fillBuffB()
+    if idxA != idxB:
+      return false
+    if toOpenArray(buffA, 0, idxA) != toOpenArray(buffB, 0, idxB):
+      return false
+    idxA = 0
+    idxB = 0
+  return riA == a.len and riB == b.len
diff --git a/tests/tests.nim b/tests/tests.nim
index 400b5d4..a6d78b6 100644
--- a/tests/tests.nim
+++ b/tests/tests.nim
@@ -1,4 +1,7 @@
 import unittest
+from unicode import Rune, toUTF8
+
+from unicodedb/casing import caseFold
 
 import unicodeplus
 
@@ -196,4 +199,122 @@ test "sanityCheck":
     discard Rune(c).isTitle()
     discard Rune(c).isUpper()
     discard Rune(c).isLower()
-    
\ No newline at end of file
+
+test "toTitle":
+  check "a".toTitle == "A"
+  check "1".toTitle == "1"
+  check "some title".toTitle == "Some Title"
+  check "The quick? (“brown”) fox can’t jump 32.3 feet, right?".toTitle ==
+    "The Quick? (“Brown”) Fox Can’t Jump 32.3 Feet, Right?"
+  check "a ⓗ Ⓗ ⓗ a".toTitle == "A Ⓗ Ⓗ Ⓗ A"
+  check "ⓗola".toTitle == "Ⓗola"
+  check "諸".toTitle == "諸"
+  check "ab 諸 ab ⓗ ab".toTitle == "Ab 諸 Ab Ⓗ Ab"
+  check "abc 弢ⒶΪ def".toTitle == "Abc 弢ⒶΪ Def"
+  check "abc 弢Ⓐ def".toTitle == "Abc 弢Ⓐ Def"
+  check "abc ⒶΪ def".toTitle == "Abc ⒶΪ Def"
+  check "abc 弢Ϊ def".toTitle == "Abc 弢Ϊ Def"
+  check "abc 弢 def".toTitle == "Abc 弢 Def"
+  check "abc Ⓐ def".toTitle == "Abc Ⓐ Def"
+  check "abc Ϊ def".toTitle == "Abc Ϊ Def"
+  check "ßear".toTitle == "Ssear"
+
+test "toUpper":
+  check "a".toUpper == "A"
+  check "1".toUpper == "1"
+  check "some title".toUpper == "SOME TITLE"
+  check "The quick? (“brown”) fox can’t jump 32.3 feet, right?".toUpper ==
+    "THE QUICK? (“BROWN”) FOX CAN’T JUMP 32.3 FEET, RIGHT?"
+  check "a ⓗ Ⓗ ⓗ a".toUpper == "A Ⓗ Ⓗ Ⓗ A"
+  check "諸".toUpper == "諸"
+  check "ab 諸 ab ⓗ ab".toUpper == "AB 諸 AB Ⓗ AB"
+  check "abc 弢ⒶΪ def".toUpper == "ABC 弢ⒶΪ DEF"
+  check "abc 弢Ⓐ def".toUpper == "ABC 弢Ⓐ DEF"
+  check "abc ⒶΪ def".toUpper == "ABC ⒶΪ DEF"
+  check "abc 弢Ϊ def".toUpper == "ABC 弢Ϊ DEF"
+  check "abc 弢 def".toUpper == "ABC 弢 DEF"
+  check "abc Ⓐ def".toUpper == "ABC Ⓐ DEF"
+  check "abc Ϊ def".toUpper == "ABC Ϊ DEF"
+  check "ßear".toUpper == "SSEAR"
+  block testLongerResult:
+    const be = "ΐeΐeΐeΐe-ΐeΐeΐeΐe-ΐeΐeΐeΐe-ΐeΐeΐeΐe"
+    const se = "Ϊ́EΪ́EΪ́EΪ́E-Ϊ́EΪ́EΪ́EΪ́E-Ϊ́EΪ́EΪ́EΪ́E-Ϊ́EΪ́EΪ́EΪ́E"
+    check be.len < se.len
+    check (be & be & be & be & be & be & be & be).toUpper ==
+          (se & se & se & se & se & se & se & se)
+
+test "toLower":
+  check "A".toLower == "a"
+  check "1".toLower == "1"
+  check "SOME TITLE".toLower == "some title"
+  check "The quIck? (“bRown”) fox cAn’T jUMp 32.3 feet, rIGHt?".toLower ==
+    "the quick? (“brown”) fox can’t jump 32.3 feet, right?"
+  check "A ⓗ Ⓗ ⓗ A".toLower == "a ⓗ ⓗ ⓗ a"
+  check "諸".toLower == "諸"
+  check "AB 諸 AB Ⓗ AB".toLower == "ab 諸 ab ⓗ ab"
+  check "ABC 弢ⒶΪ DEF".toLower == "abc 弢ⓐϊ def"
+  check "ABC 弢Ⓐ DEF".toLower == "abc 弢ⓐ def"
+  check "ABC ⒶΪ DEF".toLower == "abc ⓐϊ def"
+  check "ABC 弢Ϊ DEF".toLower == "abc 弢ϊ def"
+  check "ABC 弢 DEF".toLower == "abc 弢 def"
+  check "ABC Ⓐ DEF".toLower == "abc ⓐ def"
+  check "ABC Ϊ DEF".toLower == "abc ϊ def"
+  block testLongerResult:
+    const be = "İEİEİEİE-İEİEİEİE-İEİEİEİE-İEİEİEİE"
+    const se = "i̇ei̇ei̇ei̇e-i̇ei̇ei̇ei̇e-i̇ei̇ei̇ei̇e-i̇ei̇ei̇ei̇e"
+    check be.len < se.len
+    check (be & be & be & be & be & be & be & be).toLower ==
+          (se & se & se & se & se & se & se & se)
+
+test "cmpCaseless":
+  check cmpCaseless("a", "a")
+  check cmpCaseless("ab", "ab")
+  check cmpCaseless("abc", "abc")
+  check cmpCaseless("ABC", "abc")
+  check(not cmpCaseless("b", "a"))
+  check(not cmpCaseless("abc", "abcd"))
+  check(not cmpCaseless("abcd", "abc"))
+  check cmpCaseless("σ", "Σ")
+  check cmpCaseless("ΐe", "Ϊ́E")
+  check cmpCaseless("Ϊ́E", "ΐe")
+  check cmpCaseless("CafÉ", "Café")
+  check cmpCaseless("Caf\u0045\u0301", "Caf\u0065\u0301")
+  check cmpCaseless("\u1FA0", "\u1F60\u03B9")  # pre-CaseFolded
+  check cmpCaseless("\u1F60\u03B9", "\u1FA0")
+  check cmpCaseless("\u1F60\u03B9\u1FA0", "\u1FA0\u1F60\u03B9")
+  check cmpCaseless("\u1FA0\u1F60\u03B9", "\u1F60\u03B9\u1FA0")
+  check cmpCaseless("\u1FA0\u1F60\u03B9", "\u1FA0\u1F60\u03B9")
+  check(not cmpCaseless("\u1FA0", "x"))
+
+test "cmpCaseless all":
+  for cp in 0 .. 0x10FFFF:
+    var s = ""
+    for r in caseFold(cp.Rune):
+      s.add r.toUtf8
+    let org = cp.Rune.toUtf8
+    check cmpCaseless(s, org)
+    check cmpCaseless(org, s)
+    check cmpCaseless(s & org, org & s)
+    check cmpCaseless(org & s, s & org)
+    check cmpCaseless(org & s & org, s & org & s)
+    check cmpCaseless(s & org & s, org & s & org)
+    check(not cmpCaseless(s & "a", org))
+    check(not cmpCaseless(org & "a", s))
+    check(not cmpCaseless(s & org & "a", org & s))
+
+test "cmpCaseless max":
+  var cMax = 0
+  var i = 0
+  for cp in 0 .. 0x10FFFF:
+    i = 0
+    for r in caseFold(cp.Rune):
+      inc i
+    cMax = max(cMax, i)
+  # we may have to increase the
+  # `cmpCaseless` buffer size if this increases
+  doAssert cMax == 3
+  # caseFold(0x1FE2) == 3 Runes
+  # caseFold(0x1FE4) == 2 Runes
+  # max filled buff is 6
+  check(not cmpCaseless("\u1FE4\u1FE2", "\u1FE2\u1FE2"))
+  check(not cmpCaseless("\u1FE2\u1FE2", "\u1FE4\u1FE2"))
diff --git a/unicodeplus.nimble b/unicodeplus.nimble
index 0a21238..fc585ee 100644
--- a/unicodeplus.nimble
+++ b/unicodeplus.nimble
@@ -1,6 +1,6 @@
 # Package
 
-version = "0.5.1"
+version = "0.8.0"
 author = "Esteban Castro Borsani (@nitely)"
 description = "Common unicode operations"
 license = "MIT"
@@ -8,11 +8,12 @@ srcDir = "src"
 
 # Dependencies
 
-requires "nim >= 0.18.0"
-requires "unicodedb >= 0.7"
+requires "nim >= 0.19.0"
+requires "unicodedb >= 0.8"
+requires "segmentation >= 0.1"
 
 task test, "Test":
   exec "nim c -r tests/tests"
 
 task docs, "Docs":
-  exec "nim doc2 -o:./docs/index.html ./src/unicodeplus.nim"
+  exec "nim doc --project -o:./docs ./src/unicodeplus.nim"

Debdiff

File lists identical (after any substitutions)

No differences were encountered in the control files

More details

Full run details