From 0e1932e70954e0b87e0f24b601f9d831a0113120 Mon Sep 17 00:00:00 2001 From: Florian Hammerschmidt Date: Fri, 6 Dec 2024 23:44:40 +0100 Subject: [PATCH] V12 docs --- data/api/latest/belt.json | 685 +- data/api/latest/core.json | 99 +- data/api/latest/js.json | 5495 +++++------------ data/api/latest/toc_tree.json | 2 +- data/sidebar_gentype_latest.json | 10 - data/sidebar_manual_latest.json | 78 - next.config.mjs | 21 +- package.json | 2 +- pages/docs/latest.js | 3 - pages/docs/manual/latest/api.mdx | 18 - pages/docs/manual/latest/api/[...slug].js | 18 - pages/docs/manual/latest/array-and-list.mdx | 219 - pages/docs/manual/latest/async-await.mdx | 326 - pages/docs/manual/latest/attribute.mdx | 90 - .../latest/bind-to-global-js-values.mdx | 145 - .../manual/latest/bind-to-js-function.mdx | 490 -- .../docs/manual/latest/bind-to-js-object.mdx | 197 - .../latest/browser-support-polyfills.mdx | 18 - .../latest/build-configuration-schema.mdx | 23 - .../manual/latest/build-configuration.mdx | 245 - .../manual/latest/build-external-stdlib.mdx | 59 - pages/docs/manual/latest/build-overview.mdx | 83 - .../docs/manual/latest/build-performance.mdx | 85 - .../latest/build-pinned-dependencies.mdx | 110 - pages/docs/manual/latest/control-flow.mdx | 211 - .../docs/manual/latest/converting-from-js.mdx | 303 - pages/docs/manual/latest/editor-plugins.mdx | 20 - .../manual/latest/embed-raw-javascript.mdx | 110 - .../manual/latest/equality-comparison.mdx | 125 - pages/docs/manual/latest/exception.mdx | 498 -- .../docs/manual/latest/extensible-variant.mdx | 75 - pages/docs/manual/latest/external.mdx | 91 - pages/docs/manual/latest/faq.mdx | 66 - pages/docs/manual/latest/function.mdx | 620 -- .../latest/generate-converters-accessors.mdx | 114 - pages/docs/manual/latest/import-export.mdx | 38 - .../latest/import-from-export-to-js.mdx | 244 - .../docs/manual/latest/inlining-constants.mdx | 78 - pages/docs/manual/latest/installation.mdx | 153 - .../docs/manual/latest/interop-cheatsheet.mdx | 274 - .../latest/interop-with-js-build-systems.mdx | 57 - pages/docs/manual/latest/introduction.mdx | 80 - pages/docs/manual/latest/json.mdx | 82 - pages/docs/manual/latest/jsx.mdx | 383 -- pages/docs/manual/latest/lazy-values.mdx | 126 - pages/docs/manual/latest/let-binding.mdx | 185 - pages/docs/manual/latest/libraries.mdx | 27 - pages/docs/manual/latest/migrate-to-v11.mdx | 131 - pages/docs/manual/latest/module-functions.mdx | 311 - pages/docs/manual/latest/module.mdx | 569 -- pages/docs/manual/latest/mutation.mdx | 75 - .../docs/manual/latest/newcomer-examples.mdx | 134 - .../manual/latest/null-undefined-option.mdx | 203 - pages/docs/manual/latest/object.mdx | 195 - pages/docs/manual/latest/overview.mdx | 255 - .../latest/pattern-matching-destructuring.mdx | 864 --- pages/docs/manual/latest/pipe.mdx | 244 - .../manual/latest/polymorphic-variant.mdx | 433 -- pages/docs/manual/latest/primitive-types.mdx | 272 - .../docs/manual/latest/project-structure.mdx | 57 - pages/docs/manual/latest/promise.mdx | 182 - pages/docs/manual/latest/record.mdx | 557 -- .../docs/manual/latest/reserved-keywords.mdx | 92 - .../latest/scoped-polymorphic-types.mdx | 100 - .../docs/manual/latest/shared-data-types.mdx | 44 - pages/docs/manual/latest/tagged-templates.mdx | 124 - pages/docs/manual/latest/try.mdx | 9 - pages/docs/manual/latest/tuple.mdx | 104 - pages/docs/manual/latest/type.mdx | 278 - .../manual/latest/typescript-integration.mdx | 283 - .../latest/use-illegal-identifier-names.mdx | 51 - pages/docs/manual/latest/variant.mdx | 798 --- pages/docs/manual/latest/warning-numbers.mdx | 15 - scripts/extract-indices.mjs | 5 +- scripts/extract-tocs.mjs | 109 +- src/ApiDocs.res | 12 +- src/DocsOverview.res | 21 +- src/common/App.res | 8 + src/common/Constants.res | 16 +- src/common/Url.res | 11 +- src/common/Url.resi | 3 + src/components/Navigation.res | 18 +- src/components/Search.res | 5 +- src/components/VersionSelect.res | 19 +- src/components/VersionSelect.resi | 1 + src/layouts/BeltDocsLayout10_0_0.res | 6 +- src/layouts/BeltDocsLayout8_0_0.res | 6 +- src/layouts/BeltDocsLayout9_0_0.res | 6 +- src/layouts/DocsLayout.res | 8 +- src/layouts/DocsLayout.resi | 1 + src/layouts/DomDocsLayout10_0_0.res | 6 +- src/layouts/DomDocsLayout8_0_0.res | 6 +- src/layouts/DomDocsLayout9_0_0.res | 6 +- src/layouts/JsDocsLayout10_0_0.res | 6 +- src/layouts/JsDocsLayout8_0_0.res | 6 +- src/layouts/JsDocsLayout9_0_0.res | 6 +- src/layouts/ManualDocsLayout.res | 90 +- src/layouts/ReactDocsLayout.res | 63 +- src/layouts/ReasonCompilerDocsLayout.res | 38 +- 99 files changed, 2273 insertions(+), 17070 deletions(-) delete mode 100644 data/sidebar_gentype_latest.json delete mode 100644 data/sidebar_manual_latest.json delete mode 100644 pages/docs/latest.js delete mode 100644 pages/docs/manual/latest/api.mdx delete mode 100644 pages/docs/manual/latest/api/[...slug].js delete mode 100644 pages/docs/manual/latest/array-and-list.mdx delete mode 100644 pages/docs/manual/latest/async-await.mdx delete mode 100644 pages/docs/manual/latest/attribute.mdx delete mode 100644 pages/docs/manual/latest/bind-to-global-js-values.mdx delete mode 100644 pages/docs/manual/latest/bind-to-js-function.mdx delete mode 100644 pages/docs/manual/latest/bind-to-js-object.mdx delete mode 100644 pages/docs/manual/latest/browser-support-polyfills.mdx delete mode 100644 pages/docs/manual/latest/build-configuration-schema.mdx delete mode 100644 pages/docs/manual/latest/build-configuration.mdx delete mode 100644 pages/docs/manual/latest/build-external-stdlib.mdx delete mode 100644 pages/docs/manual/latest/build-overview.mdx delete mode 100644 pages/docs/manual/latest/build-performance.mdx delete mode 100644 pages/docs/manual/latest/build-pinned-dependencies.mdx delete mode 100644 pages/docs/manual/latest/control-flow.mdx delete mode 100644 pages/docs/manual/latest/converting-from-js.mdx delete mode 100644 pages/docs/manual/latest/editor-plugins.mdx delete mode 100644 pages/docs/manual/latest/embed-raw-javascript.mdx delete mode 100644 pages/docs/manual/latest/equality-comparison.mdx delete mode 100644 pages/docs/manual/latest/exception.mdx delete mode 100644 pages/docs/manual/latest/extensible-variant.mdx delete mode 100644 pages/docs/manual/latest/external.mdx delete mode 100644 pages/docs/manual/latest/faq.mdx delete mode 100644 pages/docs/manual/latest/function.mdx delete mode 100644 pages/docs/manual/latest/generate-converters-accessors.mdx delete mode 100644 pages/docs/manual/latest/import-export.mdx delete mode 100644 pages/docs/manual/latest/import-from-export-to-js.mdx delete mode 100644 pages/docs/manual/latest/inlining-constants.mdx delete mode 100644 pages/docs/manual/latest/installation.mdx delete mode 100644 pages/docs/manual/latest/interop-cheatsheet.mdx delete mode 100644 pages/docs/manual/latest/interop-with-js-build-systems.mdx delete mode 100644 pages/docs/manual/latest/introduction.mdx delete mode 100644 pages/docs/manual/latest/json.mdx delete mode 100644 pages/docs/manual/latest/jsx.mdx delete mode 100644 pages/docs/manual/latest/lazy-values.mdx delete mode 100644 pages/docs/manual/latest/let-binding.mdx delete mode 100644 pages/docs/manual/latest/libraries.mdx delete mode 100644 pages/docs/manual/latest/migrate-to-v11.mdx delete mode 100644 pages/docs/manual/latest/module-functions.mdx delete mode 100644 pages/docs/manual/latest/module.mdx delete mode 100644 pages/docs/manual/latest/mutation.mdx delete mode 100644 pages/docs/manual/latest/newcomer-examples.mdx delete mode 100644 pages/docs/manual/latest/null-undefined-option.mdx delete mode 100644 pages/docs/manual/latest/object.mdx delete mode 100644 pages/docs/manual/latest/overview.mdx delete mode 100644 pages/docs/manual/latest/pattern-matching-destructuring.mdx delete mode 100644 pages/docs/manual/latest/pipe.mdx delete mode 100644 pages/docs/manual/latest/polymorphic-variant.mdx delete mode 100644 pages/docs/manual/latest/primitive-types.mdx delete mode 100644 pages/docs/manual/latest/project-structure.mdx delete mode 100644 pages/docs/manual/latest/promise.mdx delete mode 100644 pages/docs/manual/latest/record.mdx delete mode 100644 pages/docs/manual/latest/reserved-keywords.mdx delete mode 100644 pages/docs/manual/latest/scoped-polymorphic-types.mdx delete mode 100644 pages/docs/manual/latest/shared-data-types.mdx delete mode 100644 pages/docs/manual/latest/tagged-templates.mdx delete mode 100644 pages/docs/manual/latest/try.mdx delete mode 100644 pages/docs/manual/latest/tuple.mdx delete mode 100644 pages/docs/manual/latest/type.mdx delete mode 100644 pages/docs/manual/latest/typescript-integration.mdx delete mode 100644 pages/docs/manual/latest/use-illegal-identifier-names.mdx delete mode 100644 pages/docs/manual/latest/variant.mdx delete mode 100644 pages/docs/manual/latest/warning-numbers.mdx diff --git a/data/api/latest/belt.json b/data/api/latest/belt.json index 86bc0c43e..94541ab50 100644 --- a/data/api/latest/belt.json +++ b/data/api/latest/belt.json @@ -3,7 +3,7 @@ "id": "Belt", "name": "Belt", "docstrings": [ - "The ReScript standard library.\n\nBelt is currently mostly covering collection types. It has no string or date functions yet, although Belt.String is in the works. In the meantime, use [Js.String](js/string) for string functions and [Js.Date](js/date) for date functions.\n\n## Motivation\n\nBelt provides:\n\n- The **highest quality** immutable data structures in JavaScript.\n- Safety by default: A Belt function will never throw exceptions, unless it is\n indicated explicitly in the function name (suffix \"Exn\").\n- Better performance and smaller code size running on the JS platform.\n- Ready for [Tree Shaking](https://webpack.js.org/guides/tree-shaking/).\n\n## Usage\n\nTo use modules from Belt, either refer to them by their fully qualified name (`Belt.List`, `Belt.Array` etc.) or open the `Belt` module by putting\n\n## Examples\n\n```rescript\nopen Belt\n```\n\nat the top of your source files. After opening Belt this way, `Array` will refer to `Belt.Array`, `List` will refer to `Belt.List` etc. in the subsequent code.\n\nIf you want to open Belt globally for all files in your project instead, you can put\n\n```json\n{\n \"bsc-flags\": [\"-open Belt\"]\n}\n```\n\ninto your `bsconfig.json`.\n\n**Note**: this is the **only** `open` we encourage.\n\nExample usage:\n\n## Examples\n\n```rescript\nlet someNumbers = [1, 1, 4, 2, 3, 6, 3, 4, 2]\n\nlet greaterThan2UniqueAndSorted =\n someNumbers\n ->Belt.Array.keep(x => x > 2)\n // convert to and from set to make values unique\n ->Belt.Set.Int.fromArray\n ->Belt.Set.Int.toArray // output is already sorted\n\nJs.log2(\"result\", greaterThan2UniqueAndSorted)\n```\n\n## Curried vs. Uncurried Callbacks\n\nFor functions taking a callback parameter, there are usually two versions\navailable:\n\n- curried (no suffix)\n- uncurried (suffixed with `U`)\n\nE.g.:\n\n## Examples\n\n```rescript\nlet forEach: (t<'a>, 'a => unit) => unit\n\nlet forEachU: (t<'a>, (. 'a) => unit) => unit\n```\n\nThe uncurried version will be faster in some cases, but for simplicity we recommend to stick with the curried version unless you need the extra performance.\n\nThe two versions can be invoked as follows:\n\n## Examples\n\n```rescript\n[\"a\", \"b\", \"c\"]->Belt.Array.forEach(x => Js.log(x))\n\n[\"a\", \"b\", \"c\"]->Belt.Array.forEachU((. x) => Js.log(x))\n```\n\n## Specialized Collections\n\nFor collections types like set or map, Belt provides both a generic module as well as specialized, more efficient implementations for string and int keys.\n\nFor example, Belt has the following set modules:\n\n- [Belt.Set](belt/set)\n- [Belt.Set.Int](belt/set/int)\n- [Belt.Set.String](belt/set/string)\n\n## Implementation Details\n\n### Array access runtime safety\n\nOne common confusion comes from the way Belt handles array access. It differs from than the default standard library's.\n\n## Examples\n\n```rescript\nlet letters = [\"a\", \"b\", \"c\"]\nlet a = letters[0] // a == \"a\"\nlet capitalA = Js.String.toUpperCase(a)\nlet k = letters[10] // Raises an exception! The 10th index doesn't exist.\n```\n\nBecause Belt avoids exceptions and returns `options` instead, this code behaves differently:\n\n## Examples\n\n```rescript\nopen Belt\nlet letters = [\"a\", \"b\", \"c\"]\nlet a = letters[0] // a == Some(\"a\")\nlet captialA = Js.String.toUpperCase(a) // Type error! This code will not compile.\nlet k = letters[10] // k == None\n```\n\nAlthough we've fixed the problem where `k` raises an exception, we now have a type error when trying to capitalize `a`. There are a few things going on here:\n\n- Reason transforms array index access to the function `Array.get`. So `letters[0]` is the same as `Array.get(letters, 0)`.\n- The compiler uses whichever `Array` module is in scope. If you `open Belt`, then it uses `Belt.Array`.\n- `Belt.Array.get` returns values wrapped in options, so `letters[0] == Some(\"a\")`.\n\nFortunately, this is easy to fix:\n\n## Examples\n\n```rescript\nopen Belt\nlet letters = [\"a\", \"b\", \"c\"]\nlet a = letters[0]\n\n// Use a switch statement:\nlet capitalA =\n switch a {\n | Some(a) => Some(Js.String.toUpperCase(a))\n | None => None\n }\n\nlet k = letters[10] // k == None\n```\n\nWith that little bit of tweaking, our code now compiles successfully and is 100% free of runtime errors!\n\n### A Special Encoding for Collection Safety\n\nWhen we create a collection library for a custom data type we need a way to provide a comparator function. Take Set for example, suppose its element type is a pair of ints, it needs a custom compare function that takes two tuples and returns their order. The Set could not just be typed as Set.t (int \\* int) , its customized compare function needs to manifest itself in the signature, otherwise, if the user creates another customized compare function, the two collection could mix which would result in runtime error.\n\nWe use a phantom type to solve the problem:\n\n## Examples\n\n```rescript\nmodule Comparable1 =\n Belt.Id.MakeComparable(\n {\n type t = (int, int)\n let cmp = ((a0, a1), (b0, b1)) =>\n switch Pervasives.compare(a0, b0) {\n | 0 => Pervasives.compare(a1, b1)\n | c => c\n }\n }\n )\n\nlet mySet1 = Belt.Set.make(~id=module(Comparable1))\n\nmodule Comparable2 =\n Belt.Id.MakeComparable(\n {\n type t = (int, int)\n let cmp = ((a0, a1), (b0, b1)) =>\n switch Pervasives.compare(a0, b0) {\n | 0 => Pervasives.compare(a1, b1)\n | c => c\n }\n }\n )\n\nlet mySet2 = Belt.Set.make(~id=module(Comparable2))\n```\n\nHere, the compiler would infer `mySet1` and `mySet2` having different type, so e.g. a `merge` operation that tries to merge these two sets will correctly fail.\n\n## Examples\n\n```rescript\nlet mySet1: t<(int, int), Comparable1.identity>\nlet mySet2: t<(int, int), Comparable2.identity>\n```\n\n`Comparable1.identity` and `Comparable2.identity` are not the same using our encoding scheme." + "The ReScript standard library.\n\nBelt is currently mostly covering collection types. It has no string or date functions yet, although Belt.String is in the works. In the meantime, use [Js.String](js/string) for string functions and [Js.Date](js/date) for date functions.\n\n## Motivation\n\nBelt provides:\n\n- The **highest quality** immutable data structures in JavaScript.\n- Safety by default: A Belt function will never throw exceptions, unless it is\n indicated explicitly in the function name (suffix \"Exn\").\n- Better performance and smaller code size running on the JS platform.\n- Ready for [Tree Shaking](https://webpack.js.org/guides/tree-shaking/).\n\n## Usage\n\nTo use modules from Belt, either refer to them by their fully qualified name (`Belt.List`, `Belt.Array` etc.) or open the `Belt` module by putting\n\n## Examples\n\n```rescript\nopen Belt\n```\n\nat the top of your source files. After opening Belt this way, `Array` will refer to `Belt.Array`, `List` will refer to `Belt.List` etc. in the subsequent code.\n\nIf you want to open Belt globally for all files in your project instead, you can put\n\n```json\n{\n \"bsc-flags\": [\"-open Belt\"]\n}\n```\n\ninto your `bsconfig.json`.\n\n**Note**: this is the **only** `open` we encourage.\n\nExample usage:\n\n## Examples\n\n```rescript\nlet someNumbers = [1, 1, 4, 2, 3, 6, 3, 4, 2]\n\nlet greaterThan2UniqueAndSorted =\n someNumbers\n ->Belt.Array.keep(x => x > 2)\n // convert to and from set to make values unique\n ->Belt.Set.Int.fromArray\n ->Belt.Set.Int.toArray // output is already sorted\n\nJs.log2(\"result\", greaterThan2UniqueAndSorted)\n```\n\n## Specialized Collections\n\nFor collections types like set or map, Belt provides both a generic module as well as specialized, more efficient implementations for string and int keys.\n\nFor example, Belt has the following set modules:\n\n- [Belt.Set](belt/set)\n- [Belt.Set.Int](belt/set/int)\n- [Belt.Set.String](belt/set/string)\n\n## Implementation Details\n\n### Array access runtime safety\n\nOne common confusion comes from the way Belt handles array access. It differs from than the default standard library's.\n\n## Examples\n\n```rescript\nlet letters = [\"a\", \"b\", \"c\"]\nlet a = letters[0] // a == \"a\"\nlet capitalA = Js.String.toUpperCase(a)\nlet k = letters[10] // Raises an exception! The 10th index doesn't exist.\n```\n\nBecause Belt avoids exceptions and returns `options` instead, this code behaves differently:\n\n## Examples\n\n```rescript\nopen Belt\nlet letters = [\"a\", \"b\", \"c\"]\nlet a = letters[0] // a == Some(\"a\")\nlet captialA = Js.String.toUpperCase(a) // Type error! This code will not compile.\nlet k = letters[10] // k == None\n```\n\nAlthough we've fixed the problem where `k` raises an exception, we now have a type error when trying to capitalize `a`. There are a few things going on here:\n\n- Reason transforms array index access to the function `Array.get`. So `letters[0]` is the same as `Array.get(letters, 0)`.\n- The compiler uses whichever `Array` module is in scope. If you `open Belt`, then it uses `Belt.Array`.\n- `Belt.Array.get` returns values wrapped in options, so `letters[0] == Some(\"a\")`.\n\nFortunately, this is easy to fix:\n\n## Examples\n\n```rescript\nopen Belt\nlet letters = [\"a\", \"b\", \"c\"]\nlet a = letters[0]\n\n// Use a switch statement:\nlet capitalA =\n switch a {\n | Some(a) => Some(Js.String.toUpperCase(a))\n | None => None\n }\n\nlet k = letters[10] // k == None\n```\n\nWith that little bit of tweaking, our code now compiles successfully and is 100% free of runtime errors!\n\n### A Special Encoding for Collection Safety\n\nWhen we create a collection library for a custom data type we need a way to provide a comparator function. Take Set for example, suppose its element type is a pair of ints, it needs a custom compare function that takes two tuples and returns their order. The Set could not just be typed as Set.t (int \\* int) , its customized compare function needs to manifest itself in the signature, otherwise, if the user creates another customized compare function, the two collection could mix which would result in runtime error.\n\nWe use a phantom type to solve the problem:\n\n## Examples\n\n```rescript\nmodule Comparable1 =\n Belt.Id.MakeComparable(\n {\n type t = (int, int)\n let cmp = ((a0, a1), (b0, b1)) =>\n switch Pervasives.compare(a0, b0) {\n | 0 => Pervasives.compare(a1, b1)\n | c => c\n }\n }\n )\n\nlet mySet1 = Belt.Set.make(~id=module(Comparable1))\n\nmodule Comparable2 =\n Belt.Id.MakeComparable(\n {\n type t = (int, int)\n let cmp = ((a0, a1), (b0, b1)) =>\n switch Pervasives.compare(a0, b0) {\n | 0 => Pervasives.compare(a1, b1)\n | c => c\n }\n }\n )\n\nlet mySet2 = Belt.Set.make(~id=module(Comparable2))\n```\n\nHere, the compiler would infer `mySet1` and `mySet2` having different type, so e.g. a `merge` operation that tries to merge these two sets will correctly fail.\n\n## Examples\n\n```rescript\nlet mySet1: t<(int, int), Comparable1.identity>\nlet mySet2: t<(int, int), Comparable2.identity>\n```\n\n`Comparable1.identity` and `Comparable2.identity` are not the same using our encoding scheme." ], "items": [] }, @@ -91,7 +91,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t<'b>, (key, 'b) => unit) => unit" + "signature": "let forEachU: (t<'b>, (key, 'b) => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.HashMap.String.forEach", @@ -105,7 +106,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t<'b>, 'c, ('c, key, 'b) => 'c) => 'c" + "signature": "let reduceU: (t<'b>, 'c, ('c, key, 'b) => 'c) => 'c", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.HashMap.String.reduce", @@ -119,7 +121,8 @@ "kind": "value", "name": "keepMapInPlaceU", "docstrings": [], - "signature": "let keepMapInPlaceU: (t<'a>, (key, 'a) => option<'a>) => unit" + "signature": "let keepMapInPlaceU: (t<'a>, (key, 'a) => option<'a>) => unit", + "deprecated": "Use `keepMapInPlace` instead" }, { "id": "Belt.HashMap.String.keepMapInPlace", @@ -270,7 +273,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t<'b>, (key, 'b) => unit) => unit" + "signature": "let forEachU: (t<'b>, (key, 'b) => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.HashMap.Int.forEach", @@ -284,7 +288,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t<'b>, 'c, ('c, key, 'b) => 'c) => 'c" + "signature": "let reduceU: (t<'b>, 'c, ('c, key, 'b) => 'c) => 'c", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.HashMap.Int.reduce", @@ -298,7 +303,8 @@ "kind": "value", "name": "keepMapInPlaceU", "docstrings": [], - "signature": "let keepMapInPlaceU: (t<'a>, (key, 'a) => option<'a>) => unit" + "signature": "let keepMapInPlaceU: (t<'a>, (key, 'a) => option<'a>) => unit", + "deprecated": "Use `keepMapInPlace` instead" }, { "id": "Belt.HashMap.Int.keepMapInPlace", @@ -441,7 +447,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t, key => unit) => unit" + "signature": "let forEachU: (t, key => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.HashSet.String.forEach", @@ -455,7 +462,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t, 'c, ('c, key) => 'c) => 'c" + "signature": "let reduceU: (t, 'c, ('c, key) => 'c) => 'c", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.HashSet.String.reduce", @@ -584,7 +592,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t, key => unit) => unit" + "signature": "let forEachU: (t, key => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.HashSet.Int.forEach", @@ -598,7 +607,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t, 'c, ('c, key) => 'c) => 'c" + "signature": "let reduceU: (t, 'c, ('c, key) => 'c) => 'c", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.HashSet.Int.reduce", @@ -703,7 +713,8 @@ "kind": "value", "name": "cmpU", "docstrings": [], - "signature": "let cmpU: (t<'a>, t<'a>, ('a, 'a) => int) => int" + "signature": "let cmpU: (t<'a>, t<'a>, ('a, 'a) => int) => int", + "deprecated": "Use `cmp` instead" }, { "id": "Belt.MutableMap.String.cmp", @@ -719,7 +730,8 @@ "kind": "value", "name": "eqU", "docstrings": [], - "signature": "let eqU: (t<'a>, t<'a>, ('a, 'a) => bool) => bool" + "signature": "let eqU: (t<'a>, t<'a>, ('a, 'a) => bool) => bool", + "deprecated": "Use `eq` instead" }, { "id": "Belt.MutableMap.String.eq", @@ -735,7 +747,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t<'a>, (key, 'a) => unit) => unit" + "signature": "let forEachU: (t<'a>, (key, 'a) => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.MutableMap.String.forEach", @@ -751,7 +764,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t<'a>, 'b, ('b, key, 'a) => 'b) => 'b" + "signature": "let reduceU: (t<'a>, 'b, ('b, key, 'a) => 'b) => 'b", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.MutableMap.String.reduce", @@ -767,7 +781,8 @@ "kind": "value", "name": "everyU", "docstrings": [], - "signature": "let everyU: (t<'a>, (key, 'a) => bool) => bool" + "signature": "let everyU: (t<'a>, (key, 'a) => bool) => bool", + "deprecated": "Use `every` instead" }, { "id": "Belt.MutableMap.String.every", @@ -783,7 +798,8 @@ "kind": "value", "name": "someU", "docstrings": [], - "signature": "let someU: (t<'a>, (key, 'a) => bool) => bool" + "signature": "let someU: (t<'a>, (key, 'a) => bool) => bool", + "deprecated": "Use `some` instead" }, { "id": "Belt.MutableMap.String.some", @@ -963,7 +979,8 @@ "kind": "value", "name": "updateU", "docstrings": [], - "signature": "let updateU: (t<'a>, key, option<'a> => option<'a>) => unit" + "signature": "let updateU: (t<'a>, key, option<'a> => option<'a>) => unit", + "deprecated": "Use `update` instead" }, { "id": "Belt.MutableMap.String.update", @@ -977,7 +994,8 @@ "kind": "value", "name": "mapU", "docstrings": [], - "signature": "let mapU: (t<'a>, 'a => 'b) => t<'b>" + "signature": "let mapU: (t<'a>, 'a => 'b) => t<'b>", + "deprecated": "Use `map` instead" }, { "id": "Belt.MutableMap.String.map", @@ -993,7 +1011,8 @@ "kind": "value", "name": "mapWithKeyU", "docstrings": [], - "signature": "let mapWithKeyU: (t<'a>, (key, 'a) => 'b) => t<'b>" + "signature": "let mapWithKeyU: (t<'a>, (key, 'a) => 'b) => t<'b>", + "deprecated": "Use `mapWithKey` instead" }, { "id": "Belt.MutableMap.String.mapWithKey", @@ -1056,7 +1075,8 @@ "kind": "value", "name": "cmpU", "docstrings": [], - "signature": "let cmpU: (t<'a>, t<'a>, ('a, 'a) => int) => int" + "signature": "let cmpU: (t<'a>, t<'a>, ('a, 'a) => int) => int", + "deprecated": "Use `cmp` instead" }, { "id": "Belt.MutableMap.Int.cmp", @@ -1072,7 +1092,8 @@ "kind": "value", "name": "eqU", "docstrings": [], - "signature": "let eqU: (t<'a>, t<'a>, ('a, 'a) => bool) => bool" + "signature": "let eqU: (t<'a>, t<'a>, ('a, 'a) => bool) => bool", + "deprecated": "Use `eq` instead" }, { "id": "Belt.MutableMap.Int.eq", @@ -1088,7 +1109,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t<'a>, (key, 'a) => unit) => unit" + "signature": "let forEachU: (t<'a>, (key, 'a) => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.MutableMap.Int.forEach", @@ -1104,7 +1126,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t<'a>, 'b, ('b, key, 'a) => 'b) => 'b" + "signature": "let reduceU: (t<'a>, 'b, ('b, key, 'a) => 'b) => 'b", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.MutableMap.Int.reduce", @@ -1120,7 +1143,8 @@ "kind": "value", "name": "everyU", "docstrings": [], - "signature": "let everyU: (t<'a>, (key, 'a) => bool) => bool" + "signature": "let everyU: (t<'a>, (key, 'a) => bool) => bool", + "deprecated": "Use `every` instead" }, { "id": "Belt.MutableMap.Int.every", @@ -1136,7 +1160,8 @@ "kind": "value", "name": "someU", "docstrings": [], - "signature": "let someU: (t<'a>, (key, 'a) => bool) => bool" + "signature": "let someU: (t<'a>, (key, 'a) => bool) => bool", + "deprecated": "Use `some` instead" }, { "id": "Belt.MutableMap.Int.some", @@ -1316,7 +1341,8 @@ "kind": "value", "name": "updateU", "docstrings": [], - "signature": "let updateU: (t<'a>, key, option<'a> => option<'a>) => unit" + "signature": "let updateU: (t<'a>, key, option<'a> => option<'a>) => unit", + "deprecated": "Use `update` instead" }, { "id": "Belt.MutableMap.Int.update", @@ -1330,7 +1356,8 @@ "kind": "value", "name": "mapU", "docstrings": [], - "signature": "let mapU: (t<'a>, 'a => 'b) => t<'b>" + "signature": "let mapU: (t<'a>, 'a => 'b) => t<'b>", + "deprecated": "Use `map` instead" }, { "id": "Belt.MutableMap.Int.map", @@ -1346,7 +1373,8 @@ "kind": "value", "name": "mapWithKeyU", "docstrings": [], - "signature": "let mapWithKeyU: (t<'a>, (key, 'a) => 'b) => t<'b>" + "signature": "let mapWithKeyU: (t<'a>, (key, 'a) => 'b) => t<'b>", + "deprecated": "Use `mapWithKey` instead" }, { "id": "Belt.MutableMap.Int.mapWithKey", @@ -1514,7 +1542,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t, value => unit) => unit" + "signature": "let forEachU: (t, value => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.MutableSet.String.forEach", @@ -1530,7 +1559,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t, 'a, ('a, value) => 'a) => 'a" + "signature": "let reduceU: (t, 'a, ('a, value) => 'a) => 'a", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.MutableSet.String.reduce", @@ -1546,7 +1576,8 @@ "kind": "value", "name": "everyU", "docstrings": [], - "signature": "let everyU: (t, value => bool) => bool" + "signature": "let everyU: (t, value => bool) => bool", + "deprecated": "Use `every` instead" }, { "id": "Belt.MutableSet.String.every", @@ -1562,7 +1593,8 @@ "kind": "value", "name": "someU", "docstrings": [], - "signature": "let someU: (t, value => bool) => bool" + "signature": "let someU: (t, value => bool) => bool", + "deprecated": "Use `some` instead" }, { "id": "Belt.MutableSet.String.some", @@ -1578,7 +1610,8 @@ "kind": "value", "name": "keepU", "docstrings": [], - "signature": "let keepU: (t, value => bool) => t" + "signature": "let keepU: (t, value => bool) => t", + "deprecated": "Use `keep` instead" }, { "id": "Belt.MutableSet.String.keep", @@ -1594,7 +1627,8 @@ "kind": "value", "name": "partitionU", "docstrings": [], - "signature": "let partitionU: (t, value => bool) => (t, t)" + "signature": "let partitionU: (t, value => bool) => (t, t)", + "deprecated": "Use `partition` instead" }, { "id": "Belt.MutableSet.String.partition", @@ -1856,7 +1890,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t, value => unit) => unit" + "signature": "let forEachU: (t, value => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.MutableSet.Int.forEach", @@ -1872,7 +1907,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t, 'a, ('a, value) => 'a) => 'a" + "signature": "let reduceU: (t, 'a, ('a, value) => 'a) => 'a", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.MutableSet.Int.reduce", @@ -1888,7 +1924,8 @@ "kind": "value", "name": "everyU", "docstrings": [], - "signature": "let everyU: (t, value => bool) => bool" + "signature": "let everyU: (t, value => bool) => bool", + "deprecated": "Use `every` instead" }, { "id": "Belt.MutableSet.Int.every", @@ -1904,7 +1941,8 @@ "kind": "value", "name": "someU", "docstrings": [], - "signature": "let someU: (t, value => bool) => bool" + "signature": "let someU: (t, value => bool) => bool", + "deprecated": "Use `some` instead" }, { "id": "Belt.MutableSet.Int.some", @@ -1920,7 +1958,8 @@ "kind": "value", "name": "keepU", "docstrings": [], - "signature": "let keepU: (t, value => bool) => t" + "signature": "let keepU: (t, value => bool) => t", + "deprecated": "Use `keep` instead" }, { "id": "Belt.MutableSet.Int.keep", @@ -1936,7 +1975,8 @@ "kind": "value", "name": "partitionU", "docstrings": [], - "signature": "let partitionU: (t, value => bool) => (t, t)" + "signature": "let partitionU: (t, value => bool) => (t, t)", + "deprecated": "Use `partition` instead" }, { "id": "Belt.MutableSet.Int.partition", @@ -2088,7 +2128,8 @@ "kind": "value", "name": "cmpU", "docstrings": [], - "signature": "let cmpU: (\n t<'k, 'v, 'id>,\n t<'k, 'v, 'id>,\n ~kcmp: cmp<'k, 'id>,\n ~vcmp: ('v, 'v) => int,\n) => int" + "signature": "let cmpU: (\n t<'k, 'v, 'id>,\n t<'k, 'v, 'id>,\n ~kcmp: cmp<'k, 'id>,\n ~vcmp: ('v, 'v) => int,\n) => int", + "deprecated": "Use `cmp` instead" }, { "id": "Belt.Map.Dict.cmp", @@ -2102,7 +2143,8 @@ "kind": "value", "name": "eqU", "docstrings": [], - "signature": "let eqU: (\n t<'k, 'a, 'id>,\n t<'k, 'a, 'id>,\n ~kcmp: cmp<'k, 'id>,\n ~veq: ('a, 'a) => bool,\n) => bool" + "signature": "let eqU: (\n t<'k, 'a, 'id>,\n t<'k, 'a, 'id>,\n ~kcmp: cmp<'k, 'id>,\n ~veq: ('a, 'a) => bool,\n) => bool", + "deprecated": "Use `eq` instead" }, { "id": "Belt.Map.Dict.eq", @@ -2118,7 +2160,8 @@ "kind": "value", "name": "findFirstByU", "docstrings": [], - "signature": "let findFirstByU: (t<'k, 'v, 'id>, ('k, 'v) => bool) => option<('k, 'v)>" + "signature": "let findFirstByU: (t<'k, 'v, 'id>, ('k, 'v) => bool) => option<('k, 'v)>", + "deprecated": "Use `findFirstBy` instead" }, { "id": "Belt.Map.Dict.findFirstBy", @@ -2134,7 +2177,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t<'k, 'a, 'id>, ('k, 'a) => unit) => unit" + "signature": "let forEachU: (t<'k, 'a, 'id>, ('k, 'a) => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.Map.Dict.forEach", @@ -2150,7 +2194,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t<'k, 'a, 'id>, 'b, ('b, 'k, 'a) => 'b) => 'b" + "signature": "let reduceU: (t<'k, 'a, 'id>, 'b, ('b, 'k, 'a) => 'b) => 'b", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.Map.Dict.reduce", @@ -2166,7 +2211,8 @@ "kind": "value", "name": "everyU", "docstrings": [], - "signature": "let everyU: (t<'k, 'a, 'id>, ('k, 'a) => bool) => bool" + "signature": "let everyU: (t<'k, 'a, 'id>, ('k, 'a) => bool) => bool", + "deprecated": "Use `every` instead" }, { "id": "Belt.Map.Dict.every", @@ -2182,7 +2228,8 @@ "kind": "value", "name": "someU", "docstrings": [], - "signature": "let someU: (t<'k, 'a, 'id>, ('k, 'a) => bool) => bool" + "signature": "let someU: (t<'k, 'a, 'id>, ('k, 'a) => bool) => bool", + "deprecated": "Use `some` instead" }, { "id": "Belt.Map.Dict.some", @@ -2358,7 +2405,8 @@ "kind": "value", "name": "updateU", "docstrings": [], - "signature": "let updateU: (\n t<'a, 'b, 'id>,\n 'a,\n option<'b> => option<'b>,\n ~cmp: cmp<'a, 'id>,\n) => t<'a, 'b, 'id>" + "signature": "let updateU: (\n t<'a, 'b, 'id>,\n 'a,\n option<'b> => option<'b>,\n ~cmp: cmp<'a, 'id>,\n) => t<'a, 'b, 'id>", + "deprecated": "Use `update` instead" }, { "id": "Belt.Map.Dict.update", @@ -2372,7 +2420,8 @@ "kind": "value", "name": "mergeU", "docstrings": [], - "signature": "let mergeU: (\n t<'a, 'b, 'id>,\n t<'a, 'c, 'id>,\n ('a, option<'b>, option<'c>) => option<'d>,\n ~cmp: cmp<'a, 'id>,\n) => t<'a, 'd, 'id>" + "signature": "let mergeU: (\n t<'a, 'b, 'id>,\n t<'a, 'c, 'id>,\n ('a, option<'b>, option<'c>) => option<'d>,\n ~cmp: cmp<'a, 'id>,\n) => t<'a, 'd, 'id>", + "deprecated": "Use `merge` instead" }, { "id": "Belt.Map.Dict.merge", @@ -2395,7 +2444,8 @@ "kind": "value", "name": "keepU", "docstrings": [], - "signature": "let keepU: (t<'k, 'a, 'id>, ('k, 'a) => bool) => t<'k, 'a, 'id>" + "signature": "let keepU: (t<'k, 'a, 'id>, ('k, 'a) => bool) => t<'k, 'a, 'id>", + "deprecated": "Use `keep` instead" }, { "id": "Belt.Map.Dict.keep", @@ -2411,7 +2461,8 @@ "kind": "value", "name": "partitionU", "docstrings": [], - "signature": "let partitionU: (\n t<'k, 'a, 'id>,\n ('k, 'a) => bool,\n) => (t<'k, 'a, 'id>, t<'k, 'a, 'id>)" + "signature": "let partitionU: (\n t<'k, 'a, 'id>,\n ('k, 'a) => bool,\n) => (t<'k, 'a, 'id>, t<'k, 'a, 'id>)", + "deprecated": "Use `partition` instead" }, { "id": "Belt.Map.Dict.partition", @@ -2436,7 +2487,8 @@ "kind": "value", "name": "mapU", "docstrings": [], - "signature": "let mapU: (t<'k, 'a, 'id>, 'a => 'b) => t<'k, 'b, 'id>" + "signature": "let mapU: (t<'k, 'a, 'id>, 'a => 'b) => t<'k, 'b, 'id>", + "deprecated": "Use `map` instead" }, { "id": "Belt.Map.Dict.map", @@ -2452,7 +2504,8 @@ "kind": "value", "name": "mapWithKeyU", "docstrings": [], - "signature": "let mapWithKeyU: (t<'k, 'a, 'id>, ('k, 'a) => 'b) => t<'k, 'b, 'id>" + "signature": "let mapWithKeyU: (t<'k, 'a, 'id>, ('k, 'a) => 'b) => t<'k, 'b, 'id>", + "deprecated": "Use `mapWithKey` instead" }, { "id": "Belt.Map.Dict.mapWithKey", @@ -2510,7 +2563,8 @@ "kind": "value", "name": "cmpU", "docstrings": [], - "signature": "let cmpU: (t<'v>, t<'v>, ('v, 'v) => int) => int" + "signature": "let cmpU: (t<'v>, t<'v>, ('v, 'v) => int) => int", + "deprecated": "Use `cmp` instead" }, { "id": "Belt.Map.String.cmp", @@ -2524,7 +2578,8 @@ "kind": "value", "name": "eqU", "docstrings": [], - "signature": "let eqU: (t<'v>, t<'v>, ('v, 'v) => bool) => bool" + "signature": "let eqU: (t<'v>, t<'v>, ('v, 'v) => bool) => bool", + "deprecated": "Use `eq` instead" }, { "id": "Belt.Map.String.eq", @@ -2540,7 +2595,8 @@ "kind": "value", "name": "findFirstByU", "docstrings": [], - "signature": "let findFirstByU: (t<'v>, (key, 'v) => bool) => option<(key, 'v)>" + "signature": "let findFirstByU: (t<'v>, (key, 'v) => bool) => option<(key, 'v)>", + "deprecated": "Use `findFirstBy` instead" }, { "id": "Belt.Map.String.findFirstBy", @@ -2556,7 +2612,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t<'v>, (key, 'v) => unit) => unit" + "signature": "let forEachU: (t<'v>, (key, 'v) => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.Map.String.forEach", @@ -2572,7 +2629,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t<'v>, 'v2, ('v2, key, 'v) => 'v2) => 'v2" + "signature": "let reduceU: (t<'v>, 'v2, ('v2, key, 'v) => 'v2) => 'v2", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.Map.String.reduce", @@ -2588,7 +2646,8 @@ "kind": "value", "name": "everyU", "docstrings": [], - "signature": "let everyU: (t<'v>, (key, 'v) => bool) => bool" + "signature": "let everyU: (t<'v>, (key, 'v) => bool) => bool", + "deprecated": "Use `every` instead" }, { "id": "Belt.Map.String.every", @@ -2604,7 +2663,8 @@ "kind": "value", "name": "someU", "docstrings": [], - "signature": "let someU: (t<'v>, (key, 'v) => bool) => bool" + "signature": "let someU: (t<'v>, (key, 'v) => bool) => bool", + "deprecated": "Use `some` instead" }, { "id": "Belt.Map.String.some", @@ -2782,7 +2842,8 @@ "kind": "value", "name": "updateU", "docstrings": [], - "signature": "let updateU: (t<'v>, key, option<'v> => option<'v>) => t<'v>" + "signature": "let updateU: (t<'v>, key, option<'v> => option<'v>) => t<'v>", + "deprecated": "Use `update` instead" }, { "id": "Belt.Map.String.update", @@ -2796,7 +2857,8 @@ "kind": "value", "name": "mergeU", "docstrings": [], - "signature": "let mergeU: (\n t<'v>,\n t<'v2>,\n (key, option<'v>, option<'v2>) => option<'c>,\n) => t<'c>" + "signature": "let mergeU: (\n t<'v>,\n t<'v2>,\n (key, option<'v>, option<'v2>) => option<'c>,\n) => t<'c>", + "deprecated": "Use `merge` instead" }, { "id": "Belt.Map.String.merge", @@ -2819,7 +2881,8 @@ "kind": "value", "name": "keepU", "docstrings": [], - "signature": "let keepU: (t<'v>, (key, 'v) => bool) => t<'v>" + "signature": "let keepU: (t<'v>, (key, 'v) => bool) => t<'v>", + "deprecated": "Use `keep` instead" }, { "id": "Belt.Map.String.keep", @@ -2835,7 +2898,8 @@ "kind": "value", "name": "partitionU", "docstrings": [], - "signature": "let partitionU: (t<'v>, (key, 'v) => bool) => (t<'v>, t<'v>)" + "signature": "let partitionU: (t<'v>, (key, 'v) => bool) => (t<'v>, t<'v>)", + "deprecated": "Use `partition` instead" }, { "id": "Belt.Map.String.partition", @@ -2860,7 +2924,8 @@ "kind": "value", "name": "mapU", "docstrings": [], - "signature": "let mapU: (t<'v>, 'v => 'v2) => t<'v2>" + "signature": "let mapU: (t<'v>, 'v => 'v2) => t<'v2>", + "deprecated": "Use `map` instead" }, { "id": "Belt.Map.String.map", @@ -2876,7 +2941,8 @@ "kind": "value", "name": "mapWithKeyU", "docstrings": [], - "signature": "let mapWithKeyU: (t<'v>, (key, 'v) => 'v2) => t<'v2>" + "signature": "let mapWithKeyU: (t<'v>, (key, 'v) => 'v2) => t<'v2>", + "deprecated": "Use `mapWithKey` instead" }, { "id": "Belt.Map.String.mapWithKey", @@ -2936,7 +3002,8 @@ "kind": "value", "name": "cmpU", "docstrings": [], - "signature": "let cmpU: (t<'v>, t<'v>, ('v, 'v) => int) => int" + "signature": "let cmpU: (t<'v>, t<'v>, ('v, 'v) => int) => int", + "deprecated": "Use `cmp` instead" }, { "id": "Belt.Map.Int.cmp", @@ -2950,7 +3017,8 @@ "kind": "value", "name": "eqU", "docstrings": [], - "signature": "let eqU: (t<'v>, t<'v>, ('v, 'v) => bool) => bool" + "signature": "let eqU: (t<'v>, t<'v>, ('v, 'v) => bool) => bool", + "deprecated": "Use `eq` instead" }, { "id": "Belt.Map.Int.eq", @@ -2966,7 +3034,8 @@ "kind": "value", "name": "findFirstByU", "docstrings": [], - "signature": "let findFirstByU: (t<'v>, (key, 'v) => bool) => option<(key, 'v)>" + "signature": "let findFirstByU: (t<'v>, (key, 'v) => bool) => option<(key, 'v)>", + "deprecated": "Use `findFirstBy` instead" }, { "id": "Belt.Map.Int.findFirstBy", @@ -2982,7 +3051,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t<'v>, (key, 'v) => unit) => unit" + "signature": "let forEachU: (t<'v>, (key, 'v) => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.Map.Int.forEach", @@ -2998,7 +3068,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t<'v>, 'v2, ('v2, key, 'v) => 'v2) => 'v2" + "signature": "let reduceU: (t<'v>, 'v2, ('v2, key, 'v) => 'v2) => 'v2", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.Map.Int.reduce", @@ -3014,7 +3085,8 @@ "kind": "value", "name": "everyU", "docstrings": [], - "signature": "let everyU: (t<'v>, (key, 'v) => bool) => bool" + "signature": "let everyU: (t<'v>, (key, 'v) => bool) => bool", + "deprecated": "Use `every` instead" }, { "id": "Belt.Map.Int.every", @@ -3030,7 +3102,8 @@ "kind": "value", "name": "someU", "docstrings": [], - "signature": "let someU: (t<'v>, (key, 'v) => bool) => bool" + "signature": "let someU: (t<'v>, (key, 'v) => bool) => bool", + "deprecated": "Use `some` instead" }, { "id": "Belt.Map.Int.some", @@ -3208,7 +3281,8 @@ "kind": "value", "name": "updateU", "docstrings": [], - "signature": "let updateU: (t<'v>, key, option<'v> => option<'v>) => t<'v>" + "signature": "let updateU: (t<'v>, key, option<'v> => option<'v>) => t<'v>", + "deprecated": "Use `update` instead" }, { "id": "Belt.Map.Int.update", @@ -3222,7 +3296,8 @@ "kind": "value", "name": "mergeU", "docstrings": [], - "signature": "let mergeU: (\n t<'v>,\n t<'v2>,\n (key, option<'v>, option<'v2>) => option<'c>,\n) => t<'c>" + "signature": "let mergeU: (\n t<'v>,\n t<'v2>,\n (key, option<'v>, option<'v2>) => option<'c>,\n) => t<'c>", + "deprecated": "Use `merge` instead" }, { "id": "Belt.Map.Int.merge", @@ -3245,7 +3320,8 @@ "kind": "value", "name": "keepU", "docstrings": [], - "signature": "let keepU: (t<'v>, (key, 'v) => bool) => t<'v>" + "signature": "let keepU: (t<'v>, (key, 'v) => bool) => t<'v>", + "deprecated": "Use `keep` instead" }, { "id": "Belt.Map.Int.keep", @@ -3261,7 +3337,8 @@ "kind": "value", "name": "partitionU", "docstrings": [], - "signature": "let partitionU: (t<'v>, (key, 'v) => bool) => (t<'v>, t<'v>)" + "signature": "let partitionU: (t<'v>, (key, 'v) => bool) => (t<'v>, t<'v>)", + "deprecated": "Use `partition` instead" }, { "id": "Belt.Map.Int.partition", @@ -3286,7 +3363,8 @@ "kind": "value", "name": "mapU", "docstrings": [], - "signature": "let mapU: (t<'v>, 'v => 'v2) => t<'v2>" + "signature": "let mapU: (t<'v>, 'v => 'v2) => t<'v2>", + "deprecated": "Use `map` instead" }, { "id": "Belt.Map.Int.map", @@ -3302,7 +3380,8 @@ "kind": "value", "name": "mapWithKeyU", "docstrings": [], - "signature": "let mapWithKeyU: (t<'v>, (key, 'v) => 'v2) => t<'v2>" + "signature": "let mapWithKeyU: (t<'v>, (key, 'v) => 'v2) => t<'v2>", + "deprecated": "Use `mapWithKey` instead" }, { "id": "Belt.Map.Int.mapWithKey", @@ -3481,7 +3560,8 @@ "docstrings": [ "Same as [forEach](##forEach) but takes uncurried functon." ], - "signature": "let forEachU: (t<'value, 'id>, 'value => unit) => unit" + "signature": "let forEachU: (t<'value, 'id>, 'value => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.Set.Dict.forEach", @@ -3497,7 +3577,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t<'value, 'id>, 'a, ('a, 'value) => 'a) => 'a" + "signature": "let reduceU: (t<'value, 'id>, 'a, ('a, 'value) => 'a) => 'a", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.Set.Dict.reduce", @@ -3513,7 +3594,8 @@ "kind": "value", "name": "everyU", "docstrings": [], - "signature": "let everyU: (t<'value, 'id>, 'value => bool) => bool" + "signature": "let everyU: (t<'value, 'id>, 'value => bool) => bool", + "deprecated": "Use `every` instead" }, { "id": "Belt.Set.Dict.every", @@ -3529,7 +3611,8 @@ "kind": "value", "name": "someU", "docstrings": [], - "signature": "let someU: (t<'value, 'id>, 'value => bool) => bool" + "signature": "let someU: (t<'value, 'id>, 'value => bool) => bool", + "deprecated": "Use `some` instead" }, { "id": "Belt.Set.Dict.some", @@ -3545,7 +3628,8 @@ "kind": "value", "name": "keepU", "docstrings": [], - "signature": "let keepU: (t<'value, 'id>, 'value => bool) => t<'value, 'id>" + "signature": "let keepU: (t<'value, 'id>, 'value => bool) => t<'value, 'id>", + "deprecated": "Use `keep` instead" }, { "id": "Belt.Set.Dict.keep", @@ -3561,7 +3645,8 @@ "kind": "value", "name": "partitionU", "docstrings": [], - "signature": "let partitionU: (\n t<'value, 'id>,\n 'value => bool,\n) => (t<'value, 'id>, t<'value, 'id>)" + "signature": "let partitionU: (\n t<'value, 'id>,\n 'value => bool,\n) => (t<'value, 'id>, t<'value, 'id>)", + "deprecated": "Use `partition` instead" }, { "id": "Belt.Set.Dict.partition", @@ -3687,7 +3772,7 @@ "name": "String", "docstrings": [ "Specialized when value type is `string`, more efficient than the generic type,\nits compare behavior is fixed using the built-in comparison", - "This module is [`Belt.Set`]() specialized with value type to be a primitive type.\n It is more efficient in general, the API is the same with [`Belt_Set`]() except its value type is fixed,\n and identity is not needed(using the built-in one)\n\n **See** [`Belt.Set`]()" + "This module is [`Belt.Set`]() specialized with value type to be a primitive type.\nIt is more efficient in general, the API is the same with [`Belt_Set`]() except its value type is fixed,\nand identity is not needed(using the built-in one)\n\n**See** [`Belt.Set`]()" ], "items": [ { @@ -3828,7 +3913,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t, value => unit) => unit" + "signature": "let forEachU: (t, value => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.Set.String.forEach", @@ -3844,7 +3930,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t, 'a, ('a, value) => 'a) => 'a" + "signature": "let reduceU: (t, 'a, ('a, value) => 'a) => 'a", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.Set.String.reduce", @@ -3860,7 +3947,8 @@ "kind": "value", "name": "everyU", "docstrings": [], - "signature": "let everyU: (t, value => bool) => bool" + "signature": "let everyU: (t, value => bool) => bool", + "deprecated": "Use `every` instead" }, { "id": "Belt.Set.String.every", @@ -3876,7 +3964,8 @@ "kind": "value", "name": "someU", "docstrings": [], - "signature": "let someU: (t, value => bool) => bool" + "signature": "let someU: (t, value => bool) => bool", + "deprecated": "Use `some` instead" }, { "id": "Belt.Set.String.some", @@ -3892,7 +3981,8 @@ "kind": "value", "name": "keepU", "docstrings": [], - "signature": "let keepU: (t, value => bool) => t" + "signature": "let keepU: (t, value => bool) => t", + "deprecated": "Use `keep` instead" }, { "id": "Belt.Set.String.keep", @@ -3908,7 +3998,8 @@ "kind": "value", "name": "partitionU", "docstrings": [], - "signature": "let partitionU: (t, value => bool) => (t, t)" + "signature": "let partitionU: (t, value => bool) => (t, t)", + "deprecated": "Use `partition` instead" }, { "id": "Belt.Set.String.partition", @@ -4016,7 +4107,7 @@ "name": "Int", "docstrings": [ "Specialized when value type is `int`, more efficient than the generic type, its\ncompare behavior is fixed using the built-in comparison", - "This module is [`Belt.Set`]() specialized with value type to be a primitive type.\n It is more efficient in general, the API is the same with [`Belt_Set`]() except its value type is fixed,\n and identity is not needed(using the built-in one)\n\n **See** [`Belt.Set`]()" + "This module is [`Belt.Set`]() specialized with value type to be a primitive type.\nIt is more efficient in general, the API is the same with [`Belt_Set`]() except its value type is fixed,\nand identity is not needed(using the built-in one)\n\n**See** [`Belt.Set`]()" ], "items": [ { @@ -4157,7 +4248,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t, value => unit) => unit" + "signature": "let forEachU: (t, value => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.Set.Int.forEach", @@ -4173,7 +4265,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t, 'a, ('a, value) => 'a) => 'a" + "signature": "let reduceU: (t, 'a, ('a, value) => 'a) => 'a", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.Set.Int.reduce", @@ -4189,7 +4282,8 @@ "kind": "value", "name": "everyU", "docstrings": [], - "signature": "let everyU: (t, value => bool) => bool" + "signature": "let everyU: (t, value => bool) => bool", + "deprecated": "Use `every` instead" }, { "id": "Belt.Set.Int.every", @@ -4205,7 +4299,8 @@ "kind": "value", "name": "someU", "docstrings": [], - "signature": "let someU: (t, value => bool) => bool" + "signature": "let someU: (t, value => bool) => bool", + "deprecated": "Use `some` instead" }, { "id": "Belt.Set.Int.some", @@ -4221,7 +4316,8 @@ "kind": "value", "name": "keepU", "docstrings": [], - "signature": "let keepU: (t, value => bool) => t" + "signature": "let keepU: (t, value => bool) => t", + "deprecated": "Use `keep` instead" }, { "id": "Belt.Set.Int.keep", @@ -4237,7 +4333,8 @@ "kind": "value", "name": "partitionU", "docstrings": [], - "signature": "let partitionU: (t, value => bool) => (t, t)" + "signature": "let partitionU: (t, value => bool) => (t, t)", + "deprecated": "Use `partition` instead" }, { "id": "Belt.Set.Int.partition", @@ -4829,7 +4926,8 @@ "kind": "value", "name": "mapWithDefaultU", "docstrings": [], - "signature": "let mapWithDefaultU: (t<'a, 'c>, 'b, 'a => 'b) => 'b" + "signature": "let mapWithDefaultU: (t<'a, 'c>, 'b, 'a => 'b) => 'b", + "deprecated": "Use `mapWithDefault` instead" }, { "id": "Belt.Result.mapWithDefault", @@ -4845,7 +4943,8 @@ "kind": "value", "name": "mapU", "docstrings": [], - "signature": "let mapU: (t<'a, 'c>, 'a => 'b) => t<'b, 'c>" + "signature": "let mapU: (t<'a, 'c>, 'a => 'b) => t<'b, 'c>", + "deprecated": "Use `map` instead" }, { "id": "Belt.Result.map", @@ -4861,7 +4960,8 @@ "kind": "value", "name": "flatMapU", "docstrings": [], - "signature": "let flatMapU: (t<'a, 'c>, 'a => t<'b, 'c>) => t<'b, 'c>" + "signature": "let flatMapU: (t<'a, 'c>, 'a => t<'b, 'c>) => t<'b, 'c>", + "deprecated": "Use `flatMap` instead" }, { "id": "Belt.Result.flatMap", @@ -4904,7 +5004,8 @@ "kind": "value", "name": "eqU", "docstrings": [], - "signature": "let eqU: (t<'a, 'c>, t<'b, 'd>, ('a, 'b) => bool) => bool" + "signature": "let eqU: (t<'a, 'c>, t<'b, 'd>, ('a, 'b) => bool) => bool", + "deprecated": "Use `eq` instead" }, { "id": "Belt.Result.eq", @@ -4920,7 +5021,8 @@ "kind": "value", "name": "cmpU", "docstrings": [], - "signature": "let cmpU: (t<'a, 'c>, t<'b, 'd>, ('a, 'b) => int) => int" + "signature": "let cmpU: (t<'a, 'c>, t<'b, 'd>, ('a, 'b) => int) => int", + "deprecated": "Use `cmp` instead" }, { "id": "Belt.Result.cmp", @@ -4948,7 +5050,8 @@ "docstrings": [ "Uncurried version of `keep`" ], - "signature": "let keepU: (option<'a>, 'a => bool) => option<'a>" + "signature": "let keepU: (option<'a>, 'a => bool) => option<'a>", + "deprecated": "Use `keep` instead" }, { "id": "Belt.Option.keep", @@ -4966,7 +5069,8 @@ "docstrings": [ "Uncurried version of `forEach`" ], - "signature": "let forEachU: (option<'a>, 'a => unit) => unit" + "signature": "let forEachU: (option<'a>, 'a => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.Option.forEach", @@ -5002,7 +5106,8 @@ "docstrings": [ "Uncurried version of `mapWithDefault`" ], - "signature": "let mapWithDefaultU: (option<'a>, 'b, 'a => 'b) => 'b" + "signature": "let mapWithDefaultU: (option<'a>, 'b, 'a => 'b) => 'b", + "deprecated": "Use `mapWithDefault` instead" }, { "id": "Belt.Option.mapWithDefault", @@ -5020,7 +5125,8 @@ "docstrings": [ "Uncurried version of `map`" ], - "signature": "let mapU: (option<'a>, 'a => 'b) => option<'b>" + "signature": "let mapU: (option<'a>, 'a => 'b) => option<'b>", + "deprecated": "Use `map` instead" }, { "id": "Belt.Option.map", @@ -5038,7 +5144,8 @@ "docstrings": [ "Uncurried version of `flatMap`" ], - "signature": "let flatMapU: (option<'a>, 'a => option<'b>) => option<'b>" + "signature": "let flatMapU: (option<'a>, 'a => option<'b>) => option<'b>", + "deprecated": "Use `flatMap` instead" }, { "id": "Belt.Option.flatMap", @@ -5092,7 +5199,8 @@ "docstrings": [ "Uncurried version of `eq`" ], - "signature": "let eqU: (option<'a>, option<'b>, ('a, 'b) => bool) => bool" + "signature": "let eqU: (option<'a>, option<'b>, ('a, 'b) => bool) => bool", + "deprecated": "Use `eq` instead" }, { "id": "Belt.Option.eq", @@ -5110,7 +5218,8 @@ "docstrings": [ "Uncurried version of `cmp`" ], - "signature": "let cmpU: (option<'a>, option<'b>, ('a, 'b) => int) => int" + "signature": "let cmpU: (option<'a>, option<'b>, ('a, 'b) => int) => int", + "deprecated": "Use `cmp` instead" }, { "id": "Belt.Option.cmp", @@ -5128,7 +5237,7 @@ "name": "HashMap", "docstrings": [ "[`Belt.HashMap`]()\n\n The top level provides generic **mutable** hash map operations.\n\n It also has two specialized inner modules\n [`Belt.HashMap.Int`]() and [`Belt.HashMap.String`]()", - "A **mutable** Hash map which allows customized [`hash`]() behavior.\n\nAll data are parameterized by not its only type but also a unique identity in\nthe time of initialization, so that two _HashMaps of ints_ initialized with different\n_hash_ functions will have different type.\n\n## Examples\n\n```rescript\ntype t = int\nmodule I0 = unpack(Belt.Id.hashableU(~hash=(. a: t) => \"&\"(a, 0xff_ff), ~eq=(. a, b) => a == b))\nlet s0: t<_, string, _> = make(~hintSize=40, ~id=module(I0))\n\nmodule I1 = unpack(Belt.Id.hashableU(~hash=(. a: t) => \"&\"(a, 0xff), ~eq=(. a, b) => a == b))\nlet s1: t<_, string, _> = make(~hintSize=40, ~id=module(I1))\n```\n\nThe invariant must be held: for two elements who are _equal_,\ntheir hashed value should be the same\n\nHere the compiler would infer `s0` and `s1` having different type so that\nit would not mix.\n\n## Examples\n\n```rescript\nlet s0: t\nlet s1: t\n```\n\nWe can add elements to the collection:\n\n## Examples\n\n```rescript\nlet () = {\n add(s1, 0, \"3\")\n add(s1, 1, \"3\")\n}\n```\n\nSince this is an mutable data strucure, `s1` will contain two pairs." + "A **mutable** Hash map which allows customized [`hash`]() behavior.\n\nAll data are parameterized by not its only type but also a unique identity in\nthe time of initialization, so that two _HashMaps of ints_ initialized with different\n_hash_ functions will have different type.\n\n## Examples\n\n```rescript\ntype t = int\nmodule I0 = unpack(Belt.Id.hashable(~hash=(a: t) => \"&\"(a, 0xff_ff), ~eq=(a, b) => a == b))\nlet s0: t<_, string, _> = make(~hintSize=40, ~id=module(I0))\n\nmodule I1 = unpack(Belt.Id.hashable(~hash=(a: t) => \"&\"(a, 0xff), ~eq=(a, b) => a == b))\nlet s1: t<_, string, _> = make(~hintSize=40, ~id=module(I1))\n```\n\nThe invariant must be held: for two elements who are _equal_,\ntheir hashed value should be the same\n\nHere the compiler would infer `s0` and `s1` having different type so that\nit would not mix.\n\n## Examples\n\n```rescript\nlet s0: t\nlet s1: t\n```\n\nWe can add elements to the collection:\n\n## Examples\n\n```rescript\nlet () = {\n add(s1, 0, \"3\")\n add(s1, 1, \"3\")\n}\n```\n\nSince this is an mutable data strucure, `s1` will contain two pairs." ], "items": [ { @@ -5228,7 +5337,8 @@ "docstrings": [ "Same as [forEach](#forEach) but takes uncurried function." ], - "signature": "let forEachU: (t<'key, 'value, 'id>, ('key, 'value) => unit) => unit" + "signature": "let forEachU: (t<'key, 'value, 'id>, ('key, 'value) => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.HashMap.forEach", @@ -5244,7 +5354,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t<'key, 'value, 'id>, 'c, ('c, 'key, 'value) => 'c) => 'c" + "signature": "let reduceU: (t<'key, 'value, 'id>, 'c, ('c, 'key, 'value) => 'c) => 'c", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.HashMap.reduce", @@ -5262,7 +5373,8 @@ "docstrings": [ "Same as [keepMapInPlace](#keepMapInPlace) but takes uncurried function." ], - "signature": "let keepMapInPlaceU: (\n t<'key, 'value, 'id>,\n ('key, 'value) => option<'value>,\n) => unit" + "signature": "let keepMapInPlaceU: (\n t<'key, 'value, 'id>,\n ('key, 'value) => option<'value>,\n) => unit", + "deprecated": "Use `keepMapInPlace` instead" }, { "id": "Belt.HashMap.keepMapInPlace", @@ -5352,7 +5464,7 @@ "name": "HashSet", "docstrings": [ "[`Belt.HashSet`]()\n\n The top level provides generic **mutable** hash set operations.\n\n It also has two specialized inner modules\n [`Belt.HashSet.Int`]() and [`Belt.HashSet.String`]()", - "A **mutable** Hash set which allows customized `hash` behavior.\n\nAll data are parameterized by not its only type but also a unique identity in\nthe time of initialization, so that two _HashSets of ints_ initialized with\ndifferent _hash_ functions will have different type.\n\n## Examples\n\n```rescript\nmodule I0 = unpack(\n Belt.Id.hashableU(\n ~hash=(. a: int) => land(a, 65535),\n ~eq=(. a, b) => a == b,\n )\n)\n\nlet s0 = Belt.HashSet.make(~id=module(I0), ~hintSize=40)\n\nmodule I1 = unpack(\n Belt.Id.hashableU(\n ~hash=(. a: int) => land(a, 255),\n ~eq=(. a, b) => a == b,\n )\n)\n\nlet s1 = Belt.HashSet.make(~id=module(I1), ~hintSize=40)\n\nBelt.HashSet.add(s1, 0)\nBelt.HashSet.add(s1, 1)\n```\n\nThe invariant must be held: for two elements who are equal, their hashed\nvalue should be the same.\n\nHere the compiler would infer `s0` and `s1` having different type so that it\nwould not mix.\n\n## Examples\n\n```rescript\nlet s0: Belt.HashSet.t\nlet s1: Belt.HashSet.t\n```\n\nWe can add elements to the collection (see last two lines in the example\nabove). Since this is an mutable data structure, `s1` will contain two pairs." + "A **mutable** Hash set which allows customized `hash` behavior.\n\nAll data are parameterized by not its only type but also a unique identity in\nthe time of initialization, so that two _HashSets of ints_ initialized with\ndifferent _hash_ functions will have different type.\n\n## Examples\n\n```rescript\nmodule I0 = unpack(\n Belt.Id.hashable(\n ~hash=(a: int) => land(a, 65535),\n ~eq=(a, b) => a == b,\n )\n)\n\nlet s0 = Belt.HashSet.make(~id=module(I0), ~hintSize=40)\n\nmodule I1 = unpack(\n Belt.Id.hashable(\n ~hash=(a: int) => land(a, 255),\n ~eq=(a, b) => a == b,\n )\n)\n\nlet s1 = Belt.HashSet.make(~id=module(I1), ~hintSize=40)\n\nBelt.HashSet.add(s1, 0)\nBelt.HashSet.add(s1, 1)\n```\n\nThe invariant must be held: for two elements who are equal, their hashed\nvalue should be the same.\n\nHere the compiler would infer `s0` and `s1` having different type so that it\nwould not mix.\n\n## Examples\n\n```rescript\nlet s0: Belt.HashSet.t\nlet s1: Belt.HashSet.t\n```\n\nWe can add elements to the collection (see last two lines in the example\nabove). Since this is an mutable data structure, `s1` will contain two pairs." ], "items": [ { @@ -5425,7 +5537,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t<'a, 'id>, 'a => unit) => unit" + "signature": "let forEachU: (t<'a, 'id>, 'a => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.HashSet.forEach", @@ -5441,7 +5554,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t<'a, 'id>, 'c, ('c, 'a) => 'c) => 'c" + "signature": "let reduceU: (t<'a, 'id>, 'c, ('c, 'a) => 'c) => 'c", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.HashSet.reduce", @@ -5550,7 +5664,8 @@ "kind": "value", "name": "cmpU", "docstrings": [], - "signature": "let cmpU: (t<'k, 'a, 'id>, t<'k, 'a, 'id>, ('a, 'a) => int) => int" + "signature": "let cmpU: (t<'k, 'a, 'id>, t<'k, 'a, 'id>, ('a, 'a) => int) => int", + "deprecated": "Use `cmp` instead" }, { "id": "Belt.MutableMap.cmp", @@ -5566,7 +5681,8 @@ "kind": "value", "name": "eqU", "docstrings": [], - "signature": "let eqU: (t<'k, 'a, 'id>, t<'k, 'a, 'id>, ('a, 'a) => bool) => bool" + "signature": "let eqU: (t<'k, 'a, 'id>, t<'k, 'a, 'id>, ('a, 'a) => bool) => bool", + "deprecated": "Use `eq` instead" }, { "id": "Belt.MutableMap.eq", @@ -5582,7 +5698,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t<'k, 'a, 'id>, ('k, 'a) => unit) => unit" + "signature": "let forEachU: (t<'k, 'a, 'id>, ('k, 'a) => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.MutableMap.forEach", @@ -5598,7 +5715,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t<'k, 'a, 'id>, 'b, ('b, 'k, 'a) => 'b) => 'b" + "signature": "let reduceU: (t<'k, 'a, 'id>, 'b, ('b, 'k, 'a) => 'b) => 'b", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.MutableMap.reduce", @@ -5614,7 +5732,8 @@ "kind": "value", "name": "everyU", "docstrings": [], - "signature": "let everyU: (t<'k, 'a, 'id>, ('k, 'a) => bool) => bool" + "signature": "let everyU: (t<'k, 'a, 'id>, ('k, 'a) => bool) => bool", + "deprecated": "Use `every` instead" }, { "id": "Belt.MutableMap.every", @@ -5630,7 +5749,8 @@ "kind": "value", "name": "someU", "docstrings": [], - "signature": "let someU: (t<'k, 'a, 'id>, ('k, 'a) => bool) => bool" + "signature": "let someU: (t<'k, 'a, 'id>, ('k, 'a) => bool) => bool", + "deprecated": "Use `some` instead" }, { "id": "Belt.MutableMap.some", @@ -5808,7 +5928,8 @@ "kind": "value", "name": "updateU", "docstrings": [], - "signature": "let updateU: (t<'k, 'a, 'id>, 'k, option<'a> => option<'a>) => unit" + "signature": "let updateU: (t<'k, 'a, 'id>, 'k, option<'a> => option<'a>) => unit", + "deprecated": "Use `update` instead" }, { "id": "Belt.MutableMap.update", @@ -5829,7 +5950,8 @@ "kind": "value", "name": "mapU", "docstrings": [], - "signature": "let mapU: (t<'k, 'a, 'id>, 'a => 'b) => t<'k, 'b, 'id>" + "signature": "let mapU: (t<'k, 'a, 'id>, 'a => 'b) => t<'k, 'b, 'id>", + "deprecated": "Use `map` instead" }, { "id": "Belt.MutableMap.map", @@ -5845,7 +5967,8 @@ "kind": "value", "name": "mapWithKeyU", "docstrings": [], - "signature": "let mapWithKeyU: (t<'k, 'a, 'id>, ('k, 'a) => 'b) => t<'k, 'b, 'id>" + "signature": "let mapWithKeyU: (t<'k, 'a, 'id>, ('k, 'a) => 'b) => t<'k, 'b, 'id>", + "deprecated": "Use `mapWithKey` instead" }, { "id": "Belt.MutableMap.mapWithKey", @@ -6047,7 +6170,8 @@ "docstrings": [ "Same as `Belt.MutableSet.forEach` but takes uncurried functon." ], - "signature": "let forEachU: (t<'value, 'id>, 'value => unit) => unit" + "signature": "let forEachU: (t<'value, 'id>, 'value => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.MutableSet.forEach", @@ -6063,7 +6187,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t<'value, 'id>, 'a, ('a, 'value) => 'a) => 'a" + "signature": "let reduceU: (t<'value, 'id>, 'a, ('a, 'value) => 'a) => 'a", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.MutableSet.reduce", @@ -6079,7 +6204,8 @@ "kind": "value", "name": "everyU", "docstrings": [], - "signature": "let everyU: (t<'value, 'id>, 'value => bool) => bool" + "signature": "let everyU: (t<'value, 'id>, 'value => bool) => bool", + "deprecated": "Use `every` instead" }, { "id": "Belt.MutableSet.every", @@ -6095,7 +6221,8 @@ "kind": "value", "name": "someU", "docstrings": [], - "signature": "let someU: (t<'value, 'id>, 'value => bool) => bool" + "signature": "let someU: (t<'value, 'id>, 'value => bool) => bool", + "deprecated": "Use `some` instead" }, { "id": "Belt.MutableSet.some", @@ -6111,7 +6238,8 @@ "kind": "value", "name": "keepU", "docstrings": [], - "signature": "let keepU: (t<'value, 'id>, 'value => bool) => t<'value, 'id>" + "signature": "let keepU: (t<'value, 'id>, 'value => bool) => t<'value, 'id>", + "deprecated": "Use `keep` instead" }, { "id": "Belt.MutableSet.keep", @@ -6127,7 +6255,8 @@ "kind": "value", "name": "partitionU", "docstrings": [], - "signature": "let partitionU: (\n t<'value, 'id>,\n 'value => bool,\n) => (t<'value, 'id>, t<'value, 'id>)" + "signature": "let partitionU: (\n t<'value, 'id>,\n 'value => bool,\n) => (t<'value, 'id>, t<'value, 'id>)", + "deprecated": "Use `partition` instead" }, { "id": "Belt.MutableSet.partition", @@ -6306,7 +6435,8 @@ "kind": "value", "name": "cmpU", "docstrings": [], - "signature": "let cmpU: (t<'k, 'v, 'id>, t<'k, 'v, 'id>, ('v, 'v) => int) => int" + "signature": "let cmpU: (t<'k, 'v, 'id>, t<'k, 'v, 'id>, ('v, 'v) => int) => int", + "deprecated": "Use `cmp` instead" }, { "id": "Belt.Map.cmp", @@ -6322,7 +6452,8 @@ "kind": "value", "name": "eqU", "docstrings": [], - "signature": "let eqU: (t<'k, 'v, 'id>, t<'k, 'v, 'id>, ('v, 'v) => bool) => bool" + "signature": "let eqU: (t<'k, 'v, 'id>, t<'k, 'v, 'id>, ('v, 'v) => bool) => bool", + "deprecated": "Use `eq` instead" }, { "id": "Belt.Map.eq", @@ -6338,7 +6469,8 @@ "kind": "value", "name": "findFirstByU", "docstrings": [], - "signature": "let findFirstByU: (t<'k, 'v, 'id>, ('k, 'v) => bool) => option<('k, 'v)>" + "signature": "let findFirstByU: (t<'k, 'v, 'id>, ('k, 'v) => bool) => option<('k, 'v)>", + "deprecated": "Use `findFirstBy` instead" }, { "id": "Belt.Map.findFirstBy", @@ -6354,7 +6486,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t<'k, 'v, 'id>, ('k, 'v) => unit) => unit" + "signature": "let forEachU: (t<'k, 'v, 'id>, ('k, 'v) => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.Map.forEach", @@ -6370,7 +6503,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t<'k, 'v, 'id>, 'acc, ('acc, 'k, 'v) => 'acc) => 'acc" + "signature": "let reduceU: (t<'k, 'v, 'id>, 'acc, ('acc, 'k, 'v) => 'acc) => 'acc", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.Map.reduce", @@ -6386,7 +6520,8 @@ "kind": "value", "name": "everyU", "docstrings": [], - "signature": "let everyU: (t<'k, 'v, 'id>, ('k, 'v) => bool) => bool" + "signature": "let everyU: (t<'k, 'v, 'id>, ('k, 'v) => bool) => bool", + "deprecated": "Use `every` instead" }, { "id": "Belt.Map.every", @@ -6402,7 +6537,8 @@ "kind": "value", "name": "someU", "docstrings": [], - "signature": "let someU: (t<'k, 'v, 'id>, ('k, 'v) => bool) => bool" + "signature": "let someU: (t<'k, 'v, 'id>, ('k, 'v) => bool) => bool", + "deprecated": "Use `some` instead" }, { "id": "Belt.Map.some", @@ -6607,7 +6743,8 @@ "kind": "value", "name": "updateU", "docstrings": [], - "signature": "let updateU: (\n t<'k, 'v, 'id>,\n 'k,\n option<'v> => option<'v>,\n) => t<'k, 'v, 'id>" + "signature": "let updateU: (\n t<'k, 'v, 'id>,\n 'k,\n option<'v> => option<'v>,\n) => t<'k, 'v, 'id>", + "deprecated": "Use `update` instead" }, { "id": "Belt.Map.update", @@ -6632,7 +6769,8 @@ "kind": "value", "name": "mergeU", "docstrings": [], - "signature": "let mergeU: (\n t<'k, 'v, 'id>,\n t<'k, 'v2, 'id>,\n ('k, option<'v>, option<'v2>) => option<'v3>,\n) => t<'k, 'v3, 'id>" + "signature": "let mergeU: (\n t<'k, 'v, 'id>,\n t<'k, 'v2, 'id>,\n ('k, option<'v>, option<'v2>) => option<'v3>,\n) => t<'k, 'v3, 'id>", + "deprecated": "Use `merge` instead" }, { "id": "Belt.Map.merge", @@ -6648,7 +6786,8 @@ "kind": "value", "name": "keepU", "docstrings": [], - "signature": "let keepU: (t<'k, 'v, 'id>, ('k, 'v) => bool) => t<'k, 'v, 'id>" + "signature": "let keepU: (t<'k, 'v, 'id>, ('k, 'v) => bool) => t<'k, 'v, 'id>", + "deprecated": "Use `keep` instead" }, { "id": "Belt.Map.keep", @@ -6664,7 +6803,8 @@ "kind": "value", "name": "partitionU", "docstrings": [], - "signature": "let partitionU: (\n t<'k, 'v, 'id>,\n ('k, 'v) => bool,\n) => (t<'k, 'v, 'id>, t<'k, 'v, 'id>)" + "signature": "let partitionU: (\n t<'k, 'v, 'id>,\n ('k, 'v) => bool,\n) => (t<'k, 'v, 'id>, t<'k, 'v, 'id>)", + "deprecated": "Use `partition` instead" }, { "id": "Belt.Map.partition", @@ -6689,7 +6829,8 @@ "kind": "value", "name": "mapU", "docstrings": [], - "signature": "let mapU: (t<'k, 'v, 'id>, 'v => 'v2) => t<'k, 'v2, 'id>" + "signature": "let mapU: (t<'k, 'v, 'id>, 'v => 'v2) => t<'k, 'v2, 'id>", + "deprecated": "Use `map` instead" }, { "id": "Belt.Map.map", @@ -6705,7 +6846,8 @@ "kind": "value", "name": "mapWithKeyU", "docstrings": [], - "signature": "let mapWithKeyU: (t<'k, 'v, 'id>, ('k, 'v) => 'v2) => t<'k, 'v2, 'id>" + "signature": "let mapWithKeyU: (t<'k, 'v, 'id>, ('k, 'v) => 'v2) => t<'k, 'v2, 'id>", + "deprecated": "Use `mapWithKey` instead" }, { "id": "Belt.Map.mapWithKey", @@ -6922,7 +7064,8 @@ "docstrings": [ "Same as [forEach](#forEach) but takes uncurried functon." ], - "signature": "let forEachU: (t<'value, 'id>, 'value => unit) => unit" + "signature": "let forEachU: (t<'value, 'id>, 'value => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.Set.forEach", @@ -6938,7 +7081,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t<'value, 'id>, 'a, ('a, 'value) => 'a) => 'a" + "signature": "let reduceU: (t<'value, 'id>, 'a, ('a, 'value) => 'a) => 'a", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.Set.reduce", @@ -6954,7 +7098,8 @@ "kind": "value", "name": "everyU", "docstrings": [], - "signature": "let everyU: (t<'value, 'id>, 'value => bool) => bool" + "signature": "let everyU: (t<'value, 'id>, 'value => bool) => bool", + "deprecated": "Use `every` instead" }, { "id": "Belt.Set.every", @@ -6970,7 +7115,8 @@ "kind": "value", "name": "someU", "docstrings": [], - "signature": "let someU: (t<'value, 'id>, 'value => bool) => bool" + "signature": "let someU: (t<'value, 'id>, 'value => bool) => bool", + "deprecated": "Use `some` instead" }, { "id": "Belt.Set.some", @@ -6986,7 +7132,8 @@ "kind": "value", "name": "keepU", "docstrings": [], - "signature": "let keepU: (t<'value, 'id>, 'value => bool) => t<'value, 'id>" + "signature": "let keepU: (t<'value, 'id>, 'value => bool) => t<'value, 'id>", + "deprecated": "Use `keep` instead" }, { "id": "Belt.Set.keep", @@ -7002,7 +7149,8 @@ "kind": "value", "name": "partitionU", "docstrings": [], - "signature": "let partitionU: (\n t<'value, 'id>,\n 'value => bool,\n) => (t<'value, 'id>, t<'value, 'id>)" + "signature": "let partitionU: (\n t<'value, 'id>,\n 'value => bool,\n) => (t<'value, 'id>, t<'value, 'id>)", + "deprecated": "Use `partition` instead" }, { "id": "Belt.Set.partition", @@ -7163,7 +7311,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (int, int, int => unit) => unit" + "signature": "let forEachU: (int, int, int => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.Range.forEach", @@ -7179,7 +7328,8 @@ "kind": "value", "name": "everyU", "docstrings": [], - "signature": "let everyU: (int, int, int => bool) => bool" + "signature": "let everyU: (int, int, int => bool) => bool", + "deprecated": "Use `every` instead" }, { "id": "Belt.Range.every", @@ -7195,7 +7345,8 @@ "kind": "value", "name": "everyByU", "docstrings": [], - "signature": "let everyByU: (int, int, ~step: int, int => bool) => bool" + "signature": "let everyByU: (int, int, ~step: int, int => bool) => bool", + "deprecated": "Use `everyBy` instead" }, { "id": "Belt.Range.everyBy", @@ -7211,7 +7362,8 @@ "kind": "value", "name": "someU", "docstrings": [], - "signature": "let someU: (int, int, int => bool) => bool" + "signature": "let someU: (int, int, int => bool) => bool", + "deprecated": "Use `some` instead" }, { "id": "Belt.Range.some", @@ -7227,7 +7379,8 @@ "kind": "value", "name": "someByU", "docstrings": [], - "signature": "let someByU: (int, int, ~step: int, int => bool) => bool" + "signature": "let someByU: (int, int, ~step: int, int => bool) => bool", + "deprecated": "Use `someBy` instead" }, { "id": "Belt.Range.someBy", @@ -7354,7 +7507,8 @@ "docstrings": [ "Uncurried version of [makeBy](#makeBy)" ], - "signature": "let makeByU: (int, int => 'a) => t<'a>" + "signature": "let makeByU: (int, int => 'a) => t<'a>", + "deprecated": "Use `makeBy` instead" }, { "id": "Belt.List.makeBy", @@ -7444,7 +7598,8 @@ "docstrings": [ "Uncurried version of [map](#map)." ], - "signature": "let mapU: (t<'a>, 'a => 'b) => t<'b>" + "signature": "let mapU: (t<'a>, 'a => 'b) => t<'b>", + "deprecated": "Use `map` instead" }, { "id": "Belt.List.map", @@ -7471,7 +7626,8 @@ "docstrings": [ "Uncurried version of [zipBy](#zipBy)." ], - "signature": "let zipByU: (t<'a>, t<'b>, ('a, 'b) => 'c) => t<'c>" + "signature": "let zipByU: (t<'a>, t<'b>, ('a, 'b) => 'c) => t<'c>", + "deprecated": "Use `zipBy` instead" }, { "id": "Belt.List.zipBy", @@ -7489,7 +7645,8 @@ "docstrings": [ "Uncurried version of [mapWithIndex](#mapWithIndex)." ], - "signature": "let mapWithIndexU: (t<'a>, (int, 'a) => 'b) => t<'b>" + "signature": "let mapWithIndexU: (t<'a>, (int, 'a) => 'b) => t<'b>", + "deprecated": "Use `mapWithIndex` instead" }, { "id": "Belt.List.mapWithIndex", @@ -7534,7 +7691,8 @@ "docstrings": [ "Uncurried version of [mapReverse](#mapReverse)." ], - "signature": "let mapReverseU: (t<'a>, 'a => 'b) => t<'b>" + "signature": "let mapReverseU: (t<'a>, 'a => 'b) => t<'b>", + "deprecated": "Use `mapReverse` instead" }, { "id": "Belt.List.mapReverse", @@ -7552,7 +7710,8 @@ "docstrings": [ "Uncurried version of [forEach](#forEach)." ], - "signature": "let forEachU: (t<'a>, 'a => 'b) => unit" + "signature": "let forEachU: (t<'a>, 'a => 'b) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.List.forEach", @@ -7570,7 +7729,8 @@ "docstrings": [ "Uncurried version of [forEachWithIndex](#forEachWithIndex)." ], - "signature": "let forEachWithIndexU: (t<'a>, (int, 'a) => 'b) => unit" + "signature": "let forEachWithIndexU: (t<'a>, (int, 'a) => 'b) => unit", + "deprecated": "Use `forEachWithIndex` instead" }, { "id": "Belt.List.forEachWithIndex", @@ -7588,7 +7748,8 @@ "docstrings": [ "Uncurried version of [reduce](#reduce)." ], - "signature": "let reduceU: (t<'a>, 'b, ('b, 'a) => 'b) => 'b" + "signature": "let reduceU: (t<'a>, 'b, ('b, 'a) => 'b) => 'b", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.List.reduce", @@ -7606,7 +7767,8 @@ "docstrings": [ "Uncurried version of [reduceWithIndex](#reduceWithIndex)." ], - "signature": "let reduceWithIndexU: (t<'a>, 'b, ('b, 'a, int) => 'b) => 'b" + "signature": "let reduceWithIndexU: (t<'a>, 'b, ('b, 'a, int) => 'b) => 'b", + "deprecated": "Use `reduceWithIndex` instead" }, { "id": "Belt.List.reduceWithIndex", @@ -7624,7 +7786,8 @@ "docstrings": [ "Uncurried version of [reduceReverse](#reduceReverse)." ], - "signature": "let reduceReverseU: (t<'a>, 'b, ('b, 'a) => 'b) => 'b" + "signature": "let reduceReverseU: (t<'a>, 'b, ('b, 'a) => 'b) => 'b", + "deprecated": "Use `reduceReverse` instead" }, { "id": "Belt.List.reduceReverse", @@ -7642,7 +7805,8 @@ "docstrings": [ "Uncurried version of [mapReverse2](#mapReverse2)." ], - "signature": "let mapReverse2U: (t<'a>, t<'b>, ('a, 'b) => 'c) => t<'c>" + "signature": "let mapReverse2U: (t<'a>, t<'b>, ('a, 'b) => 'c) => t<'c>", + "deprecated": "Use `mapReverse2` instead" }, { "id": "Belt.List.mapReverse2", @@ -7660,7 +7824,8 @@ "docstrings": [ "Uncurried version of [forEach2](#forEach2)." ], - "signature": "let forEach2U: (t<'a>, t<'b>, ('a, 'b) => 'c) => unit" + "signature": "let forEach2U: (t<'a>, t<'b>, ('a, 'b) => 'c) => unit", + "deprecated": "Use `forEach2` instead" }, { "id": "Belt.List.forEach2", @@ -7678,7 +7843,8 @@ "docstrings": [ "Uncurried version of [reduce2](#reduce2)." ], - "signature": "let reduce2U: (t<'b>, t<'c>, 'a, ('a, 'b, 'c) => 'a) => 'a" + "signature": "let reduce2U: (t<'b>, t<'c>, 'a, ('a, 'b, 'c) => 'a) => 'a", + "deprecated": "Use `reduce2` instead" }, { "id": "Belt.List.reduce2", @@ -7696,7 +7862,8 @@ "docstrings": [ "Uncurried version of [reduceReverse2](#reduceReverse2)." ], - "signature": "let reduceReverse2U: (t<'a>, t<'b>, 'c, ('c, 'a, 'b) => 'c) => 'c" + "signature": "let reduceReverse2U: (t<'a>, t<'b>, 'c, ('c, 'a, 'b) => 'c) => 'c", + "deprecated": "Use `reduceReverse2` instead" }, { "id": "Belt.List.reduceReverse2", @@ -7714,7 +7881,8 @@ "docstrings": [ "Uncurried version of [every](#every)." ], - "signature": "let everyU: (t<'a>, 'a => bool) => bool" + "signature": "let everyU: (t<'a>, 'a => bool) => bool", + "deprecated": "Use `every` instead" }, { "id": "Belt.List.every", @@ -7732,7 +7900,8 @@ "docstrings": [ "Uncurried version of [some](#some)." ], - "signature": "let someU: (t<'a>, 'a => bool) => bool" + "signature": "let someU: (t<'a>, 'a => bool) => bool", + "deprecated": "Use `some` instead" }, { "id": "Belt.List.some", @@ -7750,7 +7919,8 @@ "docstrings": [ "Uncurried version of [every2](#every2)." ], - "signature": "let every2U: (t<'a>, t<'b>, ('a, 'b) => bool) => bool" + "signature": "let every2U: (t<'a>, t<'b>, ('a, 'b) => bool) => bool", + "deprecated": "Use `every2` instead" }, { "id": "Belt.List.every2", @@ -7768,7 +7938,8 @@ "docstrings": [ "Uncurried version of [some2](#some2)." ], - "signature": "let some2U: (t<'a>, t<'b>, ('a, 'b) => bool) => bool" + "signature": "let some2U: (t<'a>, t<'b>, ('a, 'b) => bool) => bool", + "deprecated": "Use `some2` instead" }, { "id": "Belt.List.some2", @@ -7795,7 +7966,8 @@ "docstrings": [ "Uncurried version of [cmp](#cmp)." ], - "signature": "let cmpU: (t<'a>, t<'a>, ('a, 'a) => int) => int" + "signature": "let cmpU: (t<'a>, t<'a>, ('a, 'a) => int) => int", + "deprecated": "Use `cmp` instead" }, { "id": "Belt.List.cmp", @@ -7813,7 +7985,8 @@ "docstrings": [ "Uncurried version of [eq](#eq)." ], - "signature": "let eqU: (t<'a>, t<'a>, ('a, 'a) => bool) => bool" + "signature": "let eqU: (t<'a>, t<'a>, ('a, 'a) => bool) => bool", + "deprecated": "Use `eq` instead" }, { "id": "Belt.List.eq", @@ -7831,7 +8004,8 @@ "docstrings": [ "Uncurried version of [has](#has)." ], - "signature": "let hasU: (t<'a>, 'b, ('a, 'b) => bool) => bool" + "signature": "let hasU: (t<'a>, 'b, ('a, 'b) => bool) => bool", + "deprecated": "Use `has` instead" }, { "id": "Belt.List.has", @@ -7849,7 +8023,8 @@ "docstrings": [ "Uncurried version of [getBy](#getBy)." ], - "signature": "let getByU: (t<'a>, 'a => bool) => option<'a>" + "signature": "let getByU: (t<'a>, 'a => bool) => option<'a>", + "deprecated": "Use `getBy` instead" }, { "id": "Belt.List.getBy", @@ -7867,7 +8042,8 @@ "docstrings": [ "Uncurried version of [keep](#keep)." ], - "signature": "let keepU: (t<'a>, 'a => bool) => t<'a>" + "signature": "let keepU: (t<'a>, 'a => bool) => t<'a>", + "deprecated": "Use `keep` instead" }, { "id": "Belt.List.keep", @@ -7895,7 +8071,8 @@ "docstrings": [ "Uncurried version of [keepWithIndex](#keepWithIndex)." ], - "signature": "let keepWithIndexU: (t<'a>, ('a, int) => bool) => t<'a>" + "signature": "let keepWithIndexU: (t<'a>, ('a, int) => bool) => t<'a>", + "deprecated": "Use `keepWithIndex` instead" }, { "id": "Belt.List.keepWithIndex", @@ -7923,7 +8100,8 @@ "docstrings": [ "Uncurried version of [keepMap](#keepMap)." ], - "signature": "let keepMapU: (t<'a>, 'a => option<'b>) => t<'b>" + "signature": "let keepMapU: (t<'a>, 'a => option<'b>) => t<'b>", + "deprecated": "Use `keepMap` instead" }, { "id": "Belt.List.keepMap", @@ -7941,7 +8119,8 @@ "docstrings": [ "Uncurried version of [partition](#partition)." ], - "signature": "let partitionU: (t<'a>, 'a => bool) => (t<'a>, t<'a>)" + "signature": "let partitionU: (t<'a>, 'a => bool) => (t<'a>, t<'a>)", + "deprecated": "Use `partition` instead" }, { "id": "Belt.List.partition", @@ -7968,7 +8147,8 @@ "docstrings": [ "Uncurried version of [getAssoc](#getAssoc)." ], - "signature": "let getAssocU: (t<('a, 'c)>, 'b, ('a, 'b) => bool) => option<'c>" + "signature": "let getAssocU: (t<('a, 'c)>, 'b, ('a, 'b) => bool) => option<'c>", + "deprecated": "Use `getAssoc` instead" }, { "id": "Belt.List.getAssoc", @@ -7986,7 +8166,8 @@ "docstrings": [ "Uncurried version of [hasAssoc](#hasAssoc)." ], - "signature": "let hasAssocU: (t<('a, 'c)>, 'b, ('a, 'b) => bool) => bool" + "signature": "let hasAssocU: (t<('a, 'c)>, 'b, ('a, 'b) => bool) => bool", + "deprecated": "Use `hasAssoc` instead" }, { "id": "Belt.List.hasAssoc", @@ -8004,7 +8185,8 @@ "docstrings": [ "Uncurried version of [removeAssoc](#removeAssoc)." ], - "signature": "let removeAssocU: (t<('a, 'c)>, 'b, ('a, 'b) => bool) => t<('a, 'c)>" + "signature": "let removeAssocU: (t<('a, 'c)>, 'b, ('a, 'b) => bool) => t<('a, 'c)>", + "deprecated": "Use `removeAssoc` instead" }, { "id": "Belt.List.removeAssoc", @@ -8022,7 +8204,8 @@ "docstrings": [ "Uncurried version of [setAssoc](#setAssoc)." ], - "signature": "let setAssocU: (t<('a, 'c)>, 'a, 'c, ('a, 'a) => bool) => t<('a, 'c)>" + "signature": "let setAssocU: (t<('a, 'c)>, 'a, 'c, ('a, 'a) => bool) => t<('a, 'c)>", + "deprecated": "Use `setAssoc` instead" }, { "id": "Belt.List.setAssoc", @@ -8040,7 +8223,8 @@ "docstrings": [ "Uncurried version of [sort](#sort)." ], - "signature": "let sortU: (t<'a>, ('a, 'a) => int) => t<'a>" + "signature": "let sortU: (t<'a>, ('a, 'a) => int) => t<'a>", + "deprecated": "Use `sort` instead" }, { "id": "Belt.List.sort", @@ -8149,7 +8333,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t<'a>, 'a => unit) => unit" + "signature": "let forEachU: (t<'a>, 'a => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.MutableStack.forEach", @@ -8163,7 +8348,8 @@ "kind": "value", "name": "dynamicPopIterU", "docstrings": [], - "signature": "let dynamicPopIterU: (t<'a>, 'a => unit) => unit" + "signature": "let dynamicPopIterU: (t<'a>, 'a => unit) => unit", + "deprecated": "Use `dynamicPopIter` instead" }, { "id": "Belt.MutableStack.dynamicPopIter", @@ -8315,7 +8501,8 @@ "kind": "value", "name": "mapU", "docstrings": [], - "signature": "let mapU: (t<'a>, 'a => 'b) => t<'b>" + "signature": "let mapU: (t<'a>, 'a => 'b) => t<'b>", + "deprecated": "Use `map` instead" }, { "id": "Belt.MutableQueue.map", @@ -8329,7 +8516,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t<'a>, 'a => unit) => unit" + "signature": "let forEachU: (t<'a>, 'a => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.MutableQueue.forEach", @@ -8345,7 +8533,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (t<'a>, 'b, ('b, 'a) => 'b) => 'b" + "signature": "let reduceU: (t<'a>, 'b, ('b, 'a) => 'b) => 'b", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.MutableQueue.reduce", @@ -8389,7 +8578,8 @@ "kind": "value", "name": "strictlySortedLengthU", "docstrings": [], - "signature": "let strictlySortedLengthU: (array<'a>, ('a, 'a) => bool) => int" + "signature": "let strictlySortedLengthU: (array<'a>, ('a, 'a) => bool) => int", + "deprecated": "Use `strictlySortedLength` instead" }, { "id": "Belt.SortArray.strictlySortedLength", @@ -8405,7 +8595,8 @@ "kind": "value", "name": "isSortedU", "docstrings": [], - "signature": "let isSortedU: (array<'a>, ('a, 'a) => int) => bool" + "signature": "let isSortedU: (array<'a>, ('a, 'a) => int) => bool", + "deprecated": "Use `isSorted` instead" }, { "id": "Belt.SortArray.isSorted", @@ -8421,7 +8612,8 @@ "kind": "value", "name": "stableSortInPlaceByU", "docstrings": [], - "signature": "let stableSortInPlaceByU: (array<'a>, ('a, 'a) => int) => unit" + "signature": "let stableSortInPlaceByU: (array<'a>, ('a, 'a) => int) => unit", + "deprecated": "Use `stableSortInPlaceBy` instead" }, { "id": "Belt.SortArray.stableSortInPlaceBy", @@ -8435,7 +8627,8 @@ "kind": "value", "name": "stableSortByU", "docstrings": [], - "signature": "let stableSortByU: (array<'a>, ('a, 'a) => int) => array<'a>" + "signature": "let stableSortByU: (array<'a>, ('a, 'a) => int) => array<'a>", + "deprecated": "Use `stableSortBy` instead" }, { "id": "Belt.SortArray.stableSortBy", @@ -8451,7 +8644,8 @@ "kind": "value", "name": "binarySearchByU", "docstrings": [], - "signature": "let binarySearchByU: (array<'a>, 'a, ('a, 'a) => int) => int" + "signature": "let binarySearchByU: (array<'a>, 'a, ('a, 'a) => int) => int", + "deprecated": "Use `binarySearchBy` instead" }, { "id": "Belt.SortArray.binarySearchBy", @@ -8467,7 +8661,8 @@ "kind": "value", "name": "unionU", "docstrings": [], - "signature": "let unionU: (\n array<'a>,\n int,\n int,\n array<'a>,\n int,\n int,\n array<'a>,\n int,\n ('a, 'a) => int,\n) => int" + "signature": "let unionU: (\n array<'a>,\n int,\n int,\n array<'a>,\n int,\n int,\n array<'a>,\n int,\n ('a, 'a) => int,\n) => int", + "deprecated": "Use `union` instead" }, { "id": "Belt.SortArray.union", @@ -8483,7 +8678,8 @@ "kind": "value", "name": "intersectU", "docstrings": [], - "signature": "let intersectU: (\n array<'a>,\n int,\n int,\n array<'a>,\n int,\n int,\n array<'a>,\n int,\n ('a, 'a) => int,\n) => int" + "signature": "let intersectU: (\n array<'a>,\n int,\n int,\n array<'a>,\n int,\n int,\n array<'a>,\n int,\n ('a, 'a) => int,\n) => int", + "deprecated": "Use `intersect` instead" }, { "id": "Belt.SortArray.intersect", @@ -8499,7 +8695,8 @@ "kind": "value", "name": "diffU", "docstrings": [], - "signature": "let diffU: (\n array<'a>,\n int,\n int,\n array<'a>,\n int,\n int,\n array<'a>,\n int,\n ('a, 'a) => int,\n) => int" + "signature": "let diffU: (\n array<'a>,\n int,\n int,\n array<'a>,\n int,\n int,\n array<'a>,\n int,\n ('a, 'a) => int,\n) => int", + "deprecated": "Use `diff` instead" }, { "id": "Belt.SortArray.diff", @@ -8690,7 +8887,8 @@ "kind": "value", "name": "makeByU", "docstrings": [], - "signature": "let makeByU: (int, int => 'a) => t<'a>" + "signature": "let makeByU: (int, int => 'a) => t<'a>", + "deprecated": "Use `makeBy` instead" }, { "id": "Belt.Array.makeBy", @@ -8706,7 +8904,8 @@ "kind": "value", "name": "makeByAndShuffleU", "docstrings": [], - "signature": "let makeByAndShuffleU: (int, int => 'a) => t<'a>" + "signature": "let makeByAndShuffleU: (int, int => 'a) => t<'a>", + "deprecated": "Use `makeByAndShuffle` instead" }, { "id": "Belt.Array.makeByAndShuffle", @@ -8731,7 +8930,8 @@ "kind": "value", "name": "zipByU", "docstrings": [], - "signature": "let zipByU: (t<'a>, array<'b>, ('a, 'b) => 'c) => array<'c>" + "signature": "let zipByU: (t<'a>, array<'b>, ('a, 'b) => 'c) => array<'c>", + "deprecated": "Use `zipBy` instead" }, { "id": "Belt.Array.zipBy", @@ -8828,7 +9028,8 @@ "kind": "value", "name": "forEachU", "docstrings": [], - "signature": "let forEachU: (t<'a>, 'a => unit) => unit" + "signature": "let forEachU: (t<'a>, 'a => unit) => unit", + "deprecated": "Use `forEach` instead" }, { "id": "Belt.Array.forEach", @@ -8844,7 +9045,8 @@ "kind": "value", "name": "mapU", "docstrings": [], - "signature": "let mapU: (t<'a>, 'a => 'b) => array<'b>" + "signature": "let mapU: (t<'a>, 'a => 'b) => array<'b>", + "deprecated": "Use `map` instead" }, { "id": "Belt.Array.map", @@ -8860,7 +9062,8 @@ "kind": "value", "name": "flatMapU", "docstrings": [], - "signature": "let flatMapU: (t<'a>, 'a => array<'b>) => array<'b>" + "signature": "let flatMapU: (t<'a>, 'a => array<'b>) => array<'b>", + "deprecated": "Use `flatMap` instead" }, { "id": "Belt.Array.flatMap", @@ -8876,7 +9079,8 @@ "kind": "value", "name": "getByU", "docstrings": [], - "signature": "let getByU: (t<'a>, 'a => bool) => option<'a>" + "signature": "let getByU: (t<'a>, 'a => bool) => option<'a>", + "deprecated": "Use `getBy` instead" }, { "id": "Belt.Array.getBy", @@ -8892,7 +9096,8 @@ "kind": "value", "name": "getIndexByU", "docstrings": [], - "signature": "let getIndexByU: (t<'a>, 'a => bool) => option" + "signature": "let getIndexByU: (t<'a>, 'a => bool) => option", + "deprecated": "Use `getIndexBy` instead" }, { "id": "Belt.Array.getIndexBy", @@ -8908,7 +9113,8 @@ "kind": "value", "name": "keepU", "docstrings": [], - "signature": "let keepU: (t<'a>, 'a => bool) => t<'a>" + "signature": "let keepU: (t<'a>, 'a => bool) => t<'a>", + "deprecated": "Use `keep` instead" }, { "id": "Belt.Array.keep", @@ -8924,7 +9130,8 @@ "kind": "value", "name": "keepWithIndexU", "docstrings": [], - "signature": "let keepWithIndexU: (t<'a>, ('a, int) => bool) => t<'a>" + "signature": "let keepWithIndexU: (t<'a>, ('a, int) => bool) => t<'a>", + "deprecated": "Use `keepWithIndex` instead" }, { "id": "Belt.Array.keepWithIndex", @@ -8940,7 +9147,8 @@ "kind": "value", "name": "keepMapU", "docstrings": [], - "signature": "let keepMapU: (t<'a>, 'a => option<'b>) => array<'b>" + "signature": "let keepMapU: (t<'a>, 'a => option<'b>) => array<'b>", + "deprecated": "Use `keepMap` instead" }, { "id": "Belt.Array.keepMap", @@ -8956,7 +9164,8 @@ "kind": "value", "name": "forEachWithIndexU", "docstrings": [], - "signature": "let forEachWithIndexU: (t<'a>, (int, 'a) => unit) => unit" + "signature": "let forEachWithIndexU: (t<'a>, (int, 'a) => unit) => unit", + "deprecated": "Use `forEachWithIndex` instead" }, { "id": "Belt.Array.forEachWithIndex", @@ -8972,7 +9181,8 @@ "kind": "value", "name": "mapWithIndexU", "docstrings": [], - "signature": "let mapWithIndexU: (t<'a>, (int, 'a) => 'b) => array<'b>" + "signature": "let mapWithIndexU: (t<'a>, (int, 'a) => 'b) => array<'b>", + "deprecated": "Use `mapWithIndex` instead" }, { "id": "Belt.Array.mapWithIndex", @@ -8988,7 +9198,8 @@ "kind": "value", "name": "partitionU", "docstrings": [], - "signature": "let partitionU: (t<'a>, 'a => bool) => (t<'a>, t<'a>)" + "signature": "let partitionU: (t<'a>, 'a => bool) => (t<'a>, t<'a>)", + "deprecated": "Use `partition` instead" }, { "id": "Belt.Array.partition", @@ -9004,7 +9215,8 @@ "kind": "value", "name": "reduceU", "docstrings": [], - "signature": "let reduceU: (array<'b>, 'a, ('a, 'b) => 'a) => 'a" + "signature": "let reduceU: (array<'b>, 'a, ('a, 'b) => 'a) => 'a", + "deprecated": "Use `reduce` instead" }, { "id": "Belt.Array.reduce", @@ -9020,7 +9232,8 @@ "kind": "value", "name": "reduceReverseU", "docstrings": [], - "signature": "let reduceReverseU: (array<'b>, 'a, ('a, 'b) => 'a) => 'a" + "signature": "let reduceReverseU: (array<'b>, 'a, ('a, 'b) => 'a) => 'a", + "deprecated": "Use `reduceReverse` instead" }, { "id": "Belt.Array.reduceReverse", @@ -9036,7 +9249,8 @@ "kind": "value", "name": "reduceReverse2U", "docstrings": [], - "signature": "let reduceReverse2U: (t<'a>, array<'b>, 'c, ('c, 'a, 'b) => 'c) => 'c" + "signature": "let reduceReverse2U: (t<'a>, array<'b>, 'c, ('c, 'a, 'b) => 'c) => 'c", + "deprecated": "Use `reduceReverse2` instead" }, { "id": "Belt.Array.reduceReverse2", @@ -9052,7 +9266,8 @@ "kind": "value", "name": "reduceWithIndexU", "docstrings": [], - "signature": "let reduceWithIndexU: (t<'a>, 'b, ('b, 'a, int) => 'b) => 'b" + "signature": "let reduceWithIndexU: (t<'a>, 'b, ('b, 'a, int) => 'b) => 'b", + "deprecated": "Use `reduceWithIndex` instead" }, { "id": "Belt.Array.reduceWithIndex", @@ -9068,7 +9283,8 @@ "kind": "value", "name": "joinWithU", "docstrings": [], - "signature": "let joinWithU: (t<'a>, string, 'a => string) => string" + "signature": "let joinWithU: (t<'a>, string, 'a => string) => string", + "deprecated": "Use `joinWith` instead" }, { "id": "Belt.Array.joinWith", @@ -9084,7 +9300,8 @@ "kind": "value", "name": "someU", "docstrings": [], - "signature": "let someU: (t<'a>, 'a => bool) => bool" + "signature": "let someU: (t<'a>, 'a => bool) => bool", + "deprecated": "Use `some` instead" }, { "id": "Belt.Array.some", @@ -9100,7 +9317,8 @@ "kind": "value", "name": "everyU", "docstrings": [], - "signature": "let everyU: (t<'a>, 'a => bool) => bool" + "signature": "let everyU: (t<'a>, 'a => bool) => bool", + "deprecated": "Use `every` instead" }, { "id": "Belt.Array.every", @@ -9116,7 +9334,8 @@ "kind": "value", "name": "every2U", "docstrings": [], - "signature": "let every2U: (t<'a>, array<'b>, ('a, 'b) => bool) => bool" + "signature": "let every2U: (t<'a>, array<'b>, ('a, 'b) => bool) => bool", + "deprecated": "Use `every2` instead" }, { "id": "Belt.Array.every2", @@ -9132,7 +9351,8 @@ "kind": "value", "name": "some2U", "docstrings": [], - "signature": "let some2U: (t<'a>, array<'b>, ('a, 'b) => bool) => bool" + "signature": "let some2U: (t<'a>, array<'b>, ('a, 'b) => bool) => bool", + "deprecated": "Use `some2` instead" }, { "id": "Belt.Array.some2", @@ -9148,7 +9368,8 @@ "kind": "value", "name": "cmpU", "docstrings": [], - "signature": "let cmpU: (t<'a>, t<'a>, ('a, 'a) => int) => int" + "signature": "let cmpU: (t<'a>, t<'a>, ('a, 'a) => int) => int", + "deprecated": "Use `cmp` instead" }, { "id": "Belt.Array.cmp", @@ -9164,7 +9385,8 @@ "kind": "value", "name": "eqU", "docstrings": [], - "signature": "let eqU: (t<'a>, t<'a>, ('a, 'a) => bool) => bool" + "signature": "let eqU: (t<'a>, t<'a>, ('a, 'a) => bool) => bool", + "deprecated": "Use `eq` instead" }, { "id": "Belt.Array.eq", @@ -9189,7 +9411,8 @@ "kind": "value", "name": "initU", "docstrings": [], - "signature": "let initU: (int, int => 'a) => t<'a>" + "signature": "let initU: (int, int => 'a) => t<'a>", + "deprecated": "Use `init` instead" }, { "id": "Belt.Array.init", @@ -9258,7 +9481,8 @@ "kind": "value", "name": "comparableU", "docstrings": [], - "signature": "let comparableU: (\n ~cmp: ('a, 'a) => int,\n) => module(Comparable with type t = 'a)" + "signature": "let comparableU: (\n ~cmp: ('a, 'a) => int,\n) => module(Comparable with type t = 'a)", + "deprecated": "Use `comparable` instead" }, { "id": "Belt.Id.comparable", @@ -9283,7 +9507,8 @@ "kind": "value", "name": "hashableU", "docstrings": [], - "signature": "let hashableU: (\n ~hash: 'a => int,\n ~eq: ('a, 'a) => bool,\n) => module(Hashable with type t = 'a)" + "signature": "let hashableU: (\n ~hash: 'a => int,\n ~eq: ('a, 'a) => bool,\n) => module(Hashable with type t = 'a)", + "deprecated": "Use `hashable` instead" }, { "id": "Belt.Id.hashable", diff --git a/data/api/latest/core.json b/data/api/latest/core.json index ab511e0aa..63938b226 100644 --- a/data/api/latest/core.json +++ b/data/api/latest/core.json @@ -2041,6 +2041,76 @@ } ] }, + "core/int/bitwise": { + "id": "Core.Int.Bitwise", + "name": "Bitwise", + "docstrings": [], + "items": [ + { + "id": "Core.Int.Bitwise.land", + "kind": "value", + "name": "land", + "docstrings": [ + "`land(n1, n2)` calculates the bitwise logical AND of two integers.\n\n ## Examples\n\n ```rescript\n Int.Bitwise.land(7, 4) == 4\n ```" + ], + "signature": "let land: (int, int) => int" + }, + { + "id": "Core.Int.Bitwise.lor", + "kind": "value", + "name": "lor", + "docstrings": [ + "`lor(n1, n2)` calculates the bitwise logical OR of two integers.\n\n ## Examples\n\n ```rescript\n Int.Bitwise.lor(7, 4) == 7\n ```" + ], + "signature": "let lor: (int, int) => int" + }, + { + "id": "Core.Int.Bitwise.lxor", + "kind": "value", + "name": "lxor", + "docstrings": [ + "`lxor(n1, n2)` calculates the bitwise logical XOR of two integers.\n\n ## Examples\n\n ```rescript\n Int.Bitwise.lxor(7, 4) == 3\n ```" + ], + "signature": "let lxor: (int, int) => int" + }, + { + "id": "Core.Int.Bitwise.lnot", + "kind": "value", + "name": "lnot", + "docstrings": [ + "`lnot(n)` calculates the bitwise logical NOT of an integer.\n\n ## Examples\n\n ```rescript\n Int.Bitwise.lnot(2) == -3\n ```" + ], + "signature": "let lnot: int => int" + }, + { + "id": "Core.Int.Bitwise.lsl", + "kind": "value", + "name": "lsl", + "docstrings": [ + "`lsl(n, length)` calculates the bitwise logical left shift of an integer `n` by `length`.\n\n ## Examples\n\n ```rescript\n Int.Bitwise.lsl(4, 1) == 8\n ```" + ], + "signature": "let lsl: (int, int) => int" + }, + { + "id": "Core.Int.Bitwise.lsr", + "kind": "value", + "name": "lsr", + "docstrings": [ + "`lsr(n, length)` calculates the bitwise logical right shift of an integer `n` by `length`.\n\n ## Examples\n\n ```rescript\n Int.Bitwise.lsr(8, 1) == 4\n ```" + ], + "signature": "let lsr: (int, int) => int" + }, + { + "id": "Core.Int.Bitwise.asr", + "kind": "value", + "name": "asr", + "docstrings": [ + "`asr(n, length)` calculates the bitwise arithmetic right shift of an integer `n` by `length`.\n\n ## Examples\n\n ```rescript\n Int.Bitwise.asr(4, 1) == 2\n ```" + ], + "signature": "let asr: (int, int) => int" + } + ] + }, "core/int/constants": { "id": "Core.Int.Constants", "name": "Constants", @@ -4742,12 +4812,39 @@ "docstrings": [], "signature": "type value<'a> = {done: bool, value: option<'a>}" }, + { + "id": "Core.AsyncIterator.make", + "kind": "value", + "name": "make", + "docstrings": [ + "`make(nextFn)`\n\n Creates an async iterator from a function that returns the next value of the iterator.\n\n ## Examples\n - A simple example, creating an async iterator that returns 1, 2, 3:\n ```rescript\n let context = ref(0)\n\n let asyncIterator = AsyncIterator.make(async () => {\n let currentValue = context.contents\n // Increment current value\n context := currentValue + 1\n \n {\n AsyncIterator.value: Some(currentValue),\n done: currentValue >= 3\n }\n })\n\n // This will log 1, 2, 3\n await asyncIterator->AsyncIterator.forEach(value =>\n switch value {\n | Some(value) => Console.log(value)\n | None => ()\n }\n )\n ```" + ], + "signature": "let make: (unit => promise>) => t<'value>" + }, + { + "id": "Core.AsyncIterator.value", + "kind": "value", + "name": "value", + "docstrings": [ + "`value(value)`\n\n Shorthand for creating a value object with the provided value, and the `done` property set to false.\n\n ## Examples\n ```rescript\n let context = ref(0)\n\n let asyncIterator = AsyncIterator.make(async () => {\n let currentValue = context.contents\n // Increment current value\n context := currentValue + 1\n \n if currentValue >= 3 {\n AsyncIterator.done()\n } else {\n AsyncIterator.value(currentValue)\n }\n })\n ```" + ], + "signature": "let value: 'value => value<'value>" + }, + { + "id": "Core.AsyncIterator.done", + "kind": "value", + "name": "done", + "docstrings": [ + "`done(~finalValue=?)`\n\n Shorthand for creating a value object with the `done` property set to true, and the provided value as the final value, if any.\n\n ## Examples\n ```rescript\n let context = ref(0)\n\n let asyncIterator = AsyncIterator.make(async () => {\n let currentValue = context.contents\n // Increment current value\n context := currentValue + 1\n \n if currentValue >= 3 {\n AsyncIterator.done()\n } else {\n AsyncIterator.value(currentValue)\n }\n })\n ```" + ], + "signature": "let done: (~finalValue: 'value=?) => value<'value>" + }, { "id": "Core.AsyncIterator.next", "kind": "value", "name": "next", "docstrings": [ - "`next(asyncIterator)`\n\nReturns the next value of the iterator, if any.\n\nSee [async iterator protocols](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Iteration_protocols#the_async_iterator_and_async_iterable_protocols) on MDN.\n\n## Examples\n- A simple example, getting the next value:\n```rescript\n@val external asyncIterator: AsyncIterator.t = \"someAsyncIterator\"\nlet {AsyncIterator.done, value} = await asyncIterator->AsyncIterator.next\n```\n\n- Complete example, including looping over all values:\n```rescript\n// Let's pretend we get an async iterator returning ints from somewhere.\n@val external asyncIterator: AsyncIterator.t = \"someAsyncIterator\"\n\n\nlet processMyAsyncIterator = async () => {\n // ReScript doesn't have `for ... of` loops, but it's easy to mimic using a while loop.\n let break = ref(false)\n\n while !break.contents {\n // Await the next iterator value\n let {value, done} = await asyncIterator->AsyncIterator.next\n\n // Exit the while loop if the iterator says it's done\n break := done\n\n // This will log the (int) value of the current async iteration, if a value was returned.\n Console.log(value)\n }\n}\n```" + "`next(asyncIterator)`\n\nReturns the next value of the iterator, if any.\n\nSee [async iterator protocols](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Iteration_protocols#the_async_iterator_and_async_iterable_protocols) on MDN.\n\n## Examples\n- A simple example, getting the next value:\n```rescript\n@val external asyncIterator: AsyncIterator.t = \"someAsyncIterator\"\nlet value = await asyncIterator->AsyncIterator.next\n```\n\n- Complete example, including looping over all values:\n```rescript\n// Let's pretend we get an async iterator returning ints from somewhere.\n@val external asyncIterator: AsyncIterator.t = \"someAsyncIterator\"\n\n\nlet processMyAsyncIterator = async () => {\n // ReScript doesn't have `for ... of` loops, but it's easy to mimic using a while loop.\n let break = ref(false)\n\n while !break.contents {\n // Await the next iterator value\n let {value, done} = await asyncIterator->AsyncIterator.next\n\n // Exit the while loop if the iterator says it's done\n break := done\n\n // This will log the (int) value of the current async iteration, if a value was returned.\n Console.log(value)\n }\n}\n```" ], "signature": "let next: t<'a> => promise>" }, diff --git a/data/api/latest/js.json b/data/api/latest/js.json index c9b052ef9..e5d10f27b 100644 --- a/data/api/latest/js.json +++ b/data/api/latest/js.json @@ -3,7 +3,7 @@ "id": "Js", "name": "Js", "docstrings": [ - "The Js module mostly contains ReScript bindings to _standard JavaScript APIs_\nlike [console.log](https://developer.mozilla.org/en-US/docs/Web/API/Console/log),\nor the JavaScript\n[String](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String),\n[Date](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Date), and\n[Promise](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise)\nclasses.\n\nIt is meant as a zero-abstraction interop layer and directly exposes JavaScript functions as they are. If you can find your API in this module, prefer this over an equivalent Belt helper. For example, prefer [Js.Array2](js/array2) over [Belt.Array](belt/array)\n\n## Argument Order\n\nFor historical reasons, some APIs in the Js namespace (e.g. [Js.String](js/string)) are\nusing the data-last argument order whereas others (e.g. [Js.Date](js/date)) are using data-first.\n\nFor more information about these argument orders and the trade-offs between them, see\n[this blog post](https://www.javierchavarri.com/data-first-and-data-last-a-comparison/).\n\n_Eventually, all modules in the Js namespace are going to be migrated to data-first though._\n\nIn the meantime, there are several options for dealing with the data-last APIs:\n\n## Examples\n\n```rescript\n/* Js.String (data-last API used with pipe last operator) */\nJs.log(\"2019-11-10\" |> Js.String.split(\"-\"))\nJs.log(\"ReScript\" |> Js.String.startsWith(\"Re\"))\n\n/* Js.String (data-last API used with pipe first operator) */\nJs.log(\"2019-11-10\"->Js.String.split(\"-\", _))\nJs.log(\"ReScript\"->Js.String.startsWith(\"Re\", _))\n\n/* Js.String (data-last API used without any piping) */\nJs.log(Js.String.split(\"-\", \"2019-11-10\"))\nJs.log(Js.String.startsWith(\"Re\", \"ReScript\"))\n```\n## Js.Xxx2 Modules\n\nPrefer `Js.Array2` over `Js.Array`, `Js.String2` over `Js.String`, etc. The latters are old modules." + "The Js module mostly contains ReScript bindings to _standard JavaScript APIs_\nlike [console.log](https://developer.mozilla.org/en-US/docs/Web/API/Console/log),\nor the JavaScript\n[String](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String),\n[Date](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Date), and\n[Promise](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise)\nclasses.\n\nIt is meant as a zero-abstraction interop layer and directly exposes JavaScript functions as they are. If you can find your API in this module, prefer this over an equivalent Belt helper. For example, prefer [Js.Array2](js/array2) over [Belt.Array](belt/array)\n\n## Argument Order\n\nFor historical reasons, some APIs in the Js namespace (e.g. [Js.String](js/string)) are\nusing the data-last argument order whereas others (e.g. [Js.Date](js/date)) are using data-first.\n\nFor more information about these argument orders and the trade-offs between them, see\n[this blog post](https://www.javierchavarri.com/data-first-and-data-last-a-comparison/).\n\n_Eventually, all modules in the Js namespace are going to be migrated to data-first though._\n\nIn the meantime, there are several options for dealing with the data-last APIs:\n\n## Examples\n\n```rescript\n/* Js.String (data-last API used with pipe last operator) */\nJs.log(\\\"2019-11-10\\\" |> Js.String.split(\\\"-\\\"))\nJs.log(\\\"ReScript\\\" |> Js.String.startsWith(\\\"Re\\\"))\n\n/* Js.String (data-last API used with pipe first operator) */\nJs.log(\\\"2019-11-10\\\"->Js.String.split(\\\"-\\\", _))\nJs.log(\\\"ReScript\\\"->Js.String.startsWith(\\\"Re\\\", _))\n\n/* Js.String (data-last API used without any piping) */\nJs.log(Js.String.split(\\\"-\\\", \\\"2019-11-10\\\"))\nJs.log(Js.String.startsWith(\\\"Re\\\", \\\"ReScript\\\"))\n```\n## Js.Xxx2 Modules\n\nPrefer `Js.Array2` over `Js.Array`, `Js.String2` over `Js.String`, etc. The latters are old modules." ], "items": [ { @@ -15,6 +15,15 @@ ], "signature": "type t<'a> = 'a\n constraint 'a = {..}" }, + { + "id": "Js.globalThis", + "kind": "value", + "name": "globalThis", + "docstrings": [ + "JS global object reference" + ], + "signature": "let globalThis: t<{..}>" + }, { "id": "Js.null", "kind": "type", @@ -4750,20 +4759,6 @@ "docstrings": [], "signature": "let byteOffset: t => int" }, - { - "id": "Js.Typed_array.Float64Array.setArray", - "kind": "value", - "name": "setArray", - "docstrings": [], - "signature": "let setArray: (array, t) => unit" - }, - { - "id": "Js.Typed_array.Float64Array.setArrayOffset", - "kind": "value", - "name": "setArrayOffset", - "docstrings": [], - "signature": "let setArrayOffset: (array, int, t) => unit" - }, { "id": "Js.Typed_array.Float64Array.length", "kind": "value", @@ -4771,292 +4766,26 @@ "docstrings": [], "signature": "let length: t => int" }, - { - "id": "Js.Typed_array.Float64Array.copyWithin", - "kind": "value", - "name": "copyWithin", - "docstrings": [], - "signature": "let copyWithin: (~to_: int, t) => t" - }, - { - "id": "Js.Typed_array.Float64Array.copyWithinFrom", - "kind": "value", - "name": "copyWithinFrom", - "docstrings": [], - "signature": "let copyWithinFrom: (~to_: int, ~from: int, t) => t" - }, { "id": "Js.Typed_array.Float64Array.copyWithinFromRange", "kind": "value", "name": "copyWithinFromRange", "docstrings": [], - "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Float64Array.fillInPlace", - "kind": "value", - "name": "fillInPlace", - "docstrings": [], - "signature": "let fillInPlace: (elt, t) => t" - }, - { - "id": "Js.Typed_array.Float64Array.fillFromInPlace", - "kind": "value", - "name": "fillFromInPlace", - "docstrings": [], - "signature": "let fillFromInPlace: (elt, ~from: int, t) => t" - }, - { - "id": "Js.Typed_array.Float64Array.fillRangeInPlace", - "kind": "value", - "name": "fillRangeInPlace", - "docstrings": [], - "signature": "let fillRangeInPlace: (elt, ~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Float64Array.reverseInPlace", - "kind": "value", - "name": "reverseInPlace", - "docstrings": [], - "signature": "let reverseInPlace: t => t" - }, - { - "id": "Js.Typed_array.Float64Array.sortInPlace", - "kind": "value", - "name": "sortInPlace", - "docstrings": [], - "signature": "let sortInPlace: t => t" - }, - { - "id": "Js.Typed_array.Float64Array.sortInPlaceWith", - "kind": "value", - "name": "sortInPlaceWith", - "docstrings": [], - "signature": "let sortInPlaceWith: ((elt, elt) => int, t) => t" - }, - { - "id": "Js.Typed_array.Float64Array.includes", - "kind": "value", - "name": "includes", - "docstrings": [], - "signature": "let includes: (elt, t) => bool" - }, - { - "id": "Js.Typed_array.Float64Array.indexOf", - "kind": "value", - "name": "indexOf", - "docstrings": [], - "signature": "let indexOf: (elt, t) => int" - }, - { - "id": "Js.Typed_array.Float64Array.indexOfFrom", - "kind": "value", - "name": "indexOfFrom", - "docstrings": [], - "signature": "let indexOfFrom: (elt, ~from: int, t) => int" - }, - { - "id": "Js.Typed_array.Float64Array.join", - "kind": "value", - "name": "join", - "docstrings": [], - "signature": "let join: t => string" - }, - { - "id": "Js.Typed_array.Float64Array.joinWith", - "kind": "value", - "name": "joinWith", - "docstrings": [], - "signature": "let joinWith: (string, t) => string" - }, - { - "id": "Js.Typed_array.Float64Array.lastIndexOf", - "kind": "value", - "name": "lastIndexOf", - "docstrings": [], - "signature": "let lastIndexOf: (elt, t) => int" - }, - { - "id": "Js.Typed_array.Float64Array.lastIndexOfFrom", - "kind": "value", - "name": "lastIndexOfFrom", - "docstrings": [], - "signature": "let lastIndexOfFrom: (elt, ~from: int, t) => int" + "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int) => t" }, { "id": "Js.Typed_array.Float64Array.slice", "kind": "value", "name": "slice", "docstrings": [], - "signature": "let slice: (~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Float64Array.copy", - "kind": "value", - "name": "copy", - "docstrings": [], - "signature": "let copy: t => t" - }, - { - "id": "Js.Typed_array.Float64Array.sliceFrom", - "kind": "value", - "name": "sliceFrom", - "docstrings": [], - "signature": "let sliceFrom: (int, t) => t" + "signature": "let slice: (~start: int, ~end_: int) => t" }, { "id": "Js.Typed_array.Float64Array.subarray", "kind": "value", "name": "subarray", "docstrings": [], - "signature": "let subarray: (~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Float64Array.subarrayFrom", - "kind": "value", - "name": "subarrayFrom", - "docstrings": [], - "signature": "let subarrayFrom: (int, t) => t" - }, - { - "id": "Js.Typed_array.Float64Array.toString", - "kind": "value", - "name": "toString", - "docstrings": [], - "signature": "let toString: t => string" - }, - { - "id": "Js.Typed_array.Float64Array.toLocaleString", - "kind": "value", - "name": "toLocaleString", - "docstrings": [], - "signature": "let toLocaleString: t => string" - }, - { - "id": "Js.Typed_array.Float64Array.every", - "kind": "value", - "name": "every", - "docstrings": [], - "signature": "let every: (elt => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Float64Array.everyi", - "kind": "value", - "name": "everyi", - "docstrings": [], - "signature": "let everyi: ((elt, int) => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Float64Array.filter", - "kind": "value", - "name": "filter", - "docstrings": [], - "signature": "let filter: (elt => bool, t) => t" - }, - { - "id": "Js.Typed_array.Float64Array.filteri", - "kind": "value", - "name": "filteri", - "docstrings": [], - "signature": "let filteri: ((elt, int) => bool, t) => t" - }, - { - "id": "Js.Typed_array.Float64Array.find", - "kind": "value", - "name": "find", - "docstrings": [], - "signature": "let find: (elt => bool, t) => Js.undefined" - }, - { - "id": "Js.Typed_array.Float64Array.findi", - "kind": "value", - "name": "findi", - "docstrings": [], - "signature": "let findi: ((elt, int) => bool, t) => Js.undefined" - }, - { - "id": "Js.Typed_array.Float64Array.findIndex", - "kind": "value", - "name": "findIndex", - "docstrings": [], - "signature": "let findIndex: (elt => bool, t) => int" - }, - { - "id": "Js.Typed_array.Float64Array.findIndexi", - "kind": "value", - "name": "findIndexi", - "docstrings": [], - "signature": "let findIndexi: ((elt, int) => bool, t) => int" - }, - { - "id": "Js.Typed_array.Float64Array.forEach", - "kind": "value", - "name": "forEach", - "docstrings": [], - "signature": "let forEach: (elt => unit, t) => unit" - }, - { - "id": "Js.Typed_array.Float64Array.forEachi", - "kind": "value", - "name": "forEachi", - "docstrings": [], - "signature": "let forEachi: ((elt, int) => unit, t) => unit" - }, - { - "id": "Js.Typed_array.Float64Array.map", - "kind": "value", - "name": "map", - "docstrings": [], - "signature": "let map: (elt => 'a, t) => typed_array<'a>" - }, - { - "id": "Js.Typed_array.Float64Array.mapi", - "kind": "value", - "name": "mapi", - "docstrings": [], - "signature": "let mapi: ((elt, int) => 'a, t) => typed_array<'a>" - }, - { - "id": "Js.Typed_array.Float64Array.reduce", - "kind": "value", - "name": "reduce", - "docstrings": [], - "signature": "let reduce: (('a, elt) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Float64Array.reducei", - "kind": "value", - "name": "reducei", - "docstrings": [], - "signature": "let reducei: (('a, elt, int) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Float64Array.reduceRight", - "kind": "value", - "name": "reduceRight", - "docstrings": [], - "signature": "let reduceRight: (('a, elt) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Float64Array.reduceRighti", - "kind": "value", - "name": "reduceRighti", - "docstrings": [], - "signature": "let reduceRighti: (('a, elt, int) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Float64Array.some", - "kind": "value", - "name": "some", - "docstrings": [], - "signature": "let some: (elt => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Float64Array.somei", - "kind": "value", - "name": "somei", - "docstrings": [], - "signature": "let somei: ((elt, int) => bool, t) => bool" + "signature": "let subarray: (~start: int, ~end_: int) => t" }, { "id": "Js.Typed_array.Float64Array._BYTES_PER_ELEMENT", @@ -5200,20 +4929,6 @@ "docstrings": [], "signature": "let byteOffset: t => int" }, - { - "id": "Js.Typed_array.Float32Array.setArray", - "kind": "value", - "name": "setArray", - "docstrings": [], - "signature": "let setArray: (array, t) => unit" - }, - { - "id": "Js.Typed_array.Float32Array.setArrayOffset", - "kind": "value", - "name": "setArrayOffset", - "docstrings": [], - "signature": "let setArrayOffset: (array, int, t) => unit" - }, { "id": "Js.Typed_array.Float32Array.length", "kind": "value", @@ -5221,309 +4936,361 @@ "docstrings": [], "signature": "let length: t => int" }, - { - "id": "Js.Typed_array.Float32Array.copyWithin", - "kind": "value", - "name": "copyWithin", - "docstrings": [], - "signature": "let copyWithin: (~to_: int, t) => t" - }, - { - "id": "Js.Typed_array.Float32Array.copyWithinFrom", - "kind": "value", - "name": "copyWithinFrom", - "docstrings": [], - "signature": "let copyWithinFrom: (~to_: int, ~from: int, t) => t" - }, { "id": "Js.Typed_array.Float32Array.copyWithinFromRange", "kind": "value", "name": "copyWithinFromRange", "docstrings": [], - "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int, t) => t" + "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Float32Array.fillInPlace", + "id": "Js.Typed_array.Float32Array.slice", "kind": "value", - "name": "fillInPlace", + "name": "slice", "docstrings": [], - "signature": "let fillInPlace: (elt, t) => t" + "signature": "let slice: (~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Float32Array.fillFromInPlace", + "id": "Js.Typed_array.Float32Array.subarray", "kind": "value", - "name": "fillFromInPlace", + "name": "subarray", "docstrings": [], - "signature": "let fillFromInPlace: (elt, ~from: int, t) => t" + "signature": "let subarray: (~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Float32Array.fillRangeInPlace", + "id": "Js.Typed_array.Float32Array._BYTES_PER_ELEMENT", "kind": "value", - "name": "fillRangeInPlace", + "name": "_BYTES_PER_ELEMENT", "docstrings": [], - "signature": "let fillRangeInPlace: (elt, ~start: int, ~end_: int, t) => t" + "signature": "let _BYTES_PER_ELEMENT: int" }, { - "id": "Js.Typed_array.Float32Array.reverseInPlace", + "id": "Js.Typed_array.Float32Array.make", "kind": "value", - "name": "reverseInPlace", + "name": "make", "docstrings": [], - "signature": "let reverseInPlace: t => t" + "signature": "let make: array => t" }, { - "id": "Js.Typed_array.Float32Array.sortInPlace", + "id": "Js.Typed_array.Float32Array.fromBuffer", "kind": "value", - "name": "sortInPlace", - "docstrings": [], - "signature": "let sortInPlace: t => t" + "name": "fromBuffer", + "docstrings": [ + "can throw" + ], + "signature": "let fromBuffer: array_buffer => t" }, { - "id": "Js.Typed_array.Float32Array.sortInPlaceWith", + "id": "Js.Typed_array.Float32Array.fromBufferOffset", "kind": "value", - "name": "sortInPlaceWith", - "docstrings": [], - "signature": "let sortInPlaceWith: ((elt, elt) => int, t) => t" + "name": "fromBufferOffset", + "docstrings": [ + "**raise** Js.Exn.Error raise Js exception\n\n **param** offset is in bytes" + ], + "signature": "let fromBufferOffset: (array_buffer, int) => t" }, { - "id": "Js.Typed_array.Float32Array.includes", + "id": "Js.Typed_array.Float32Array.fromBufferRange", "kind": "value", - "name": "includes", - "docstrings": [], - "signature": "let includes: (elt, t) => bool" + "name": "fromBufferRange", + "docstrings": [ + "**raise** Js.Exn.Error raises Js exception\n\n **param** offset is in bytes, length in elements" + ], + "signature": "let fromBufferRange: (array_buffer, ~offset: int, ~length: int) => t" }, { - "id": "Js.Typed_array.Float32Array.indexOf", + "id": "Js.Typed_array.Float32Array.fromLength", "kind": "value", - "name": "indexOf", + "name": "fromLength", "docstrings": [], - "signature": "let indexOf: (elt, t) => int" + "signature": "let fromLength: int => t" }, { - "id": "Js.Typed_array.Float32Array.indexOfFrom", + "id": "Js.Typed_array.Float32Array.from", "kind": "value", - "name": "indexOfFrom", + "name": "from", "docstrings": [], - "signature": "let indexOfFrom: (elt, ~from: int, t) => int" + "signature": "let from: array_like => t" }, { - "id": "Js.Typed_array.Float32Array.join", + "id": "Js.Typed_array.Float32Array.create", "kind": "value", - "name": "join", + "name": "create", "docstrings": [], - "signature": "let join: t => string" + "signature": "let create: array => t", + "deprecated": "use `make` instead" }, { - "id": "Js.Typed_array.Float32Array.joinWith", + "id": "Js.Typed_array.Float32Array.of_buffer", "kind": "value", - "name": "joinWith", + "name": "of_buffer", "docstrings": [], - "signature": "let joinWith: (string, t) => string" + "signature": "let of_buffer: array_buffer => t", + "deprecated": "use `fromBuffer` instead" + } + ] + }, + "js/typed_array/uint32array": { + "id": "Js.Typed_array.Uint32Array", + "name": "Uint32Array", + "docstrings": [], + "items": [ + { + "id": "Js.Typed_array.Uint32Array.elt", + "kind": "type", + "name": "elt", + "docstrings": [ + "" + ], + "signature": "type elt = int" }, { - "id": "Js.Typed_array.Float32Array.lastIndexOf", - "kind": "value", - "name": "lastIndexOf", + "id": "Js.Typed_array.Uint32Array.typed_array", + "kind": "type", + "name": "typed_array", "docstrings": [], - "signature": "let lastIndexOf: (elt, t) => int" + "signature": "type typed_array<\n 'a,\n> = Js_typed_array2.Uint32Array.typed_array<'a>" }, { - "id": "Js.Typed_array.Float32Array.lastIndexOfFrom", - "kind": "value", - "name": "lastIndexOfFrom", + "id": "Js.Typed_array.Uint32Array.t", + "kind": "type", + "name": "t", "docstrings": [], - "signature": "let lastIndexOfFrom: (elt, ~from: int, t) => int" + "signature": "type t = typed_array" }, { - "id": "Js.Typed_array.Float32Array.slice", + "id": "Js.Typed_array.Uint32Array.unsafe_get", "kind": "value", - "name": "slice", + "name": "unsafe_get", "docstrings": [], - "signature": "let slice: (~start: int, ~end_: int, t) => t" + "signature": "let unsafe_get: (t, int) => elt" }, { - "id": "Js.Typed_array.Float32Array.copy", + "id": "Js.Typed_array.Uint32Array.unsafe_set", "kind": "value", - "name": "copy", + "name": "unsafe_set", "docstrings": [], - "signature": "let copy: t => t" + "signature": "let unsafe_set: (t, int, elt) => unit" }, { - "id": "Js.Typed_array.Float32Array.sliceFrom", + "id": "Js.Typed_array.Uint32Array.buffer", "kind": "value", - "name": "sliceFrom", + "name": "buffer", "docstrings": [], - "signature": "let sliceFrom: (int, t) => t" + "signature": "let buffer: t => array_buffer" }, { - "id": "Js.Typed_array.Float32Array.subarray", + "id": "Js.Typed_array.Uint32Array.byteLength", "kind": "value", - "name": "subarray", + "name": "byteLength", "docstrings": [], - "signature": "let subarray: (~start: int, ~end_: int, t) => t" + "signature": "let byteLength: t => int" }, { - "id": "Js.Typed_array.Float32Array.subarrayFrom", + "id": "Js.Typed_array.Uint32Array.byteOffset", "kind": "value", - "name": "subarrayFrom", + "name": "byteOffset", "docstrings": [], - "signature": "let subarrayFrom: (int, t) => t" + "signature": "let byteOffset: t => int" }, { - "id": "Js.Typed_array.Float32Array.toString", + "id": "Js.Typed_array.Uint32Array.length", "kind": "value", - "name": "toString", + "name": "length", "docstrings": [], - "signature": "let toString: t => string" + "signature": "let length: t => int" }, { - "id": "Js.Typed_array.Float32Array.toLocaleString", + "id": "Js.Typed_array.Uint32Array.copyWithinFromRange", "kind": "value", - "name": "toLocaleString", + "name": "copyWithinFromRange", "docstrings": [], - "signature": "let toLocaleString: t => string" + "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Float32Array.every", + "id": "Js.Typed_array.Uint32Array.slice", "kind": "value", - "name": "every", + "name": "slice", "docstrings": [], - "signature": "let every: (elt => bool, t) => bool" + "signature": "let slice: (~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Float32Array.everyi", + "id": "Js.Typed_array.Uint32Array.subarray", "kind": "value", - "name": "everyi", + "name": "subarray", "docstrings": [], - "signature": "let everyi: ((elt, int) => bool, t) => bool" + "signature": "let subarray: (~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Float32Array.filter", + "id": "Js.Typed_array.Uint32Array._BYTES_PER_ELEMENT", "kind": "value", - "name": "filter", + "name": "_BYTES_PER_ELEMENT", "docstrings": [], - "signature": "let filter: (elt => bool, t) => t" + "signature": "let _BYTES_PER_ELEMENT: int" }, { - "id": "Js.Typed_array.Float32Array.filteri", + "id": "Js.Typed_array.Uint32Array.make", "kind": "value", - "name": "filteri", + "name": "make", "docstrings": [], - "signature": "let filteri: ((elt, int) => bool, t) => t" + "signature": "let make: array => t" }, { - "id": "Js.Typed_array.Float32Array.find", + "id": "Js.Typed_array.Uint32Array.fromBuffer", "kind": "value", - "name": "find", - "docstrings": [], - "signature": "let find: (elt => bool, t) => Js.undefined" + "name": "fromBuffer", + "docstrings": [ + "can throw" + ], + "signature": "let fromBuffer: array_buffer => t" }, { - "id": "Js.Typed_array.Float32Array.findi", + "id": "Js.Typed_array.Uint32Array.fromBufferOffset", "kind": "value", - "name": "findi", - "docstrings": [], - "signature": "let findi: ((elt, int) => bool, t) => Js.undefined" + "name": "fromBufferOffset", + "docstrings": [ + "**raise** Js.Exn.Error raise Js exception\n\n **param** offset is in bytes" + ], + "signature": "let fromBufferOffset: (array_buffer, int) => t" }, { - "id": "Js.Typed_array.Float32Array.findIndex", + "id": "Js.Typed_array.Uint32Array.fromBufferRange", "kind": "value", - "name": "findIndex", - "docstrings": [], - "signature": "let findIndex: (elt => bool, t) => int" + "name": "fromBufferRange", + "docstrings": [ + "**raise** Js.Exn.Error raises Js exception\n\n **param** offset is in bytes, length in elements" + ], + "signature": "let fromBufferRange: (array_buffer, ~offset: int, ~length: int) => t" }, { - "id": "Js.Typed_array.Float32Array.findIndexi", + "id": "Js.Typed_array.Uint32Array.fromLength", "kind": "value", - "name": "findIndexi", + "name": "fromLength", "docstrings": [], - "signature": "let findIndexi: ((elt, int) => bool, t) => int" + "signature": "let fromLength: int => t" }, { - "id": "Js.Typed_array.Float32Array.forEach", + "id": "Js.Typed_array.Uint32Array.from", "kind": "value", - "name": "forEach", + "name": "from", + "docstrings": [], + "signature": "let from: array_like => t" + } + ] + }, + "js/typed_array/int32_array": { + "id": "Js.Typed_array.Int32_array", + "name": "Int32_array", + "docstrings": [], + "items": [] + }, + "js/typed_array/int32array": { + "id": "Js.Typed_array.Int32Array", + "name": "Int32Array", + "docstrings": [], + "items": [ + { + "id": "Js.Typed_array.Int32Array.elt", + "kind": "type", + "name": "elt", + "docstrings": [ + "" + ], + "signature": "type elt = int" + }, + { + "id": "Js.Typed_array.Int32Array.typed_array", + "kind": "type", + "name": "typed_array", "docstrings": [], - "signature": "let forEach: (elt => unit, t) => unit" + "signature": "type typed_array<\n 'a,\n> = Js_typed_array2.Int32Array.typed_array<'a>" + }, + { + "id": "Js.Typed_array.Int32Array.t", + "kind": "type", + "name": "t", + "docstrings": [], + "signature": "type t = typed_array" }, { - "id": "Js.Typed_array.Float32Array.forEachi", + "id": "Js.Typed_array.Int32Array.unsafe_get", "kind": "value", - "name": "forEachi", + "name": "unsafe_get", "docstrings": [], - "signature": "let forEachi: ((elt, int) => unit, t) => unit" + "signature": "let unsafe_get: (t, int) => elt" }, { - "id": "Js.Typed_array.Float32Array.map", + "id": "Js.Typed_array.Int32Array.unsafe_set", "kind": "value", - "name": "map", + "name": "unsafe_set", "docstrings": [], - "signature": "let map: (elt => 'a, t) => typed_array<'a>" + "signature": "let unsafe_set: (t, int, elt) => unit" }, { - "id": "Js.Typed_array.Float32Array.mapi", + "id": "Js.Typed_array.Int32Array.buffer", "kind": "value", - "name": "mapi", + "name": "buffer", "docstrings": [], - "signature": "let mapi: ((elt, int) => 'a, t) => typed_array<'a>" + "signature": "let buffer: t => array_buffer" }, { - "id": "Js.Typed_array.Float32Array.reduce", + "id": "Js.Typed_array.Int32Array.byteLength", "kind": "value", - "name": "reduce", + "name": "byteLength", "docstrings": [], - "signature": "let reduce: (('a, elt) => 'a, 'a, t) => 'a" + "signature": "let byteLength: t => int" }, { - "id": "Js.Typed_array.Float32Array.reducei", + "id": "Js.Typed_array.Int32Array.byteOffset", "kind": "value", - "name": "reducei", + "name": "byteOffset", "docstrings": [], - "signature": "let reducei: (('a, elt, int) => 'a, 'a, t) => 'a" + "signature": "let byteOffset: t => int" }, { - "id": "Js.Typed_array.Float32Array.reduceRight", + "id": "Js.Typed_array.Int32Array.length", "kind": "value", - "name": "reduceRight", + "name": "length", "docstrings": [], - "signature": "let reduceRight: (('a, elt) => 'a, 'a, t) => 'a" + "signature": "let length: t => int" }, { - "id": "Js.Typed_array.Float32Array.reduceRighti", + "id": "Js.Typed_array.Int32Array.copyWithinFromRange", "kind": "value", - "name": "reduceRighti", + "name": "copyWithinFromRange", "docstrings": [], - "signature": "let reduceRighti: (('a, elt, int) => 'a, 'a, t) => 'a" + "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Float32Array.some", + "id": "Js.Typed_array.Int32Array.slice", "kind": "value", - "name": "some", + "name": "slice", "docstrings": [], - "signature": "let some: (elt => bool, t) => bool" + "signature": "let slice: (~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Float32Array.somei", + "id": "Js.Typed_array.Int32Array.subarray", "kind": "value", - "name": "somei", + "name": "subarray", "docstrings": [], - "signature": "let somei: ((elt, int) => bool, t) => bool" + "signature": "let subarray: (~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Float32Array._BYTES_PER_ELEMENT", + "id": "Js.Typed_array.Int32Array._BYTES_PER_ELEMENT", "kind": "value", "name": "_BYTES_PER_ELEMENT", "docstrings": [], "signature": "let _BYTES_PER_ELEMENT: int" }, { - "id": "Js.Typed_array.Float32Array.make", + "id": "Js.Typed_array.Int32Array.make", "kind": "value", "name": "make", "docstrings": [], "signature": "let make: array => t" }, { - "id": "Js.Typed_array.Float32Array.fromBuffer", + "id": "Js.Typed_array.Int32Array.fromBuffer", "kind": "value", "name": "fromBuffer", "docstrings": [ @@ -5532,7 +5299,7 @@ "signature": "let fromBuffer: array_buffer => t" }, { - "id": "Js.Typed_array.Float32Array.fromBufferOffset", + "id": "Js.Typed_array.Int32Array.fromBufferOffset", "kind": "value", "name": "fromBufferOffset", "docstrings": [ @@ -5541,7 +5308,7 @@ "signature": "let fromBufferOffset: (array_buffer, int) => t" }, { - "id": "Js.Typed_array.Float32Array.fromBufferRange", + "id": "Js.Typed_array.Int32Array.fromBufferRange", "kind": "value", "name": "fromBufferRange", "docstrings": [ @@ -5550,29 +5317,29 @@ "signature": "let fromBufferRange: (array_buffer, ~offset: int, ~length: int) => t" }, { - "id": "Js.Typed_array.Float32Array.fromLength", + "id": "Js.Typed_array.Int32Array.fromLength", "kind": "value", "name": "fromLength", "docstrings": [], "signature": "let fromLength: int => t" }, { - "id": "Js.Typed_array.Float32Array.from", + "id": "Js.Typed_array.Int32Array.from", "kind": "value", "name": "from", "docstrings": [], "signature": "let from: array_like => t" }, { - "id": "Js.Typed_array.Float32Array.create", + "id": "Js.Typed_array.Int32Array.create", "kind": "value", "name": "create", "docstrings": [], - "signature": "let create: array => t", + "signature": "let create: array => t", "deprecated": "use `make` instead" }, { - "id": "Js.Typed_array.Float32Array.of_buffer", + "id": "Js.Typed_array.Int32Array.of_buffer", "kind": "value", "name": "of_buffer", "docstrings": [], @@ -5581,13 +5348,13 @@ } ] }, - "js/typed_array/uint32array": { - "id": "Js.Typed_array.Uint32Array", - "name": "Uint32Array", + "js/typed_array/uint16array": { + "id": "Js.Typed_array.Uint16Array", + "name": "Uint16Array", "docstrings": [], "items": [ { - "id": "Js.Typed_array.Uint32Array.elt", + "id": "Js.Typed_array.Uint16Array.elt", "kind": "type", "name": "elt", "docstrings": [ @@ -5596,378 +5363,394 @@ "signature": "type elt = int" }, { - "id": "Js.Typed_array.Uint32Array.typed_array", + "id": "Js.Typed_array.Uint16Array.typed_array", "kind": "type", "name": "typed_array", "docstrings": [], - "signature": "type typed_array<\n 'a,\n> = Js_typed_array2.Uint32Array.typed_array<'a>" + "signature": "type typed_array<\n 'a,\n> = Js_typed_array2.Uint16Array.typed_array<'a>" }, { - "id": "Js.Typed_array.Uint32Array.t", + "id": "Js.Typed_array.Uint16Array.t", "kind": "type", "name": "t", "docstrings": [], "signature": "type t = typed_array" }, { - "id": "Js.Typed_array.Uint32Array.unsafe_get", + "id": "Js.Typed_array.Uint16Array.unsafe_get", "kind": "value", "name": "unsafe_get", "docstrings": [], "signature": "let unsafe_get: (t, int) => elt" }, { - "id": "Js.Typed_array.Uint32Array.unsafe_set", + "id": "Js.Typed_array.Uint16Array.unsafe_set", "kind": "value", "name": "unsafe_set", "docstrings": [], "signature": "let unsafe_set: (t, int, elt) => unit" }, { - "id": "Js.Typed_array.Uint32Array.buffer", + "id": "Js.Typed_array.Uint16Array.buffer", "kind": "value", "name": "buffer", "docstrings": [], "signature": "let buffer: t => array_buffer" }, { - "id": "Js.Typed_array.Uint32Array.byteLength", + "id": "Js.Typed_array.Uint16Array.byteLength", "kind": "value", "name": "byteLength", "docstrings": [], "signature": "let byteLength: t => int" }, { - "id": "Js.Typed_array.Uint32Array.byteOffset", + "id": "Js.Typed_array.Uint16Array.byteOffset", "kind": "value", "name": "byteOffset", "docstrings": [], "signature": "let byteOffset: t => int" }, { - "id": "Js.Typed_array.Uint32Array.setArray", + "id": "Js.Typed_array.Uint16Array.length", "kind": "value", - "name": "setArray", - "docstrings": [], - "signature": "let setArray: (array, t) => unit" - }, - { - "id": "Js.Typed_array.Uint32Array.setArrayOffset", - "kind": "value", - "name": "setArrayOffset", - "docstrings": [], - "signature": "let setArrayOffset: (array, int, t) => unit" - }, - { - "id": "Js.Typed_array.Uint32Array.length", - "kind": "value", - "name": "length", + "name": "length", "docstrings": [], "signature": "let length: t => int" }, { - "id": "Js.Typed_array.Uint32Array.copyWithin", + "id": "Js.Typed_array.Uint16Array.copyWithinFromRange", "kind": "value", - "name": "copyWithin", + "name": "copyWithinFromRange", "docstrings": [], - "signature": "let copyWithin: (~to_: int, t) => t" + "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Uint32Array.copyWithinFrom", + "id": "Js.Typed_array.Uint16Array.slice", "kind": "value", - "name": "copyWithinFrom", + "name": "slice", "docstrings": [], - "signature": "let copyWithinFrom: (~to_: int, ~from: int, t) => t" + "signature": "let slice: (~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Uint32Array.copyWithinFromRange", + "id": "Js.Typed_array.Uint16Array.subarray", "kind": "value", - "name": "copyWithinFromRange", + "name": "subarray", "docstrings": [], - "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int, t) => t" + "signature": "let subarray: (~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Uint32Array.fillInPlace", + "id": "Js.Typed_array.Uint16Array._BYTES_PER_ELEMENT", "kind": "value", - "name": "fillInPlace", + "name": "_BYTES_PER_ELEMENT", "docstrings": [], - "signature": "let fillInPlace: (elt, t) => t" + "signature": "let _BYTES_PER_ELEMENT: int" }, { - "id": "Js.Typed_array.Uint32Array.fillFromInPlace", + "id": "Js.Typed_array.Uint16Array.make", "kind": "value", - "name": "fillFromInPlace", + "name": "make", "docstrings": [], - "signature": "let fillFromInPlace: (elt, ~from: int, t) => t" + "signature": "let make: array => t" }, { - "id": "Js.Typed_array.Uint32Array.fillRangeInPlace", + "id": "Js.Typed_array.Uint16Array.fromBuffer", "kind": "value", - "name": "fillRangeInPlace", - "docstrings": [], - "signature": "let fillRangeInPlace: (elt, ~start: int, ~end_: int, t) => t" + "name": "fromBuffer", + "docstrings": [ + "can throw" + ], + "signature": "let fromBuffer: array_buffer => t" }, { - "id": "Js.Typed_array.Uint32Array.reverseInPlace", + "id": "Js.Typed_array.Uint16Array.fromBufferOffset", "kind": "value", - "name": "reverseInPlace", - "docstrings": [], - "signature": "let reverseInPlace: t => t" + "name": "fromBufferOffset", + "docstrings": [ + "**raise** Js.Exn.Error raise Js exception\n\n **param** offset is in bytes" + ], + "signature": "let fromBufferOffset: (array_buffer, int) => t" }, { - "id": "Js.Typed_array.Uint32Array.sortInPlace", + "id": "Js.Typed_array.Uint16Array.fromBufferRange", "kind": "value", - "name": "sortInPlace", - "docstrings": [], - "signature": "let sortInPlace: t => t" + "name": "fromBufferRange", + "docstrings": [ + "**raise** Js.Exn.Error raises Js exception\n\n **param** offset is in bytes, length in elements" + ], + "signature": "let fromBufferRange: (array_buffer, ~offset: int, ~length: int) => t" }, { - "id": "Js.Typed_array.Uint32Array.sortInPlaceWith", + "id": "Js.Typed_array.Uint16Array.fromLength", "kind": "value", - "name": "sortInPlaceWith", + "name": "fromLength", "docstrings": [], - "signature": "let sortInPlaceWith: ((elt, elt) => int, t) => t" + "signature": "let fromLength: int => t" }, { - "id": "Js.Typed_array.Uint32Array.includes", + "id": "Js.Typed_array.Uint16Array.from", "kind": "value", - "name": "includes", + "name": "from", "docstrings": [], - "signature": "let includes: (elt, t) => bool" - }, + "signature": "let from: array_like => t" + } + ] + }, + "js/typed_array/int16array": { + "id": "Js.Typed_array.Int16Array", + "name": "Int16Array", + "docstrings": [], + "items": [ { - "id": "Js.Typed_array.Uint32Array.indexOf", - "kind": "value", - "name": "indexOf", - "docstrings": [], - "signature": "let indexOf: (elt, t) => int" + "id": "Js.Typed_array.Int16Array.elt", + "kind": "type", + "name": "elt", + "docstrings": [ + "" + ], + "signature": "type elt = int" }, { - "id": "Js.Typed_array.Uint32Array.indexOfFrom", - "kind": "value", - "name": "indexOfFrom", + "id": "Js.Typed_array.Int16Array.typed_array", + "kind": "type", + "name": "typed_array", "docstrings": [], - "signature": "let indexOfFrom: (elt, ~from: int, t) => int" + "signature": "type typed_array<\n 'a,\n> = Js_typed_array2.Int16Array.typed_array<'a>" }, { - "id": "Js.Typed_array.Uint32Array.join", - "kind": "value", - "name": "join", + "id": "Js.Typed_array.Int16Array.t", + "kind": "type", + "name": "t", "docstrings": [], - "signature": "let join: t => string" + "signature": "type t = typed_array" }, { - "id": "Js.Typed_array.Uint32Array.joinWith", + "id": "Js.Typed_array.Int16Array.unsafe_get", "kind": "value", - "name": "joinWith", + "name": "unsafe_get", "docstrings": [], - "signature": "let joinWith: (string, t) => string" + "signature": "let unsafe_get: (t, int) => elt" }, { - "id": "Js.Typed_array.Uint32Array.lastIndexOf", + "id": "Js.Typed_array.Int16Array.unsafe_set", "kind": "value", - "name": "lastIndexOf", + "name": "unsafe_set", "docstrings": [], - "signature": "let lastIndexOf: (elt, t) => int" + "signature": "let unsafe_set: (t, int, elt) => unit" }, { - "id": "Js.Typed_array.Uint32Array.lastIndexOfFrom", + "id": "Js.Typed_array.Int16Array.buffer", "kind": "value", - "name": "lastIndexOfFrom", + "name": "buffer", "docstrings": [], - "signature": "let lastIndexOfFrom: (elt, ~from: int, t) => int" + "signature": "let buffer: t => array_buffer" }, { - "id": "Js.Typed_array.Uint32Array.slice", + "id": "Js.Typed_array.Int16Array.byteLength", "kind": "value", - "name": "slice", + "name": "byteLength", "docstrings": [], - "signature": "let slice: (~start: int, ~end_: int, t) => t" + "signature": "let byteLength: t => int" }, { - "id": "Js.Typed_array.Uint32Array.copy", + "id": "Js.Typed_array.Int16Array.byteOffset", "kind": "value", - "name": "copy", + "name": "byteOffset", "docstrings": [], - "signature": "let copy: t => t" + "signature": "let byteOffset: t => int" }, { - "id": "Js.Typed_array.Uint32Array.sliceFrom", + "id": "Js.Typed_array.Int16Array.length", "kind": "value", - "name": "sliceFrom", + "name": "length", "docstrings": [], - "signature": "let sliceFrom: (int, t) => t" + "signature": "let length: t => int" }, { - "id": "Js.Typed_array.Uint32Array.subarray", + "id": "Js.Typed_array.Int16Array.copyWithinFromRange", "kind": "value", - "name": "subarray", + "name": "copyWithinFromRange", "docstrings": [], - "signature": "let subarray: (~start: int, ~end_: int, t) => t" + "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Uint32Array.subarrayFrom", + "id": "Js.Typed_array.Int16Array.slice", "kind": "value", - "name": "subarrayFrom", + "name": "slice", "docstrings": [], - "signature": "let subarrayFrom: (int, t) => t" + "signature": "let slice: (~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Uint32Array.toString", + "id": "Js.Typed_array.Int16Array.subarray", "kind": "value", - "name": "toString", + "name": "subarray", "docstrings": [], - "signature": "let toString: t => string" + "signature": "let subarray: (~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Uint32Array.toLocaleString", + "id": "Js.Typed_array.Int16Array._BYTES_PER_ELEMENT", "kind": "value", - "name": "toLocaleString", + "name": "_BYTES_PER_ELEMENT", "docstrings": [], - "signature": "let toLocaleString: t => string" + "signature": "let _BYTES_PER_ELEMENT: int" }, { - "id": "Js.Typed_array.Uint32Array.every", + "id": "Js.Typed_array.Int16Array.make", "kind": "value", - "name": "every", + "name": "make", "docstrings": [], - "signature": "let every: (elt => bool, t) => bool" + "signature": "let make: array => t" }, { - "id": "Js.Typed_array.Uint32Array.everyi", + "id": "Js.Typed_array.Int16Array.fromBuffer", "kind": "value", - "name": "everyi", - "docstrings": [], - "signature": "let everyi: ((elt, int) => bool, t) => bool" + "name": "fromBuffer", + "docstrings": [ + "can throw" + ], + "signature": "let fromBuffer: array_buffer => t" }, { - "id": "Js.Typed_array.Uint32Array.filter", + "id": "Js.Typed_array.Int16Array.fromBufferOffset", "kind": "value", - "name": "filter", - "docstrings": [], - "signature": "let filter: (elt => bool, t) => t" + "name": "fromBufferOffset", + "docstrings": [ + "**raise** Js.Exn.Error raise Js exception\n\n **param** offset is in bytes" + ], + "signature": "let fromBufferOffset: (array_buffer, int) => t" }, { - "id": "Js.Typed_array.Uint32Array.filteri", + "id": "Js.Typed_array.Int16Array.fromBufferRange", "kind": "value", - "name": "filteri", - "docstrings": [], - "signature": "let filteri: ((elt, int) => bool, t) => t" + "name": "fromBufferRange", + "docstrings": [ + "**raise** Js.Exn.Error raises Js exception\n\n **param** offset is in bytes, length in elements" + ], + "signature": "let fromBufferRange: (array_buffer, ~offset: int, ~length: int) => t" }, { - "id": "Js.Typed_array.Uint32Array.find", + "id": "Js.Typed_array.Int16Array.fromLength", "kind": "value", - "name": "find", + "name": "fromLength", "docstrings": [], - "signature": "let find: (elt => bool, t) => Js.undefined" + "signature": "let fromLength: int => t" }, { - "id": "Js.Typed_array.Uint32Array.findi", + "id": "Js.Typed_array.Int16Array.from", "kind": "value", - "name": "findi", + "name": "from", "docstrings": [], - "signature": "let findi: ((elt, int) => bool, t) => Js.undefined" - }, + "signature": "let from: array_like => t" + } + ] + }, + "js/typed_array/uint8clampedarray": { + "id": "Js.Typed_array.Uint8ClampedArray", + "name": "Uint8ClampedArray", + "docstrings": [], + "items": [ { - "id": "Js.Typed_array.Uint32Array.findIndex", - "kind": "value", - "name": "findIndex", - "docstrings": [], - "signature": "let findIndex: (elt => bool, t) => int" + "id": "Js.Typed_array.Uint8ClampedArray.elt", + "kind": "type", + "name": "elt", + "docstrings": [ + "" + ], + "signature": "type elt = int" }, { - "id": "Js.Typed_array.Uint32Array.findIndexi", - "kind": "value", - "name": "findIndexi", + "id": "Js.Typed_array.Uint8ClampedArray.typed_array", + "kind": "type", + "name": "typed_array", "docstrings": [], - "signature": "let findIndexi: ((elt, int) => bool, t) => int" + "signature": "type typed_array<\n 'a,\n> = Js_typed_array2.Uint8ClampedArray.typed_array<'a>" }, { - "id": "Js.Typed_array.Uint32Array.forEach", - "kind": "value", - "name": "forEach", + "id": "Js.Typed_array.Uint8ClampedArray.t", + "kind": "type", + "name": "t", "docstrings": [], - "signature": "let forEach: (elt => unit, t) => unit" + "signature": "type t = typed_array" }, { - "id": "Js.Typed_array.Uint32Array.forEachi", + "id": "Js.Typed_array.Uint8ClampedArray.unsafe_get", "kind": "value", - "name": "forEachi", + "name": "unsafe_get", "docstrings": [], - "signature": "let forEachi: ((elt, int) => unit, t) => unit" + "signature": "let unsafe_get: (t, int) => elt" }, { - "id": "Js.Typed_array.Uint32Array.map", + "id": "Js.Typed_array.Uint8ClampedArray.unsafe_set", "kind": "value", - "name": "map", + "name": "unsafe_set", "docstrings": [], - "signature": "let map: (elt => 'a, t) => typed_array<'a>" + "signature": "let unsafe_set: (t, int, elt) => unit" }, { - "id": "Js.Typed_array.Uint32Array.mapi", + "id": "Js.Typed_array.Uint8ClampedArray.buffer", "kind": "value", - "name": "mapi", + "name": "buffer", "docstrings": [], - "signature": "let mapi: ((elt, int) => 'a, t) => typed_array<'a>" + "signature": "let buffer: t => array_buffer" }, { - "id": "Js.Typed_array.Uint32Array.reduce", + "id": "Js.Typed_array.Uint8ClampedArray.byteLength", "kind": "value", - "name": "reduce", + "name": "byteLength", "docstrings": [], - "signature": "let reduce: (('a, elt) => 'a, 'a, t) => 'a" + "signature": "let byteLength: t => int" }, { - "id": "Js.Typed_array.Uint32Array.reducei", + "id": "Js.Typed_array.Uint8ClampedArray.byteOffset", "kind": "value", - "name": "reducei", + "name": "byteOffset", "docstrings": [], - "signature": "let reducei: (('a, elt, int) => 'a, 'a, t) => 'a" + "signature": "let byteOffset: t => int" }, { - "id": "Js.Typed_array.Uint32Array.reduceRight", + "id": "Js.Typed_array.Uint8ClampedArray.length", "kind": "value", - "name": "reduceRight", + "name": "length", "docstrings": [], - "signature": "let reduceRight: (('a, elt) => 'a, 'a, t) => 'a" + "signature": "let length: t => int" }, { - "id": "Js.Typed_array.Uint32Array.reduceRighti", + "id": "Js.Typed_array.Uint8ClampedArray.copyWithinFromRange", "kind": "value", - "name": "reduceRighti", + "name": "copyWithinFromRange", "docstrings": [], - "signature": "let reduceRighti: (('a, elt, int) => 'a, 'a, t) => 'a" + "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Uint32Array.some", + "id": "Js.Typed_array.Uint8ClampedArray.slice", "kind": "value", - "name": "some", + "name": "slice", "docstrings": [], - "signature": "let some: (elt => bool, t) => bool" + "signature": "let slice: (~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Uint32Array.somei", + "id": "Js.Typed_array.Uint8ClampedArray.subarray", "kind": "value", - "name": "somei", + "name": "subarray", "docstrings": [], - "signature": "let somei: ((elt, int) => bool, t) => bool" + "signature": "let subarray: (~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Uint32Array._BYTES_PER_ELEMENT", + "id": "Js.Typed_array.Uint8ClampedArray._BYTES_PER_ELEMENT", "kind": "value", "name": "_BYTES_PER_ELEMENT", "docstrings": [], "signature": "let _BYTES_PER_ELEMENT: int" }, { - "id": "Js.Typed_array.Uint32Array.make", + "id": "Js.Typed_array.Uint8ClampedArray.make", "kind": "value", "name": "make", "docstrings": [], "signature": "let make: array => t" }, { - "id": "Js.Typed_array.Uint32Array.fromBuffer", + "id": "Js.Typed_array.Uint8ClampedArray.fromBuffer", "kind": "value", "name": "fromBuffer", "docstrings": [ @@ -5976,7 +5759,7 @@ "signature": "let fromBuffer: array_buffer => t" }, { - "id": "Js.Typed_array.Uint32Array.fromBufferOffset", + "id": "Js.Typed_array.Uint8ClampedArray.fromBufferOffset", "kind": "value", "name": "fromBufferOffset", "docstrings": [ @@ -5985,7 +5768,7 @@ "signature": "let fromBufferOffset: (array_buffer, int) => t" }, { - "id": "Js.Typed_array.Uint32Array.fromBufferRange", + "id": "Js.Typed_array.Uint8ClampedArray.fromBufferRange", "kind": "value", "name": "fromBufferRange", "docstrings": [ @@ -5994,14 +5777,14 @@ "signature": "let fromBufferRange: (array_buffer, ~offset: int, ~length: int) => t" }, { - "id": "Js.Typed_array.Uint32Array.fromLength", + "id": "Js.Typed_array.Uint8ClampedArray.fromLength", "kind": "value", "name": "fromLength", "docstrings": [], "signature": "let fromLength: int => t" }, { - "id": "Js.Typed_array.Uint32Array.from", + "id": "Js.Typed_array.Uint8ClampedArray.from", "kind": "value", "name": "from", "docstrings": [], @@ -6009,19 +5792,13 @@ } ] }, - "js/typed_array/int32_array": { - "id": "Js.Typed_array.Int32_array", - "name": "Int32_array", - "docstrings": [], - "items": [] - }, - "js/typed_array/int32array": { - "id": "Js.Typed_array.Int32Array", - "name": "Int32Array", + "js/typed_array/uint8array": { + "id": "Js.Typed_array.Uint8Array", + "name": "Uint8Array", "docstrings": [], "items": [ { - "id": "Js.Typed_array.Int32Array.elt", + "id": "Js.Typed_array.Uint8Array.elt", "kind": "type", "name": "elt", "docstrings": [ @@ -6030,3601 +5807,684 @@ "signature": "type elt = int" }, { - "id": "Js.Typed_array.Int32Array.typed_array", + "id": "Js.Typed_array.Uint8Array.typed_array", "kind": "type", "name": "typed_array", "docstrings": [], - "signature": "type typed_array<\n 'a,\n> = Js_typed_array2.Int32Array.typed_array<'a>" + "signature": "type typed_array<\n 'a,\n> = Js_typed_array2.Uint8Array.typed_array<'a>" }, { - "id": "Js.Typed_array.Int32Array.t", + "id": "Js.Typed_array.Uint8Array.t", "kind": "type", "name": "t", "docstrings": [], "signature": "type t = typed_array" }, { - "id": "Js.Typed_array.Int32Array.unsafe_get", + "id": "Js.Typed_array.Uint8Array.unsafe_get", "kind": "value", "name": "unsafe_get", "docstrings": [], "signature": "let unsafe_get: (t, int) => elt" }, { - "id": "Js.Typed_array.Int32Array.unsafe_set", + "id": "Js.Typed_array.Uint8Array.unsafe_set", "kind": "value", "name": "unsafe_set", "docstrings": [], "signature": "let unsafe_set: (t, int, elt) => unit" }, { - "id": "Js.Typed_array.Int32Array.buffer", + "id": "Js.Typed_array.Uint8Array.buffer", "kind": "value", "name": "buffer", "docstrings": [], "signature": "let buffer: t => array_buffer" }, { - "id": "Js.Typed_array.Int32Array.byteLength", + "id": "Js.Typed_array.Uint8Array.byteLength", "kind": "value", "name": "byteLength", "docstrings": [], "signature": "let byteLength: t => int" }, { - "id": "Js.Typed_array.Int32Array.byteOffset", + "id": "Js.Typed_array.Uint8Array.byteOffset", "kind": "value", "name": "byteOffset", "docstrings": [], "signature": "let byteOffset: t => int" }, { - "id": "Js.Typed_array.Int32Array.setArray", - "kind": "value", - "name": "setArray", - "docstrings": [], - "signature": "let setArray: (array, t) => unit" - }, - { - "id": "Js.Typed_array.Int32Array.setArrayOffset", - "kind": "value", - "name": "setArrayOffset", - "docstrings": [], - "signature": "let setArrayOffset: (array, int, t) => unit" - }, - { - "id": "Js.Typed_array.Int32Array.length", + "id": "Js.Typed_array.Uint8Array.length", "kind": "value", "name": "length", "docstrings": [], "signature": "let length: t => int" }, { - "id": "Js.Typed_array.Int32Array.copyWithin", + "id": "Js.Typed_array.Uint8Array.copyWithinFromRange", "kind": "value", - "name": "copyWithin", + "name": "copyWithinFromRange", "docstrings": [], - "signature": "let copyWithin: (~to_: int, t) => t" + "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Int32Array.copyWithinFrom", + "id": "Js.Typed_array.Uint8Array.slice", "kind": "value", - "name": "copyWithinFrom", + "name": "slice", "docstrings": [], - "signature": "let copyWithinFrom: (~to_: int, ~from: int, t) => t" + "signature": "let slice: (~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Int32Array.copyWithinFromRange", + "id": "Js.Typed_array.Uint8Array.subarray", "kind": "value", - "name": "copyWithinFromRange", + "name": "subarray", "docstrings": [], - "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int, t) => t" + "signature": "let subarray: (~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Int32Array.fillInPlace", + "id": "Js.Typed_array.Uint8Array._BYTES_PER_ELEMENT", "kind": "value", - "name": "fillInPlace", + "name": "_BYTES_PER_ELEMENT", "docstrings": [], - "signature": "let fillInPlace: (elt, t) => t" + "signature": "let _BYTES_PER_ELEMENT: int" }, { - "id": "Js.Typed_array.Int32Array.fillFromInPlace", + "id": "Js.Typed_array.Uint8Array.make", "kind": "value", - "name": "fillFromInPlace", + "name": "make", "docstrings": [], - "signature": "let fillFromInPlace: (elt, ~from: int, t) => t" + "signature": "let make: array => t" }, { - "id": "Js.Typed_array.Int32Array.fillRangeInPlace", + "id": "Js.Typed_array.Uint8Array.fromBuffer", "kind": "value", - "name": "fillRangeInPlace", - "docstrings": [], - "signature": "let fillRangeInPlace: (elt, ~start: int, ~end_: int, t) => t" + "name": "fromBuffer", + "docstrings": [ + "can throw" + ], + "signature": "let fromBuffer: array_buffer => t" }, { - "id": "Js.Typed_array.Int32Array.reverseInPlace", + "id": "Js.Typed_array.Uint8Array.fromBufferOffset", "kind": "value", - "name": "reverseInPlace", - "docstrings": [], - "signature": "let reverseInPlace: t => t" + "name": "fromBufferOffset", + "docstrings": [ + "**raise** Js.Exn.Error raise Js exception\n\n **param** offset is in bytes" + ], + "signature": "let fromBufferOffset: (array_buffer, int) => t" }, { - "id": "Js.Typed_array.Int32Array.sortInPlace", + "id": "Js.Typed_array.Uint8Array.fromBufferRange", "kind": "value", - "name": "sortInPlace", - "docstrings": [], - "signature": "let sortInPlace: t => t" + "name": "fromBufferRange", + "docstrings": [ + "**raise** Js.Exn.Error raises Js exception\n\n **param** offset is in bytes, length in elements" + ], + "signature": "let fromBufferRange: (array_buffer, ~offset: int, ~length: int) => t" }, { - "id": "Js.Typed_array.Int32Array.sortInPlaceWith", + "id": "Js.Typed_array.Uint8Array.fromLength", "kind": "value", - "name": "sortInPlaceWith", + "name": "fromLength", "docstrings": [], - "signature": "let sortInPlaceWith: ((elt, elt) => int, t) => t" + "signature": "let fromLength: int => t" }, { - "id": "Js.Typed_array.Int32Array.includes", + "id": "Js.Typed_array.Uint8Array.from", "kind": "value", - "name": "includes", + "name": "from", "docstrings": [], - "signature": "let includes: (elt, t) => bool" - }, + "signature": "let from: array_like => t" + } + ] + }, + "js/typed_array/int8array": { + "id": "Js.Typed_array.Int8Array", + "name": "Int8Array", + "docstrings": [], + "items": [ { - "id": "Js.Typed_array.Int32Array.indexOf", - "kind": "value", - "name": "indexOf", - "docstrings": [], - "signature": "let indexOf: (elt, t) => int" + "id": "Js.Typed_array.Int8Array.elt", + "kind": "type", + "name": "elt", + "docstrings": [ + "" + ], + "signature": "type elt = int" }, { - "id": "Js.Typed_array.Int32Array.indexOfFrom", - "kind": "value", - "name": "indexOfFrom", + "id": "Js.Typed_array.Int8Array.typed_array", + "kind": "type", + "name": "typed_array", "docstrings": [], - "signature": "let indexOfFrom: (elt, ~from: int, t) => int" + "signature": "type typed_array<\n 'a,\n> = Js_typed_array2.Int8Array.typed_array<'a>" }, { - "id": "Js.Typed_array.Int32Array.join", - "kind": "value", - "name": "join", + "id": "Js.Typed_array.Int8Array.t", + "kind": "type", + "name": "t", "docstrings": [], - "signature": "let join: t => string" + "signature": "type t = typed_array" }, { - "id": "Js.Typed_array.Int32Array.joinWith", + "id": "Js.Typed_array.Int8Array.unsafe_get", "kind": "value", - "name": "joinWith", + "name": "unsafe_get", "docstrings": [], - "signature": "let joinWith: (string, t) => string" + "signature": "let unsafe_get: (t, int) => elt" }, { - "id": "Js.Typed_array.Int32Array.lastIndexOf", + "id": "Js.Typed_array.Int8Array.unsafe_set", "kind": "value", - "name": "lastIndexOf", + "name": "unsafe_set", "docstrings": [], - "signature": "let lastIndexOf: (elt, t) => int" + "signature": "let unsafe_set: (t, int, elt) => unit" }, { - "id": "Js.Typed_array.Int32Array.lastIndexOfFrom", + "id": "Js.Typed_array.Int8Array.buffer", "kind": "value", - "name": "lastIndexOfFrom", + "name": "buffer", "docstrings": [], - "signature": "let lastIndexOfFrom: (elt, ~from: int, t) => int" + "signature": "let buffer: t => array_buffer" }, { - "id": "Js.Typed_array.Int32Array.slice", + "id": "Js.Typed_array.Int8Array.byteLength", "kind": "value", - "name": "slice", + "name": "byteLength", "docstrings": [], - "signature": "let slice: (~start: int, ~end_: int, t) => t" + "signature": "let byteLength: t => int" }, { - "id": "Js.Typed_array.Int32Array.copy", + "id": "Js.Typed_array.Int8Array.byteOffset", "kind": "value", - "name": "copy", + "name": "byteOffset", "docstrings": [], - "signature": "let copy: t => t" + "signature": "let byteOffset: t => int" }, { - "id": "Js.Typed_array.Int32Array.sliceFrom", + "id": "Js.Typed_array.Int8Array.length", "kind": "value", - "name": "sliceFrom", + "name": "length", "docstrings": [], - "signature": "let sliceFrom: (int, t) => t" + "signature": "let length: t => int" }, { - "id": "Js.Typed_array.Int32Array.subarray", + "id": "Js.Typed_array.Int8Array.copyWithinFromRange", "kind": "value", - "name": "subarray", + "name": "copyWithinFromRange", "docstrings": [], - "signature": "let subarray: (~start: int, ~end_: int, t) => t" + "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Int32Array.subarrayFrom", + "id": "Js.Typed_array.Int8Array.slice", "kind": "value", - "name": "subarrayFrom", + "name": "slice", "docstrings": [], - "signature": "let subarrayFrom: (int, t) => t" + "signature": "let slice: (~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Int32Array.toString", + "id": "Js.Typed_array.Int8Array.subarray", "kind": "value", - "name": "toString", + "name": "subarray", "docstrings": [], - "signature": "let toString: t => string" + "signature": "let subarray: (~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Int32Array.toLocaleString", + "id": "Js.Typed_array.Int8Array._BYTES_PER_ELEMENT", "kind": "value", - "name": "toLocaleString", + "name": "_BYTES_PER_ELEMENT", "docstrings": [], - "signature": "let toLocaleString: t => string" + "signature": "let _BYTES_PER_ELEMENT: int" }, { - "id": "Js.Typed_array.Int32Array.every", + "id": "Js.Typed_array.Int8Array.make", "kind": "value", - "name": "every", + "name": "make", "docstrings": [], - "signature": "let every: (elt => bool, t) => bool" + "signature": "let make: array => t" }, { - "id": "Js.Typed_array.Int32Array.everyi", + "id": "Js.Typed_array.Int8Array.fromBuffer", "kind": "value", - "name": "everyi", - "docstrings": [], - "signature": "let everyi: ((elt, int) => bool, t) => bool" + "name": "fromBuffer", + "docstrings": [ + "can throw" + ], + "signature": "let fromBuffer: array_buffer => t" }, { - "id": "Js.Typed_array.Int32Array.filter", + "id": "Js.Typed_array.Int8Array.fromBufferOffset", "kind": "value", - "name": "filter", - "docstrings": [], - "signature": "let filter: (elt => bool, t) => t" + "name": "fromBufferOffset", + "docstrings": [ + "raise Js.Exn.Error raise Js exception\n\n param offset is in bytes" + ], + "signature": "let fromBufferOffset: (array_buffer, int) => t" }, { - "id": "Js.Typed_array.Int32Array.filteri", + "id": "Js.Typed_array.Int8Array.fromBufferRange", "kind": "value", - "name": "filteri", - "docstrings": [], - "signature": "let filteri: ((elt, int) => bool, t) => t" + "name": "fromBufferRange", + "docstrings": [ + "raise Js.Exn.Error raises Js exception\n\n param offset is in bytes, length in elements" + ], + "signature": "let fromBufferRange: (array_buffer, ~offset: int, ~length: int) => t" }, { - "id": "Js.Typed_array.Int32Array.find", + "id": "Js.Typed_array.Int8Array.fromLength", "kind": "value", - "name": "find", + "name": "fromLength", "docstrings": [], - "signature": "let find: (elt => bool, t) => Js.undefined" + "signature": "let fromLength: int => t" }, { - "id": "Js.Typed_array.Int32Array.findi", + "id": "Js.Typed_array.Int8Array.from", "kind": "value", - "name": "findi", + "name": "from", "docstrings": [], - "signature": "let findi: ((elt, int) => bool, t) => Js.undefined" - }, + "signature": "let from: array_like => t" + } + ] + }, + "js/typed_array/s": { + "id": "Js.Typed_array.S", + "name": "S", + "docstrings": [], + "items": [ { - "id": "Js.Typed_array.Int32Array.findIndex", - "kind": "value", - "name": "findIndex", + "id": "Js.Typed_array.S.elt", + "kind": "type", + "name": "elt", "docstrings": [], - "signature": "let findIndex: (elt => bool, t) => int" + "signature": "type elt" }, { - "id": "Js.Typed_array.Int32Array.findIndexi", - "kind": "value", - "name": "findIndexi", + "id": "Js.Typed_array.S.typed_array", + "kind": "type", + "name": "typed_array", "docstrings": [], - "signature": "let findIndexi: ((elt, int) => bool, t) => int" + "signature": "type typed_array<'a>" }, { - "id": "Js.Typed_array.Int32Array.forEach", - "kind": "value", - "name": "forEach", + "id": "Js.Typed_array.S.t", + "kind": "type", + "name": "t", "docstrings": [], - "signature": "let forEach: (elt => unit, t) => unit" + "signature": "type t = typed_array" }, { - "id": "Js.Typed_array.Int32Array.forEachi", + "id": "Js.Typed_array.S.unsafe_get", "kind": "value", - "name": "forEachi", + "name": "unsafe_get", "docstrings": [], - "signature": "let forEachi: ((elt, int) => unit, t) => unit" + "signature": "let unsafe_get: (t, int) => elt" }, { - "id": "Js.Typed_array.Int32Array.map", + "id": "Js.Typed_array.S.unsafe_set", "kind": "value", - "name": "map", + "name": "unsafe_set", "docstrings": [], - "signature": "let map: (elt => 'a, t) => typed_array<'a>" + "signature": "let unsafe_set: (t, int, elt) => unit" }, { - "id": "Js.Typed_array.Int32Array.mapi", + "id": "Js.Typed_array.S.buffer", "kind": "value", - "name": "mapi", + "name": "buffer", "docstrings": [], - "signature": "let mapi: ((elt, int) => 'a, t) => typed_array<'a>" + "signature": "let buffer: t => array_buffer" }, { - "id": "Js.Typed_array.Int32Array.reduce", + "id": "Js.Typed_array.S.byteLength", "kind": "value", - "name": "reduce", + "name": "byteLength", "docstrings": [], - "signature": "let reduce: (('a, elt) => 'a, 'a, t) => 'a" + "signature": "let byteLength: t => int" }, { - "id": "Js.Typed_array.Int32Array.reducei", + "id": "Js.Typed_array.S.byteOffset", "kind": "value", - "name": "reducei", + "name": "byteOffset", "docstrings": [], - "signature": "let reducei: (('a, elt, int) => 'a, 'a, t) => 'a" + "signature": "let byteOffset: t => int" }, { - "id": "Js.Typed_array.Int32Array.reduceRight", + "id": "Js.Typed_array.S.length", "kind": "value", - "name": "reduceRight", + "name": "length", "docstrings": [], - "signature": "let reduceRight: (('a, elt) => 'a, 'a, t) => 'a" + "signature": "let length: t => int" }, { - "id": "Js.Typed_array.Int32Array.reduceRighti", + "id": "Js.Typed_array.S.copyWithinFromRange", "kind": "value", - "name": "reduceRighti", + "name": "copyWithinFromRange", "docstrings": [], - "signature": "let reduceRighti: (('a, elt, int) => 'a, 'a, t) => 'a" + "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int) => t" }, { - "id": "Js.Typed_array.Int32Array.some", + "id": "Js.Typed_array.S.includes", "kind": "value", - "name": "some", + "name": "includes", "docstrings": [], - "signature": "let some: (elt => bool, t) => bool" + "signature": "let includes: elt => bool" }, { - "id": "Js.Typed_array.Int32Array.somei", - "kind": "value", - "name": "somei", - "docstrings": [], - "signature": "let somei: ((elt, int) => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Int32Array._BYTES_PER_ELEMENT", - "kind": "value", - "name": "_BYTES_PER_ELEMENT", - "docstrings": [], - "signature": "let _BYTES_PER_ELEMENT: int" - }, - { - "id": "Js.Typed_array.Int32Array.make", - "kind": "value", - "name": "make", - "docstrings": [], - "signature": "let make: array => t" - }, - { - "id": "Js.Typed_array.Int32Array.fromBuffer", + "id": "Js.Typed_array.S.filter", "kind": "value", - "name": "fromBuffer", + "name": "filter", "docstrings": [ - "can throw" + "should we use `bool` or `boolean` seems they are intechangeable here" ], - "signature": "let fromBuffer: array_buffer => t" - }, + "signature": "let filter: (elt => bool) => t" + } + ] + }, + "js/typed_array/arraybuffer": { + "id": "Js.Typed_array.ArrayBuffer", + "name": "ArrayBuffer", + "docstrings": [ + "The underlying buffer that the typed arrays provide views of\n\n **see** [MDN](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/ArrayBuffer)" + ], + "items": [ { - "id": "Js.Typed_array.Int32Array.fromBufferOffset", - "kind": "value", - "name": "fromBufferOffset", - "docstrings": [ - "**raise** Js.Exn.Error raise Js exception\n\n **param** offset is in bytes" - ], - "signature": "let fromBufferOffset: (array_buffer, int) => t" + "id": "Js.Typed_array.ArrayBuffer.t", + "kind": "type", + "name": "t", + "docstrings": [], + "signature": "type t = array_buffer" }, { - "id": "Js.Typed_array.Int32Array.fromBufferRange", + "id": "Js.Typed_array.ArrayBuffer.make", "kind": "value", - "name": "fromBufferRange", + "name": "make", "docstrings": [ - "**raise** Js.Exn.Error raises Js exception\n\n **param** offset is in bytes, length in elements" + "takes length. initializes elements to 0" ], - "signature": "let fromBufferRange: (array_buffer, ~offset: int, ~length: int) => t" - }, - { - "id": "Js.Typed_array.Int32Array.fromLength", - "kind": "value", - "name": "fromLength", - "docstrings": [], - "signature": "let fromLength: int => t" - }, - { - "id": "Js.Typed_array.Int32Array.from", - "kind": "value", - "name": "from", - "docstrings": [], - "signature": "let from: array_like => t" - }, - { - "id": "Js.Typed_array.Int32Array.create", - "kind": "value", - "name": "create", - "docstrings": [], - "signature": "let create: array => t", - "deprecated": "use `make` instead" + "signature": "let make: int => t" }, { - "id": "Js.Typed_array.Int32Array.of_buffer", + "id": "Js.Typed_array.ArrayBuffer.byteLength", "kind": "value", - "name": "of_buffer", + "name": "byteLength", "docstrings": [], - "signature": "let of_buffer: array_buffer => t", - "deprecated": "use `fromBuffer` instead" + "signature": "let byteLength: t => int" } ] }, - "js/typed_array/uint16array": { - "id": "Js.Typed_array.Uint16Array", - "name": "Uint16Array", + "js/typed_array/type": { + "id": "Js.Typed_array.Type", + "name": "Type", "docstrings": [], "items": [ { - "id": "Js.Typed_array.Uint16Array.elt", + "id": "Js.Typed_array.Type.t", "kind": "type", - "name": "elt", - "docstrings": [ - "" - ], - "signature": "type elt = int" - }, + "name": "t", + "docstrings": [], + "signature": "type t" + } + ] + }, + "js/json/kind": { + "id": "Js.Json.Kind", + "name": "Kind", + "docstrings": [], + "items": [ { - "id": "Js.Typed_array.Uint16Array.typed_array", + "id": "Js.Json.Kind.json", "kind": "type", - "name": "typed_array", + "name": "json", "docstrings": [], - "signature": "type typed_array<\n 'a,\n> = Js_typed_array2.Uint16Array.typed_array<'a>" + "signature": "type json = t" }, { - "id": "Js.Typed_array.Uint16Array.t", + "id": "Js.Json.Kind.t", "kind": "type", "name": "t", - "docstrings": [], - "signature": "type t = typed_array" - }, - { - "id": "Js.Typed_array.Uint16Array.unsafe_get", - "kind": "value", - "name": "unsafe_get", - "docstrings": [], - "signature": "let unsafe_get: (t, int) => elt" - }, - { - "id": "Js.Typed_array.Uint16Array.unsafe_set", - "kind": "value", - "name": "unsafe_set", - "docstrings": [], - "signature": "let unsafe_set: (t, int, elt) => unit" - }, - { - "id": "Js.Typed_array.Uint16Array.buffer", - "kind": "value", - "name": "buffer", - "docstrings": [], - "signature": "let buffer: t => array_buffer" - }, - { - "id": "Js.Typed_array.Uint16Array.byteLength", - "kind": "value", - "name": "byteLength", - "docstrings": [], - "signature": "let byteLength: t => int" - }, - { - "id": "Js.Typed_array.Uint16Array.byteOffset", - "kind": "value", - "name": "byteOffset", - "docstrings": [], - "signature": "let byteOffset: t => int" - }, - { - "id": "Js.Typed_array.Uint16Array.setArray", - "kind": "value", - "name": "setArray", - "docstrings": [], - "signature": "let setArray: (array, t) => unit" - }, - { - "id": "Js.Typed_array.Uint16Array.setArrayOffset", - "kind": "value", - "name": "setArrayOffset", - "docstrings": [], - "signature": "let setArrayOffset: (array, int, t) => unit" - }, - { - "id": "Js.Typed_array.Uint16Array.length", - "kind": "value", - "name": "length", - "docstrings": [], - "signature": "let length: t => int" - }, - { - "id": "Js.Typed_array.Uint16Array.copyWithin", - "kind": "value", - "name": "copyWithin", - "docstrings": [], - "signature": "let copyWithin: (~to_: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint16Array.copyWithinFrom", - "kind": "value", - "name": "copyWithinFrom", - "docstrings": [], - "signature": "let copyWithinFrom: (~to_: int, ~from: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint16Array.copyWithinFromRange", - "kind": "value", - "name": "copyWithinFromRange", - "docstrings": [], - "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint16Array.fillInPlace", - "kind": "value", - "name": "fillInPlace", - "docstrings": [], - "signature": "let fillInPlace: (elt, t) => t" - }, - { - "id": "Js.Typed_array.Uint16Array.fillFromInPlace", - "kind": "value", - "name": "fillFromInPlace", - "docstrings": [], - "signature": "let fillFromInPlace: (elt, ~from: int, t) => t" - }, + "docstrings": [ + "Underlying type of a JSON value" + ], + "signature": "type t<_> =\n | String: t\n | Number: t\n | Object: t>\n | Array: t>\n | Boolean: t\n | Null: t" + } + ] + }, + "js/weakmap": { + "id": "Js.WeakMap", + "name": "WeakMap", + "docstrings": [ + "Provides bindings for ES6 WeakMap", + "ES6 WeakMap API" + ], + "items": [ { - "id": "Js.Typed_array.Uint16Array.fillRangeInPlace", - "kind": "value", - "name": "fillRangeInPlace", + "id": "Js.WeakMap.t", + "kind": "type", + "name": "t", "docstrings": [], - "signature": "let fillRangeInPlace: (elt, ~start: int, ~end_: int, t) => t" - }, + "signature": "type t<'k, 'v>" + } + ] + }, + "js/map": { + "id": "Js.Map", + "name": "Map", + "docstrings": [ + "Provides bindings for ES6 Map", + "ES6 Map API" + ], + "items": [ { - "id": "Js.Typed_array.Uint16Array.reverseInPlace", - "kind": "value", - "name": "reverseInPlace", + "id": "Js.Map.t", + "kind": "type", + "name": "t", "docstrings": [], - "signature": "let reverseInPlace: t => t" - }, + "signature": "type t<'k, 'v>" + } + ] + }, + "js/weakset": { + "id": "Js.WeakSet", + "name": "WeakSet", + "docstrings": [ + "Provides bindings for ES6 WeakSet", + "ES6 WeakSet API" + ], + "items": [ { - "id": "Js.Typed_array.Uint16Array.sortInPlace", - "kind": "value", - "name": "sortInPlace", + "id": "Js.WeakSet.t", + "kind": "type", + "name": "t", "docstrings": [], - "signature": "let sortInPlace: t => t" - }, - { - "id": "Js.Typed_array.Uint16Array.sortInPlaceWith", - "kind": "value", - "name": "sortInPlaceWith", - "docstrings": [], - "signature": "let sortInPlaceWith: ((elt, elt) => int, t) => t" - }, - { - "id": "Js.Typed_array.Uint16Array.includes", - "kind": "value", - "name": "includes", - "docstrings": [], - "signature": "let includes: (elt, t) => bool" - }, - { - "id": "Js.Typed_array.Uint16Array.indexOf", - "kind": "value", - "name": "indexOf", - "docstrings": [], - "signature": "let indexOf: (elt, t) => int" - }, - { - "id": "Js.Typed_array.Uint16Array.indexOfFrom", - "kind": "value", - "name": "indexOfFrom", - "docstrings": [], - "signature": "let indexOfFrom: (elt, ~from: int, t) => int" - }, - { - "id": "Js.Typed_array.Uint16Array.join", - "kind": "value", - "name": "join", - "docstrings": [], - "signature": "let join: t => string" - }, - { - "id": "Js.Typed_array.Uint16Array.joinWith", - "kind": "value", - "name": "joinWith", - "docstrings": [], - "signature": "let joinWith: (string, t) => string" - }, - { - "id": "Js.Typed_array.Uint16Array.lastIndexOf", - "kind": "value", - "name": "lastIndexOf", - "docstrings": [], - "signature": "let lastIndexOf: (elt, t) => int" - }, - { - "id": "Js.Typed_array.Uint16Array.lastIndexOfFrom", - "kind": "value", - "name": "lastIndexOfFrom", - "docstrings": [], - "signature": "let lastIndexOfFrom: (elt, ~from: int, t) => int" - }, - { - "id": "Js.Typed_array.Uint16Array.slice", - "kind": "value", - "name": "slice", - "docstrings": [], - "signature": "let slice: (~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint16Array.copy", - "kind": "value", - "name": "copy", - "docstrings": [], - "signature": "let copy: t => t" - }, - { - "id": "Js.Typed_array.Uint16Array.sliceFrom", - "kind": "value", - "name": "sliceFrom", - "docstrings": [], - "signature": "let sliceFrom: (int, t) => t" - }, - { - "id": "Js.Typed_array.Uint16Array.subarray", - "kind": "value", - "name": "subarray", - "docstrings": [], - "signature": "let subarray: (~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint16Array.subarrayFrom", - "kind": "value", - "name": "subarrayFrom", - "docstrings": [], - "signature": "let subarrayFrom: (int, t) => t" - }, - { - "id": "Js.Typed_array.Uint16Array.toString", - "kind": "value", - "name": "toString", - "docstrings": [], - "signature": "let toString: t => string" - }, - { - "id": "Js.Typed_array.Uint16Array.toLocaleString", - "kind": "value", - "name": "toLocaleString", - "docstrings": [], - "signature": "let toLocaleString: t => string" - }, - { - "id": "Js.Typed_array.Uint16Array.every", - "kind": "value", - "name": "every", - "docstrings": [], - "signature": "let every: (elt => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Uint16Array.everyi", - "kind": "value", - "name": "everyi", - "docstrings": [], - "signature": "let everyi: ((elt, int) => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Uint16Array.filter", - "kind": "value", - "name": "filter", - "docstrings": [], - "signature": "let filter: (elt => bool, t) => t" - }, - { - "id": "Js.Typed_array.Uint16Array.filteri", - "kind": "value", - "name": "filteri", - "docstrings": [], - "signature": "let filteri: ((elt, int) => bool, t) => t" - }, - { - "id": "Js.Typed_array.Uint16Array.find", - "kind": "value", - "name": "find", - "docstrings": [], - "signature": "let find: (elt => bool, t) => Js.undefined" - }, - { - "id": "Js.Typed_array.Uint16Array.findi", - "kind": "value", - "name": "findi", - "docstrings": [], - "signature": "let findi: ((elt, int) => bool, t) => Js.undefined" - }, - { - "id": "Js.Typed_array.Uint16Array.findIndex", - "kind": "value", - "name": "findIndex", - "docstrings": [], - "signature": "let findIndex: (elt => bool, t) => int" - }, - { - "id": "Js.Typed_array.Uint16Array.findIndexi", - "kind": "value", - "name": "findIndexi", - "docstrings": [], - "signature": "let findIndexi: ((elt, int) => bool, t) => int" - }, - { - "id": "Js.Typed_array.Uint16Array.forEach", - "kind": "value", - "name": "forEach", - "docstrings": [], - "signature": "let forEach: (elt => unit, t) => unit" - }, - { - "id": "Js.Typed_array.Uint16Array.forEachi", - "kind": "value", - "name": "forEachi", - "docstrings": [], - "signature": "let forEachi: ((elt, int) => unit, t) => unit" - }, - { - "id": "Js.Typed_array.Uint16Array.map", - "kind": "value", - "name": "map", - "docstrings": [], - "signature": "let map: (elt => 'a, t) => typed_array<'a>" - }, - { - "id": "Js.Typed_array.Uint16Array.mapi", - "kind": "value", - "name": "mapi", - "docstrings": [], - "signature": "let mapi: ((elt, int) => 'a, t) => typed_array<'a>" - }, - { - "id": "Js.Typed_array.Uint16Array.reduce", - "kind": "value", - "name": "reduce", - "docstrings": [], - "signature": "let reduce: (('a, elt) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Uint16Array.reducei", - "kind": "value", - "name": "reducei", - "docstrings": [], - "signature": "let reducei: (('a, elt, int) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Uint16Array.reduceRight", - "kind": "value", - "name": "reduceRight", - "docstrings": [], - "signature": "let reduceRight: (('a, elt) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Uint16Array.reduceRighti", - "kind": "value", - "name": "reduceRighti", - "docstrings": [], - "signature": "let reduceRighti: (('a, elt, int) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Uint16Array.some", - "kind": "value", - "name": "some", - "docstrings": [], - "signature": "let some: (elt => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Uint16Array.somei", - "kind": "value", - "name": "somei", - "docstrings": [], - "signature": "let somei: ((elt, int) => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Uint16Array._BYTES_PER_ELEMENT", - "kind": "value", - "name": "_BYTES_PER_ELEMENT", - "docstrings": [], - "signature": "let _BYTES_PER_ELEMENT: int" - }, - { - "id": "Js.Typed_array.Uint16Array.make", - "kind": "value", - "name": "make", - "docstrings": [], - "signature": "let make: array => t" - }, - { - "id": "Js.Typed_array.Uint16Array.fromBuffer", - "kind": "value", - "name": "fromBuffer", - "docstrings": [ - "can throw" - ], - "signature": "let fromBuffer: array_buffer => t" - }, - { - "id": "Js.Typed_array.Uint16Array.fromBufferOffset", - "kind": "value", - "name": "fromBufferOffset", - "docstrings": [ - "**raise** Js.Exn.Error raise Js exception\n\n **param** offset is in bytes" - ], - "signature": "let fromBufferOffset: (array_buffer, int) => t" - }, - { - "id": "Js.Typed_array.Uint16Array.fromBufferRange", - "kind": "value", - "name": "fromBufferRange", - "docstrings": [ - "**raise** Js.Exn.Error raises Js exception\n\n **param** offset is in bytes, length in elements" - ], - "signature": "let fromBufferRange: (array_buffer, ~offset: int, ~length: int) => t" - }, - { - "id": "Js.Typed_array.Uint16Array.fromLength", - "kind": "value", - "name": "fromLength", - "docstrings": [], - "signature": "let fromLength: int => t" - }, - { - "id": "Js.Typed_array.Uint16Array.from", - "kind": "value", - "name": "from", - "docstrings": [], - "signature": "let from: array_like => t" - } - ] - }, - "js/typed_array/int16array": { - "id": "Js.Typed_array.Int16Array", - "name": "Int16Array", - "docstrings": [], - "items": [ - { - "id": "Js.Typed_array.Int16Array.elt", - "kind": "type", - "name": "elt", - "docstrings": [ - "" - ], - "signature": "type elt = int" - }, - { - "id": "Js.Typed_array.Int16Array.typed_array", - "kind": "type", - "name": "typed_array", - "docstrings": [], - "signature": "type typed_array<\n 'a,\n> = Js_typed_array2.Int16Array.typed_array<'a>" - }, - { - "id": "Js.Typed_array.Int16Array.t", - "kind": "type", - "name": "t", - "docstrings": [], - "signature": "type t = typed_array" - }, - { - "id": "Js.Typed_array.Int16Array.unsafe_get", - "kind": "value", - "name": "unsafe_get", - "docstrings": [], - "signature": "let unsafe_get: (t, int) => elt" - }, - { - "id": "Js.Typed_array.Int16Array.unsafe_set", - "kind": "value", - "name": "unsafe_set", - "docstrings": [], - "signature": "let unsafe_set: (t, int, elt) => unit" - }, - { - "id": "Js.Typed_array.Int16Array.buffer", - "kind": "value", - "name": "buffer", - "docstrings": [], - "signature": "let buffer: t => array_buffer" - }, - { - "id": "Js.Typed_array.Int16Array.byteLength", - "kind": "value", - "name": "byteLength", - "docstrings": [], - "signature": "let byteLength: t => int" - }, - { - "id": "Js.Typed_array.Int16Array.byteOffset", - "kind": "value", - "name": "byteOffset", - "docstrings": [], - "signature": "let byteOffset: t => int" - }, - { - "id": "Js.Typed_array.Int16Array.setArray", - "kind": "value", - "name": "setArray", - "docstrings": [], - "signature": "let setArray: (array, t) => unit" - }, - { - "id": "Js.Typed_array.Int16Array.setArrayOffset", - "kind": "value", - "name": "setArrayOffset", - "docstrings": [], - "signature": "let setArrayOffset: (array, int, t) => unit" - }, - { - "id": "Js.Typed_array.Int16Array.length", - "kind": "value", - "name": "length", - "docstrings": [], - "signature": "let length: t => int" - }, - { - "id": "Js.Typed_array.Int16Array.copyWithin", - "kind": "value", - "name": "copyWithin", - "docstrings": [], - "signature": "let copyWithin: (~to_: int, t) => t" - }, - { - "id": "Js.Typed_array.Int16Array.copyWithinFrom", - "kind": "value", - "name": "copyWithinFrom", - "docstrings": [], - "signature": "let copyWithinFrom: (~to_: int, ~from: int, t) => t" - }, - { - "id": "Js.Typed_array.Int16Array.copyWithinFromRange", - "kind": "value", - "name": "copyWithinFromRange", - "docstrings": [], - "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Int16Array.fillInPlace", - "kind": "value", - "name": "fillInPlace", - "docstrings": [], - "signature": "let fillInPlace: (elt, t) => t" - }, - { - "id": "Js.Typed_array.Int16Array.fillFromInPlace", - "kind": "value", - "name": "fillFromInPlace", - "docstrings": [], - "signature": "let fillFromInPlace: (elt, ~from: int, t) => t" - }, - { - "id": "Js.Typed_array.Int16Array.fillRangeInPlace", - "kind": "value", - "name": "fillRangeInPlace", - "docstrings": [], - "signature": "let fillRangeInPlace: (elt, ~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Int16Array.reverseInPlace", - "kind": "value", - "name": "reverseInPlace", - "docstrings": [], - "signature": "let reverseInPlace: t => t" - }, - { - "id": "Js.Typed_array.Int16Array.sortInPlace", - "kind": "value", - "name": "sortInPlace", - "docstrings": [], - "signature": "let sortInPlace: t => t" - }, - { - "id": "Js.Typed_array.Int16Array.sortInPlaceWith", - "kind": "value", - "name": "sortInPlaceWith", - "docstrings": [], - "signature": "let sortInPlaceWith: ((elt, elt) => int, t) => t" - }, - { - "id": "Js.Typed_array.Int16Array.includes", - "kind": "value", - "name": "includes", - "docstrings": [], - "signature": "let includes: (elt, t) => bool" - }, - { - "id": "Js.Typed_array.Int16Array.indexOf", - "kind": "value", - "name": "indexOf", - "docstrings": [], - "signature": "let indexOf: (elt, t) => int" - }, - { - "id": "Js.Typed_array.Int16Array.indexOfFrom", - "kind": "value", - "name": "indexOfFrom", - "docstrings": [], - "signature": "let indexOfFrom: (elt, ~from: int, t) => int" - }, - { - "id": "Js.Typed_array.Int16Array.join", - "kind": "value", - "name": "join", - "docstrings": [], - "signature": "let join: t => string" - }, - { - "id": "Js.Typed_array.Int16Array.joinWith", - "kind": "value", - "name": "joinWith", - "docstrings": [], - "signature": "let joinWith: (string, t) => string" - }, - { - "id": "Js.Typed_array.Int16Array.lastIndexOf", - "kind": "value", - "name": "lastIndexOf", - "docstrings": [], - "signature": "let lastIndexOf: (elt, t) => int" - }, - { - "id": "Js.Typed_array.Int16Array.lastIndexOfFrom", - "kind": "value", - "name": "lastIndexOfFrom", - "docstrings": [], - "signature": "let lastIndexOfFrom: (elt, ~from: int, t) => int" - }, - { - "id": "Js.Typed_array.Int16Array.slice", - "kind": "value", - "name": "slice", - "docstrings": [], - "signature": "let slice: (~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Int16Array.copy", - "kind": "value", - "name": "copy", - "docstrings": [], - "signature": "let copy: t => t" - }, - { - "id": "Js.Typed_array.Int16Array.sliceFrom", - "kind": "value", - "name": "sliceFrom", - "docstrings": [], - "signature": "let sliceFrom: (int, t) => t" - }, - { - "id": "Js.Typed_array.Int16Array.subarray", - "kind": "value", - "name": "subarray", - "docstrings": [], - "signature": "let subarray: (~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Int16Array.subarrayFrom", - "kind": "value", - "name": "subarrayFrom", - "docstrings": [], - "signature": "let subarrayFrom: (int, t) => t" - }, - { - "id": "Js.Typed_array.Int16Array.toString", - "kind": "value", - "name": "toString", - "docstrings": [], - "signature": "let toString: t => string" - }, - { - "id": "Js.Typed_array.Int16Array.toLocaleString", - "kind": "value", - "name": "toLocaleString", - "docstrings": [], - "signature": "let toLocaleString: t => string" - }, - { - "id": "Js.Typed_array.Int16Array.every", - "kind": "value", - "name": "every", - "docstrings": [], - "signature": "let every: (elt => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Int16Array.everyi", - "kind": "value", - "name": "everyi", - "docstrings": [], - "signature": "let everyi: ((elt, int) => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Int16Array.filter", - "kind": "value", - "name": "filter", - "docstrings": [], - "signature": "let filter: (elt => bool, t) => t" - }, - { - "id": "Js.Typed_array.Int16Array.filteri", - "kind": "value", - "name": "filteri", - "docstrings": [], - "signature": "let filteri: ((elt, int) => bool, t) => t" - }, - { - "id": "Js.Typed_array.Int16Array.find", - "kind": "value", - "name": "find", - "docstrings": [], - "signature": "let find: (elt => bool, t) => Js.undefined" - }, - { - "id": "Js.Typed_array.Int16Array.findi", - "kind": "value", - "name": "findi", - "docstrings": [], - "signature": "let findi: ((elt, int) => bool, t) => Js.undefined" - }, - { - "id": "Js.Typed_array.Int16Array.findIndex", - "kind": "value", - "name": "findIndex", - "docstrings": [], - "signature": "let findIndex: (elt => bool, t) => int" - }, - { - "id": "Js.Typed_array.Int16Array.findIndexi", - "kind": "value", - "name": "findIndexi", - "docstrings": [], - "signature": "let findIndexi: ((elt, int) => bool, t) => int" - }, - { - "id": "Js.Typed_array.Int16Array.forEach", - "kind": "value", - "name": "forEach", - "docstrings": [], - "signature": "let forEach: (elt => unit, t) => unit" - }, - { - "id": "Js.Typed_array.Int16Array.forEachi", - "kind": "value", - "name": "forEachi", - "docstrings": [], - "signature": "let forEachi: ((elt, int) => unit, t) => unit" - }, - { - "id": "Js.Typed_array.Int16Array.map", - "kind": "value", - "name": "map", - "docstrings": [], - "signature": "let map: (elt => 'a, t) => typed_array<'a>" - }, - { - "id": "Js.Typed_array.Int16Array.mapi", - "kind": "value", - "name": "mapi", - "docstrings": [], - "signature": "let mapi: ((elt, int) => 'a, t) => typed_array<'a>" - }, - { - "id": "Js.Typed_array.Int16Array.reduce", - "kind": "value", - "name": "reduce", - "docstrings": [], - "signature": "let reduce: (('a, elt) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Int16Array.reducei", - "kind": "value", - "name": "reducei", - "docstrings": [], - "signature": "let reducei: (('a, elt, int) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Int16Array.reduceRight", - "kind": "value", - "name": "reduceRight", - "docstrings": [], - "signature": "let reduceRight: (('a, elt) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Int16Array.reduceRighti", - "kind": "value", - "name": "reduceRighti", - "docstrings": [], - "signature": "let reduceRighti: (('a, elt, int) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Int16Array.some", - "kind": "value", - "name": "some", - "docstrings": [], - "signature": "let some: (elt => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Int16Array.somei", - "kind": "value", - "name": "somei", - "docstrings": [], - "signature": "let somei: ((elt, int) => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Int16Array._BYTES_PER_ELEMENT", - "kind": "value", - "name": "_BYTES_PER_ELEMENT", - "docstrings": [], - "signature": "let _BYTES_PER_ELEMENT: int" - }, - { - "id": "Js.Typed_array.Int16Array.make", - "kind": "value", - "name": "make", - "docstrings": [], - "signature": "let make: array => t" - }, - { - "id": "Js.Typed_array.Int16Array.fromBuffer", - "kind": "value", - "name": "fromBuffer", - "docstrings": [ - "can throw" - ], - "signature": "let fromBuffer: array_buffer => t" - }, - { - "id": "Js.Typed_array.Int16Array.fromBufferOffset", - "kind": "value", - "name": "fromBufferOffset", - "docstrings": [ - "**raise** Js.Exn.Error raise Js exception\n\n **param** offset is in bytes" - ], - "signature": "let fromBufferOffset: (array_buffer, int) => t" - }, - { - "id": "Js.Typed_array.Int16Array.fromBufferRange", - "kind": "value", - "name": "fromBufferRange", - "docstrings": [ - "**raise** Js.Exn.Error raises Js exception\n\n **param** offset is in bytes, length in elements" - ], - "signature": "let fromBufferRange: (array_buffer, ~offset: int, ~length: int) => t" - }, - { - "id": "Js.Typed_array.Int16Array.fromLength", - "kind": "value", - "name": "fromLength", - "docstrings": [], - "signature": "let fromLength: int => t" - }, - { - "id": "Js.Typed_array.Int16Array.from", - "kind": "value", - "name": "from", - "docstrings": [], - "signature": "let from: array_like => t" - } - ] - }, - "js/typed_array/uint8clampedarray": { - "id": "Js.Typed_array.Uint8ClampedArray", - "name": "Uint8ClampedArray", - "docstrings": [], - "items": [ - { - "id": "Js.Typed_array.Uint8ClampedArray.elt", - "kind": "type", - "name": "elt", - "docstrings": [ - "" - ], - "signature": "type elt = int" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.typed_array", - "kind": "type", - "name": "typed_array", - "docstrings": [], - "signature": "type typed_array<\n 'a,\n> = Js_typed_array2.Uint8ClampedArray.typed_array<'a>" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.t", - "kind": "type", - "name": "t", - "docstrings": [], - "signature": "type t = typed_array" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.unsafe_get", - "kind": "value", - "name": "unsafe_get", - "docstrings": [], - "signature": "let unsafe_get: (t, int) => elt" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.unsafe_set", - "kind": "value", - "name": "unsafe_set", - "docstrings": [], - "signature": "let unsafe_set: (t, int, elt) => unit" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.buffer", - "kind": "value", - "name": "buffer", - "docstrings": [], - "signature": "let buffer: t => array_buffer" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.byteLength", - "kind": "value", - "name": "byteLength", - "docstrings": [], - "signature": "let byteLength: t => int" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.byteOffset", - "kind": "value", - "name": "byteOffset", - "docstrings": [], - "signature": "let byteOffset: t => int" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.setArray", - "kind": "value", - "name": "setArray", - "docstrings": [], - "signature": "let setArray: (array, t) => unit" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.setArrayOffset", - "kind": "value", - "name": "setArrayOffset", - "docstrings": [], - "signature": "let setArrayOffset: (array, int, t) => unit" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.length", - "kind": "value", - "name": "length", - "docstrings": [], - "signature": "let length: t => int" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.copyWithin", - "kind": "value", - "name": "copyWithin", - "docstrings": [], - "signature": "let copyWithin: (~to_: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.copyWithinFrom", - "kind": "value", - "name": "copyWithinFrom", - "docstrings": [], - "signature": "let copyWithinFrom: (~to_: int, ~from: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.copyWithinFromRange", - "kind": "value", - "name": "copyWithinFromRange", - "docstrings": [], - "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.fillInPlace", - "kind": "value", - "name": "fillInPlace", - "docstrings": [], - "signature": "let fillInPlace: (elt, t) => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.fillFromInPlace", - "kind": "value", - "name": "fillFromInPlace", - "docstrings": [], - "signature": "let fillFromInPlace: (elt, ~from: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.fillRangeInPlace", - "kind": "value", - "name": "fillRangeInPlace", - "docstrings": [], - "signature": "let fillRangeInPlace: (elt, ~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.reverseInPlace", - "kind": "value", - "name": "reverseInPlace", - "docstrings": [], - "signature": "let reverseInPlace: t => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.sortInPlace", - "kind": "value", - "name": "sortInPlace", - "docstrings": [], - "signature": "let sortInPlace: t => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.sortInPlaceWith", - "kind": "value", - "name": "sortInPlaceWith", - "docstrings": [], - "signature": "let sortInPlaceWith: ((elt, elt) => int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.includes", - "kind": "value", - "name": "includes", - "docstrings": [], - "signature": "let includes: (elt, t) => bool" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.indexOf", - "kind": "value", - "name": "indexOf", - "docstrings": [], - "signature": "let indexOf: (elt, t) => int" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.indexOfFrom", - "kind": "value", - "name": "indexOfFrom", - "docstrings": [], - "signature": "let indexOfFrom: (elt, ~from: int, t) => int" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.join", - "kind": "value", - "name": "join", - "docstrings": [], - "signature": "let join: t => string" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.joinWith", - "kind": "value", - "name": "joinWith", - "docstrings": [], - "signature": "let joinWith: (string, t) => string" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.lastIndexOf", - "kind": "value", - "name": "lastIndexOf", - "docstrings": [], - "signature": "let lastIndexOf: (elt, t) => int" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.lastIndexOfFrom", - "kind": "value", - "name": "lastIndexOfFrom", - "docstrings": [], - "signature": "let lastIndexOfFrom: (elt, ~from: int, t) => int" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.slice", - "kind": "value", - "name": "slice", - "docstrings": [], - "signature": "let slice: (~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.copy", - "kind": "value", - "name": "copy", - "docstrings": [], - "signature": "let copy: t => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.sliceFrom", - "kind": "value", - "name": "sliceFrom", - "docstrings": [], - "signature": "let sliceFrom: (int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.subarray", - "kind": "value", - "name": "subarray", - "docstrings": [], - "signature": "let subarray: (~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.subarrayFrom", - "kind": "value", - "name": "subarrayFrom", - "docstrings": [], - "signature": "let subarrayFrom: (int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.toString", - "kind": "value", - "name": "toString", - "docstrings": [], - "signature": "let toString: t => string" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.toLocaleString", - "kind": "value", - "name": "toLocaleString", - "docstrings": [], - "signature": "let toLocaleString: t => string" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.every", - "kind": "value", - "name": "every", - "docstrings": [], - "signature": "let every: (elt => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.everyi", - "kind": "value", - "name": "everyi", - "docstrings": [], - "signature": "let everyi: ((elt, int) => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.filter", - "kind": "value", - "name": "filter", - "docstrings": [], - "signature": "let filter: (elt => bool, t) => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.filteri", - "kind": "value", - "name": "filteri", - "docstrings": [], - "signature": "let filteri: ((elt, int) => bool, t) => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.find", - "kind": "value", - "name": "find", - "docstrings": [], - "signature": "let find: (elt => bool, t) => Js.undefined" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.findi", - "kind": "value", - "name": "findi", - "docstrings": [], - "signature": "let findi: ((elt, int) => bool, t) => Js.undefined" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.findIndex", - "kind": "value", - "name": "findIndex", - "docstrings": [], - "signature": "let findIndex: (elt => bool, t) => int" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.findIndexi", - "kind": "value", - "name": "findIndexi", - "docstrings": [], - "signature": "let findIndexi: ((elt, int) => bool, t) => int" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.forEach", - "kind": "value", - "name": "forEach", - "docstrings": [], - "signature": "let forEach: (elt => unit, t) => unit" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.forEachi", - "kind": "value", - "name": "forEachi", - "docstrings": [], - "signature": "let forEachi: ((elt, int) => unit, t) => unit" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.map", - "kind": "value", - "name": "map", - "docstrings": [], - "signature": "let map: (elt => 'a, t) => typed_array<'a>" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.mapi", - "kind": "value", - "name": "mapi", - "docstrings": [], - "signature": "let mapi: ((elt, int) => 'a, t) => typed_array<'a>" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.reduce", - "kind": "value", - "name": "reduce", - "docstrings": [], - "signature": "let reduce: (('a, elt) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.reducei", - "kind": "value", - "name": "reducei", - "docstrings": [], - "signature": "let reducei: (('a, elt, int) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.reduceRight", - "kind": "value", - "name": "reduceRight", - "docstrings": [], - "signature": "let reduceRight: (('a, elt) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.reduceRighti", - "kind": "value", - "name": "reduceRighti", - "docstrings": [], - "signature": "let reduceRighti: (('a, elt, int) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.some", - "kind": "value", - "name": "some", - "docstrings": [], - "signature": "let some: (elt => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.somei", - "kind": "value", - "name": "somei", - "docstrings": [], - "signature": "let somei: ((elt, int) => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray._BYTES_PER_ELEMENT", - "kind": "value", - "name": "_BYTES_PER_ELEMENT", - "docstrings": [], - "signature": "let _BYTES_PER_ELEMENT: int" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.make", - "kind": "value", - "name": "make", - "docstrings": [], - "signature": "let make: array => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.fromBuffer", - "kind": "value", - "name": "fromBuffer", - "docstrings": [ - "can throw" - ], - "signature": "let fromBuffer: array_buffer => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.fromBufferOffset", - "kind": "value", - "name": "fromBufferOffset", - "docstrings": [ - "**raise** Js.Exn.Error raise Js exception\n\n **param** offset is in bytes" - ], - "signature": "let fromBufferOffset: (array_buffer, int) => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.fromBufferRange", - "kind": "value", - "name": "fromBufferRange", - "docstrings": [ - "**raise** Js.Exn.Error raises Js exception\n\n **param** offset is in bytes, length in elements" - ], - "signature": "let fromBufferRange: (array_buffer, ~offset: int, ~length: int) => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.fromLength", - "kind": "value", - "name": "fromLength", - "docstrings": [], - "signature": "let fromLength: int => t" - }, - { - "id": "Js.Typed_array.Uint8ClampedArray.from", - "kind": "value", - "name": "from", - "docstrings": [], - "signature": "let from: array_like => t" - } - ] - }, - "js/typed_array/uint8array": { - "id": "Js.Typed_array.Uint8Array", - "name": "Uint8Array", - "docstrings": [], - "items": [ - { - "id": "Js.Typed_array.Uint8Array.elt", - "kind": "type", - "name": "elt", - "docstrings": [ - "" - ], - "signature": "type elt = int" - }, - { - "id": "Js.Typed_array.Uint8Array.typed_array", - "kind": "type", - "name": "typed_array", - "docstrings": [], - "signature": "type typed_array<\n 'a,\n> = Js_typed_array2.Uint8Array.typed_array<'a>" - }, - { - "id": "Js.Typed_array.Uint8Array.t", - "kind": "type", - "name": "t", - "docstrings": [], - "signature": "type t = typed_array" - }, - { - "id": "Js.Typed_array.Uint8Array.unsafe_get", - "kind": "value", - "name": "unsafe_get", - "docstrings": [], - "signature": "let unsafe_get: (t, int) => elt" - }, - { - "id": "Js.Typed_array.Uint8Array.unsafe_set", - "kind": "value", - "name": "unsafe_set", - "docstrings": [], - "signature": "let unsafe_set: (t, int, elt) => unit" - }, - { - "id": "Js.Typed_array.Uint8Array.buffer", - "kind": "value", - "name": "buffer", - "docstrings": [], - "signature": "let buffer: t => array_buffer" - }, - { - "id": "Js.Typed_array.Uint8Array.byteLength", - "kind": "value", - "name": "byteLength", - "docstrings": [], - "signature": "let byteLength: t => int" - }, - { - "id": "Js.Typed_array.Uint8Array.byteOffset", - "kind": "value", - "name": "byteOffset", - "docstrings": [], - "signature": "let byteOffset: t => int" - }, - { - "id": "Js.Typed_array.Uint8Array.setArray", - "kind": "value", - "name": "setArray", - "docstrings": [], - "signature": "let setArray: (array, t) => unit" - }, - { - "id": "Js.Typed_array.Uint8Array.setArrayOffset", - "kind": "value", - "name": "setArrayOffset", - "docstrings": [], - "signature": "let setArrayOffset: (array, int, t) => unit" - }, - { - "id": "Js.Typed_array.Uint8Array.length", - "kind": "value", - "name": "length", - "docstrings": [], - "signature": "let length: t => int" - }, - { - "id": "Js.Typed_array.Uint8Array.copyWithin", - "kind": "value", - "name": "copyWithin", - "docstrings": [], - "signature": "let copyWithin: (~to_: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8Array.copyWithinFrom", - "kind": "value", - "name": "copyWithinFrom", - "docstrings": [], - "signature": "let copyWithinFrom: (~to_: int, ~from: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8Array.copyWithinFromRange", - "kind": "value", - "name": "copyWithinFromRange", - "docstrings": [], - "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8Array.fillInPlace", - "kind": "value", - "name": "fillInPlace", - "docstrings": [], - "signature": "let fillInPlace: (elt, t) => t" - }, - { - "id": "Js.Typed_array.Uint8Array.fillFromInPlace", - "kind": "value", - "name": "fillFromInPlace", - "docstrings": [], - "signature": "let fillFromInPlace: (elt, ~from: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8Array.fillRangeInPlace", - "kind": "value", - "name": "fillRangeInPlace", - "docstrings": [], - "signature": "let fillRangeInPlace: (elt, ~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8Array.reverseInPlace", - "kind": "value", - "name": "reverseInPlace", - "docstrings": [], - "signature": "let reverseInPlace: t => t" - }, - { - "id": "Js.Typed_array.Uint8Array.sortInPlace", - "kind": "value", - "name": "sortInPlace", - "docstrings": [], - "signature": "let sortInPlace: t => t" - }, - { - "id": "Js.Typed_array.Uint8Array.sortInPlaceWith", - "kind": "value", - "name": "sortInPlaceWith", - "docstrings": [], - "signature": "let sortInPlaceWith: ((elt, elt) => int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8Array.includes", - "kind": "value", - "name": "includes", - "docstrings": [], - "signature": "let includes: (elt, t) => bool" - }, - { - "id": "Js.Typed_array.Uint8Array.indexOf", - "kind": "value", - "name": "indexOf", - "docstrings": [], - "signature": "let indexOf: (elt, t) => int" - }, - { - "id": "Js.Typed_array.Uint8Array.indexOfFrom", - "kind": "value", - "name": "indexOfFrom", - "docstrings": [], - "signature": "let indexOfFrom: (elt, ~from: int, t) => int" - }, - { - "id": "Js.Typed_array.Uint8Array.join", - "kind": "value", - "name": "join", - "docstrings": [], - "signature": "let join: t => string" - }, - { - "id": "Js.Typed_array.Uint8Array.joinWith", - "kind": "value", - "name": "joinWith", - "docstrings": [], - "signature": "let joinWith: (string, t) => string" - }, - { - "id": "Js.Typed_array.Uint8Array.lastIndexOf", - "kind": "value", - "name": "lastIndexOf", - "docstrings": [], - "signature": "let lastIndexOf: (elt, t) => int" - }, - { - "id": "Js.Typed_array.Uint8Array.lastIndexOfFrom", - "kind": "value", - "name": "lastIndexOfFrom", - "docstrings": [], - "signature": "let lastIndexOfFrom: (elt, ~from: int, t) => int" - }, - { - "id": "Js.Typed_array.Uint8Array.slice", - "kind": "value", - "name": "slice", - "docstrings": [], - "signature": "let slice: (~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8Array.copy", - "kind": "value", - "name": "copy", - "docstrings": [], - "signature": "let copy: t => t" - }, - { - "id": "Js.Typed_array.Uint8Array.sliceFrom", - "kind": "value", - "name": "sliceFrom", - "docstrings": [], - "signature": "let sliceFrom: (int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8Array.subarray", - "kind": "value", - "name": "subarray", - "docstrings": [], - "signature": "let subarray: (~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8Array.subarrayFrom", - "kind": "value", - "name": "subarrayFrom", - "docstrings": [], - "signature": "let subarrayFrom: (int, t) => t" - }, - { - "id": "Js.Typed_array.Uint8Array.toString", - "kind": "value", - "name": "toString", - "docstrings": [], - "signature": "let toString: t => string" - }, - { - "id": "Js.Typed_array.Uint8Array.toLocaleString", - "kind": "value", - "name": "toLocaleString", - "docstrings": [], - "signature": "let toLocaleString: t => string" - }, - { - "id": "Js.Typed_array.Uint8Array.every", - "kind": "value", - "name": "every", - "docstrings": [], - "signature": "let every: (elt => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Uint8Array.everyi", - "kind": "value", - "name": "everyi", - "docstrings": [], - "signature": "let everyi: ((elt, int) => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Uint8Array.filter", - "kind": "value", - "name": "filter", - "docstrings": [], - "signature": "let filter: (elt => bool, t) => t" - }, - { - "id": "Js.Typed_array.Uint8Array.filteri", - "kind": "value", - "name": "filteri", - "docstrings": [], - "signature": "let filteri: ((elt, int) => bool, t) => t" - }, - { - "id": "Js.Typed_array.Uint8Array.find", - "kind": "value", - "name": "find", - "docstrings": [], - "signature": "let find: (elt => bool, t) => Js.undefined" - }, - { - "id": "Js.Typed_array.Uint8Array.findi", - "kind": "value", - "name": "findi", - "docstrings": [], - "signature": "let findi: ((elt, int) => bool, t) => Js.undefined" - }, - { - "id": "Js.Typed_array.Uint8Array.findIndex", - "kind": "value", - "name": "findIndex", - "docstrings": [], - "signature": "let findIndex: (elt => bool, t) => int" - }, - { - "id": "Js.Typed_array.Uint8Array.findIndexi", - "kind": "value", - "name": "findIndexi", - "docstrings": [], - "signature": "let findIndexi: ((elt, int) => bool, t) => int" - }, - { - "id": "Js.Typed_array.Uint8Array.forEach", - "kind": "value", - "name": "forEach", - "docstrings": [], - "signature": "let forEach: (elt => unit, t) => unit" - }, - { - "id": "Js.Typed_array.Uint8Array.forEachi", - "kind": "value", - "name": "forEachi", - "docstrings": [], - "signature": "let forEachi: ((elt, int) => unit, t) => unit" - }, - { - "id": "Js.Typed_array.Uint8Array.map", - "kind": "value", - "name": "map", - "docstrings": [], - "signature": "let map: (elt => 'a, t) => typed_array<'a>" - }, - { - "id": "Js.Typed_array.Uint8Array.mapi", - "kind": "value", - "name": "mapi", - "docstrings": [], - "signature": "let mapi: ((elt, int) => 'a, t) => typed_array<'a>" - }, - { - "id": "Js.Typed_array.Uint8Array.reduce", - "kind": "value", - "name": "reduce", - "docstrings": [], - "signature": "let reduce: (('a, elt) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Uint8Array.reducei", - "kind": "value", - "name": "reducei", - "docstrings": [], - "signature": "let reducei: (('a, elt, int) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Uint8Array.reduceRight", - "kind": "value", - "name": "reduceRight", - "docstrings": [], - "signature": "let reduceRight: (('a, elt) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Uint8Array.reduceRighti", - "kind": "value", - "name": "reduceRighti", - "docstrings": [], - "signature": "let reduceRighti: (('a, elt, int) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Uint8Array.some", - "kind": "value", - "name": "some", - "docstrings": [], - "signature": "let some: (elt => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Uint8Array.somei", - "kind": "value", - "name": "somei", - "docstrings": [], - "signature": "let somei: ((elt, int) => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Uint8Array._BYTES_PER_ELEMENT", - "kind": "value", - "name": "_BYTES_PER_ELEMENT", - "docstrings": [], - "signature": "let _BYTES_PER_ELEMENT: int" - }, - { - "id": "Js.Typed_array.Uint8Array.make", - "kind": "value", - "name": "make", - "docstrings": [], - "signature": "let make: array => t" - }, - { - "id": "Js.Typed_array.Uint8Array.fromBuffer", - "kind": "value", - "name": "fromBuffer", - "docstrings": [ - "can throw" - ], - "signature": "let fromBuffer: array_buffer => t" - }, - { - "id": "Js.Typed_array.Uint8Array.fromBufferOffset", - "kind": "value", - "name": "fromBufferOffset", - "docstrings": [ - "**raise** Js.Exn.Error raise Js exception\n\n **param** offset is in bytes" - ], - "signature": "let fromBufferOffset: (array_buffer, int) => t" - }, - { - "id": "Js.Typed_array.Uint8Array.fromBufferRange", - "kind": "value", - "name": "fromBufferRange", - "docstrings": [ - "**raise** Js.Exn.Error raises Js exception\n\n **param** offset is in bytes, length in elements" - ], - "signature": "let fromBufferRange: (array_buffer, ~offset: int, ~length: int) => t" - }, - { - "id": "Js.Typed_array.Uint8Array.fromLength", - "kind": "value", - "name": "fromLength", - "docstrings": [], - "signature": "let fromLength: int => t" - }, - { - "id": "Js.Typed_array.Uint8Array.from", - "kind": "value", - "name": "from", - "docstrings": [], - "signature": "let from: array_like => t" - } - ] - }, - "js/typed_array/int8array": { - "id": "Js.Typed_array.Int8Array", - "name": "Int8Array", - "docstrings": [], - "items": [ - { - "id": "Js.Typed_array.Int8Array.elt", - "kind": "type", - "name": "elt", - "docstrings": [ - "" - ], - "signature": "type elt = int" - }, - { - "id": "Js.Typed_array.Int8Array.typed_array", - "kind": "type", - "name": "typed_array", - "docstrings": [], - "signature": "type typed_array<\n 'a,\n> = Js_typed_array2.Int8Array.typed_array<'a>" - }, - { - "id": "Js.Typed_array.Int8Array.t", - "kind": "type", - "name": "t", - "docstrings": [], - "signature": "type t = typed_array" - }, - { - "id": "Js.Typed_array.Int8Array.unsafe_get", - "kind": "value", - "name": "unsafe_get", - "docstrings": [], - "signature": "let unsafe_get: (t, int) => elt" - }, - { - "id": "Js.Typed_array.Int8Array.unsafe_set", - "kind": "value", - "name": "unsafe_set", - "docstrings": [], - "signature": "let unsafe_set: (t, int, elt) => unit" - }, - { - "id": "Js.Typed_array.Int8Array.buffer", - "kind": "value", - "name": "buffer", - "docstrings": [], - "signature": "let buffer: t => array_buffer" - }, - { - "id": "Js.Typed_array.Int8Array.byteLength", - "kind": "value", - "name": "byteLength", - "docstrings": [], - "signature": "let byteLength: t => int" - }, - { - "id": "Js.Typed_array.Int8Array.byteOffset", - "kind": "value", - "name": "byteOffset", - "docstrings": [], - "signature": "let byteOffset: t => int" - }, - { - "id": "Js.Typed_array.Int8Array.setArray", - "kind": "value", - "name": "setArray", - "docstrings": [], - "signature": "let setArray: (array, t) => unit" - }, - { - "id": "Js.Typed_array.Int8Array.setArrayOffset", - "kind": "value", - "name": "setArrayOffset", - "docstrings": [], - "signature": "let setArrayOffset: (array, int, t) => unit" - }, - { - "id": "Js.Typed_array.Int8Array.length", - "kind": "value", - "name": "length", - "docstrings": [], - "signature": "let length: t => int" - }, - { - "id": "Js.Typed_array.Int8Array.copyWithin", - "kind": "value", - "name": "copyWithin", - "docstrings": [], - "signature": "let copyWithin: (~to_: int, t) => t" - }, - { - "id": "Js.Typed_array.Int8Array.copyWithinFrom", - "kind": "value", - "name": "copyWithinFrom", - "docstrings": [], - "signature": "let copyWithinFrom: (~to_: int, ~from: int, t) => t" - }, - { - "id": "Js.Typed_array.Int8Array.copyWithinFromRange", - "kind": "value", - "name": "copyWithinFromRange", - "docstrings": [], - "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Int8Array.fillInPlace", - "kind": "value", - "name": "fillInPlace", - "docstrings": [], - "signature": "let fillInPlace: (elt, t) => t" - }, - { - "id": "Js.Typed_array.Int8Array.fillFromInPlace", - "kind": "value", - "name": "fillFromInPlace", - "docstrings": [], - "signature": "let fillFromInPlace: (elt, ~from: int, t) => t" - }, - { - "id": "Js.Typed_array.Int8Array.fillRangeInPlace", - "kind": "value", - "name": "fillRangeInPlace", - "docstrings": [], - "signature": "let fillRangeInPlace: (elt, ~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Int8Array.reverseInPlace", - "kind": "value", - "name": "reverseInPlace", - "docstrings": [], - "signature": "let reverseInPlace: t => t" - }, - { - "id": "Js.Typed_array.Int8Array.sortInPlace", - "kind": "value", - "name": "sortInPlace", - "docstrings": [], - "signature": "let sortInPlace: t => t" - }, - { - "id": "Js.Typed_array.Int8Array.sortInPlaceWith", - "kind": "value", - "name": "sortInPlaceWith", - "docstrings": [], - "signature": "let sortInPlaceWith: ((elt, elt) => int, t) => t" - }, - { - "id": "Js.Typed_array.Int8Array.includes", - "kind": "value", - "name": "includes", - "docstrings": [], - "signature": "let includes: (elt, t) => bool" - }, - { - "id": "Js.Typed_array.Int8Array.indexOf", - "kind": "value", - "name": "indexOf", - "docstrings": [], - "signature": "let indexOf: (elt, t) => int" - }, - { - "id": "Js.Typed_array.Int8Array.indexOfFrom", - "kind": "value", - "name": "indexOfFrom", - "docstrings": [], - "signature": "let indexOfFrom: (elt, ~from: int, t) => int" - }, - { - "id": "Js.Typed_array.Int8Array.join", - "kind": "value", - "name": "join", - "docstrings": [], - "signature": "let join: t => string" - }, - { - "id": "Js.Typed_array.Int8Array.joinWith", - "kind": "value", - "name": "joinWith", - "docstrings": [], - "signature": "let joinWith: (string, t) => string" - }, - { - "id": "Js.Typed_array.Int8Array.lastIndexOf", - "kind": "value", - "name": "lastIndexOf", - "docstrings": [], - "signature": "let lastIndexOf: (elt, t) => int" - }, - { - "id": "Js.Typed_array.Int8Array.lastIndexOfFrom", - "kind": "value", - "name": "lastIndexOfFrom", - "docstrings": [], - "signature": "let lastIndexOfFrom: (elt, ~from: int, t) => int" - }, - { - "id": "Js.Typed_array.Int8Array.slice", - "kind": "value", - "name": "slice", - "docstrings": [], - "signature": "let slice: (~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Int8Array.copy", - "kind": "value", - "name": "copy", - "docstrings": [], - "signature": "let copy: t => t" - }, - { - "id": "Js.Typed_array.Int8Array.sliceFrom", - "kind": "value", - "name": "sliceFrom", - "docstrings": [], - "signature": "let sliceFrom: (int, t) => t" - }, - { - "id": "Js.Typed_array.Int8Array.subarray", - "kind": "value", - "name": "subarray", - "docstrings": [], - "signature": "let subarray: (~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.Int8Array.subarrayFrom", - "kind": "value", - "name": "subarrayFrom", - "docstrings": [], - "signature": "let subarrayFrom: (int, t) => t" - }, - { - "id": "Js.Typed_array.Int8Array.toString", - "kind": "value", - "name": "toString", - "docstrings": [], - "signature": "let toString: t => string" - }, - { - "id": "Js.Typed_array.Int8Array.toLocaleString", - "kind": "value", - "name": "toLocaleString", - "docstrings": [], - "signature": "let toLocaleString: t => string" - }, - { - "id": "Js.Typed_array.Int8Array.every", - "kind": "value", - "name": "every", - "docstrings": [], - "signature": "let every: (elt => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Int8Array.everyi", - "kind": "value", - "name": "everyi", - "docstrings": [], - "signature": "let everyi: ((elt, int) => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Int8Array.filter", - "kind": "value", - "name": "filter", - "docstrings": [], - "signature": "let filter: (elt => bool, t) => t" - }, - { - "id": "Js.Typed_array.Int8Array.filteri", - "kind": "value", - "name": "filteri", - "docstrings": [], - "signature": "let filteri: ((elt, int) => bool, t) => t" - }, - { - "id": "Js.Typed_array.Int8Array.find", - "kind": "value", - "name": "find", - "docstrings": [], - "signature": "let find: (elt => bool, t) => Js.undefined" - }, - { - "id": "Js.Typed_array.Int8Array.findi", - "kind": "value", - "name": "findi", - "docstrings": [], - "signature": "let findi: ((elt, int) => bool, t) => Js.undefined" - }, - { - "id": "Js.Typed_array.Int8Array.findIndex", - "kind": "value", - "name": "findIndex", - "docstrings": [], - "signature": "let findIndex: (elt => bool, t) => int" - }, - { - "id": "Js.Typed_array.Int8Array.findIndexi", - "kind": "value", - "name": "findIndexi", - "docstrings": [], - "signature": "let findIndexi: ((elt, int) => bool, t) => int" - }, - { - "id": "Js.Typed_array.Int8Array.forEach", - "kind": "value", - "name": "forEach", - "docstrings": [], - "signature": "let forEach: (elt => unit, t) => unit" - }, - { - "id": "Js.Typed_array.Int8Array.forEachi", - "kind": "value", - "name": "forEachi", - "docstrings": [], - "signature": "let forEachi: ((elt, int) => unit, t) => unit" - }, - { - "id": "Js.Typed_array.Int8Array.map", - "kind": "value", - "name": "map", - "docstrings": [], - "signature": "let map: (elt => 'a, t) => typed_array<'a>" - }, - { - "id": "Js.Typed_array.Int8Array.mapi", - "kind": "value", - "name": "mapi", - "docstrings": [], - "signature": "let mapi: ((elt, int) => 'a, t) => typed_array<'a>" - }, - { - "id": "Js.Typed_array.Int8Array.reduce", - "kind": "value", - "name": "reduce", - "docstrings": [], - "signature": "let reduce: (('a, elt) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Int8Array.reducei", - "kind": "value", - "name": "reducei", - "docstrings": [], - "signature": "let reducei: (('a, elt, int) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Int8Array.reduceRight", - "kind": "value", - "name": "reduceRight", - "docstrings": [], - "signature": "let reduceRight: (('a, elt) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Int8Array.reduceRighti", - "kind": "value", - "name": "reduceRighti", - "docstrings": [], - "signature": "let reduceRighti: (('a, elt, int) => 'a, 'a, t) => 'a" - }, - { - "id": "Js.Typed_array.Int8Array.some", - "kind": "value", - "name": "some", - "docstrings": [], - "signature": "let some: (elt => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Int8Array.somei", - "kind": "value", - "name": "somei", - "docstrings": [], - "signature": "let somei: ((elt, int) => bool, t) => bool" - }, - { - "id": "Js.Typed_array.Int8Array._BYTES_PER_ELEMENT", - "kind": "value", - "name": "_BYTES_PER_ELEMENT", - "docstrings": [], - "signature": "let _BYTES_PER_ELEMENT: int" - }, - { - "id": "Js.Typed_array.Int8Array.make", - "kind": "value", - "name": "make", - "docstrings": [], - "signature": "let make: array => t" - }, - { - "id": "Js.Typed_array.Int8Array.fromBuffer", - "kind": "value", - "name": "fromBuffer", - "docstrings": [ - "can throw" - ], - "signature": "let fromBuffer: array_buffer => t" - }, - { - "id": "Js.Typed_array.Int8Array.fromBufferOffset", - "kind": "value", - "name": "fromBufferOffset", - "docstrings": [ - "raise Js.Exn.Error raise Js exception\n\n param offset is in bytes" - ], - "signature": "let fromBufferOffset: (array_buffer, int) => t" - }, - { - "id": "Js.Typed_array.Int8Array.fromBufferRange", - "kind": "value", - "name": "fromBufferRange", - "docstrings": [ - "raise Js.Exn.Error raises Js exception\n\n param offset is in bytes, length in elements" - ], - "signature": "let fromBufferRange: (array_buffer, ~offset: int, ~length: int) => t" - }, - { - "id": "Js.Typed_array.Int8Array.fromLength", - "kind": "value", - "name": "fromLength", - "docstrings": [], - "signature": "let fromLength: int => t" - }, - { - "id": "Js.Typed_array.Int8Array.from", - "kind": "value", - "name": "from", - "docstrings": [], - "signature": "let from: array_like => t" - } - ] - }, - "js/typed_array/s": { - "id": "Js.Typed_array.S", - "name": "S", - "docstrings": [], - "items": [ - { - "id": "Js.Typed_array.S.elt", - "kind": "type", - "name": "elt", - "docstrings": [], - "signature": "type elt" - }, - { - "id": "Js.Typed_array.S.typed_array", - "kind": "type", - "name": "typed_array", - "docstrings": [], - "signature": "type typed_array<'a>" - }, - { - "id": "Js.Typed_array.S.t", - "kind": "type", - "name": "t", - "docstrings": [], - "signature": "type t = typed_array" - }, - { - "id": "Js.Typed_array.S.unsafe_get", - "kind": "value", - "name": "unsafe_get", - "docstrings": [], - "signature": "let unsafe_get: (t, int) => elt" - }, - { - "id": "Js.Typed_array.S.unsafe_set", - "kind": "value", - "name": "unsafe_set", - "docstrings": [], - "signature": "let unsafe_set: (t, int, elt) => unit" - }, - { - "id": "Js.Typed_array.S.buffer", - "kind": "value", - "name": "buffer", - "docstrings": [], - "signature": "let buffer: t => array_buffer" - }, - { - "id": "Js.Typed_array.S.byteLength", - "kind": "value", - "name": "byteLength", - "docstrings": [], - "signature": "let byteLength: t => int" - }, - { - "id": "Js.Typed_array.S.byteOffset", - "kind": "value", - "name": "byteOffset", - "docstrings": [], - "signature": "let byteOffset: t => int" - }, - { - "id": "Js.Typed_array.S.setArray", - "kind": "value", - "name": "setArray", - "docstrings": [], - "signature": "let setArray: (array, t) => unit" - }, - { - "id": "Js.Typed_array.S.setArrayOffset", - "kind": "value", - "name": "setArrayOffset", - "docstrings": [], - "signature": "let setArrayOffset: (array, int, t) => unit" - }, - { - "id": "Js.Typed_array.S.length", - "kind": "value", - "name": "length", - "docstrings": [], - "signature": "let length: t => int" - }, - { - "id": "Js.Typed_array.S.copyWithin", - "kind": "value", - "name": "copyWithin", - "docstrings": [], - "signature": "let copyWithin: (~to_: int, t) => t" - }, - { - "id": "Js.Typed_array.S.copyWithinFrom", - "kind": "value", - "name": "copyWithinFrom", - "docstrings": [], - "signature": "let copyWithinFrom: (~to_: int, ~from: int, t) => t" - }, - { - "id": "Js.Typed_array.S.copyWithinFromRange", - "kind": "value", - "name": "copyWithinFromRange", - "docstrings": [], - "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.S.fillInPlace", - "kind": "value", - "name": "fillInPlace", - "docstrings": [], - "signature": "let fillInPlace: (elt, t) => t" - }, - { - "id": "Js.Typed_array.S.fillFromInPlace", - "kind": "value", - "name": "fillFromInPlace", - "docstrings": [], - "signature": "let fillFromInPlace: (elt, ~from: int, t) => t" - }, - { - "id": "Js.Typed_array.S.fillRangeInPlace", - "kind": "value", - "name": "fillRangeInPlace", - "docstrings": [], - "signature": "let fillRangeInPlace: (elt, ~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.S.reverseInPlace", - "kind": "value", - "name": "reverseInPlace", - "docstrings": [], - "signature": "let reverseInPlace: t => t" - }, - { - "id": "Js.Typed_array.S.sortInPlace", - "kind": "value", - "name": "sortInPlace", - "docstrings": [], - "signature": "let sortInPlace: t => t" - }, - { - "id": "Js.Typed_array.S.sortInPlaceWith", - "kind": "value", - "name": "sortInPlaceWith", - "docstrings": [], - "signature": "let sortInPlaceWith: ((elt, elt) => int, t) => t" - }, - { - "id": "Js.Typed_array.S.includes", - "kind": "value", - "name": "includes", - "docstrings": [], - "signature": "let includes: (elt, t) => bool" - }, - { - "id": "Js.Typed_array.S.indexOf", - "kind": "value", - "name": "indexOf", - "docstrings": [], - "signature": "let indexOf: (elt, t) => int" - }, - { - "id": "Js.Typed_array.S.indexOfFrom", - "kind": "value", - "name": "indexOfFrom", - "docstrings": [], - "signature": "let indexOfFrom: (elt, ~from: int, t) => int" - }, - { - "id": "Js.Typed_array.S.join", - "kind": "value", - "name": "join", - "docstrings": [], - "signature": "let join: t => string" - }, - { - "id": "Js.Typed_array.S.joinWith", - "kind": "value", - "name": "joinWith", - "docstrings": [], - "signature": "let joinWith: (string, t) => string" - }, - { - "id": "Js.Typed_array.S.lastIndexOf", - "kind": "value", - "name": "lastIndexOf", - "docstrings": [], - "signature": "let lastIndexOf: (elt, t) => int" - }, - { - "id": "Js.Typed_array.S.lastIndexOfFrom", - "kind": "value", - "name": "lastIndexOfFrom", - "docstrings": [], - "signature": "let lastIndexOfFrom: (elt, ~from: int, t) => int" - }, - { - "id": "Js.Typed_array.S.slice", - "kind": "value", - "name": "slice", - "docstrings": [], - "signature": "let slice: (~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.S.copy", - "kind": "value", - "name": "copy", - "docstrings": [], - "signature": "let copy: t => t" - }, - { - "id": "Js.Typed_array.S.sliceFrom", - "kind": "value", - "name": "sliceFrom", - "docstrings": [], - "signature": "let sliceFrom: (int, t) => t" - }, - { - "id": "Js.Typed_array.S.subarray", - "kind": "value", - "name": "subarray", - "docstrings": [], - "signature": "let subarray: (~start: int, ~end_: int, t) => t" - }, - { - "id": "Js.Typed_array.S.subarrayFrom", - "kind": "value", - "name": "subarrayFrom", - "docstrings": [], - "signature": "let subarrayFrom: (int, t) => t" - }, - { - "id": "Js.Typed_array.S.toString", - "kind": "value", - "name": "toString", - "docstrings": [], - "signature": "let toString: t => string" - }, - { - "id": "Js.Typed_array.S.toLocaleString", - "kind": "value", - "name": "toLocaleString", - "docstrings": [], - "signature": "let toLocaleString: t => string" - }, - { - "id": "Js.Typed_array.S.every", - "kind": "value", - "name": "every", - "docstrings": [], - "signature": "let every: (elt => bool, t) => bool" - }, - { - "id": "Js.Typed_array.S.everyi", - "kind": "value", - "name": "everyi", - "docstrings": [], - "signature": "let everyi: ((elt, int) => bool, t) => bool" - }, - { - "id": "Js.Typed_array.S.filter", - "kind": "value", - "name": "filter", - "docstrings": [], - "signature": "let filter: (elt => bool, t) => t" - }, - { - "id": "Js.Typed_array.S.filteri", - "kind": "value", - "name": "filteri", - "docstrings": [], - "signature": "let filteri: ((elt, int) => bool, t) => t" - }, - { - "id": "Js.Typed_array.S.find", - "kind": "value", - "name": "find", - "docstrings": [], - "signature": "let find: (elt => bool, t) => Js.undefined" - }, - { - "id": "Js.Typed_array.S.findi", - "kind": "value", - "name": "findi", - "docstrings": [], - "signature": "let findi: ((elt, int) => bool, t) => Js.undefined" - }, - { - "id": "Js.Typed_array.S.findIndex", - "kind": "value", - "name": "findIndex", - "docstrings": [], - "signature": "let findIndex: (elt => bool, t) => int" - }, - { - "id": "Js.Typed_array.S.findIndexi", - "kind": "value", - "name": "findIndexi", - "docstrings": [], - "signature": "let findIndexi: ((elt, int) => bool, t) => int" - }, - { - "id": "Js.Typed_array.S.forEach", - "kind": "value", - "name": "forEach", - "docstrings": [], - "signature": "let forEach: (elt => unit, t) => unit" - }, - { - "id": "Js.Typed_array.S.forEachi", - "kind": "value", - "name": "forEachi", - "docstrings": [], - "signature": "let forEachi: ((elt, int) => unit, t) => unit" - }, - { - "id": "Js.Typed_array.S.map", - "kind": "value", - "name": "map", - "docstrings": [], - "signature": "let map: (elt => 'b, t) => typed_array<'b>" - }, - { - "id": "Js.Typed_array.S.mapi", - "kind": "value", - "name": "mapi", - "docstrings": [], - "signature": "let mapi: ((elt, int) => 'b, t) => typed_array<'b>" - }, - { - "id": "Js.Typed_array.S.reduce", - "kind": "value", - "name": "reduce", - "docstrings": [], - "signature": "let reduce: (('b, elt) => 'b, 'b, t) => 'b" - }, - { - "id": "Js.Typed_array.S.reducei", - "kind": "value", - "name": "reducei", - "docstrings": [], - "signature": "let reducei: (('b, elt, int) => 'b, 'b, t) => 'b" - }, - { - "id": "Js.Typed_array.S.reduceRight", - "kind": "value", - "name": "reduceRight", - "docstrings": [], - "signature": "let reduceRight: (('b, elt) => 'b, 'b, t) => 'b" - }, - { - "id": "Js.Typed_array.S.reduceRighti", - "kind": "value", - "name": "reduceRighti", - "docstrings": [], - "signature": "let reduceRighti: (('b, elt, int) => 'b, 'b, t) => 'b" - }, - { - "id": "Js.Typed_array.S.some", - "kind": "value", - "name": "some", - "docstrings": [], - "signature": "let some: (elt => bool, t) => bool" - }, - { - "id": "Js.Typed_array.S.somei", - "kind": "value", - "name": "somei", - "docstrings": [], - "signature": "let somei: ((elt, int) => bool, t) => bool" - } - ] - }, - "js/typed_array/arraybuffer": { - "id": "Js.Typed_array.ArrayBuffer", - "name": "ArrayBuffer", - "docstrings": [ - "The underlying buffer that the typed arrays provide views of\n\n **see** [MDN](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/ArrayBuffer)" - ], - "items": [ - { - "id": "Js.Typed_array.ArrayBuffer.t", - "kind": "type", - "name": "t", - "docstrings": [], - "signature": "type t = array_buffer" - }, - { - "id": "Js.Typed_array.ArrayBuffer.make", - "kind": "value", - "name": "make", - "docstrings": [ - "takes length. initializes elements to 0" - ], - "signature": "let make: int => t" - }, - { - "id": "Js.Typed_array.ArrayBuffer.byteLength", - "kind": "value", - "name": "byteLength", - "docstrings": [], - "signature": "let byteLength: t => int" - }, - { - "id": "Js.Typed_array.ArrayBuffer.slice", - "kind": "value", - "name": "slice", - "docstrings": [], - "signature": "let slice: (~start: int, ~end_: int, t) => array_buffer" - }, - { - "id": "Js.Typed_array.ArrayBuffer.sliceFrom", - "kind": "value", - "name": "sliceFrom", - "docstrings": [], - "signature": "let sliceFrom: (int, t) => array_buffer" - } - ] - }, - "js/typed_array/type": { - "id": "Js.Typed_array.Type", - "name": "Type", - "docstrings": [], - "items": [ - { - "id": "Js.Typed_array.Type.t", - "kind": "type", - "name": "t", - "docstrings": [], - "signature": "type t" - } - ] - }, - "js/json/kind": { - "id": "Js.Json.Kind", - "name": "Kind", - "docstrings": [], - "items": [ - { - "id": "Js.Json.Kind.json", - "kind": "type", - "name": "json", - "docstrings": [], - "signature": "type json = t" - }, - { - "id": "Js.Json.Kind.t", - "kind": "type", - "name": "t", - "docstrings": [ - "Underlying type of a JSON value" - ], - "signature": "type t<_> =\n | String: t\n | Number: t\n | Object: t>\n | Array: t>\n | Boolean: t\n | Null: t" - } - ] - }, - "js/weakmap": { - "id": "Js.WeakMap", - "name": "WeakMap", - "docstrings": [ - "Provides bindings for ES6 WeakMap", - "ES6 WeakMap API" - ], - "items": [ - { - "id": "Js.WeakMap.t", - "kind": "type", - "name": "t", - "docstrings": [], - "signature": "type t<'k, 'v>" - } - ] - }, - "js/map": { - "id": "Js.Map", - "name": "Map", - "docstrings": [ - "Provides bindings for ES6 Map", - "ES6 Map API" - ], - "items": [ - { - "id": "Js.Map.t", - "kind": "type", - "name": "t", - "docstrings": [], - "signature": "type t<'k, 'v>" - } - ] - }, - "js/weakset": { - "id": "Js.WeakSet", - "name": "WeakSet", - "docstrings": [ - "Provides bindings for ES6 WeakSet", - "ES6 WeakSet API" - ], - "items": [ - { - "id": "Js.WeakSet.t", - "kind": "type", - "name": "t", - "docstrings": [], - "signature": "type t<'a>" - } - ] - }, - "js/set": { - "id": "Js.Set", - "name": "Set", - "docstrings": [ - "Provides bindings for ES6 Set", - "ES6 Set API" - ], - "items": [ - { - "id": "Js.Set.t", - "kind": "type", - "name": "t", - "docstrings": [], - "signature": "type t<'a>" - } - ] - }, - "js/console": { - "id": "Js.Console", - "name": "Console", - "docstrings": [ - "Provides bindings for console" - ], - "items": [ - { - "id": "Js.Console.log", - "kind": "value", - "name": "log", - "docstrings": [], - "signature": "let log: 'a => unit" - }, - { - "id": "Js.Console.log2", - "kind": "value", - "name": "log2", - "docstrings": [], - "signature": "let log2: ('a, 'b) => unit" - }, - { - "id": "Js.Console.log3", - "kind": "value", - "name": "log3", - "docstrings": [], - "signature": "let log3: ('a, 'b, 'c) => unit" - }, - { - "id": "Js.Console.log4", - "kind": "value", - "name": "log4", - "docstrings": [], - "signature": "let log4: ('a, 'b, 'c, 'd) => unit" - }, - { - "id": "Js.Console.logMany", - "kind": "value", - "name": "logMany", - "docstrings": [], - "signature": "let logMany: array<'a> => unit" - }, - { - "id": "Js.Console.info", - "kind": "value", - "name": "info", - "docstrings": [], - "signature": "let info: 'a => unit" - }, - { - "id": "Js.Console.info2", - "kind": "value", - "name": "info2", - "docstrings": [], - "signature": "let info2: ('a, 'b) => unit" - }, - { - "id": "Js.Console.info3", - "kind": "value", - "name": "info3", - "docstrings": [], - "signature": "let info3: ('a, 'b, 'c) => unit" - }, - { - "id": "Js.Console.info4", - "kind": "value", - "name": "info4", - "docstrings": [], - "signature": "let info4: ('a, 'b, 'c, 'd) => unit" - }, - { - "id": "Js.Console.infoMany", - "kind": "value", - "name": "infoMany", - "docstrings": [], - "signature": "let infoMany: array<'a> => unit" - }, - { - "id": "Js.Console.warn", - "kind": "value", - "name": "warn", - "docstrings": [], - "signature": "let warn: 'a => unit" - }, - { - "id": "Js.Console.warn2", - "kind": "value", - "name": "warn2", - "docstrings": [], - "signature": "let warn2: ('a, 'b) => unit" - }, - { - "id": "Js.Console.warn3", - "kind": "value", - "name": "warn3", - "docstrings": [], - "signature": "let warn3: ('a, 'b, 'c) => unit" - }, - { - "id": "Js.Console.warn4", - "kind": "value", - "name": "warn4", - "docstrings": [], - "signature": "let warn4: ('a, 'b, 'c, 'd) => unit" - }, - { - "id": "Js.Console.warnMany", - "kind": "value", - "name": "warnMany", - "docstrings": [], - "signature": "let warnMany: array<'a> => unit" - }, - { - "id": "Js.Console.error", - "kind": "value", - "name": "error", - "docstrings": [], - "signature": "let error: 'a => unit" - }, - { - "id": "Js.Console.error2", - "kind": "value", - "name": "error2", - "docstrings": [], - "signature": "let error2: ('a, 'b) => unit" - }, - { - "id": "Js.Console.error3", - "kind": "value", - "name": "error3", - "docstrings": [], - "signature": "let error3: ('a, 'b, 'c) => unit" - }, - { - "id": "Js.Console.error4", - "kind": "value", - "name": "error4", - "docstrings": [], - "signature": "let error4: ('a, 'b, 'c, 'd) => unit" - }, - { - "id": "Js.Console.errorMany", - "kind": "value", - "name": "errorMany", - "docstrings": [], - "signature": "let errorMany: array<'a> => unit" - }, - { - "id": "Js.Console.trace", - "kind": "value", - "name": "trace", - "docstrings": [], - "signature": "let trace: unit => unit" - }, - { - "id": "Js.Console.timeStart", - "kind": "value", - "name": "timeStart", - "docstrings": [], - "signature": "let timeStart: string => unit" - }, - { - "id": "Js.Console.timeEnd", - "kind": "value", - "name": "timeEnd", - "docstrings": [], - "signature": "let timeEnd: string => unit" + "signature": "type t<'a>" } ] }, - "js/vector": { - "id": "Js.Vector", - "name": "Vector", + "js/set": { + "id": "Js.Set", + "name": "Set", "docstrings": [ - "Provides bindings for JS Vector" + "Provides bindings for ES6 Set", + "ES6 Set API" ], - "items": [ - { - "id": "Js.Vector.t", - "kind": "type", - "name": "t", - "docstrings": [], - "signature": "type t<'a> = array<'a>" - }, - { - "id": "Js.Vector.filterInPlace", - "kind": "value", - "name": "filterInPlace", - "docstrings": [], - "signature": "let filterInPlace: ('a => bool, t<'a>) => unit" - }, - { - "id": "Js.Vector.empty", - "kind": "value", - "name": "empty", - "docstrings": [], - "signature": "let empty: t<'a> => unit" - }, - { - "id": "Js.Vector.pushBack", - "kind": "value", - "name": "pushBack", - "docstrings": [], - "signature": "let pushBack: ('a, t<'a>) => unit" - }, - { - "id": "Js.Vector.copy", - "kind": "value", - "name": "copy", - "docstrings": [ - "shallow copy" - ], - "signature": "let copy: t<'a> => t<'a>" - }, - { - "id": "Js.Vector.memByRef", - "kind": "value", - "name": "memByRef", - "docstrings": [], - "signature": "let memByRef: ('a, t<'a>) => bool" - }, - { - "id": "Js.Vector.iter", - "kind": "value", - "name": "iter", - "docstrings": [], - "signature": "let iter: ('a => unit, t<'a>) => unit" - }, - { - "id": "Js.Vector.iteri", - "kind": "value", - "name": "iteri", - "docstrings": [], - "signature": "let iteri: ((int, 'a) => unit, t<'a>) => unit" - }, - { - "id": "Js.Vector.toList", - "kind": "value", - "name": "toList", - "docstrings": [], - "signature": "let toList: t<'a> => list<'a>" - }, - { - "id": "Js.Vector.map", - "kind": "value", - "name": "map", - "docstrings": [], - "signature": "let map: ('a => 'b, t<'a>) => t<'b>" - }, - { - "id": "Js.Vector.mapi", - "kind": "value", - "name": "mapi", - "docstrings": [], - "signature": "let mapi: ((int, 'a) => 'b, t<'a>) => t<'b>" - }, - { - "id": "Js.Vector.foldLeft", - "kind": "value", - "name": "foldLeft", - "docstrings": [], - "signature": "let foldLeft: (('a, 'b) => 'a, 'a, t<'b>) => 'a" - }, - { - "id": "Js.Vector.foldRight", - "kind": "value", - "name": "foldRight", - "docstrings": [], - "signature": "let foldRight: (('b, 'a) => 'a, t<'b>, 'a) => 'a" - }, - { - "id": "Js.Vector.length", - "kind": "value", - "name": "length", - "docstrings": [ - "Return the length (number of elements) of the given array." - ], - "signature": "let length: t<'a> => int" - }, - { - "id": "Js.Vector.get", - "kind": "value", - "name": "get", - "docstrings": [ - "`Vector.get(a, n)` returns the element number `n` of vector `a`. The first\nelement has number 0. The last element has number `Vector.length(a) - 1`. You\ncan also write `a[n]` instead of `Vector.get(a, n)`. Raise `Invalid_argument\n\"index out of bounds\"` if `n` is outside the range 0 to (`Array.length(a) -\n1`)." - ], - "signature": "let get: (t<'a>, int) => 'a" - }, - { - "id": "Js.Vector.set", - "kind": "value", - "name": "set", - "docstrings": [ - "`Vector.set(a, n, x)` modifies vector `a` in place, replacing element number\n`n` with `x`. Raise `Invalid_argument \"index out of bounds\"` if `n` is outside\nthe range 0 to `Array.length(a) - 1`." - ], - "signature": "let set: (t<'a>, int, 'a) => unit" - }, - { - "id": "Js.Vector.make", - "kind": "value", - "name": "make", - "docstrings": [ - "`Vector.make(n, x)` returns a fresh vector of length `n`, initialized with `x`.\nAll the elements of this new vector are initially physically equal to `x` (in\nthe sense of the `==` predicate). Consequently, if `x` is mutable, it is shared\namong all elements of the array, and modifying `x` through one of the array\nentries will modify all other entries at the same time. Raise\n`Invalid_argument` if `n < 0` or `n > Sys.max_array_length`. If the value of\n`x` is a floating-point number, then the maximum size is only\n`Sys.max_array_length / 2`." - ], - "signature": "let make: (int, 'a) => t<'a>" - }, - { - "id": "Js.Vector.init", - "kind": "value", - "name": "init", - "docstrings": [ - "Raises `RangeError` when n is negative.\nn : size" - ], - "signature": "let init: (int, int => 'a) => t<'a>" - }, - { - "id": "Js.Vector.append", - "kind": "value", - "name": "append", - "docstrings": [ - "`append(x, a)` returns a fresh vector with `x` appended to `a`." - ], - "signature": "let append: ('a, t<'a>) => t<'a>" - }, - { - "id": "Js.Vector.unsafe_get", - "kind": "value", - "name": "unsafe_get", - "docstrings": [], - "signature": "let unsafe_get: (t<'a>, int) => 'a" - }, + "items": [ { - "id": "Js.Vector.unsafe_set", - "kind": "value", - "name": "unsafe_set", + "id": "Js.Set.t", + "kind": "type", + "name": "t", "docstrings": [], - "signature": "let unsafe_set: (t<'a>, int, 'a) => unit" + "signature": "type t<'a>" } ] }, - "js/list": { - "id": "Js.List", - "name": "List", + "js/console": { + "id": "Js.Console", + "name": "Console", "docstrings": [ - "Provide utilities for list" + "Provides bindings for console" ], "items": [ { - "id": "Js.List.t", - "kind": "type", - "name": "t", + "id": "Js.Console.log", + "kind": "value", + "name": "log", "docstrings": [], - "signature": "type t<'a> = list<'a>" + "signature": "let log: 'a => unit" }, { - "id": "Js.List.length", + "id": "Js.Console.log2", "kind": "value", - "name": "length", + "name": "log2", "docstrings": [], - "signature": "let length: t<'a> => int" + "signature": "let log2: ('a, 'b) => unit" }, { - "id": "Js.List.cons", + "id": "Js.Console.log3", "kind": "value", - "name": "cons", + "name": "log3", "docstrings": [], - "signature": "let cons: ('a, t<'a>) => t<'a>" + "signature": "let log3: ('a, 'b, 'c) => unit" }, { - "id": "Js.List.isEmpty", + "id": "Js.Console.log4", "kind": "value", - "name": "isEmpty", + "name": "log4", "docstrings": [], - "signature": "let isEmpty: t<'a> => bool" + "signature": "let log4: ('a, 'b, 'c, 'd) => unit" }, { - "id": "Js.List.hd", + "id": "Js.Console.logMany", "kind": "value", - "name": "hd", + "name": "logMany", "docstrings": [], - "signature": "let hd: t<'a> => option<'a>" + "signature": "let logMany: array<'a> => unit" }, { - "id": "Js.List.tl", + "id": "Js.Console.info", "kind": "value", - "name": "tl", + "name": "info", "docstrings": [], - "signature": "let tl: t<'a> => option>" + "signature": "let info: 'a => unit" }, { - "id": "Js.List.nth", + "id": "Js.Console.info2", "kind": "value", - "name": "nth", + "name": "info2", "docstrings": [], - "signature": "let nth: (t<'a>, int) => option<'a>" + "signature": "let info2: ('a, 'b) => unit" }, { - "id": "Js.List.revAppend", + "id": "Js.Console.info3", "kind": "value", - "name": "revAppend", + "name": "info3", "docstrings": [], - "signature": "let revAppend: (t<'a>, t<'a>) => t<'a>" + "signature": "let info3: ('a, 'b, 'c) => unit" }, { - "id": "Js.List.rev", + "id": "Js.Console.info4", "kind": "value", - "name": "rev", + "name": "info4", "docstrings": [], - "signature": "let rev: t<'a> => t<'a>" + "signature": "let info4: ('a, 'b, 'c, 'd) => unit" }, { - "id": "Js.List.mapRev", + "id": "Js.Console.infoMany", "kind": "value", - "name": "mapRev", + "name": "infoMany", "docstrings": [], - "signature": "let mapRev: ('a => 'b, t<'a>) => t<'b>" + "signature": "let infoMany: array<'a> => unit" }, { - "id": "Js.List.map", + "id": "Js.Console.warn", "kind": "value", - "name": "map", + "name": "warn", "docstrings": [], - "signature": "let map: ('a => 'b, t<'a>) => t<'b>" + "signature": "let warn: 'a => unit" }, { - "id": "Js.List.iter", + "id": "Js.Console.warn2", "kind": "value", - "name": "iter", + "name": "warn2", "docstrings": [], - "signature": "let iter: ('a => unit, t<'a>) => unit" + "signature": "let warn2: ('a, 'b) => unit" }, { - "id": "Js.List.iteri", + "id": "Js.Console.warn3", "kind": "value", - "name": "iteri", + "name": "warn3", "docstrings": [], - "signature": "let iteri: ((int, 'a) => unit, t<'a>) => unit" + "signature": "let warn3: ('a, 'b, 'c) => unit" }, { - "id": "Js.List.foldLeft", + "id": "Js.Console.warn4", "kind": "value", - "name": "foldLeft", - "docstrings": [ - "Application order is left to right, tail recurisve" - ], - "signature": "let foldLeft: (('a, 'b) => 'a, 'a, list<'b>) => 'a" + "name": "warn4", + "docstrings": [], + "signature": "let warn4: ('a, 'b, 'c, 'd) => unit" + }, + { + "id": "Js.Console.warnMany", + "kind": "value", + "name": "warnMany", + "docstrings": [], + "signature": "let warnMany: array<'a> => unit" }, { - "id": "Js.List.foldRight", + "id": "Js.Console.error", "kind": "value", - "name": "foldRight", - "docstrings": [ - "Application order is right to left tail-recursive." - ], - "signature": "let foldRight: (('a, 'b) => 'b, list<'a>, 'b) => 'b" + "name": "error", + "docstrings": [], + "signature": "let error: 'a => unit" }, { - "id": "Js.List.flatten", + "id": "Js.Console.error2", "kind": "value", - "name": "flatten", + "name": "error2", "docstrings": [], - "signature": "let flatten: t> => t<'a>" + "signature": "let error2: ('a, 'b) => unit" }, { - "id": "Js.List.filter", + "id": "Js.Console.error3", "kind": "value", - "name": "filter", + "name": "error3", "docstrings": [], - "signature": "let filter: ('a => bool, t<'a>) => t<'a>" + "signature": "let error3: ('a, 'b, 'c) => unit" }, { - "id": "Js.List.filterMap", + "id": "Js.Console.error4", "kind": "value", - "name": "filterMap", + "name": "error4", "docstrings": [], - "signature": "let filterMap: ('a => option<'b>, t<'a>) => t<'b>" + "signature": "let error4: ('a, 'b, 'c, 'd) => unit" }, { - "id": "Js.List.countBy", + "id": "Js.Console.errorMany", "kind": "value", - "name": "countBy", + "name": "errorMany", "docstrings": [], - "signature": "let countBy: ('a => bool, list<'a>) => int" + "signature": "let errorMany: array<'a> => unit" }, { - "id": "Js.List.init", + "id": "Js.Console.trace", "kind": "value", - "name": "init", + "name": "trace", "docstrings": [], - "signature": "let init: (int, int => 'a) => t<'a>" + "signature": "let trace: unit => unit" }, { - "id": "Js.List.toVector", + "id": "Js.Console.timeStart", "kind": "value", - "name": "toVector", + "name": "timeStart", "docstrings": [], - "signature": "let toVector: t<'a> => array<'a>" + "signature": "let timeStart: string => unit" }, { - "id": "Js.List.equal", + "id": "Js.Console.timeEnd", "kind": "value", - "name": "equal", + "name": "timeEnd", "docstrings": [], - "signature": "let equal: (('a, 'a) => bool, list<'a>, list<'a>) => bool" + "signature": "let timeEnd: string => unit" } ] }, @@ -10142,15 +7002,6 @@ ], "signature": "type symbol" }, - { - "id": "Js.Types.bigint_val", - "kind": "type", - "name": "bigint_val", - "docstrings": [ - "Js bigint type only available in ES2020" - ], - "signature": "type bigint_val" - }, { "id": "Js.Types.obj_val", "kind": "type", @@ -10188,7 +7039,7 @@ "kind": "type", "name": "t", "docstrings": [], - "signature": "type t<_> =\n | Undefined: t\n | Null: t\n | Boolean: t\n | Number: t\n | String: t\n | Function: t\n | Object: t\n | Symbol: t\n | BigInt: t" + "signature": "type t<_> =\n | Undefined: t\n | Null: t\n | Boolean: t\n | Number: t\n | String: t\n | Function: t\n | Object: t\n | Symbol: t\n | BigInt: t" }, { "id": "Js.Types.test", @@ -10204,7 +7055,7 @@ "kind": "type", "name": "tagged_t", "docstrings": [], - "signature": "type tagged_t =\n | JSFalse\n | JSTrue\n | JSNull\n | JSUndefined\n | JSNumber(float)\n | JSString(string)\n | JSFunction(function_val)\n | JSObject(obj_val)\n | JSSymbol(symbol)\n | JSBigInt(bigint_val)" + "signature": "type tagged_t =\n | JSFalse\n | JSTrue\n | JSNull\n | JSUndefined\n | JSNumber(float)\n | JSString(string)\n | JSFunction(function_val)\n | JSObject(obj_val)\n | JSSymbol(symbol)\n | JSBigInt(bigint)" }, { "id": "Js.Types.classify", @@ -12338,28 +9189,28 @@ "kind": "value", "name": "then_", "docstrings": [], - "signature": "let then_: ('a => promise<'b>, promise<'a>) => promise<'b>" + "signature": "let then_: (promise<'a>, 'a => promise<'b>) => promise<'b>" }, { - "id": "Js.Promise.catch", + "id": "Js.Promise.then_", "kind": "value", - "name": "catch", + "name": "then_", "docstrings": [], - "signature": "let catch: (error => promise<'a>, promise<'a>) => promise<'a>" + "signature": "let then_: ('a => promise<'b>, promise<'a>) => promise<'b>" }, { - "id": "Js.Promise.unsafe_async", + "id": "Js.Promise.catch", "kind": "value", - "name": "unsafe_async", + "name": "catch", "docstrings": [], - "signature": "let unsafe_async: 'a => promise<'a>" + "signature": "let catch: (promise<'a>, error => promise<'a>) => promise<'a>" }, { - "id": "Js.Promise.unsafe_await", + "id": "Js.Promise.catch", "kind": "value", - "name": "unsafe_await", + "name": "catch", "docstrings": [], - "signature": "let unsafe_await: promise<'a> => 'a" + "signature": "let catch: (error => promise<'a>, promise<'a>) => promise<'a>" } ] }, @@ -12429,7 +9280,7 @@ "kind": "value", "name": "fromString", "docstrings": [ - "Constructs a RegExp object (Js.Re.t) from a `string`.\nRegex literals `%re(\"/.../\")` should generally be preferred, but `fromString`\nis useful when you need to dynamically construct a regex using strings,\nexactly like when you do so in JavaScript.\n\n## Examples\n\n```rescript\nlet firstReScriptFileExtension = (filename, content) => {\n let result = Js.Re.fromString(filename ++ \"\\.(res|resi)\")->Js.Re.exec_(content)\n switch result {\n | Some(r) => Js.Nullable.toOption(Js.Re.captures(r)[1])\n | None => None\n }\n}\n\n// outputs \"res\"\nfirstReScriptFileExtension(\"School\", \"School.res School.resi Main.js School.bs.js\")\n```" + "Constructs a RegExp object (Js.Re.t) from a `string`.\nRegex literals `/.../` should generally be preferred, but `fromString`\nis useful when you need to dynamically construct a regex using strings,\nexactly like when you do so in JavaScript.\n\n## Examples\n\n```rescript\nlet firstReScriptFileExtension = (filename, content) => {\n let result = Js.Re.fromString(filename ++ \"\\.(res|resi)\")->Js.Re.exec_(content)\n switch result {\n | Some(r) => Js.Nullable.toOption(Js.Re.captures(r)[1])\n | None => None\n }\n}\n\n// outputs \"res\"\nfirstReScriptFileExtension(\"School\", \"School.res School.resi Main.js School.bs.js\")\n```" ], "signature": "let fromString: string => t" }, @@ -12474,7 +9325,7 @@ "kind": "value", "name": "lastIndex", "docstrings": [ - "Returns the index where the next match will start its search. This property\nwill be modified when the RegExp object is used, if the global (\"g\") flag is\nset.\n\n## Examples\n\n```rescript\nlet re = %re(\"/ab*TODO/g\")\nlet str = \"abbcdefabh\"\n\nlet break = ref(false)\nwhile !break.contents {\n switch Js.Re.exec_(re, str) {\n | Some(result) => Js.Nullable.iter(Js.Re.captures(result)[0], (. match_) => {\n let next = Belt.Int.toString(Js.Re.lastIndex(re))\n Js.log(\"Found \" ++ (match_ ++ (\". Next match starts at \" ++ next)))\n })\n | None => break := true\n }\n}\n```\n\nSee\n[`RegExp: lastIndex`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/RegExp/lastIndex)\non MDN." + "Returns the index where the next match will start its search. This property\nwill be modified when the RegExp object is used, if the global (\"g\") flag is\nset.\n\n## Examples\n\n```rescript\nlet re = /ab*TODO/g\nlet str = \"abbcdefabh\"\n\nlet break = ref(false)\nwhile !break.contents {\n switch Js.Re.exec_(re, str) {\n | Some(result) => Js.Nullable.iter(Js.Re.captures(result)[0], (. match_) => {\n let next = Belt.Int.toString(Js.Re.lastIndex(re))\n Js.log(\"Found \" ++ (match_ ++ (\". Next match starts at \" ++ next)))\n })\n | None => break := true\n }\n}\n```\n\nSee\n[`RegExp: lastIndex`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/RegExp/lastIndex)\non MDN." ], "signature": "let lastIndex: t => int" }, @@ -12528,7 +9379,7 @@ "kind": "value", "name": "exec_", "docstrings": [ - "Executes a search on a given string using the given RegExp object.\nReturns `Some(Js.Re.result)` if a match is found, `None` otherwise.\n\n## Examples\n\n```rescript\n/* Match \"quick brown\" followed by \"jumps\", ignoring characters in between\n * Remember \"brown\" and \"jumps\"\n * Ignore case\n */\n\nlet re = %re(\"/quick\\s(brown).+?(jumps)/ig\")\nlet result = Js.Re.exec_(re, \"The Quick Brown Fox Jumps Over The Lazy Dog\")\n```\n\nSee [`RegExp.prototype.exec()`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/RegExp/exec)\non MDN." + "Executes a search on a given string using the given RegExp object.\nReturns `Some(Js.Re.result)` if a match is found, `None` otherwise.\n\n## Examples\n\n```rescript\n/* Match \"quick brown\" followed by \"jumps\", ignoring characters in between\n * Remember \"brown\" and \"jumps\"\n * Ignore case\n */\n\nlet re = /quick\\s(brown).+?(jumps)/ig\nlet result = Js.Re.exec_(re, \"The Quick Brown Fox Jumps Over The Lazy Dog\")\n```\n\nSee [`RegExp.prototype.exec()`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/RegExp/exec)\non MDN." ], "signature": "let exec_: (t, string) => option" }, @@ -12752,7 +9603,7 @@ "kind": "value", "name": "match_", "docstrings": [ - "`match(str, regexp)` matches a `string` against the given `regexp`. If there is\nno match, it returns `None`. For regular expressions without the g modifier, if\n there is a match, the return value is `Some(array)` where the array contains:\n- The entire matched string\n- Any capture groups if the regexp had parentheses\nFor regular expressions with the g modifier, a matched expression returns\n`Some(array)` with all the matched substrings and no capture groups.\n\nSee [`String.match`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/match)\non MDN.\n\n## Examples\n\n```rescript\nJs.String2.match_(\"The better bats\", %re(\"/b[aeiou]t/\")) == Some([\"bet\"])\nJs.String2.match_(\"The better bats\", %re(\"/b[aeiou]t/g\")) == Some([\"bet\", \"bat\"])\nJs.String2.match_(\"Today is 2018-04-05.\", %re(\"/(\\d+)-(\\d+)-(\\d+)/\")) ==\n Some([\"2018-04-05\", \"2018\", \"04\", \"05\"])\nJs.String2.match_(\"The large container.\", %re(\"/b[aeiou]g/\")) == None\n```" + "`match(str, regexp)` matches a `string` against the given `regexp`. If there is\nno match, it returns `None`. For regular expressions without the g modifier, if\n there is a match, the return value is `Some(array)` where the array contains:\n- The entire matched string\n- Any capture groups if the regexp had parentheses\nFor regular expressions with the g modifier, a matched expression returns\n`Some(array)` with all the matched substrings and no capture groups.\n\nSee [`String.match`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/match)\non MDN.\n\n## Examples\n\n```rescript\nJs.String2.match_(\"The better bats\", /b[aeiou]t/) == Some([\"bet\"])\nJs.String2.match_(\"The better bats\", /b[aeiou]t/g) == Some([\"bet\", \"bat\"])\nJs.String2.match_(\"Today is 2018-04-05.\", /(\\d+)-(\\d+)-(\\d+)/) ==\n Some([\"2018-04-05\", \"2018\", \"04\", \"05\"])\nJs.String2.match_(\"The large container.\", /b[aeiou]g/) == None\n```" ], "signature": "let match_: (t, Js_re.t) => option>>" }, @@ -12797,7 +9648,7 @@ "kind": "value", "name": "replaceByRe", "docstrings": [ - "`replaceByRe(str, regex, replacement)` returns a new `string` where occurrences\nmatching regex have been replaced by `replacement`.\n\nSee [`String.replace`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/replace)\non MDN.\n\n## Examples\n\n```rescript\nJs.String2.replaceByRe(\"vowels be gone\", %re(\"/[aeiou]/g\"), \"x\") == \"vxwxls bx gxnx\"\nJs.String2.replaceByRe(\"Juan Fulano\", %re(\"/(\\w+) (\\w+)/\"), \"$2, $1\") == \"Fulano, Juan\"\n```" + "`replaceByRe(str, regex, replacement)` returns a new `string` where occurrences\nmatching regex have been replaced by `replacement`.\n\nSee [`String.replace`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/replace)\non MDN.\n\n## Examples\n\n```rescript\nJs.String2.replaceByRe(\"vowels be gone\", /[aeiou]/g, \"x\") == \"vxwxls bx gxnx\"\nJs.String2.replaceByRe(\"Juan Fulano\", /(\\w+) (\\w+)/, \"$2, $1\") == \"Fulano, Juan\"\n```" ], "signature": "let replaceByRe: (t, Js_re.t, t) => t" }, @@ -12806,7 +9657,7 @@ "kind": "value", "name": "unsafeReplaceBy0", "docstrings": [ - "Returns a new `string` with some or all matches of a pattern with no capturing\nparentheses replaced by the value returned from the given function. The\nfunction receives as its parameters the matched string, the offset at which the\nmatch begins, and the whole string being matched.\n\nSee [`String.replace`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/replace)\non MDN.\n\n## Examples\n\n```rescript\nlet str = \"beautiful vowels\"\nlet re = %re(\"/[aeiou]/g\")\nlet matchFn = (matchPart, _offset, _wholeString) => Js.String2.toUpperCase(matchPart)\n\nJs.String2.unsafeReplaceBy0(str, re, matchFn) == \"bEAUtIfUl vOwEls\"\n```" + "Returns a new `string` with some or all matches of a pattern with no capturing\nparentheses replaced by the value returned from the given function. The\nfunction receives as its parameters the matched string, the offset at which the\nmatch begins, and the whole string being matched.\n\nSee [`String.replace`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/replace)\non MDN.\n\n## Examples\n\n```rescript\nlet str = \"beautiful vowels\"\nlet re = /[aeiou]/g\nlet matchFn = (matchPart, _offset, _wholeString) => Js.String2.toUpperCase(matchPart)\n\nJs.String2.unsafeReplaceBy0(str, re, matchFn) == \"bEAUtIfUl vOwEls\"\n```" ], "signature": "let unsafeReplaceBy0: (t, Js_re.t, (t, int, t) => t) => t" }, @@ -12815,7 +9666,7 @@ "kind": "value", "name": "unsafeReplaceBy1", "docstrings": [ - "Returns a new `string` with some or all matches of a pattern with one set of\ncapturing parentheses replaced by the value returned from the given function.\nThe function receives as its parameters the matched string, the captured\nstring, the offset at which the match begins, and the whole string being\nmatched.\n\nSee [`String.replace`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/replace)\non MDN.\n\n## Examples\n\n```rescript\nlet str = \"Jony is 40\"\nlet re = %re(\"/(Jony is )\\d+/g\")\nlet matchFn = (_match, part1, _offset, _wholeString) => {\n part1 ++ \"41\"\n}\n\nJs.String2.unsafeReplaceBy1(str, re, matchFn) == \"Jony is 41\"\n```" + "Returns a new `string` with some or all matches of a pattern with one set of\ncapturing parentheses replaced by the value returned from the given function.\nThe function receives as its parameters the matched string, the captured\nstring, the offset at which the match begins, and the whole string being\nmatched.\n\nSee [`String.replace`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/replace)\non MDN.\n\n## Examples\n\n```rescript\nlet str = \"Jony is 40\"\nlet re = /(Jony is )\\d+/g\nlet matchFn = (_match, part1, _offset, _wholeString) => {\n part1 ++ \"41\"\n}\n\nJs.String2.unsafeReplaceBy1(str, re, matchFn) == \"Jony is 41\"\n```" ], "signature": "let unsafeReplaceBy1: (t, Js_re.t, (t, t, int, t) => t) => t" }, @@ -12824,7 +9675,7 @@ "kind": "value", "name": "unsafeReplaceBy2", "docstrings": [ - "Returns a new `string` with some or all matches of a pattern with two sets of\ncapturing parentheses replaced by the value returned from the given function.\nThe function receives as its parameters the matched string, the captured\nstrings, the offset at which the match begins, and the whole string being\nmatched.\n\nSee [`String.replace`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/replace)\non MDN.\n\n## Examples\n\n```rescript\nlet str = \"7 times 6\"\nlet re = %re(\"/(\\d+) times (\\d+)/\")\nlet matchFn = (_match, p1, p2, _offset, _wholeString) => {\n switch (Belt.Int.fromString(p1), Belt.Int.fromString(p2)) {\n | (Some(x), Some(y)) => Belt.Int.toString(x * y)\n | _ => \"???\"\n }\n}\n\nJs.String2.unsafeReplaceBy2(str, re, matchFn) == \"42\"\n```" + "Returns a new `string` with some or all matches of a pattern with two sets of\ncapturing parentheses replaced by the value returned from the given function.\nThe function receives as its parameters the matched string, the captured\nstrings, the offset at which the match begins, and the whole string being\nmatched.\n\nSee [`String.replace`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/replace)\non MDN.\n\n## Examples\n\n```rescript\nlet str = \"7 times 6\"\nlet re = /(\\d+) times (\\d+)/\nlet matchFn = (_match, p1, p2, _offset, _wholeString) => {\n switch (Belt.Int.fromString(p1), Belt.Int.fromString(p2)) {\n | (Some(x), Some(y)) => Belt.Int.toString(x * y)\n | _ => \"???\"\n }\n}\n\nJs.String2.unsafeReplaceBy2(str, re, matchFn) == \"42\"\n```" ], "signature": "let unsafeReplaceBy2: (t, Js_re.t, (t, t, t, int, t) => t) => t" }, @@ -12842,7 +9693,7 @@ "kind": "value", "name": "search", "docstrings": [ - "`search(str, regexp)` returns the starting position of the first match of\n`regexp` in the given `str`, or -1 if there is no match.\n\nSee [`String.search`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/search)\non MDN.\n\n## Examples\n\n```rescript\nJs.String2.search(\"testing 1 2 3\", %re(\"/\\d+/\")) == 8\nJs.String2.search(\"no numbers\", %re(\"/\\d+/\")) == -1\n```" + "`search(str, regexp)` returns the starting position of the first match of\n`regexp` in the given `str`, or -1 if there is no match.\n\nSee [`String.search`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/search)\non MDN.\n\n## Examples\n\n```rescript\nJs.String2.search(\"testing 1 2 3\", /\\d+/) == 8\nJs.String2.search(\"no numbers\", /\\d+/) == -1\n```" ], "signature": "let search: (t, Js_re.t) => int" }, @@ -12887,7 +9738,7 @@ "kind": "value", "name": "splitByRe", "docstrings": [ - "`splitByRe(str, regex)` splits the given `str` at every occurrence of `regex`\nand returns an array of the resulting substrings.\n\nSee [`String.split`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/split)\non MDN.\n\n## Examples\n\n```rescript\nJs.String2.splitByRe(\"art; bed , cog ;dad\", %re(\"/\\s*[,;]\\s*TODO/\")) == [\n Some(\"art\"),\n Some(\"bed\"),\n Some(\"cog\"),\n Some(\"dad\"),\n ]\n```" + "`splitByRe(str, regex)` splits the given `str` at every occurrence of `regex`\nand returns an array of the resulting substrings.\n\nSee [`String.split`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/split)\non MDN.\n\n## Examples\n\n```rescript\nJs.String2.splitByRe(\"art; bed , cog ;dad\", /\\s*[,;]\\s*TODO/) == [\n Some(\"art\"),\n Some(\"bed\"),\n Some(\"cog\"),\n Some(\"dad\"),\n ]\n```" ], "signature": "let splitByRe: (t, Js_re.t) => array>" }, @@ -12896,7 +9747,7 @@ "kind": "value", "name": "splitByReAtMost", "docstrings": [ - "`splitByReAtMost(str, regex, ~limit:n)` splits the given `str` at every\noccurrence of `regex` and returns an array of the first `n` resulting\nsubstrings. If `n` is negative or greater than the number of substrings, the\narray will contain all the substrings.\n\nSee [`String.split`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/split)\non MDN.\n\n## Examples\n\n```rescript\nJs.String2.splitByReAtMost(\"one: two: three: four\", %re(\"/\\s*:\\s*TODO/\"), ~limit=3) == [\n Some(\"one\"),\n Some(\"two\"),\n Some(\"three\"),\n ]\n\nJs.String2.splitByReAtMost(\"one: two: three: four\", %re(\"/\\s*:\\s*TODO/\"), ~limit=0) == []\n\nJs.String2.splitByReAtMost(\"one: two: three: four\", %re(\"/\\s*:\\s*TODO/\"), ~limit=8) == [\n Some(\"one\"),\n Some(\"two\"),\n Some(\"three\"),\n Some(\"four\"),\n ]\n```" + "`splitByReAtMost(str, regex, ~limit:n)` splits the given `str` at every\noccurrence of `regex` and returns an array of the first `n` resulting\nsubstrings. If `n` is negative or greater than the number of substrings, the\narray will contain all the substrings.\n\nSee [`String.split`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/split)\non MDN.\n\n## Examples\n\n```rescript\nJs.String2.splitByReAtMost(\"one: two: three: four\", /\\s*:\\s*TODO/, ~limit=3) == [\n Some(\"one\"),\n Some(\"two\"),\n Some(\"three\"),\n ]\n\nJs.String2.splitByReAtMost(\"one: two: three: four\", /\\s*:\\s*TODO/, ~limit=0) == []\n\nJs.String2.splitByReAtMost(\"one: two: three: four\", /\\s*:\\s*TODO/, ~limit=8) == [\n Some(\"one\"),\n Some(\"two\"),\n Some(\"three\"),\n Some(\"four\"),\n ]\n```" ], "signature": "let splitByReAtMost: (t, Js_re.t, ~limit: int) => array>" }, @@ -13106,6 +9957,15 @@ ], "signature": "let get: (t, int) => t" }, + { + "id": "Js.String.charAt", + "kind": "value", + "name": "charAt", + "docstrings": [ + "`charAt(n, s)` gets the character at index `n` within string `s`. If `n` is\nnegative or greater than the length of `s`, it returns the empty string. If the\nstring contains characters outside the range \\u0000-\\uffff, it will return the\nfirst 16-bit value at that position in the string.\n\nSee [`String.charAt`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/charAt)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.charAt(0, \"Reason\") == \"R\"\nJs.String.charAt(12, \"Reason\") == \"\"\nJs.String.charAt(5, `Rẽasöń`) == `ń`\n```" + ], + "signature": "let charAt: (t, int) => t" + }, { "id": "Js.String.charAt", "kind": "value", @@ -13113,6 +9973,15 @@ "docstrings": [], "signature": "let charAt: (int, t) => t" }, + { + "id": "Js.String.charCodeAt", + "kind": "value", + "name": "charCodeAt", + "docstrings": [ + "`charCodeAt(n, s)` returns the character code at position `n` in string `s`;\nthe result is in the range 0-65535, unlke `codePointAt`, so it will not work\ncorrectly for characters with code points greater than or equal to 0x10000. The\nreturn type is `float` because this function returns NaN if `n` is less than\nzero or greater than the length of the string.\n\nSee [`String.charCodeAt`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/charCodeAt)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.charCodeAt(0, `😺`) == 0xd83d->Belt.Int.toFloat\nJs.String.codePointAt(0, `😺`) == Some(0x1f63a)\n```" + ], + "signature": "let charCodeAt: (t, int) => float" + }, { "id": "Js.String.charCodeAt", "kind": "value", @@ -13120,6 +9989,15 @@ "docstrings": [], "signature": "let charCodeAt: (int, t) => float" }, + { + "id": "Js.String.codePointAt", + "kind": "value", + "name": "codePointAt", + "docstrings": [ + "`codePointAt(n, s)` returns the code point at position `n` within string `s` as\na `Some(value)`. The return value handles code points greater than or equal to\n0x10000. If there is no code point at the given position, the function returns\n`None`.\n\nSee [`String.codePointAt`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/codePointAt)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.codePointAt(1, `¿😺?`) == Some(0x1f63a)\nJs.String.codePointAt(5, \"abc\") == None\n```" + ], + "signature": "let codePointAt: (t, int) => option" + }, { "id": "Js.String.codePointAt", "kind": "value", @@ -13127,6 +10005,15 @@ "docstrings": [], "signature": "let codePointAt: (int, t) => option" }, + { + "id": "Js.String.concat", + "kind": "value", + "name": "concat", + "docstrings": [ + "`concat(append, original)` returns a new `string` with `append` added after\n`original`.\n\nSee [`String.concat`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/concat)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.concat(\"bell\", \"cow\") == \"cowbell\"\n```" + ], + "signature": "let concat: (t, t) => t" + }, { "id": "Js.String.concat", "kind": "value", @@ -13134,6 +10021,15 @@ "docstrings": [], "signature": "let concat: (t, t) => t" }, + { + "id": "Js.String.concatMany", + "kind": "value", + "name": "concatMany", + "docstrings": [ + "`concat(arr, original)` returns a new `string` consisting of each item of an\narray of strings added to the `original` string.\n\nSee [`String.concat`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/concat)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.concatMany([\"2nd\", \"3rd\", \"4th\"], \"1st\") == \"1st2nd3rd4th\"\n```" + ], + "signature": "let concatMany: (t, array) => t" + }, { "id": "Js.String.concatMany", "kind": "value", @@ -13141,6 +10037,15 @@ "docstrings": [], "signature": "let concatMany: (array, t) => t" }, + { + "id": "Js.String.endsWith", + "kind": "value", + "name": "endsWith", + "docstrings": [ + "ES2015: `endsWith(substr, str)` returns `true` if the `str` ends with `substr`,\n`false` otherwise.\n\nSee [`String.endsWith`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/endsWith)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.endsWith(\"Script\", \"ReScript\") == true\nJs.String.endsWith(\"Script\", \"C++\") == false\n```" + ], + "signature": "let endsWith: (t, t) => bool" + }, { "id": "Js.String.endsWith", "kind": "value", @@ -13148,6 +10053,15 @@ "docstrings": [], "signature": "let endsWith: (t, t) => bool" }, + { + "id": "Js.String.endsWithFrom", + "kind": "value", + "name": "endsWithFrom", + "docstrings": [ + "`endsWithFrom(ending, len, str)` returns `true` if the first len characters of\n`str` end with `ending`, `false` otherwise. If `len` is greater than or equal\nto the length of `str`, then it works like `endsWith`. (Honestly, this should\nhave been named endsWithAt, but oh well.)\n\nSee [`String.endsWith`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/endsWith)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.endsWithFrom(\"cd\", 4, \"abcd\") == true\nJs.String.endsWithFrom(\"cd\", 3, \"abcde\") == false\nJs.String.endsWithFrom(\"cde\", 99, \"abcde\") == true\nJs.String.endsWithFrom(\"ple\", 7, \"example.dat\") == true\n```" + ], + "signature": "let endsWithFrom: (t, t, int) => bool" + }, { "id": "Js.String.endsWithFrom", "kind": "value", @@ -13155,6 +10069,15 @@ "docstrings": [], "signature": "let endsWithFrom: (t, int, t) => bool" }, + { + "id": "Js.String.includes", + "kind": "value", + "name": "includes", + "docstrings": [ + "ES2015: `includes(searchValue, str)` returns `true` if `searchValue` is found\nanywhere within `str`, false otherwise.\n\nSee [`String.includes`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/includes)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.includes(\"gram\", \"programmer\") == true\nJs.String.includes(\"er\", \"programmer\") == true\nJs.String.includes(\"pro\", \"programmer\") == true\nJs.String.includes(\"xyz\", \"programmer.dat\") == false\n```" + ], + "signature": "let includes: (t, t) => bool" + }, { "id": "Js.String.includes", "kind": "value", @@ -13162,6 +10085,15 @@ "docstrings": [], "signature": "let includes: (t, t) => bool" }, + { + "id": "Js.String.includesFrom", + "kind": "value", + "name": "includesFrom", + "docstrings": [ + "ES2015: `includes(searchValue start, str)` returns `true` if `searchValue` is\nfound anywhere within `str` starting at character number `start` (where 0 is\nthe first character), `false` otherwise.\n\nSee [`String.includes`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/includes)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.includesFrom(\"gram\", 1, \"programmer\") == true\nJs.String.includesFrom(\"gram\", 4, \"programmer\") == false\nJs.String.includesFrom(`한`, 1, `대한민국`) == true\n```" + ], + "signature": "let includesFrom: (t, t, int) => bool" + }, { "id": "Js.String.includesFrom", "kind": "value", @@ -13169,6 +10101,15 @@ "docstrings": [], "signature": "let includesFrom: (t, int, t) => bool" }, + { + "id": "Js.String.indexOf", + "kind": "value", + "name": "indexOf", + "docstrings": [ + "ES2015: `indexOf(searchValue, str)` returns the position at which `searchValue`\nwas first found within `str`, or -1 if `searchValue` is not in `str`.\n\nSee [`String.indexOf`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/indexOf)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.indexOf(\"ok\", \"bookseller\") == 2\nJs.String.indexOf(\"sell\", \"bookseller\") == 4\nJs.String.indexOf(\"ee\", \"beekeeper\") == 1\nJs.String.indexOf(\"xyz\", \"bookseller\") == -1\n```" + ], + "signature": "let indexOf: (t, t) => int" + }, { "id": "Js.String.indexOf", "kind": "value", @@ -13176,6 +10117,15 @@ "docstrings": [], "signature": "let indexOf: (t, t) => int" }, + { + "id": "Js.String.indexOfFrom", + "kind": "value", + "name": "indexOfFrom", + "docstrings": [ + "`indexOfFrom(searchValue, start, str)` returns the position at which\n`searchValue` was found within `str` starting at character position `start`, or\n-1 if `searchValue` is not found in that portion of `str`. The return value is\nrelative to the beginning of the string, no matter where the search started\nfrom.\n\nSee [`String.indexOf`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/indexOf)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.indexOfFrom(\"ok\", 1, \"bookseller\") == 2\nJs.String.indexOfFrom(\"sell\", 2, \"bookseller\") == 4\nJs.String.indexOfFrom(\"sell\", 5, \"bookseller\") == -1\n```" + ], + "signature": "let indexOfFrom: (t, t, int) => int" + }, { "id": "Js.String.indexOfFrom", "kind": "value", @@ -13183,6 +10133,15 @@ "docstrings": [], "signature": "let indexOfFrom: (t, int, t) => int" }, + { + "id": "Js.String.lastIndexOf", + "kind": "value", + "name": "lastIndexOf", + "docstrings": [ + "`lastIndexOf(searchValue, str)` returns the position of the last occurrence of\n`searchValue` within `str`, searching backwards from the end of the string.\nReturns -1 if `searchValue` is not in `str`. The return value is always\nrelative to the beginning of the string.\n\nSee [`String.lastIndexOf`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/lastIndexOf)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.lastIndexOf(\"ok\", \"bookseller\") == 2\nJs.String.lastIndexOf(\"ee\", \"beekeeper\") == 4\nJs.String.lastIndexOf(\"xyz\", \"abcdefg\") == -1\n```" + ], + "signature": "let lastIndexOf: (t, t) => int" + }, { "id": "Js.String.lastIndexOf", "kind": "value", @@ -13190,6 +10149,15 @@ "docstrings": [], "signature": "let lastIndexOf: (t, t) => int" }, + { + "id": "Js.String.lastIndexOfFrom", + "kind": "value", + "name": "lastIndexOfFrom", + "docstrings": [ + "`lastIndexOfFrom(searchValue, start, str)` returns the position of the last\noccurrence of `searchValue` within `str`, searching backwards from the given\nstart position. Returns -1 if `searchValue` is not in `str`. The return value\nis always relative to the beginning of the string.\n\nSee [`String.lastIndexOf`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/lastIndexOf)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.lastIndexOfFrom(\"ok\", 6, \"bookseller\") == 2\nJs.String.lastIndexOfFrom(\"ee\", 8, \"beekeeper\") == 4\nJs.String.lastIndexOfFrom(\"ee\", 3, \"beekeeper\") == 1\nJs.String.lastIndexOfFrom(\"xyz\", 4, \"abcdefg\") == -1\n```" + ], + "signature": "let lastIndexOfFrom: (t, t, int) => int" + }, { "id": "Js.String.lastIndexOfFrom", "kind": "value", @@ -13197,6 +10165,15 @@ "docstrings": [], "signature": "let lastIndexOfFrom: (t, int, t) => int" }, + { + "id": "Js.String.localeCompare", + "kind": "value", + "name": "localeCompare", + "docstrings": [ + "`localeCompare(comparison, reference)` returns\n- a negative value if reference comes before comparison in sort order\n- zero if reference and comparison have the same sort order\n- a positive value if reference comes after comparison in sort order\n\nSee [`String.localeCompare`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/localeCompare) on MDN.\n\n## Examples\n\n```rescript\nJs.String.localeCompare(\"ant\", \"zebra\") > 0.0\nJs.String.localeCompare(\"zebra\", \"ant\") < 0.0\nJs.String.localeCompare(\"cat\", \"cat\") == 0.0\nJs.String.localeCompare(\"cat\", \"CAT\") > 0.0\n```" + ], + "signature": "let localeCompare: (t, t) => float" + }, { "id": "Js.String.localeCompare", "kind": "value", @@ -13204,6 +10181,15 @@ "docstrings": [], "signature": "let localeCompare: (t, t) => float" }, + { + "id": "Js.String.match_", + "kind": "value", + "name": "match_", + "docstrings": [ + "`match(regexp, str)` matches a `string` against the given `regexp`. If there is\nno match, it returns `None`. For regular expressions without the g modifier, if\n there is a match, the return value is `Some(array)` where the array contains:\n- The entire matched string\n- Any capture groups if the regexp had parentheses\n\nFor regular expressions with the g modifier, a matched expression returns\n`Some(array)` with all the matched substrings and no capture groups.\n\nSee [`String.match`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/match)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.match_(/b[aeiou]t/, \"The better bats\") == Some([\"bet\"])\nJs.String.match_(/b[aeiou]t/g, \"The better bats\") == Some([\"bet\", \"bat\"])\nJs.String.match_(/(\\d+)-(\\d+)-(\\d+)/, \"Today is 2018-04-05.\") ==\n Some([\"2018-04-05\", \"2018\", \"04\", \"05\"])\nJs.String.match_(/b[aeiou]g/, \"The large container.\") == None\n```" + ], + "signature": "let match_: (t, Js_re.t) => option>>" + }, { "id": "Js.String.match_", "kind": "value", @@ -13223,9 +10209,27 @@ { "id": "Js.String.normalizeByForm", "kind": "value", - "name": "normalizeByForm", - "docstrings": [], - "signature": "let normalizeByForm: (t, t) => t" + "name": "normalizeByForm", + "docstrings": [ + "ES2015: `normalize(form, str)` returns the normalized Unicode string using the specified form of normalization, which may be one of:\n- \"NFC\" — Normalization Form Canonical Composition.\n- \"NFD\" — Normalization Form Canonical Decomposition.\n- \"NFKC\" — Normalization Form Compatibility Composition.\n- \"NFKD\" — Normalization Form Compatibility Decomposition.\n\nSee [`String.normalize`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/normalize) on MDN.\n\nSee also [Unicode technical report #15](https://unicode.org/reports/tr15/) for details." + ], + "signature": "let normalizeByForm: (t, t) => t" + }, + { + "id": "Js.String.normalizeByForm", + "kind": "value", + "name": "normalizeByForm", + "docstrings": [], + "signature": "let normalizeByForm: (t, t) => t" + }, + { + "id": "Js.String.repeat", + "kind": "value", + "name": "repeat", + "docstrings": [ + "`repeat(n, str)` returns a `string` that consists of `n` repetitions of `str`.\nRaises `RangeError` if `n` is negative.\n\nSee [`String.repeat`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/repeat)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.repeat(3, \"ha\") == \"hahaha\"\nJs.String.repeat(0, \"empty\") == \"\"\n```" + ], + "signature": "let repeat: (t, int) => t" }, { "id": "Js.String.repeat", @@ -13234,6 +10238,15 @@ "docstrings": [], "signature": "let repeat: (int, t) => t" }, + { + "id": "Js.String.replace", + "kind": "value", + "name": "replace", + "docstrings": [ + "ES2015: `replace(substr, newSubstr, str)` returns a new `string` which is\nidentical to `str` except with the first matching instance of `substr` replaced\nby `newSubstr`. `substr` is treated as a verbatim string to match, not a\nregular expression.\n\nSee [`String.replace`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/replace)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.replace(\"old\", \"new\", \"old string\") == \"new string\"\nJs.String.replace(\"the\", \"this\", \"the cat and the dog\") == \"this cat and the dog\"\n```" + ], + "signature": "let replace: (t, t, t) => t" + }, { "id": "Js.String.replace", "kind": "value", @@ -13241,6 +10254,15 @@ "docstrings": [], "signature": "let replace: (t, t, t) => t" }, + { + "id": "Js.String.replaceByRe", + "kind": "value", + "name": "replaceByRe", + "docstrings": [ + "`replaceByRe(regex, replacement, str)` returns a new `string` where occurrences\nmatching regex have been replaced by `replacement`.\n\nSee [`String.replace`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/replace)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.replaceByRe(/[aeiou]/g, \"x\", \"vowels be gone\") == \"vxwxls bx gxnx\"\nJs.String.replaceByRe(/(\\w+) (\\w+)/, \"$2, $1\", \"Juan Fulano\") == \"Fulano, Juan\"\n```" + ], + "signature": "let replaceByRe: (t, Js_re.t, t) => t" + }, { "id": "Js.String.replaceByRe", "kind": "value", @@ -13248,6 +10270,15 @@ "docstrings": [], "signature": "let replaceByRe: (Js_re.t, t, t) => t" }, + { + "id": "Js.String.unsafeReplaceBy0", + "kind": "value", + "name": "unsafeReplaceBy0", + "docstrings": [ + "Returns a new `string` with some or all matches of a pattern with no capturing\nparentheses replaced by the value returned from the given function. The\nfunction receives as its parameters the matched string, the offset at which the\nmatch begins, and the whole string being matched.\n\nSee [`String.replace`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/replace)\non MDN.\n\n## Examples\n\n```rescript\nlet str = \"beautiful vowels\"\nlet re = /[aeiou]/g\nlet matchFn = (matchPart, _offset, _wholeString) => Js.String.toUpperCase(matchPart)\n\nJs.String.unsafeReplaceBy0(re, matchFn, str) == \"bEAUtIfUl vOwEls\"\n```" + ], + "signature": "let unsafeReplaceBy0: (t, Js_re.t, (t, int, t) => t) => t" + }, { "id": "Js.String.unsafeReplaceBy0", "kind": "value", @@ -13255,6 +10286,15 @@ "docstrings": [], "signature": "let unsafeReplaceBy0: (Js_re.t, (t, int, t) => t, t) => t" }, + { + "id": "Js.String.unsafeReplaceBy1", + "kind": "value", + "name": "unsafeReplaceBy1", + "docstrings": [ + "Returns a new `string` with some or all matches of a pattern with one set of\ncapturing parentheses replaced by the value returned from the given function.\nThe function receives as its parameters the matched string, the captured\nstring, the offset at which the match begins, and the whole string being\nmatched.\n\nSee [`String.replace`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/replace)\non MDN.\n\n## Examples\n\n```rescript\nlet str = \"Jony is 40\"\nlet re = /(Jony is )\\d+/g\nlet matchFn = (_match, part1, _offset, _wholeString) => {\n part1 ++ \"41\"\n}\n\nJs.String.unsafeReplaceBy1(re, matchFn, str) == \"Jony is 41\"\n```" + ], + "signature": "let unsafeReplaceBy1: (t, Js_re.t, (t, t, int, t) => t) => t" + }, { "id": "Js.String.unsafeReplaceBy1", "kind": "value", @@ -13262,6 +10302,15 @@ "docstrings": [], "signature": "let unsafeReplaceBy1: (Js_re.t, (t, t, int, t) => t, t) => t" }, + { + "id": "Js.String.unsafeReplaceBy2", + "kind": "value", + "name": "unsafeReplaceBy2", + "docstrings": [ + "Returns a new `string` with some or all matches of a pattern with two sets of\ncapturing parentheses replaced by the value returned from the given function.\nThe function receives as its parameters the matched string, the captured\nstrings, the offset at which the match begins, and the whole string being\nmatched.\n\nSee [`String.replace`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/replace)\non MDN.\n\n## Examples\n\n```rescript\nlet str = \"7 times 6\"\nlet re = /(\\d+) times (\\d+)/\nlet matchFn = (_match, p1, p2, _offset, _wholeString) => {\n switch (Belt.Int.fromString(p1), Belt.Int.fromString(p2)) {\n | (Some(x), Some(y)) => Belt.Int.toString(x * y)\n | _ => \"???\"\n }\n}\n\nJs.String.unsafeReplaceBy2(re, matchFn, str) == \"42\"\n```" + ], + "signature": "let unsafeReplaceBy2: (t, Js_re.t, (t, t, t, int, t) => t) => t" + }, { "id": "Js.String.unsafeReplaceBy2", "kind": "value", @@ -13269,6 +10318,15 @@ "docstrings": [], "signature": "let unsafeReplaceBy2: (Js_re.t, (t, t, t, int, t) => t, t) => t" }, + { + "id": "Js.String.unsafeReplaceBy3", + "kind": "value", + "name": "unsafeReplaceBy3", + "docstrings": [ + "Returns a new `string` with some or all matches of a pattern with three sets of\ncapturing parentheses replaced by the value returned from the given function.\nThe function receives as its parameters the matched string, the captured\nstrings, the offset at which the match begins, and the whole string being\nmatched.\n\nSee [`String.replace`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/replace)\non MDN." + ], + "signature": "let unsafeReplaceBy3: (t, Js_re.t, (t, t, t, t, int, t) => t) => t" + }, { "id": "Js.String.unsafeReplaceBy3", "kind": "value", @@ -13276,6 +10334,15 @@ "docstrings": [], "signature": "let unsafeReplaceBy3: (Js_re.t, (t, t, t, t, int, t) => t, t) => t" }, + { + "id": "Js.String.search", + "kind": "value", + "name": "search", + "docstrings": [ + "`search(regexp, str)` returns the starting position of the first match of\n`regexp` in the given `str`, or -1 if there is no match.\n\nSee [`String.search`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/search)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.search(/\\d+/, \"testing 1 2 3\") == 8\nJs.String.search(/\\d+/, \"no numbers\") == -1\n```" + ], + "signature": "let search: (t, Js_re.t) => int" + }, { "id": "Js.String.search", "kind": "value", @@ -13283,6 +10350,15 @@ "docstrings": [], "signature": "let search: (Js_re.t, t) => int" }, + { + "id": "Js.String.slice", + "kind": "value", + "name": "slice", + "docstrings": [ + "`slice(from:n1, to_:n2, str)` returns the substring of `str` starting at\ncharacter `n1` up to but not including `n2`.\n- If either `n1` or `n2` is negative, then it is evaluated as `length(str - n1)` or `length(str - n2)`.\n- If `n2` is greater than the length of `str`, then it is treated as `length(str)`.\n- If `n1` is greater than `n2`, slice returns the empty string.\n\nSee [`String.slice`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/slice) on MDN.\n\n## Examples\n\n```rescript\nJs.String.slice(~from=2, ~to_=5, \"abcdefg\") == \"cde\"\nJs.String.slice(~from=2, ~to_=9, \"abcdefg\") == \"cdefg\"\nJs.String.slice(~from=-4, ~to_=-2, \"abcdefg\") == \"de\"\nJs.String.slice(~from=5, ~to_=1, \"abcdefg\") == \"\"\n```" + ], + "signature": "let slice: (t, ~from: int, ~to_: int) => t" + }, { "id": "Js.String.slice", "kind": "value", @@ -13290,6 +10366,15 @@ "docstrings": [], "signature": "let slice: (~from: int, ~to_: int, t) => t" }, + { + "id": "Js.String.sliceToEnd", + "kind": "value", + "name": "sliceToEnd", + "docstrings": [ + "`sliceToEnd(str, from:n)` returns the substring of `str` starting at character\n`n` to the end of the string.\n- If `n` is negative, then it is evaluated as `length(str - n)`.\n- If `n` is greater than the length of `str`, then sliceToEnd returns the empty string.\n\nSee [`String.slice`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/slice) on MDN.\n\n## Examples\n\n```rescript\nJs.String.sliceToEnd(~from=4, \"abcdefg\") == \"efg\"\nJs.String.sliceToEnd(~from=-2, \"abcdefg\") == \"fg\"\nJs.String.sliceToEnd(~from=7, \"abcdefg\") == \"\"\n```" + ], + "signature": "let sliceToEnd: (t, ~from: int) => t" + }, { "id": "Js.String.sliceToEnd", "kind": "value", @@ -13297,6 +10382,15 @@ "docstrings": [], "signature": "let sliceToEnd: (~from: int, t) => t" }, + { + "id": "Js.String.split", + "kind": "value", + "name": "split", + "docstrings": [ + "`split(delimiter, str)` splits the given `str` at every occurrence of\n`delimiter` and returns an array of the resulting substrings.\n\nSee [`String.split`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/split)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.split(\"-\", \"2018-01-02\") == [\"2018\", \"01\", \"02\"]\nJs.String.split(\",\", \"a,b,,c\") == [\"a\", \"b\", \"\", \"c\"]\nJs.String.split(\"::\", \"good::bad as great::awful\") == [\"good\", \"bad as great\", \"awful\"]\nJs.String.split(\";\", \"has-no-delimiter\") == [\"has-no-delimiter\"]\n```" + ], + "signature": "let split: (t, t) => array" + }, { "id": "Js.String.split", "kind": "value", @@ -13304,6 +10398,15 @@ "docstrings": [], "signature": "let split: (t, t) => array" }, + { + "id": "Js.String.splitAtMost", + "kind": "value", + "name": "splitAtMost", + "docstrings": [ + "`splitAtMost(delimiter, ~limit:n, str)` splits the given `str` at every\noccurrence of `delimiter` and returns an array of the first `n` resulting\nsubstrings. If `n` is negative or greater than the number of substrings, the\narray will contain all the substrings.\n\nSee [`String.split`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/split)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.splitAtMost(\"/\", ~limit=3, \"ant/bee/cat/dog/elk\") == [\"ant\", \"bee\", \"cat\"]\nJs.String.splitAtMost(\"/\", ~limit=0, \"ant/bee/cat/dog/elk\") == []\nJs.String.splitAtMost(\"/\", ~limit=9, \"ant/bee/cat/dog/elk\") == [\"ant\", \"bee\", \"cat\", \"dog\", \"elk\"]\n```" + ], + "signature": "let splitAtMost: (t, t, ~limit: int) => array" + }, { "id": "Js.String.splitAtMost", "kind": "value", @@ -13311,6 +10414,15 @@ "docstrings": [], "signature": "let splitAtMost: (t, ~limit: int, t) => array" }, + { + "id": "Js.String.splitByRe", + "kind": "value", + "name": "splitByRe", + "docstrings": [ + "`splitByRe(regex, str)` splits the given `str` at every occurrence of `regex`\nand returns an array of the resulting substrings.\n\nSee [`String.split`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/split)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.splitByRe(%re(\"/\\s*[,;]\\s*TODO/\"), \"art; bed , cog ;dad\") == [\n Some(\"art\"),\n Some(\"bed\"),\n Some(\"cog\"),\n Some(\"dad\"),\n ]\n```" + ], + "signature": "let splitByRe: (t, Js_re.t) => array>" + }, { "id": "Js.String.splitByRe", "kind": "value", @@ -13318,6 +10430,15 @@ "docstrings": [], "signature": "let splitByRe: (Js_re.t, t) => array>" }, + { + "id": "Js.String.splitByReAtMost", + "kind": "value", + "name": "splitByReAtMost", + "docstrings": [ + "`splitByReAtMost(regex, ~limit:n, str)` splits the given `str` at every\noccurrence of `regex` and returns an array of the first `n` resulting\nsubstrings. If `n` is negative or greater than the number of substrings, the\narray will contain all the substrings.\n\nSee [`String.split`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/split)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.splitByReAtMost(%re(\"/\\s*[,;]\\s*TODO/\"), ~limit=3, \"one: two: three: four\") == [\n Some(\"one\"),\n Some(\"two\"),\n Some(\"three\"),\n ]\n\nJs.String.splitByReAtMost(%re(\"/\\s*[,;]\\s*TODO/\"), ~limit=0, \"one: two: three: four\") == []\n\nJs.String.splitByReAtMost(%re(\"/\\s*[,;]\\s*TODO/\"), ~limit=8, \"one: two: three: four\") == [\n Some(\"one\"),\n Some(\"two\"),\n Some(\"three\"),\n Some(\"four\"),\n ]\n```" + ], + "signature": "let splitByReAtMost: (t, Js_re.t, ~limit: int) => array>" + }, { "id": "Js.String.splitByReAtMost", "kind": "value", @@ -13325,6 +10446,15 @@ "docstrings": [], "signature": "let splitByReAtMost: (Js_re.t, ~limit: int, t) => array>" }, + { + "id": "Js.String.startsWith", + "kind": "value", + "name": "startsWith", + "docstrings": [ + "ES2015: `startsWith(substr, str)` returns `true` if the `str` starts with\n`substr`, `false` otherwise.\n\nSee [`String.startsWith`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/startsWith)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.startsWith(\"Re\", \"ReScript\") == true\nJs.String.startsWith(\"\", \"ReScript\") == true\nJs.String.startsWith(\"Re\", \"JavaScript\") == false\n```" + ], + "signature": "let startsWith: (t, t) => bool" + }, { "id": "Js.String.startsWith", "kind": "value", @@ -13332,6 +10462,15 @@ "docstrings": [], "signature": "let startsWith: (t, t) => bool" }, + { + "id": "Js.String.startsWithFrom", + "kind": "value", + "name": "startsWithFrom", + "docstrings": [ + "ES2015: `startsWithFrom(substr, n, str)` returns `true` if the `str` starts\nwith `substr` starting at position `n`, false otherwise. If `n` is negative,\nthe search starts at the beginning of `str`.\n\nSee [`String.startsWith`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/startsWith)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.startsWithFrom(\"Scri\", 2, \"ReScript\") == true\nJs.String.startsWithFrom(\"\", 2, \"ReScript\") == true\nJs.String.startsWithFrom(\"Scri\", 2, \"JavaScript\") == false\n```" + ], + "signature": "let startsWithFrom: (t, t, int) => bool" + }, { "id": "Js.String.startsWithFrom", "kind": "value", @@ -13339,6 +10478,15 @@ "docstrings": [], "signature": "let startsWithFrom: (t, int, t) => bool" }, + { + "id": "Js.String.substr", + "kind": "value", + "name": "substr", + "docstrings": [ + "`substr(~from:n, str)` returns the substring of `str` from position `n` to the\nend of the string.\n- If `n` is less than zero, the starting position is the length of `str - n`.\n- If `n` is greater than or equal to the length of `str`, returns the empty string.\n\nJavaScript’s `String.substr()` is a legacy function. When possible, use\n`substring()` instead.\n\nSee [`String.substr`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/substr)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.substr(~from=3, \"abcdefghij\") == \"defghij\"\nJs.String.substr(~from=-3, \"abcdefghij\") == \"hij\"\nJs.String.substr(~from=12, \"abcdefghij\") == \"\"\n```" + ], + "signature": "let substr: (t, ~from: int) => t" + }, { "id": "Js.String.substr", "kind": "value", @@ -13346,6 +10494,15 @@ "docstrings": [], "signature": "let substr: (~from: int, t) => t" }, + { + "id": "Js.String.substrAtMost", + "kind": "value", + "name": "substrAtMost", + "docstrings": [ + "`substrAtMost(~from: pos, ~length: n, str)` returns the substring of `str` of\nlength `n` starting at position `pos`.\n- If `pos` is less than zero, the starting position is the length of `str - pos`.\n- If `pos` is greater than or equal to the length of `str`, returns the empty string.\n- If `n` is less than or equal to zero, returns the empty string.\n\nJavaScript’s `String.substr()` is a legacy function. When possible, use\n`substring()` instead.\n\nSee [`String.substr`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/substr)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.substrAtMost(~from=3, ~length=4, \"abcdefghij\") == \"defg\"\nJs.String.substrAtMost(~from=-3, ~length=4, \"abcdefghij\") == \"hij\"\nJs.String.substrAtMost(~from=12, ~length=2, \"abcdefghij\") == \"\"\n```" + ], + "signature": "let substrAtMost: (t, ~from: int, ~length: int) => t" + }, { "id": "Js.String.substrAtMost", "kind": "value", @@ -13353,6 +10510,15 @@ "docstrings": [], "signature": "let substrAtMost: (~from: int, ~length: int, t) => t" }, + { + "id": "Js.String.substring", + "kind": "value", + "name": "substring", + "docstrings": [ + "`substring(~from: start, ~to_: finish, str)` returns characters `start` up to\nbut not including finish from `str`.\n- If `start` is less than zero, it is treated as zero.\n- If `finish` is zero or negative, the empty string is returned.\n- If `start` is greater than `finish`, the `start` and `finish` points are swapped.\n\nSee [`String.substring`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/substring) on MDN.\n\n## Examples\n\n```rescript\nJs.String.substring(~from=3, ~to_=6, \"playground\") == \"ygr\"\nJs.String.substring(~from=6, ~to_=3, \"playground\") == \"ygr\"\nJs.String.substring(~from=4, ~to_=12, \"playground\") == \"ground\"\n```" + ], + "signature": "let substring: (t, ~from: int, ~to_: int) => t" + }, { "id": "Js.String.substring", "kind": "value", @@ -13360,6 +10526,15 @@ "docstrings": [], "signature": "let substring: (~from: int, ~to_: int, t) => t" }, + { + "id": "Js.String.substringToEnd", + "kind": "value", + "name": "substringToEnd", + "docstrings": [ + "`substringToEnd(~from: start, str)` returns the substring of `str` from\nposition `start` to the end.\n- If `start` is less than or equal to zero, the entire string is returned.\n- If `start` is greater than or equal to the length of `str`, the empty string is returned.\n\nSee [`String.substring`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/substring) on MDN.\n\n## Examples\n\n```rescript\nJs.String.substringToEnd(~from=4, \"playground\") == \"ground\"\nJs.String.substringToEnd(~from=-3, \"playground\") == \"playground\"\nJs.String.substringToEnd(~from=12, \"playground\") == \"\"\n```" + ], + "signature": "let substringToEnd: (t, ~from: int) => t" + }, { "id": "Js.String.substringToEnd", "kind": "value", @@ -13412,6 +10587,15 @@ ], "signature": "let trim: t => t" }, + { + "id": "Js.String.anchor", + "kind": "value", + "name": "anchor", + "docstrings": [ + "`anchor(anchorName, anchorText)` creates a string with an HTML `` element\nwith name attribute of `anchorName` and `anchorText` as its content. Please do\nnot use this method, as it has been removed from the relevant web standards.\n\nSee [`String.anchor`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/anchor)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.anchor(\"page1\", \"Page One\") == \"Page One\"\n```" + ], + "signature": "let anchor: (t, t) => t" + }, { "id": "Js.String.anchor", "kind": "value", @@ -13419,6 +10603,15 @@ "docstrings": [], "signature": "let anchor: (t, t) => t" }, + { + "id": "Js.String.link", + "kind": "value", + "name": "link", + "docstrings": [ + "ES2015: `link(urlText, linkText)` creates a string with an HTML `` element\nwith href attribute of `urlText` and `linkText` as its content. Please do not\nuse this method, as it has been removed from the relevant web standards.\n\nSee [`String.link`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/link)\non MDN.\n\n## Examples\n\n```rescript\nJs.String.link(\"page2.html\", \"Go to page two\") == \"Go to page two\"\n```" + ], + "signature": "let link: (t, t) => t" + }, { "id": "Js.String.link", "kind": "value", @@ -14028,47 +11221,101 @@ ], "signature": "let length: array<'a> => int" }, + { + "id": "Js.Array.copyWithin", + "kind": "value", + "name": "copyWithin", + "docstrings": [ + "Copies from the first element in the given array to the designated `~to_` position, returning the resulting array. *This function modifies the original array.* See [`Array.copyWithin`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/copyWithin) on MDN.\n\n## Examples\n\n```rescript\nlet arr = [100, 101, 102, 103, 104]\nJs.Array.copyWithin(~to_=2, arr) == [100, 101, 100, 101, 102]\narr == [100, 101, 100, 101, 102]\n```" + ], + "signature": "let copyWithin: (t<'a>, ~to_: int) => 'this" + }, { "id": "Js.Array.copyWithin", "kind": "value", "name": "copyWithin", "docstrings": [], - "signature": "let copyWithin: (~to_: int, t<'a>) => t<'a>" + "signature": "let copyWithin: (~to_: int, t<'a>) => 'b" + }, + { + "id": "Js.Array.copyWithinFrom", + "kind": "value", + "name": "copyWithinFrom", + "docstrings": [ + "Copies starting at element `~from` in the given array to the designated `~to_` position, returning the resulting array. *This function modifies the original array.* See [`Array.copyWithin`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/copyWithin) on MDN.\n\n## Examples\n\n```rescript\nlet arr = [100, 101, 102, 103, 104]\nJs.Array.copyWithinFrom(~from=2, ~to_=0, arr) == [102, 103, 104, 103, 104]\narr == [102, 103, 104, 103, 104]\n```" + ], + "signature": "let copyWithinFrom: (t<'a>, ~to_: int, ~from: int) => 'this" }, { "id": "Js.Array.copyWithinFrom", "kind": "value", "name": "copyWithinFrom", "docstrings": [], - "signature": "let copyWithinFrom: (~to_: int, ~from: int, t<'a>) => t<'a>" + "signature": "let copyWithinFrom: (~to_: int, ~from: int, t<'a>) => 'b" + }, + { + "id": "Js.Array.copyWithinFromRange", + "kind": "value", + "name": "copyWithinFromRange", + "docstrings": [ + "Copies starting at element `~start` in the given array up to but not including `~end_` to the designated `~to_` position, returning the resulting array. *This function modifies the original array.* See [`Array.copyWithin`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/copyWithin) on MDN.\n\n## Examples\n\n```rescript\nlet arr = [100, 101, 102, 103, 104, 105]\nJs.Array.copyWithinFromRange(~start=2, ~end_=5, ~to_=1, arr) == [100, 102, 103, 104, 104, 105]\narr == [100, 102, 103, 104, 104, 105]\n```" + ], + "signature": "let copyWithinFromRange: (t<'a>, ~to_: int, ~start: int, ~end_: int) => 'this" }, { "id": "Js.Array.copyWithinFromRange", "kind": "value", "name": "copyWithinFromRange", "docstrings": [], - "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int, t<'a>) => t<'a>" + "signature": "let copyWithinFromRange: (~to_: int, ~start: int, ~end_: int, t<'a>) => 'b" + }, + { + "id": "Js.Array.fillInPlace", + "kind": "value", + "name": "fillInPlace", + "docstrings": [ + "Sets all elements of the given array (the second arumgent) to the designated value (the first argument), returning the resulting array. *This function modifies the original array.* See [`Array.fill`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/fill) on MDN.\n\n## Examples\n\n```rescript\nlet arr = [100, 101, 102, 103, 104]\nJs.Array.fillInPlace(99, arr) == [99, 99, 99, 99, 99]\narr == [99, 99, 99, 99, 99]\n```" + ], + "signature": "let fillInPlace: (t<'a>, 'a) => 'this" }, { "id": "Js.Array.fillInPlace", "kind": "value", "name": "fillInPlace", "docstrings": [], - "signature": "let fillInPlace: ('a, t<'a>) => t<'a>" + "signature": "let fillInPlace: ('a, t<'a>) => 'b" + }, + { + "id": "Js.Array.fillFromInPlace", + "kind": "value", + "name": "fillFromInPlace", + "docstrings": [ + "Sets all elements of the given array (the last arumgent) from position `~from` to the end to the designated value (the first argument), returning the resulting array. *This function modifies the original array.* See [`Array.fill`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/fill) on MDN.\n\n## Examples\n\n```rescript\nlet arr = [100, 101, 102, 103, 104]\nJs.Array.fillFromInPlace(99, ~from=2, arr) == [100, 101, 99, 99, 99]\narr == [100, 101, 99, 99, 99]\n```" + ], + "signature": "let fillFromInPlace: (t<'a>, 'a, ~from: int) => 'this" }, { "id": "Js.Array.fillFromInPlace", "kind": "value", "name": "fillFromInPlace", "docstrings": [], - "signature": "let fillFromInPlace: ('a, ~from: int, t<'a>) => t<'a>" + "signature": "let fillFromInPlace: ('a, ~from: int, t<'a>) => 'b" + }, + { + "id": "Js.Array.fillRangeInPlace", + "kind": "value", + "name": "fillRangeInPlace", + "docstrings": [ + "Sets the elements of the given array (the last arumgent) from position `~start` up to but not including position `~end_` to the designated value (the first argument), returning the resulting array. *This function modifies the original array.* See [`Array.fill`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/fill) on MDN.\n\n## Examples\n\n```rescript\nlet arr = [100, 101, 102, 103, 104]\nJs.Array.fillRangeInPlace(99, ~start=1, ~end_=4, arr) == [100, 99, 99, 99, 104]\narr == [100, 99, 99, 99, 104]\n```" + ], + "signature": "let fillRangeInPlace: (t<'a>, 'a, ~start: int, ~end_: int) => 'this" }, { "id": "Js.Array.fillRangeInPlace", "kind": "value", "name": "fillRangeInPlace", "docstrings": [], - "signature": "let fillRangeInPlace: ('a, ~start: int, ~end_: int, t<'a>) => t<'a>" + "signature": "let fillRangeInPlace: ('a, ~start: int, ~end_: int, t<'a>) => 'b" }, { "id": "Js.Array.pop", @@ -14079,6 +11326,15 @@ ], "signature": "let pop: t<'a> => option<'a>" }, + { + "id": "Js.Array.push", + "kind": "value", + "name": "push", + "docstrings": [ + "Appends the given value to the array, returning the number of elements in the updated array. *This function modifies the original array.* See [`Array.push`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/push) on MDN.\n\n## Examples\n\n```rescript\nlet arr = [\"ant\", \"bee\", \"cat\"]\nJs.Array.push(\"dog\", arr) == 4\narr == [\"ant\", \"bee\", \"cat\", \"dog\"]\n```" + ], + "signature": "let push: (t<'a>, 'a) => int" + }, { "id": "Js.Array.push", "kind": "value", @@ -14086,6 +11342,15 @@ "docstrings": [], "signature": "let push: ('a, t<'a>) => int" }, + { + "id": "Js.Array.pushMany", + "kind": "value", + "name": "pushMany", + "docstrings": [ + "Appends the values from one array (the first argument) to another (the second argument), returning the number of elements in the updated array. *This function modifies the original array.* See [`Array.push`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/push) on MDN.\n\n## Examples\n\n```rescript\nlet arr = [\"ant\", \"bee\", \"cat\"]\nJs.Array.pushMany([\"dog\", \"elk\"], arr) == 5\narr == [\"ant\", \"bee\", \"cat\", \"dog\", \"elk\"]\n```" + ], + "signature": "let pushMany: (t<'a>, array<'a>) => int" + }, { "id": "Js.Array.pushMany", "kind": "value", @@ -14100,7 +11365,7 @@ "docstrings": [ "Returns an array with the elements of the input array in reverse order. *This function modifies the original array.* See [`Array.reverse`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/reverse) on MDN.\n\n## Examples\n\n```rescript\nlet arr = [\"ant\", \"bee\", \"cat\"]\nJs.Array.reverseInPlace(arr) == [\"cat\", \"bee\", \"ant\"]\narr == [\"cat\", \"bee\", \"ant\"]\n```" ], - "signature": "let reverseInPlace: t<'a> => t<'a>" + "signature": "let reverseInPlace: t<'a> => 'this" }, { "id": "Js.Array.shift", @@ -14116,37 +11381,82 @@ "kind": "value", "name": "sortInPlace", "docstrings": [ - "Sorts the given array in place and returns the sorted array. JavaScript sorts the array by converting the arguments to UTF-16 strings and sorting them. See the second example with sorting numbers, which does not do a numeric sort. *This function modifies the original array.* See [`Array.sort`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort) on MDN.\n\n## Examples\n\n```rescript\nlet words = [\"bee\", \"dog\", \"ant\", \"cat\"]\nJs.Array.sortInPlace(words) == [\"ant\", \"bee\", \"cat\", \"dog\"]\nwords == [\"ant\", \"bee\", \"cat\", \"dog\"]\n\nlet numbers = [3, 30, 10, 1, 20, 2]\nJs.Array.sortInPlace(numbers) == [1, 10, 2, 20, 3, 30]\nnumbers == [1, 10, 2, 20, 3, 30]\n```" + "Sorts the given array in place and returns the sorted array. JavaScript sorts the array by converting the arguments to UTF-16 strings and sorting them. See the second example with sorting numbers, which does not do a numeric sort. *This function modifies the original array.* See [`Array.sort`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort) on MDN.\n\n## Examples\n\n```rescript\nlet words = [\"bee\", \"dog\", \"ant\", \"cat\"]\nJs.Array.sortInPlace(words) == [\"ant\", \"bee\", \"cat\", \"dog\"]\nwords == [\"ant\", \"bee\", \"cat\", \"dog\"]\n\nlet numbers = [3, 30, 10, 1, 20, 2]\nJs.Array.sortInPlace(numbers) == [1, 10, 2, 20, 3, 30]\nnumbers == [1, 10, 2, 20, 3, 30]\n```" + ], + "signature": "let sortInPlace: t<'a> => 'this" + }, + { + "id": "Js.Array.sortInPlaceWith", + "kind": "value", + "name": "sortInPlaceWith", + "docstrings": [ + "Sorts the given array in place and returns the sorted array. *This function modifies the original array.*\n\nThe first argument to `sortInPlaceWith()` is a function that compares two items from the array and returns:\n\n* an integer less than zero if the first item is less than the second item\n* zero if the items are equal\n* an integer greater than zero if the first item is greater than the second item\n\nSee [`Array.sort`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort) on MDN.\n\n## Examples\n\n```rescript\n// sort by word length\nlet words = [\"horse\", \"aardvark\", \"dog\", \"camel\"]\nlet byLength = (s1, s2) => Js.String.length(s1) - Js.String.length(s2)\n\nJs.Array.sortInPlaceWith(byLength, words) == [\"dog\", \"horse\", \"camel\", \"aardvark\"]\n\n// sort in reverse numeric order\nlet numbers = [3, 30, 10, 1, 20, 2]\nlet reverseNumeric = (n1, n2) => n2 - n1\nJs.Array.sortInPlaceWith(reverseNumeric, numbers) == [30, 20, 10, 3, 2, 1]\n```" + ], + "signature": "let sortInPlaceWith: (t<'a>, ('a, 'a) => int) => 'this" + }, + { + "id": "Js.Array.sortInPlaceWith", + "kind": "value", + "name": "sortInPlaceWith", + "docstrings": [], + "signature": "let sortInPlaceWith: (('a, 'a) => int, t<'a>) => 'b" + }, + { + "id": "Js.Array.spliceInPlace", + "kind": "value", + "name": "spliceInPlace", + "docstrings": [ + "Starting at position `~pos`, remove `~remove` elements and then add the\nelements from the `~add` array. Returns an array consisting of the removed\nitems. *This function modifies the original array.* See\n[`Array.splice`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/splice)\non MDN.\n\n## Examples\n\n```rescript\nlet arr = [\"a\", \"b\", \"c\", \"d\", \"e\", \"f\"]\nJs.Array.spliceInPlace(~pos=2, ~remove=2, ~add=[\"x\", \"y\", \"z\"], arr) == [\"c\", \"d\"]\narr == [\"a\", \"b\", \"x\", \"y\", \"z\", \"e\", \"f\"]\n\nlet arr2 = [\"a\", \"b\", \"c\", \"d\"]\nJs.Array.spliceInPlace(~pos=3, ~remove=0, ~add=[\"x\", \"y\"], arr2) == []\narr2 == [\"a\", \"b\", \"c\", \"x\", \"y\", \"d\"]\n\nlet arr3 = [\"a\", \"b\", \"c\", \"d\", \"e\", \"f\"]\nJs.Array.spliceInPlace(~pos=9, ~remove=2, ~add=[\"x\", \"y\", \"z\"], arr3) == []\narr3 == [\"a\", \"b\", \"c\", \"d\", \"e\", \"f\", \"x\", \"y\", \"z\"]\n```" ], - "signature": "let sortInPlace: t<'a> => t<'a>" + "signature": "let spliceInPlace: (t<'a>, ~pos: int, ~remove: int, ~add: array<'a>) => 'this" }, { - "id": "Js.Array.sortInPlaceWith", + "id": "Js.Array.spliceInPlace", "kind": "value", - "name": "sortInPlaceWith", + "name": "spliceInPlace", "docstrings": [], - "signature": "let sortInPlaceWith: (('a, 'a) => int, t<'a>) => t<'a>" + "signature": "let spliceInPlace: (~pos: int, ~remove: int, ~add: array<'a>, t<'a>) => 'b" }, { - "id": "Js.Array.spliceInPlace", + "id": "Js.Array.removeFromInPlace", "kind": "value", - "name": "spliceInPlace", - "docstrings": [], - "signature": "let spliceInPlace: (~pos: int, ~remove: int, ~add: array<'a>, t<'a>) => t<'a>" + "name": "removeFromInPlace", + "docstrings": [ + "Removes elements from the given array starting at position `~pos` to the end\nof the array, returning the removed elements. *This function modifies the\noriginal array.* See\n[`Array.splice`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/splice)\non MDN.\n\n## Examples\n\n```rescript\nlet arr = [\"a\", \"b\", \"c\", \"d\", \"e\", \"f\"]\nJs.Array.removeFromInPlace(~pos=4, arr) == [\"e\", \"f\"]\narr == [\"a\", \"b\", \"c\", \"d\"]\n```" + ], + "signature": "let removeFromInPlace: (t<'a>, ~pos: int) => 'this" }, { "id": "Js.Array.removeFromInPlace", "kind": "value", "name": "removeFromInPlace", "docstrings": [], - "signature": "let removeFromInPlace: (~pos: int, t<'a>) => t<'a>" + "signature": "let removeFromInPlace: (~pos: int, t<'a>) => 'b" + }, + { + "id": "Js.Array.removeCountInPlace", + "kind": "value", + "name": "removeCountInPlace", + "docstrings": [ + "Removes `~count` elements from the given array starting at position `~pos`,\nreturning the removed elements. *This function modifies the original array.*\nSee\n[`Array.splice`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/splice)\non MDN.\n\n## Examples\n\n```rescript\nlet arr = [\"a\", \"b\", \"c\", \"d\", \"e\", \"f\"]\nJs.Array.removeCountInPlace(~pos=2, ~count=3, arr) == [\"c\", \"d\", \"e\"]\narr == [\"a\", \"b\", \"f\"]\n```" + ], + "signature": "let removeCountInPlace: (t<'a>, ~pos: int, ~count: int) => 'this" }, { "id": "Js.Array.removeCountInPlace", "kind": "value", "name": "removeCountInPlace", "docstrings": [], - "signature": "let removeCountInPlace: (~pos: int, ~count: int, t<'a>) => t<'a>" + "signature": "let removeCountInPlace: (~pos: int, ~count: int, t<'a>) => 'b" + }, + { + "id": "Js.Array.unshift", + "kind": "value", + "name": "unshift", + "docstrings": [ + "Adds the given element to the array, returning the new number of elements in\nthe array. *This function modifies the original array.* See\n[`Array.unshift`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/unshift)\non MDN.\n\n## Examples\n\n```rescript\nlet arr = [\"b\", \"c\", \"d\"]\nJs.Array.unshift(\"a\", arr) == 4\narr == [\"a\", \"b\", \"c\", \"d\"]\n```" + ], + "signature": "let unshift: (t<'a>, 'a) => int" }, { "id": "Js.Array.unshift", @@ -14155,6 +11465,15 @@ "docstrings": [], "signature": "let unshift: ('a, t<'a>) => int" }, + { + "id": "Js.Array.unshiftMany", + "kind": "value", + "name": "unshiftMany", + "docstrings": [ + "Adds the elements in the first array argument at the beginning of the second\narray argument, returning the new number of elements in the array. *This\nfunction modifies the original array.* See\n[`Array.unshift`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/unshift)\non MDN.\n\n## Examples\n\n```rescript\nlet arr = [\"d\", \"e\"]\nJs.Array.unshiftMany([\"a\", \"b\", \"c\"], arr) == 5\narr == [\"a\", \"b\", \"c\", \"d\", \"e\"]\n```" + ], + "signature": "let unshiftMany: (t<'a>, array<'a>) => int" + }, { "id": "Js.Array.unshiftMany", "kind": "value", @@ -14162,19 +11481,46 @@ "docstrings": [], "signature": "let unshiftMany: (array<'a>, t<'a>) => int" }, + { + "id": "Js.Array.concat", + "kind": "value", + "name": "concat", + "docstrings": [ + "Concatenates the first array argument to the second array argument, returning\na new array. The original arrays are not modified. See\n[`Array.concat`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/concat)\non MDN.\n\n## Examples\n\n```rescript\nJs.Array.concat([\"c\", \"d\", \"e\"], [\"a\", \"b\"]) == [\"a\", \"b\", \"c\", \"d\", \"e\"]\n```" + ], + "signature": "let concat: (t<'a>, 'this) => 'this" + }, { "id": "Js.Array.concat", "kind": "value", "name": "concat", "docstrings": [], - "signature": "let concat: (t<'a>, t<'a>) => t<'a>" + "signature": "let concat: ('a, t<'b>) => 'a" + }, + { + "id": "Js.Array.concatMany", + "kind": "value", + "name": "concatMany", + "docstrings": [ + "The first argument to `concatMany()` is an array of arrays; these are added\nat the end of the second argument, returning a new array. See\n[`Array.concat`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/concat)\non MDN.\n\n## Examples\n\n```rescript\nJs.Array.concatMany([[\"d\", \"e\"], [\"f\", \"g\", \"h\"]], [\"a\", \"b\", \"c\"]) == [\n \"a\",\n \"b\",\n \"c\",\n \"d\",\n \"e\",\n \"f\",\n \"g\",\n \"h\",\n ]\n```" + ], + "signature": "let concatMany: (t<'a>, array<'this>) => 'this" }, { "id": "Js.Array.concatMany", "kind": "value", "name": "concatMany", "docstrings": [], - "signature": "let concatMany: (array>, t<'a>) => t<'a>" + "signature": "let concatMany: (array<'a>, t<'b>) => 'a" + }, + { + "id": "Js.Array.includes", + "kind": "value", + "name": "includes", + "docstrings": [ + "Returns true if the given value is in the array, `false` otherwise. See\n[`Array.includes`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/includes)\non MDN.\n\n## Examples\n\n```rescript\nJs.Array.includes(\"b\", [\"a\", \"b\", \"c\"]) == true\nJs.Array.includes(\"x\", [\"a\", \"b\", \"c\"]) == false\n```" + ], + "signature": "let includes: (t<'a>, 'a) => bool" }, { "id": "Js.Array.includes", @@ -14183,6 +11529,15 @@ "docstrings": [], "signature": "let includes: ('a, t<'a>) => bool" }, + { + "id": "Js.Array.indexOf", + "kind": "value", + "name": "indexOf", + "docstrings": [ + "Returns the index of the first element in the array that has the given value.\nIf the value is not in the array, returns -1. See\n[`Array.indexOf`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/indexOf)\non MDN.\n\n## Examples\n\n```rescript\nJs.Array.indexOf(102, [100, 101, 102, 103]) == 2\nJs.Array.indexOf(999, [100, 101, 102, 103]) == -1\n```" + ], + "signature": "let indexOf: (t<'a>, 'a) => int" + }, { "id": "Js.Array.indexOf", "kind": "value", @@ -14190,6 +11545,15 @@ "docstrings": [], "signature": "let indexOf: ('a, t<'a>) => int" }, + { + "id": "Js.Array.indexOfFrom", + "kind": "value", + "name": "indexOfFrom", + "docstrings": [ + "Returns the index of the first element in the array with the given value. The\nsearch starts at position `~from`. See\n[`Array.indexOf`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/indexOf)\non MDN.\n\n## Examples\n\n```rescript\nJs.Array.indexOfFrom(\"a\", ~from=2, [\"a\", \"b\", \"a\", \"c\", \"a\"]) == 2\nJs.Array.indexOfFrom(\"a\", ~from=3, [\"a\", \"b\", \"a\", \"c\", \"a\"]) == 4\nJs.Array.indexOfFrom(\"b\", ~from=2, [\"a\", \"b\", \"a\", \"c\", \"a\"]) == -1\n```" + ], + "signature": "let indexOfFrom: (t<'a>, 'a, ~from: int) => int" + }, { "id": "Js.Array.indexOfFrom", "kind": "value", @@ -14205,6 +11569,15 @@ "signature": "let join: t<'a> => string", "deprecated": "please use joinWith instead" }, + { + "id": "Js.Array.joinWith", + "kind": "value", + "name": "joinWith", + "docstrings": [ + "This function converts each element of the array to a string (via JavaScript)\nand concatenates them, separated by the string given in the first argument,\ninto a single string. See\n[`Array.join`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/join)\non MDN.\n\n## Examples\n\n```rescript\nJs.Array.joinWith(\"--\", [\"ant\", \"bee\", \"cat\"]) == \"ant--bee--cat\"\nJs.Array.joinWith(\"\", [\"door\", \"bell\"]) == \"doorbell\"\nJs.Array.joinWith(\"/\", [2020, 9, 4]) == \"2020/9/4\"\nJs.Array.joinWith(\";\", [2.5, 3.6, 3e-2]) == \"2.5;3.6;0.03\"\n```" + ], + "signature": "let joinWith: (t<'a>, string) => string" + }, { "id": "Js.Array.joinWith", "kind": "value", @@ -14212,6 +11585,15 @@ "docstrings": [], "signature": "let joinWith: (string, t<'a>) => string" }, + { + "id": "Js.Array.lastIndexOf", + "kind": "value", + "name": "lastIndexOf", + "docstrings": [ + "Returns the index of the last element in the array that has the given value.\nIf the value is not in the array, returns -1. See\n[`Array.lastIndexOf`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/lastIndexOf)\non MDN.\n\n## Examples\n\n```rescript\nJs.Array.lastIndexOf(\"a\", [\"a\", \"b\", \"a\", \"c\"]) == 2\nJs.Array.lastIndexOf(\"x\", [\"a\", \"b\", \"a\", \"c\"]) == -1\n```" + ], + "signature": "let lastIndexOf: (t<'a>, 'a) => int" + }, { "id": "Js.Array.lastIndexOf", "kind": "value", @@ -14219,6 +11601,15 @@ "docstrings": [], "signature": "let lastIndexOf: ('a, t<'a>) => int" }, + { + "id": "Js.Array.lastIndexOfFrom", + "kind": "value", + "name": "lastIndexOfFrom", + "docstrings": [ + "Returns the index of the last element in the array that has the given value,\nsearching from position `~from` down to the start of the array. If the value\nis not in the array, returns -1. See\n[`Array.lastIndexOf`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/lastIndexOf)\non MDN.\n\n## Examples\n\n```rescript\nJs.Array.lastIndexOfFrom(\"a\", ~from=3, [\"a\", \"b\", \"a\", \"c\", \"a\", \"d\"]) == 2\nJs.Array.lastIndexOfFrom(\"c\", ~from=2, [\"a\", \"b\", \"a\", \"c\", \"a\", \"d\"]) == -1\n```" + ], + "signature": "let lastIndexOfFrom: (t<'a>, 'a, ~from: int) => int" + }, { "id": "Js.Array.lastIndexOfFrom", "kind": "value", @@ -14226,12 +11617,21 @@ "docstrings": [], "signature": "let lastIndexOfFrom: ('a, ~from: int, t<'a>) => int" }, + { + "id": "Js.Array.slice", + "kind": "value", + "name": "slice", + "docstrings": [ + "Returns a shallow copy of the given array from the `~start` index up to but\nnot including the `~end_` position. Negative numbers indicate an offset from\nthe end of the array. See\n[`Array.slice`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/slice)\non MDN.\n\n## Examples\n\n```rescript\nlet arr = [100, 101, 102, 103, 104, 105, 106]\nJs.Array.slice(~start=2, ~end_=5, arr) == [102, 103, 104]\nJs.Array.slice(~start=-3, ~end_=-1, arr) == [104, 105]\nJs.Array.slice(~start=9, ~end_=10, arr) == []\n```" + ], + "signature": "let slice: (t<'a>, ~start: int, ~end_: int) => 'this" + }, { "id": "Js.Array.slice", "kind": "value", "name": "slice", "docstrings": [], - "signature": "let slice: (~start: int, ~end_: int, t<'a>) => t<'a>" + "signature": "let slice: (int, ~end_: int, ~obj: t<'a>) => 'b" }, { "id": "Js.Array.copy", @@ -14240,14 +11640,23 @@ "docstrings": [ "Returns a copy of the entire array. Same as `Js.Array.Slice(~start=0,\n~end_=Js.Array.length(arr), arr)`. See\n[`Array.slice`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/slice)\non MDN." ], - "signature": "let copy: t<'a> => t<'a>" + "signature": "let copy: t<'a> => 'this" + }, + { + "id": "Js.Array.sliceFrom", + "kind": "value", + "name": "sliceFrom", + "docstrings": [ + "Returns a shallow copy of the given array from the given index to the end. \nSee [`Array.slice`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/slice) on MDN.\n\n## Examples\n\n```rescript\nJs.Array.sliceFrom(2, [100, 101, 102, 103, 104]) == [102, 103, 104]\n```" + ], + "signature": "let sliceFrom: (t<'a>, int) => 'this" }, { "id": "Js.Array.sliceFrom", "kind": "value", "name": "sliceFrom", "docstrings": [], - "signature": "let sliceFrom: (int, t<'a>) => t<'a>" + "signature": "let sliceFrom: (int, t<'a>) => 'b" }, { "id": "Js.Array.toString", @@ -14267,6 +11676,15 @@ ], "signature": "let toLocaleString: t<'a> => string" }, + { + "id": "Js.Array.every", + "kind": "value", + "name": "every", + "docstrings": [ + "The first argument to `every()` is a predicate function that returns a boolean. The `every()` function returns `true` if the predicate function is true for all items in the given array. If given an empty array, returns `true`. See [`Array.every`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/every) on MDN.\n\n## Examples\n\n```rescript\nlet isEven = x => mod(x, 2) == 0\nJs.Array.every(isEven, [6, 22, 8, 4]) == true\nJs.Array.every(isEven, [6, 22, 7, 4]) == false\n```" + ], + "signature": "let every: (t<'a>, 'a => bool) => bool" + }, { "id": "Js.Array.every", "kind": "value", @@ -14274,6 +11692,15 @@ "docstrings": [], "signature": "let every: ('a => bool, t<'a>) => bool" }, + { + "id": "Js.Array.everyi", + "kind": "value", + "name": "everyi", + "docstrings": [ + "The first argument to `everyi()` is a predicate function with two arguments: an array element and that element’s index; it returns a boolean. The `everyi()` function returns `true` if the predicate function is true for all items in the given array. If given an empty array, returns `true`. See [`Array.every`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/every) on MDN.\n\n## Examples\n\n```rescript\n// determine if all even-index items are positive\nlet evenIndexPositive = (item, index) => mod(index, 2) == 0 ? item > 0 : true\n\nJs.Array.everyi(evenIndexPositive, [6, -3, 5, 8]) == true\nJs.Array.everyi(evenIndexPositive, [6, 3, -5, 8]) == false\n```" + ], + "signature": "let everyi: (t<'a>, ('a, int) => bool) => bool" + }, { "id": "Js.Array.everyi", "kind": "value", @@ -14281,19 +11708,46 @@ "docstrings": [], "signature": "let everyi: (('a, int) => bool, t<'a>) => bool" }, + { + "id": "Js.Array.filter", + "kind": "value", + "name": "filter", + "docstrings": [ + "Applies the given predicate function to each element in the array; the result is an array of those elements for which the predicate function returned `true`. See [`Array.filter`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/filter) on MDN.\n\n## Examples\n\n```rescript\nlet nonEmpty = s => s != \"\"\nJs.Array.filter(nonEmpty, [\"abc\", \"\", \"\", \"def\", \"ghi\"]) == [\"abc\", \"def\", \"ghi\"]\n```" + ], + "signature": "let filter: (t<'a>, 'a => bool) => 'this" + }, { "id": "Js.Array.filter", "kind": "value", "name": "filter", "docstrings": [], - "signature": "let filter: ('a => bool, t<'a>) => t<'a>" + "signature": "let filter: ('a => bool, t<'a>) => 'b" + }, + { + "id": "Js.Array.filteri", + "kind": "value", + "name": "filteri", + "docstrings": [ + "Each element of the given array are passed to the predicate function. The\nreturn value is an array of all those elements for which the predicate\nfunction returned `true`. See\n[`Array.filter`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/filter)\non MDN.\n\n## Examples\n\n```rescript\n// keep only positive elements at odd indices\nlet positiveOddElement = (item, index) => mod(index, 2) == 1 && item > 0\n\nJs.Array.filteri(positiveOddElement, [6, 3, 5, 8, 7, -4, 1]) == [3, 8]\n```" + ], + "signature": "let filteri: (t<'a>, ('a, int) => bool) => 'this" }, { "id": "Js.Array.filteri", "kind": "value", "name": "filteri", "docstrings": [], - "signature": "let filteri: (('a, int) => bool, t<'a>) => t<'a>" + "signature": "let filteri: (('a, int) => bool, t<'a>) => 'b" + }, + { + "id": "Js.Array.find", + "kind": "value", + "name": "find", + "docstrings": [ + "Returns `Some(value)` for the first element in the array that satisifies the\ngiven predicate function, or `None` if no element satisifies the predicate.\nSee\n[`Array.find`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/find)\non MDN.\n\n## Examples\n\n```rescript\n// find first negative element\nJs.Array.find(x => x < 0, [33, 22, -55, 77, -44]) == Some(-55)\nJs.Array.find(x => x < 0, [33, 22, 55, 77, 44]) == None\n```" + ], + "signature": "let find: (t<'a>, 'a => bool) => option<'a>" }, { "id": "Js.Array.find", @@ -14302,6 +11756,15 @@ "docstrings": [], "signature": "let find: ('a => bool, t<'a>) => option<'a>" }, + { + "id": "Js.Array.findi", + "kind": "value", + "name": "findi", + "docstrings": [ + "Returns `Some(value)` for the first element in the array that satisifies the given predicate function, or `None` if no element satisifies the predicate. The predicate function takes an array element and an index as its parameters. See [`Array.find`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/find) on MDN.\n\n## Examples\n\n```rescript\n// find first positive item at an odd index\nlet positiveOddElement = (item, index) => mod(index, 2) == 1 && item > 0\n\nJs.Array.findi(positiveOddElement, [66, -33, 55, 88, 22]) == Some(88)\nJs.Array.findi(positiveOddElement, [66, -33, 55, -88, 22]) == None\n```" + ], + "signature": "let findi: (t<'a>, ('a, int) => bool) => option<'a>" + }, { "id": "Js.Array.findi", "kind": "value", @@ -14309,6 +11772,15 @@ "docstrings": [], "signature": "let findi: (('a, int) => bool, t<'a>) => option<'a>" }, + { + "id": "Js.Array.findIndex", + "kind": "value", + "name": "findIndex", + "docstrings": [ + "Returns the index of the first element in the array that satisifies the given predicate function, or -1 if no element satisifies the predicate. See [`Array.find`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/find) on MDN.\n\n## Examples\n\n```rescript\nJs.Array.findIndex(x => x < 0, [33, 22, -55, 77, -44]) == 2\nJs.Array.findIndex(x => x < 0, [33, 22, 55, 77, 44]) == -1\n```" + ], + "signature": "let findIndex: (t<'a>, 'a => bool) => int" + }, { "id": "Js.Array.findIndex", "kind": "value", @@ -14316,6 +11788,15 @@ "docstrings": [], "signature": "let findIndex: ('a => bool, t<'a>) => int" }, + { + "id": "Js.Array.findIndexi", + "kind": "value", + "name": "findIndexi", + "docstrings": [ + "Returns `Some(value)` for the first element in the array that satisifies the given predicate function, or `None` if no element satisifies the predicate. The predicate function takes an array element and an index as its parameters. See [`Array.find`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/find) on MDN.\n\n## Examples\n\n```rescript\n// find index of first positive item at an odd index\nlet positiveOddElement = (item, index) => mod(index, 2) == 1 && item > 0\n\nJs.Array.findIndexi(positiveOddElement, [66, -33, 55, 88, 22]) == 3\nJs.Array.findIndexi(positiveOddElement, [66, -33, 55, -88, 22]) == -1\n```" + ], + "signature": "let findIndexi: (t<'a>, ('a, int) => bool) => int" + }, { "id": "Js.Array.findIndexi", "kind": "value", @@ -14323,6 +11804,15 @@ "docstrings": [], "signature": "let findIndexi: (('a, int) => bool, t<'a>) => int" }, + { + "id": "Js.Array.forEach", + "kind": "value", + "name": "forEach", + "docstrings": [ + "The `forEach()` function applies the function given as the first argument to each element in the array. The function you provide returns `unit`, and the `forEach()` function also returns `unit`. You use `forEach()` when you need to process each element in the array but not return any new array or value; for example, to print the items in an array. See [`Array.forEach`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/forEach) on MDN.\n\n## Examples\n\n```rescript\n// display all elements in an array\nJs.Array.forEach(x => Js.log(x), [\"a\", \"b\", \"c\"]) == ()\n```" + ], + "signature": "let forEach: (t<'a>, 'a => unit) => unit" + }, { "id": "Js.Array.forEach", "kind": "value", @@ -14330,6 +11820,15 @@ "docstrings": [], "signature": "let forEach: ('a => unit, t<'a>) => unit" }, + { + "id": "Js.Array.forEachi", + "kind": "value", + "name": "forEachi", + "docstrings": [ + "The `forEachi()` function applies the function given as the first argument to each element in the array. The function you provide takes an item in the array and its index number, and returns `unit`. The `forEachi()` function also returns `unit`. You use `forEachi()` when you need to process each element in the array but not return any new array or value; for example, to print the items in an array. See [`Array.forEach`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/forEach) on MDN.\n\n## Examples\n\n```rescript\n// display all elements in an array as a numbered list\nJs.Array.forEachi((item, index) => Js.log2(index + 1, item), [\"a\", \"b\", \"c\"]) == ()\n```" + ], + "signature": "let forEachi: (t<'a>, ('a, int) => unit) => unit" + }, { "id": "Js.Array.forEachi", "kind": "value", @@ -14337,6 +11836,15 @@ "docstrings": [], "signature": "let forEachi: (('a, int) => unit, t<'a>) => unit" }, + { + "id": "Js.Array.map", + "kind": "value", + "name": "map", + "docstrings": [ + "Applies the function (given as the first argument) to each item in the array,\nreturning a new array. The result array does not have to have elements of the\nsame type as the input array. See\n[`Array.map`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/map)\non MDN.\n\n## Examples\n\n```rescript\nJs.Array.map(x => x * x, [12, 4, 8]) == [144, 16, 64]\nJs.Array.map(Js.String.length, [\"animal\", \"vegetable\", \"mineral\"]) == [6, 9, 7]\n```" + ], + "signature": "let map: (t<'a>, 'a => 'b) => t<'b>" + }, { "id": "Js.Array.map", "kind": "value", @@ -14344,6 +11852,15 @@ "docstrings": [], "signature": "let map: ('a => 'b, t<'a>) => t<'b>" }, + { + "id": "Js.Array.mapi", + "kind": "value", + "name": "mapi", + "docstrings": [ + "Applies the function (given as the first argument) to each item in the array,\nreturning a new array. The function acceps two arguments: an item from the\narray and its index number. The result array does not have to have elements\nof the same type as the input array. See\n[`Array.map`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/map)\non MDN.\n\n## Examples\n\n```rescript\n// multiply each item in array by its position\nlet product = (item, index) => item * index\nJs.Array.mapi(product, [10, 11, 12]) == [0, 11, 24]\n```" + ], + "signature": "let mapi: (t<'a>, ('a, int) => 'b) => t<'b>" + }, { "id": "Js.Array.mapi", "kind": "value", @@ -14351,6 +11868,15 @@ "docstrings": [], "signature": "let mapi: (('a, int) => 'b, t<'a>) => t<'b>" }, + { + "id": "Js.Array.reduce", + "kind": "value", + "name": "reduce", + "docstrings": [ + "The `reduce()` function takes three parameters: a *reducer function*, a\nbeginning accumulator value, and an array. The reducer function has two\nparameters: an accumulated value and an element of the array.\n\n`reduce()` first calls the reducer function with the beginning value and the\nfirst element in the array. The result becomes the new accumulator value, which\nis passed in to the reducer function along with the second element in the\narray. `reduce()` proceeds through the array, passing in the result of each\nstage as the accumulator to the reducer function.\n\nWhen all array elements are processed, the final value of the accumulator\nbecomes the return value of `reduce()`. See\n[`Array.reduce`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/reduce)\non MDN.\n\n## Examples\n\n```rescript\nlet sumOfSquares = (accumulator, item) => accumulator + item * item\n\nJs.Array.reduce(sumOfSquares, 0, [10, 2, 4]) == 120\nJs.Array.reduce(\\\"*\", 1, [10, 2, 4]) == 80\nJs.Array.reduce(\n (acc, item) => acc + Js.String.length(item),\n 0,\n [\"animal\", \"vegetable\", \"mineral\"],\n) == 22 // 6 + 9 + 7\nJs.Array.reduce((acc, item) => item /. acc, 1.0, [2.0, 4.0]) == 2.0 // 4.0 / (2.0 / 1.0)\n```" + ], + "signature": "let reduce: (t<'a>, ('b, 'a) => 'b, 'b) => 'b" + }, { "id": "Js.Array.reduce", "kind": "value", @@ -14358,6 +11884,15 @@ "docstrings": [], "signature": "let reduce: (('a, 'b) => 'a, 'a, t<'b>) => 'a" }, + { + "id": "Js.Array.reducei", + "kind": "value", + "name": "reducei", + "docstrings": [ + "The `reducei()` function takes three parameters: a *reducer function*, a\nbeginning accumulator value, and an array. The reducer function has three\nparameters: an accumulated value, an element of the array, and the index of\nthat element.\n\n`reducei()` first calls the reducer function with the beginning value, the\nfirst element in the array, and zero (its index). The result becomes the new\naccumulator value, which is passed to the reducer function along with the\nsecond element in the array and one (its index). `reducei()` proceeds from left\nto right through the array, passing in the result of each stage as the\naccumulator to the reducer function.\n\nWhen all array elements are processed, the final value of the accumulator\nbecomes the return value of `reducei()`. See\n[`Array.reduce`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/reduce)\non MDN.\n\n## Examples\n\n```rescript\n// find sum of even-index elements in array\nlet sumOfEvens = (accumulator, item, index) =>\n if mod(index, 2) == 0 {\n accumulator + item\n } else {\n accumulator\n }\n\nJs.Array.reducei(sumOfEvens, 0, [2, 5, 1, 4, 3]) == 6\n```" + ], + "signature": "let reducei: (t<'a>, ('b, 'a, int) => 'b, 'b) => 'b" + }, { "id": "Js.Array.reducei", "kind": "value", @@ -14365,6 +11900,15 @@ "docstrings": [], "signature": "let reducei: (('a, 'b, int) => 'a, 'a, t<'b>) => 'a" }, + { + "id": "Js.Array.reduceRight", + "kind": "value", + "name": "reduceRight", + "docstrings": [ + "The `reduceRight()` function takes three parameters: a *reducer function*, a\nbeginning accumulator value, and an array. The reducer function has two\nparameters: an accumulated value and an element of the array.\n\n`reduceRight()` first calls the reducer function with the beginning value and\nthe last element in the array. The result becomes the new accumulator value,\nwhich is passed in to the reducer function along with the next-to-last element\nin the array. `reduceRight()` proceeds from right to left through the array,\npassing in the result of each stage as the accumulator to the reducer function.\n\nWhen all array elements are processed, the final value of the accumulator\nbecomes the return value of `reduceRight()`. See\n[`Array.reduceRight`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/reduceRight)\non MDN.\n\n**NOTE:** In many cases, `reduce()` and `reduceRight()` give the same result. However, see the last example here and compare it to the example from `reduce()`, where order makes a difference.\n\n## Examples\n\n```rescript\nlet sumOfSquares = (accumulator, item) => accumulator + item * item\n\nJs.Array.reduceRight(sumOfSquares, 0, [10, 2, 4]) == 120\nJs.Array.reduceRight((acc, item) => item /. acc, 1.0, [2.0, 4.0]) == 0.5 // 2.0 / (4.0 / 1.0)\n```" + ], + "signature": "let reduceRight: (t<'a>, ('b, 'a) => 'b, 'b) => 'b" + }, { "id": "Js.Array.reduceRight", "kind": "value", @@ -14372,6 +11916,15 @@ "docstrings": [], "signature": "let reduceRight: (('a, 'b) => 'a, 'a, t<'b>) => 'a" }, + { + "id": "Js.Array.reduceRighti", + "kind": "value", + "name": "reduceRighti", + "docstrings": [ + "The `reduceRighti()` function takes three parameters: a *reducer function*, a\nbeginning accumulator value, and an array. The reducer function has three\nparameters: an accumulated value, an element of the array, and the index of\nthat element. `reduceRighti()` first calls the reducer function with the\nbeginning value, the last element in the array, and its index (length of array\nminus one). The result becomes the new accumulator value, which is passed in to\nthe reducer function along with the second element in the array and one (its\nindex). `reduceRighti()` proceeds from right to left through the array, passing\nin the result of each stage as the accumulator to the reducer function.\n\nWhen all array elements are processed, the final value of the accumulator\nbecomes the return value of `reduceRighti()`. See\n[`Array.reduceRight`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/reduceRight)\non MDN.\n\n**NOTE:** In many cases, `reducei()` and `reduceRighti()` give the same result.\nHowever, there are cases where the order in which items are processed makes a\ndifference.\n\n## Examples\n\n```rescript\n// find sum of even-index elements in array\nlet sumOfEvens = (accumulator, item, index) =>\n if mod(index, 2) == 0 {\n accumulator + item\n } else {\n accumulator\n }\n\nJs.Array.reduceRighti(sumOfEvens, 0, [2, 5, 1, 4, 3]) == 6\n```" + ], + "signature": "let reduceRighti: (t<'a>, ('b, 'a, int) => 'b, 'b) => 'b" + }, { "id": "Js.Array.reduceRighti", "kind": "value", @@ -14379,6 +11932,15 @@ "docstrings": [], "signature": "let reduceRighti: (('a, 'b, int) => 'a, 'a, t<'b>) => 'a" }, + { + "id": "Js.Array.some", + "kind": "value", + "name": "some", + "docstrings": [ + "Returns `true` if the predicate function given as the first argument to\n`some()` returns `true` for any element in the array; `false` otherwise.\n\n## Examples\n\n```rescript\nlet isEven = x => mod(x, 2) == 0\n\nJs.Array.some(isEven, [3, 7, 5, 2, 9]) == true\nJs.Array.some(isEven, [3, 7, 5, 1, 9]) == false\n```" + ], + "signature": "let some: (t<'a>, 'a => bool) => bool" + }, { "id": "Js.Array.some", "kind": "value", @@ -14386,6 +11948,15 @@ "docstrings": [], "signature": "let some: ('a => bool, t<'a>) => bool" }, + { + "id": "Js.Array.somei", + "kind": "value", + "name": "somei", + "docstrings": [ + "Returns `true` if the predicate function given as the first argument to\n`somei()` returns `true` for any element in the array; `false` otherwise. The\npredicate function has two arguments: an item from the array and the index\nvalue\n\n## Examples\n\n```rescript\n// Does any string in the array\n// have the same length as its index?\n\nlet sameLength = (str, index) => Js.String.length(str) == index\n\n// \"ef\" has length 2 and is it at index 2\nJs.Array.somei(sameLength, [\"ab\", \"cd\", \"ef\", \"gh\"]) == true\n// no item has the same length as its index\nJs.Array.somei(sameLength, [\"a\", \"bc\", \"def\", \"gh\"]) == false\n```" + ], + "signature": "let somei: (t<'a>, ('a, int) => bool) => bool" + }, { "id": "Js.Array.somei", "kind": "value", diff --git a/data/api/latest/toc_tree.json b/data/api/latest/toc_tree.json index faf8bb122..a88e89b50 100644 --- a/data/api/latest/toc_tree.json +++ b/data/api/latest/toc_tree.json @@ -1 +1 @@ -{"js":{"name":"Js","path":["js"],"children":[{"name":"WeakMap","path":["js","weakmap"],"children":[]},{"name":"Map","path":["js","map"],"children":[]},{"name":"WeakSet","path":["js","weakset"],"children":[]},{"name":"Set","path":["js","set"],"children":[]},{"name":"Console","path":["js","console"],"children":[]},{"name":"Vector","path":["js","vector"],"children":[]},{"name":"List","path":["js","list"],"children":[]},{"name":"Result","path":["js","result"],"children":[]},{"name":"Option","path":["js","option"],"children":[]},{"name":"Blob","path":["js","blob"],"children":[]},{"name":"File","path":["js","file"],"children":[]},{"name":"BigInt","path":["js","bigint"],"children":[]},{"name":"Int","path":["js","int"],"children":[]},{"name":"Float","path":["js","float"],"children":[]},{"name":"Types","path":["js","types"],"children":[]},{"name":"TypedArray2","path":["js","typedarray2"],"children":[{"name":"DataView","path":["js","typedarray2","dataview"],"children":[]},{"name":"Float64Array","path":["js","typedarray2","float64array"],"children":[]},{"name":"Float32Array","path":["js","typedarray2","float32array"],"children":[]},{"name":"Uint32Array","path":["js","typedarray2","uint32array"],"children":[]},{"name":"Int32Array","path":["js","typedarray2","int32array"],"children":[]},{"name":"Uint16Array","path":["js","typedarray2","uint16array"],"children":[]},{"name":"Int16Array","path":["js","typedarray2","int16array"],"children":[]},{"name":"Uint8ClampedArray","path":["js","typedarray2","uint8clampedarray"],"children":[]},{"name":"Uint8Array","path":["js","typedarray2","uint8array"],"children":[]},{"name":"Int8Array","path":["js","typedarray2","int8array"],"children":[]},{"name":"ArrayBuffer","path":["js","typedarray2","arraybuffer"],"children":[]}]},{"name":"Typed_array","path":["js","typed_array"],"children":[{"name":"DataView","path":["js","typed_array","dataview"],"children":[]},{"name":"Float64_array","path":["js","typed_array","float64_array"],"children":[]},{"name":"Float64Array","path":["js","typed_array","float64array"],"children":[]},{"name":"Float32_array","path":["js","typed_array","float32_array"],"children":[]},{"name":"Float32Array","path":["js","typed_array","float32array"],"children":[]},{"name":"Uint32Array","path":["js","typed_array","uint32array"],"children":[]},{"name":"Int32_array","path":["js","typed_array","int32_array"],"children":[]},{"name":"Int32Array","path":["js","typed_array","int32array"],"children":[]},{"name":"Uint16Array","path":["js","typed_array","uint16array"],"children":[]},{"name":"Int16Array","path":["js","typed_array","int16array"],"children":[]},{"name":"Uint8ClampedArray","path":["js","typed_array","uint8clampedarray"],"children":[]},{"name":"Uint8Array","path":["js","typed_array","uint8array"],"children":[]},{"name":"Int8Array","path":["js","typed_array","int8array"],"children":[]},{"name":"S","path":["js","typed_array","s"],"children":[]},{"name":"ArrayBuffer","path":["js","typed_array","arraybuffer"],"children":[]},{"name":"Type","path":["js","typed_array","type"],"children":[]}]},{"name":"Obj","path":["js","obj"],"children":[]},{"name":"Math","path":["js","math"],"children":[]},{"name":"Json","path":["js","json"],"children":[{"name":"Kind","path":["js","json","kind"],"children":[]}]},{"name":"Global","path":["js","global"],"children":[]},{"name":"Dict","path":["js","dict"],"children":[]},{"name":"Date","path":["js","date"],"children":[]},{"name":"Promise2","path":["js","promise2"],"children":[]},{"name":"Promise","path":["js","promise"],"children":[]},{"name":"Re","path":["js","re"],"children":[]},{"name":"String2","path":["js","string2"],"children":[]},{"name":"String","path":["js","string"],"children":[]},{"name":"Array2","path":["js","array2"],"children":[]},{"name":"Array","path":["js","array"],"children":[]},{"name":"Exn","path":["js","exn"],"children":[]},{"name":"Null_undefined","path":["js","null_undefined"],"children":[]},{"name":"Nullable","path":["js","nullable"],"children":[]},{"name":"Undefined","path":["js","undefined"],"children":[]},{"name":"Null","path":["js","null"],"children":[]}]},"belt":{"name":"Belt","path":["belt"],"children":[{"name":"Float","path":["belt","float"],"children":[]},{"name":"Int","path":["belt","int"],"children":[]},{"name":"Result","path":["belt","result"],"children":[]},{"name":"Option","path":["belt","option"],"children":[]},{"name":"HashMap","path":["belt","hashmap"],"children":[{"name":"String","path":["belt","hashmap","string"],"children":[]},{"name":"Int","path":["belt","hashmap","int"],"children":[]}]},{"name":"HashSet","path":["belt","hashset"],"children":[{"name":"String","path":["belt","hashset","string"],"children":[]},{"name":"Int","path":["belt","hashset","int"],"children":[]}]},{"name":"MutableMap","path":["belt","mutablemap"],"children":[{"name":"String","path":["belt","mutablemap","string"],"children":[]},{"name":"Int","path":["belt","mutablemap","int"],"children":[]}]},{"name":"MutableSet","path":["belt","mutableset"],"children":[{"name":"String","path":["belt","mutableset","string"],"children":[]},{"name":"Int","path":["belt","mutableset","int"],"children":[]}]},{"name":"Map","path":["belt","map"],"children":[{"name":"Dict","path":["belt","map","dict"],"children":[]},{"name":"String","path":["belt","map","string"],"children":[]},{"name":"Int","path":["belt","map","int"],"children":[]}]},{"name":"Set","path":["belt","set"],"children":[{"name":"Dict","path":["belt","set","dict"],"children":[]},{"name":"String","path":["belt","set","string"],"children":[]},{"name":"Int","path":["belt","set","int"],"children":[]}]},{"name":"Range","path":["belt","range"],"children":[]},{"name":"List","path":["belt","list"],"children":[]},{"name":"MutableStack","path":["belt","mutablestack"],"children":[]},{"name":"MutableQueue","path":["belt","mutablequeue"],"children":[]},{"name":"SortArray","path":["belt","sortarray"],"children":[{"name":"String","path":["belt","sortarray","string"],"children":[]},{"name":"Int","path":["belt","sortarray","int"],"children":[]}]},{"name":"Array","path":["belt","array"],"children":[]},{"name":"Id","path":["belt","id"],"children":[{"name":"MakeHashable","path":["belt","id","makehashable"],"children":[]},{"name":"MakeHashableU","path":["belt","id","makehashableu"],"children":[]},{"name":"MakeComparable","path":["belt","id","makecomparable"],"children":[]},{"name":"MakeComparableU","path":["belt","id","makecomparableu"],"children":[]}]}]},"dom":{"name":"Dom","path":["dom"],"children":[{"name":"Storage2","path":["dom","storage2"],"children":[]},{"name":"Storage","path":["dom","storage"],"children":[]}]},"core":{"name":"Core","path":["core"],"children":[{"name":"Result","path":["core","result"],"children":[]},{"name":"List","path":["core","list"],"children":[]},{"name":"Option","path":["core","option"],"children":[]},{"name":"Exn","path":["core","exn"],"children":[]},{"name":"Intl","path":["core","intl"],"children":[{"name":"Segments","path":["core","intl","segments"],"children":[]},{"name":"Segmenter","path":["core","intl","segmenter"],"children":[]},{"name":"RelativeTimeFormat","path":["core","intl","relativetimeformat"],"children":[]},{"name":"PluralRules","path":["core","intl","pluralrules"],"children":[]},{"name":"NumberFormat","path":["core","intl","numberformat"],"children":[{"name":"Grouping","path":["core","intl","numberformat","grouping"],"children":[]}]},{"name":"Locale","path":["core","intl","locale"],"children":[]},{"name":"ListFormat","path":["core","intl","listformat"],"children":[]},{"name":"DateTimeFormat","path":["core","intl","datetimeformat"],"children":[]},{"name":"Collator","path":["core","intl","collator"],"children":[]},{"name":"Common","path":["core","intl","common"],"children":[]}]},{"name":"BigUint64Array","path":["core","biguint64array"],"children":[{"name":"Constants","path":["core","biguint64array","constants"],"children":[]}]},{"name":"BigInt64Array","path":["core","bigint64array"],"children":[{"name":"Constants","path":["core","bigint64array","constants"],"children":[]}]},{"name":"Uint8ClampedArray","path":["core","uint8clampedarray"],"children":[{"name":"Constants","path":["core","uint8clampedarray","constants"],"children":[]}]},{"name":"Uint32Array","path":["core","uint32array"],"children":[{"name":"Constants","path":["core","uint32array","constants"],"children":[]}]},{"name":"Uint16Array","path":["core","uint16array"],"children":[{"name":"Constants","path":["core","uint16array","constants"],"children":[]}]},{"name":"Uint8Array","path":["core","uint8array"],"children":[{"name":"Constants","path":["core","uint8array","constants"],"children":[]}]},{"name":"Int32Array","path":["core","int32array"],"children":[{"name":"Constants","path":["core","int32array","constants"],"children":[]}]},{"name":"Int16Array","path":["core","int16array"],"children":[{"name":"Constants","path":["core","int16array","constants"],"children":[]}]},{"name":"Int8Array","path":["core","int8array"],"children":[{"name":"Constants","path":["core","int8array","constants"],"children":[]}]},{"name":"Float64Array","path":["core","float64array"],"children":[{"name":"Constants","path":["core","float64array","constants"],"children":[]}]},{"name":"Float32Array","path":["core","float32array"],"children":[{"name":"Constants","path":["core","float32array","constants"],"children":[]}]},{"name":"TypedArray","path":["core","typedarray"],"children":[]},{"name":"ArrayBuffer","path":["core","arraybuffer"],"children":[]},{"name":"WeakSet","path":["core","weakset"],"children":[]},{"name":"Set","path":["core","set"],"children":[]},{"name":"WeakMap","path":["core","weakmap"],"children":[]},{"name":"Map","path":["core","map"],"children":[]},{"name":"AsyncIterator","path":["core","asynciterator"],"children":[]},{"name":"Iterator","path":["core","iterator"],"children":[]},{"name":"JSON","path":["core","json"],"children":[{"name":"Decode","path":["core","json","decode"],"children":[]},{"name":"Encode","path":["core","json","encode"],"children":[]},{"name":"Classify","path":["core","json","classify"],"children":[]}]},{"name":"Type","path":["core","type"],"children":[{"name":"Classify","path":["core","type","classify"],"children":[]}]},{"name":"Symbol","path":["core","symbol"],"children":[]},{"name":"String","path":["core","string"],"children":[]},{"name":"RegExp","path":["core","regexp"],"children":[{"name":"Result","path":["core","regexp","result"],"children":[]}]},{"name":"Promise","path":["core","promise"],"children":[]},{"name":"Ordering","path":["core","ordering"],"children":[]},{"name":"Object","path":["core","object"],"children":[]},{"name":"Nullable","path":["core","nullable"],"children":[]},{"name":"Null","path":["core","null"],"children":[]},{"name":"Math","path":["core","math"],"children":[{"name":"Int","path":["core","math","int"],"children":[]},{"name":"Constants","path":["core","math","constants"],"children":[]}]},{"name":"BigInt","path":["core","bigint"],"children":[]},{"name":"Int","path":["core","int"],"children":[{"name":"Constants","path":["core","int","constants"],"children":[]}]},{"name":"Float","path":["core","float"],"children":[{"name":"Constants","path":["core","float","constants"],"children":[]}]},{"name":"Error","path":["core","error"],"children":[{"name":"URIError","path":["core","error","urierror"],"children":[]},{"name":"TypeError","path":["core","error","typeerror"],"children":[]},{"name":"SyntaxError","path":["core","error","syntaxerror"],"children":[]},{"name":"ReferenceError","path":["core","error","referenceerror"],"children":[]},{"name":"RangeError","path":["core","error","rangeerror"],"children":[]},{"name":"EvalError","path":["core","error","evalerror"],"children":[]}]},{"name":"Dict","path":["core","dict"],"children":[]},{"name":"Date","path":["core","date"],"children":[{"name":"UTC","path":["core","date","utc"],"children":[]}]},{"name":"DataView","path":["core","dataview"],"children":[]},{"name":"Console","path":["core","console"],"children":[]},{"name":"Array","path":["core","array"],"children":[]}]}} \ No newline at end of file +{"js":{"name":"Js","path":["js"],"children":[{"name":"WeakMap","path":["js","weakmap"],"children":[]},{"name":"Map","path":["js","map"],"children":[]},{"name":"WeakSet","path":["js","weakset"],"children":[]},{"name":"Set","path":["js","set"],"children":[]},{"name":"Console","path":["js","console"],"children":[]},{"name":"Result","path":["js","result"],"children":[]},{"name":"Option","path":["js","option"],"children":[]},{"name":"Blob","path":["js","blob"],"children":[]},{"name":"File","path":["js","file"],"children":[]},{"name":"BigInt","path":["js","bigint"],"children":[]},{"name":"Int","path":["js","int"],"children":[]},{"name":"Float","path":["js","float"],"children":[]},{"name":"Types","path":["js","types"],"children":[]},{"name":"TypedArray2","path":["js","typedarray2"],"children":[{"name":"DataView","path":["js","typedarray2","dataview"],"children":[]},{"name":"Float64Array","path":["js","typedarray2","float64array"],"children":[]},{"name":"Float32Array","path":["js","typedarray2","float32array"],"children":[]},{"name":"Uint32Array","path":["js","typedarray2","uint32array"],"children":[]},{"name":"Int32Array","path":["js","typedarray2","int32array"],"children":[]},{"name":"Uint16Array","path":["js","typedarray2","uint16array"],"children":[]},{"name":"Int16Array","path":["js","typedarray2","int16array"],"children":[]},{"name":"Uint8ClampedArray","path":["js","typedarray2","uint8clampedarray"],"children":[]},{"name":"Uint8Array","path":["js","typedarray2","uint8array"],"children":[]},{"name":"Int8Array","path":["js","typedarray2","int8array"],"children":[]},{"name":"ArrayBuffer","path":["js","typedarray2","arraybuffer"],"children":[]}]},{"name":"Typed_array","path":["js","typed_array"],"children":[{"name":"DataView","path":["js","typed_array","dataview"],"children":[]},{"name":"Float64_array","path":["js","typed_array","float64_array"],"children":[]},{"name":"Float64Array","path":["js","typed_array","float64array"],"children":[]},{"name":"Float32_array","path":["js","typed_array","float32_array"],"children":[]},{"name":"Float32Array","path":["js","typed_array","float32array"],"children":[]},{"name":"Uint32Array","path":["js","typed_array","uint32array"],"children":[]},{"name":"Int32_array","path":["js","typed_array","int32_array"],"children":[]},{"name":"Int32Array","path":["js","typed_array","int32array"],"children":[]},{"name":"Uint16Array","path":["js","typed_array","uint16array"],"children":[]},{"name":"Int16Array","path":["js","typed_array","int16array"],"children":[]},{"name":"Uint8ClampedArray","path":["js","typed_array","uint8clampedarray"],"children":[]},{"name":"Uint8Array","path":["js","typed_array","uint8array"],"children":[]},{"name":"Int8Array","path":["js","typed_array","int8array"],"children":[]},{"name":"S","path":["js","typed_array","s"],"children":[]},{"name":"ArrayBuffer","path":["js","typed_array","arraybuffer"],"children":[]},{"name":"Type","path":["js","typed_array","type"],"children":[]}]},{"name":"Obj","path":["js","obj"],"children":[]},{"name":"Math","path":["js","math"],"children":[]},{"name":"Json","path":["js","json"],"children":[{"name":"Kind","path":["js","json","kind"],"children":[]}]},{"name":"Global","path":["js","global"],"children":[]},{"name":"Dict","path":["js","dict"],"children":[]},{"name":"Date","path":["js","date"],"children":[]},{"name":"Promise2","path":["js","promise2"],"children":[]},{"name":"Promise","path":["js","promise"],"children":[]},{"name":"Re","path":["js","re"],"children":[]},{"name":"String2","path":["js","string2"],"children":[]},{"name":"String","path":["js","string"],"children":[]},{"name":"Array2","path":["js","array2"],"children":[]},{"name":"Array","path":["js","array"],"children":[]},{"name":"Exn","path":["js","exn"],"children":[]},{"name":"Null_undefined","path":["js","null_undefined"],"children":[]},{"name":"Nullable","path":["js","nullable"],"children":[]},{"name":"Undefined","path":["js","undefined"],"children":[]},{"name":"Null","path":["js","null"],"children":[]}]},"belt":{"name":"Belt","path":["belt"],"children":[{"name":"Float","path":["belt","float"],"children":[]},{"name":"Int","path":["belt","int"],"children":[]},{"name":"Result","path":["belt","result"],"children":[]},{"name":"Option","path":["belt","option"],"children":[]},{"name":"HashMap","path":["belt","hashmap"],"children":[{"name":"String","path":["belt","hashmap","string"],"children":[]},{"name":"Int","path":["belt","hashmap","int"],"children":[]}]},{"name":"HashSet","path":["belt","hashset"],"children":[{"name":"String","path":["belt","hashset","string"],"children":[]},{"name":"Int","path":["belt","hashset","int"],"children":[]}]},{"name":"MutableMap","path":["belt","mutablemap"],"children":[{"name":"String","path":["belt","mutablemap","string"],"children":[]},{"name":"Int","path":["belt","mutablemap","int"],"children":[]}]},{"name":"MutableSet","path":["belt","mutableset"],"children":[{"name":"String","path":["belt","mutableset","string"],"children":[]},{"name":"Int","path":["belt","mutableset","int"],"children":[]}]},{"name":"Map","path":["belt","map"],"children":[{"name":"Dict","path":["belt","map","dict"],"children":[]},{"name":"String","path":["belt","map","string"],"children":[]},{"name":"Int","path":["belt","map","int"],"children":[]}]},{"name":"Set","path":["belt","set"],"children":[{"name":"Dict","path":["belt","set","dict"],"children":[]},{"name":"String","path":["belt","set","string"],"children":[]},{"name":"Int","path":["belt","set","int"],"children":[]}]},{"name":"Range","path":["belt","range"],"children":[]},{"name":"List","path":["belt","list"],"children":[]},{"name":"MutableStack","path":["belt","mutablestack"],"children":[]},{"name":"MutableQueue","path":["belt","mutablequeue"],"children":[]},{"name":"SortArray","path":["belt","sortarray"],"children":[{"name":"String","path":["belt","sortarray","string"],"children":[]},{"name":"Int","path":["belt","sortarray","int"],"children":[]}]},{"name":"Array","path":["belt","array"],"children":[]},{"name":"Id","path":["belt","id"],"children":[{"name":"MakeHashable","path":["belt","id","makehashable"],"children":[]},{"name":"MakeHashableU","path":["belt","id","makehashableu"],"children":[]},{"name":"MakeComparable","path":["belt","id","makecomparable"],"children":[]},{"name":"MakeComparableU","path":["belt","id","makecomparableu"],"children":[]}]}]},"dom":{"name":"Dom","path":["dom"],"children":[{"name":"Storage2","path":["dom","storage2"],"children":[]},{"name":"Storage","path":["dom","storage"],"children":[]}]},"core":{"name":"Core","path":["core"],"children":[{"name":"Result","path":["core","result"],"children":[]},{"name":"List","path":["core","list"],"children":[]},{"name":"Option","path":["core","option"],"children":[]},{"name":"Exn","path":["core","exn"],"children":[]},{"name":"Intl","path":["core","intl"],"children":[{"name":"Segments","path":["core","intl","segments"],"children":[]},{"name":"Segmenter","path":["core","intl","segmenter"],"children":[]},{"name":"RelativeTimeFormat","path":["core","intl","relativetimeformat"],"children":[]},{"name":"PluralRules","path":["core","intl","pluralrules"],"children":[]},{"name":"NumberFormat","path":["core","intl","numberformat"],"children":[{"name":"Grouping","path":["core","intl","numberformat","grouping"],"children":[]}]},{"name":"Locale","path":["core","intl","locale"],"children":[]},{"name":"ListFormat","path":["core","intl","listformat"],"children":[]},{"name":"DateTimeFormat","path":["core","intl","datetimeformat"],"children":[]},{"name":"Collator","path":["core","intl","collator"],"children":[]},{"name":"Common","path":["core","intl","common"],"children":[]}]},{"name":"BigUint64Array","path":["core","biguint64array"],"children":[{"name":"Constants","path":["core","biguint64array","constants"],"children":[]}]},{"name":"BigInt64Array","path":["core","bigint64array"],"children":[{"name":"Constants","path":["core","bigint64array","constants"],"children":[]}]},{"name":"Uint8ClampedArray","path":["core","uint8clampedarray"],"children":[{"name":"Constants","path":["core","uint8clampedarray","constants"],"children":[]}]},{"name":"Uint32Array","path":["core","uint32array"],"children":[{"name":"Constants","path":["core","uint32array","constants"],"children":[]}]},{"name":"Uint16Array","path":["core","uint16array"],"children":[{"name":"Constants","path":["core","uint16array","constants"],"children":[]}]},{"name":"Uint8Array","path":["core","uint8array"],"children":[{"name":"Constants","path":["core","uint8array","constants"],"children":[]}]},{"name":"Int32Array","path":["core","int32array"],"children":[{"name":"Constants","path":["core","int32array","constants"],"children":[]}]},{"name":"Int16Array","path":["core","int16array"],"children":[{"name":"Constants","path":["core","int16array","constants"],"children":[]}]},{"name":"Int8Array","path":["core","int8array"],"children":[{"name":"Constants","path":["core","int8array","constants"],"children":[]}]},{"name":"Float64Array","path":["core","float64array"],"children":[{"name":"Constants","path":["core","float64array","constants"],"children":[]}]},{"name":"Float32Array","path":["core","float32array"],"children":[{"name":"Constants","path":["core","float32array","constants"],"children":[]}]},{"name":"TypedArray","path":["core","typedarray"],"children":[]},{"name":"ArrayBuffer","path":["core","arraybuffer"],"children":[]},{"name":"WeakSet","path":["core","weakset"],"children":[]},{"name":"Set","path":["core","set"],"children":[]},{"name":"WeakMap","path":["core","weakmap"],"children":[]},{"name":"Map","path":["core","map"],"children":[]},{"name":"AsyncIterator","path":["core","asynciterator"],"children":[]},{"name":"Iterator","path":["core","iterator"],"children":[]},{"name":"JSON","path":["core","json"],"children":[{"name":"Decode","path":["core","json","decode"],"children":[]},{"name":"Encode","path":["core","json","encode"],"children":[]},{"name":"Classify","path":["core","json","classify"],"children":[]}]},{"name":"Type","path":["core","type"],"children":[{"name":"Classify","path":["core","type","classify"],"children":[]}]},{"name":"Symbol","path":["core","symbol"],"children":[]},{"name":"String","path":["core","string"],"children":[]},{"name":"RegExp","path":["core","regexp"],"children":[{"name":"Result","path":["core","regexp","result"],"children":[]}]},{"name":"Promise","path":["core","promise"],"children":[]},{"name":"Ordering","path":["core","ordering"],"children":[]},{"name":"Object","path":["core","object"],"children":[]},{"name":"Nullable","path":["core","nullable"],"children":[]},{"name":"Null","path":["core","null"],"children":[]},{"name":"Math","path":["core","math"],"children":[{"name":"Int","path":["core","math","int"],"children":[]},{"name":"Constants","path":["core","math","constants"],"children":[]}]},{"name":"BigInt","path":["core","bigint"],"children":[]},{"name":"Int","path":["core","int"],"children":[{"name":"Bitwise","path":["core","int","bitwise"],"children":[]},{"name":"Constants","path":["core","int","constants"],"children":[]}]},{"name":"Float","path":["core","float"],"children":[{"name":"Constants","path":["core","float","constants"],"children":[]}]},{"name":"Error","path":["core","error"],"children":[{"name":"URIError","path":["core","error","urierror"],"children":[]},{"name":"TypeError","path":["core","error","typeerror"],"children":[]},{"name":"SyntaxError","path":["core","error","syntaxerror"],"children":[]},{"name":"ReferenceError","path":["core","error","referenceerror"],"children":[]},{"name":"RangeError","path":["core","error","rangeerror"],"children":[]},{"name":"EvalError","path":["core","error","evalerror"],"children":[]}]},{"name":"Dict","path":["core","dict"],"children":[]},{"name":"Date","path":["core","date"],"children":[{"name":"UTC","path":["core","date","utc"],"children":[]}]},{"name":"DataView","path":["core","dataview"],"children":[]},{"name":"Console","path":["core","console"],"children":[]},{"name":"Array","path":["core","array"],"children":[]}]}} \ No newline at end of file diff --git a/data/sidebar_gentype_latest.json b/data/sidebar_gentype_latest.json deleted file mode 100644 index d9a57bf9c..000000000 --- a/data/sidebar_gentype_latest.json +++ /dev/null @@ -1,10 +0,0 @@ -{ - "Overview": [ - "introduction", - "getting-started", - "usage" - ], - "Advanced": [ - "supported-types" - ] -} diff --git a/data/sidebar_manual_latest.json b/data/sidebar_manual_latest.json deleted file mode 100644 index 6e6c86255..000000000 --- a/data/sidebar_manual_latest.json +++ /dev/null @@ -1,78 +0,0 @@ -{ - "Overview": [ - "introduction", - "installation", - "migrate-to-v11", - "editor-plugins", - "try" - ], - "Language Features": [ - "overview", - "let-binding", - "type", - "primitive-types", - "tuple", - "record", - "object", - "variant", - "polymorphic-variant", - "null-undefined-option", - "array-and-list", - "function", - "control-flow", - "pipe", - "pattern-matching-destructuring", - "mutation", - "jsx", - "exception", - "lazy-values", - "promise", - "async-await", - "tagged-templates", - "module", - "import-export", - "attribute", - "reserved-keywords", - "equality-comparison" - ], - "Advanced Features": [ - "extensible-variant", - "scoped-polymorphic-types", - "module-functions" - ], - "JavaScript Interop": [ - "interop-cheatsheet", - "embed-raw-javascript", - "shared-data-types", - "external", - "bind-to-js-object", - "bind-to-js-function", - "import-from-export-to-js", - "bind-to-global-js-values", - "json", - "inlining-constants", - "use-illegal-identifier-names", - "generate-converters-accessors", - "browser-support-polyfills", - "libraries", - "typescript-integration" - ], - "Build System": [ - "build-overview", - "build-configuration", - "build-configuration-schema", - "build-external-stdlib", - "build-pinned-dependencies", - "interop-with-js-build-systems", - "build-performance", - "warning-numbers" - ], - "Guides": [ - "converting-from-js" - ], - "Extra": [ - "newcomer-examples", - "project-structure", - "faq" - ] -} \ No newline at end of file diff --git a/next.config.mjs b/next.config.mjs index a2d4e8df0..e8f0ff144 100644 --- a/next.config.mjs +++ b/next.config.mjs @@ -1,3 +1,4 @@ +// @ts-check import fs from "fs"; import webpack from "webpack"; import rehypeSlug from "rehype-slug"; @@ -7,17 +8,19 @@ import remarkFrontmatter from "remark-frontmatter"; import remarkMdxFrontmatter from "remark-mdx-frontmatter"; import { createLoader } from "simple-functional-loader"; -const bsconfig = JSON.parse(fs.readFileSync("./rescript.json")); +const bsconfig = JSON.parse(fs.readFileSync("./rescript.json").toString()); const { ProvidePlugin } = webpack; const transpileModules = ["rescript"].concat(bsconfig["bs-dependencies"]); const config = { - output: process.env.BUILD_STATIC === 'true' ? 'export' : undefined, + output: process.env.BUILD_STATIC === "true" ? "export" : undefined, pageExtensions: ["jsx", "js", "bs.js", "mdx", "mjs"], env: { ENV: process.env.NODE_ENV, + VERSION_LATEST: process.env.VERSION_LATEST, + VERSION_NEXT: process.env.VERSION_NEXT, }, swcMinify: false, webpack: (config, options) => { @@ -81,6 +84,20 @@ const config = { config.plugins.push(new ProvidePlugin({ React: "react" })); return config; }, + async rewrites() { + return { + beforeFiles: [ + { + source: "/docs/manual/latest/:slug", + destination: `/docs/manual/${process.env.VERSION_LATEST}/:slug`, + }, + { + source: "/docs/manual/next/:slug", + destination: `/docs/manual/${process.env.VERSION_NEXT}/:slug`, + }, + ], + }; + }, async redirects() { return [ { diff --git a/package.json b/package.json index 094138d55..75d4f00f5 100644 --- a/package.json +++ b/package.json @@ -57,7 +57,7 @@ "unified": "^8.4.0" }, "scripts": { - "dev": "next", + "dev": "NODE_OPTIONS='--inspect' next dev", "build": "rescript && npm run update-index && next build", "test": "node scripts/test-examples.mjs && node scripts/test-hrefs.mjs", "reanalyze": "reanalyze -all-cmt .", diff --git a/pages/docs/latest.js b/pages/docs/latest.js deleted file mode 100644 index 8762e51e7..000000000 --- a/pages/docs/latest.js +++ /dev/null @@ -1,3 +0,0 @@ -import make from "src/DocsOverview.mjs"; - -export default make; diff --git a/pages/docs/manual/latest/api.mdx b/pages/docs/manual/latest/api.mdx deleted file mode 100644 index d166f925c..000000000 --- a/pages/docs/manual/latest/api.mdx +++ /dev/null @@ -1,18 +0,0 @@ -# Overview - -## ReScript Core - -[Core](api/core) is ReScript's new standard library. It replaces the complete `Js` module as well as some of the more frequently used modules from `Belt` and is recommended to use with uncurried mode. - -In ReScript 11, it is shipped as a separate npm package `@rescript/core` that is added to your project as per the [installation instructions](/docs/manual/latest/installation). In future ReScript versions, it will be included with the `rescript` npm package itself. - -## Additional Libraries - -ReScript ships with these two additional modules in its standard library: - -- [Belt](api/belt): immutable collections and extra helpers not available in JavaScript / [Core](api/core). -- [Dom](api/dom): Dom related types and modules. Contains our standardized types used by various userland DOM bindings. - -## Legacy Modules - -The [Js](api/js) module is superseded by [Core](api/core). diff --git a/pages/docs/manual/latest/api/[...slug].js b/pages/docs/manual/latest/api/[...slug].js deleted file mode 100644 index 2098f7129..000000000 --- a/pages/docs/manual/latest/api/[...slug].js +++ /dev/null @@ -1,18 +0,0 @@ -import { - getStaticPathsByVersion, - getStaticPropsByVersion -} from "src/ApiDocs.mjs"; - -import APIDocs from "src/ApiDocs.mjs"; - -export async function getStaticProps(ctx) { - return await getStaticPropsByVersion({ ...ctx, version: "latest" }); -} - -export async function getStaticPaths(ctx) { - return await getStaticPathsByVersion("latest"); -} - -export default function Comp(props) { - return ; -} diff --git a/pages/docs/manual/latest/array-and-list.mdx b/pages/docs/manual/latest/array-and-list.mdx deleted file mode 100644 index c411fe964..000000000 --- a/pages/docs/manual/latest/array-and-list.mdx +++ /dev/null @@ -1,219 +0,0 @@ ---- -title: "Array & List" -description: "Arrays and List data structures" -canonical: "/docs/manual/latest/array-and-list" ---- - -# Array and List - -## Array - -Arrays are our main ordered data structure. They work the same way as JavaScript arrays: they can be randomly accessed, dynamically resized, updated, etc. - - - -```res example -let myArray = ["hello", "world", "how are you"] -``` -```js -var myArray = ["hello", "world", "how are you"]; -``` - - - -ReScript arrays' items must have the same type, i.e. homogeneous. - -### Usage -#### Access -Accessing items in an array will return an `option` and can be done like so: - - - -```res example -let myArray = ["hello", "world", "how are you"] - -let firstItem = myArray[0] // Some("hello") - -let tenthItem = myArray->Array.get(10) // None -``` -```js -var myArray = ["hello", "world", "how are you"]; - -var firstItem = myArray[0]; - -var tenthItem = myArray[10]; -``` - - - -The behavior of returning an `option` is new to V11 when you have [Core](api/core) open. -It provides a safer way to access array items, which is especially useful when you're not sure if the index is out of bounds. -If you would like to not use an `option`, you can use [`Array.getUnsafe`](api/core/array#value-getUnsafe). - -#### Update -Items in an array can be updated by assigning a value to an index or using a function: - - - -```res example -let myArray = ["hello", "world", "how are you"] - -myArray[0] = "hey" // now ["hey", "world", "how are you"] - -myArray->Array.push("?") // ["hey", "world", "how are you", "?"] - -myArray->Array.set(0, "bye") // ["bye", "world", "how are you", "?"] -``` -```js -var myArray = ["hello", "world", "how are you"]; - -myArray[0] = "hey"; - -myArray.push("?"); - -myArray[0] = "bye"; -``` - - - -### Array spreads - -**Since 11.1** - - -You can spread arrays of the the same type into new arrays, just like in JavaScript: - - - -```res example -let y = [1, 2] -let x = [4, 5, ...y] -let x2 = [4, 5, ...y, 7, ...y] -let x3 = [...y] -``` - -```javascript -var Belt_Array = require("rescript/lib/js/belt_Array.js"); - -var y = [ - 1, - 2 -]; - -var x = Belt_Array.concatMany([ - [ - 4, - 5 - ], - y - ]); - -var x2 = Belt_Array.concatMany([ - [ - 4, - 5 - ], - y, - [7], - y - ]); - -var x3 = Belt_Array.concatMany([y]); -``` - - -> Note that array spreads compiles to `Belt.Array.concatMany` right now. This is likely to change to native array spreads in the future. - -## List - -ReScript provides a singly linked list too. Lists are: - -- immutable -- fast at prepending items -- fast at getting the head -- slow at everything else - - - -```res example -let myList = list{1, 2, 3} -``` -```js -var myList = { - hd: 1, - tl: { - hd: 2, - tl: { - hd: 3, - tl: 0 - } - } -}; -``` - - - -Like arrays, lists' items need to be of the same type. - -### Usage - -You'd use list for its resizability, its fast prepend (adding at the head), and its fast split, all of which are immutable and relatively efficient. - -Do **not** use list if you need to randomly access an item or insert at non-head position. Your code would end up obtuse and/or slow. - -The standard lib provides a [List module](api/core/list). - -#### Immutable Prepend - -Use the spread syntax: - - - -```res prelude -let myList = list{1, 2, 3} -let anotherList = list{0, ...myList} -``` -```js -var myList = { - hd: 1, - tl: { - hd: 2, - tl: { - hd: 3, - tl: 0 - } - } -}; - -var anotherList = { - hd: 0, - tl: myList -}; -``` - - - -`myList` didn't mutate. `anotherList` is now `list{0, 1, 2, 3}`. This is efficient (constant time, not linear). `anotherList`'s last 3 elements are shared with `myList`! - -**Note that `list{a, ...b, ...c}` was a syntax error** before compiler v10.1. In general, the pattern should be used with care as its performance and allocation overhead are linear (`O(n)`). - -#### Access - -`switch` (described in the [pattern matching section](pattern-matching-destructuring.md)) is usually used to access list items: - - - -```res example -let message = - switch myList { - | list{} => "This list is empty" - | list{a, ...rest} => "The head of the list is the string " ++ Int.toString(a) - } -``` -```js -var message = myList - ? "The head of the list is the string " + (1).toString() - : "This list is empty"; -``` - - diff --git a/pages/docs/manual/latest/async-await.mdx b/pages/docs/manual/latest/async-await.mdx deleted file mode 100644 index 5688e41b3..000000000 --- a/pages/docs/manual/latest/async-await.mdx +++ /dev/null @@ -1,326 +0,0 @@ ---- -title: "Async / Await" -description: "Async / await for asynchronous operations" -canonical: "/docs/manual/latest/async-await" ---- - - -
- -```res prelude -@val external fetchUserMail: string => promise = "GlobalAPI.fetchUserMail" -@val external sendAnalytics: string => promise = "GlobalAPI.sendAnalytics" -``` - -
- - - -# Async / Await - -ReScript comes with `async` / `await` support to make asynchronous, `Promise` based code easier to read and write. This feature is very similar to its JS equivalent, so if you are already familiar with JS' `async` / `await`, you will feel right at home. - -## How it looks - -Let's start with a quick example to show-case the syntax: - - - - -```res -// Some fictive functionality that offers asynchronous network actions -@val external fetchUserMail: string => promise = "GlobalAPI.fetchUserMail" -@val external sendAnalytics: string => promise = "GlobalAPI.sendAnalytics" - -// We use the `async` keyword to allow the use of `await` in the function body -let logUserDetails = async (userId: string) => { - // We use `await` to fetch the user email from our fictive user endpoint - let email = await fetchUserMail(userId) - - await sendAnalytics(`User details have been logged for ${userId}`) - - Console.log(`Email address for user ${userId}: ${email}`) -} -``` - -```js -async function logUserDetails(userId) { - var email = await GlobalAPI.fetchUserMail(userId); - await GlobalAPI.sendAnalytics("User details have been logged for " + userId + ""); - console.log("Email address for user " + userId + ": " + email + ""); -} -``` - - - -As we can see above, an `async` function is defined via the `async` keyword right before the function's parameter list. In the function body, we are now able to use the `await` keyword to explicitly wait for a `Promise` value and assign its content to a let binding `email`. - -You will probably notice that this looks very similar to `async` / `await` in JS, but there are still a few details that are specific to ReScript. The next few sections will go through all the details that are specific to the ReScript type system. - -## Basics - -- You may only use `await` in `async` function bodies -- `await` may only be called on a `promise` value -- `await` calls are expressions, therefore they can be used in pattern matching (`switch`) -- A function returning a `promise<'a>` is equivalent to an `async` function returning a value `'a` (important for writing signature files and bindings) -- `promise` values and types returned from an `async` function don't auto-collapse into a "flat promise" like in JS (more on this later) - - -## Types and `async` functions - -### `async` function type signatures - -Function type signatures (i.e defined in signature files) don't require any special keywords for `async` usage. Whenever you want to type an `async` function, use a `promise` return type. - -```resi -// Demo.resi - -let fetchUserMail: string => promise -``` - -The same logic applies to type definitions in `.res` files: - -```res example -// function type -type someAsyncFn = int => promise - -// Function type annotation -let fetchData: string => promise = async (userId) => { - await fetchUserMail(userId) -} -``` - -**BUT:** When typing `async` functions in your implementation files, you need to omit the `promise<'a>` type: - -```res -// This function is compiled into a `string => promise` type. -// The promise<...> part is implicitly added by the compiler. -let fetchData = async (userId: string): string => { - await fetchUserMail("test") -} -``` - -For completeness reasons, let's expand the full signature and inline type definitions in one code snippet: - -```res -// Note how the inline return type uses `string`, while the type definition uses `promise` -let fetchData: string => promise = async (userId: string): string { - await fetchUserMail(userId) -} -``` - -**Note:** In a practical scenario you'd either use a type signature, or inline types, not both at the same time. In case you are interested in the design decisions, check out [this discussion](https://github.com/rescript-lang/rescript/pull/5913#issuecomment-1359003870). - -### Promises don't auto-collapse in async functions - -In JS, nested promises (i.e. `promise>`) will automatically collapse into a flat promise (`promise<'a>`). This is not the case in ReScript. Use the `await` function to manually unwrap any nested promises within an `async` function instead. - -```res -let fetchData = async (userId: string): string => { - // We can't just return the result of `fetchUserMail`, otherwise we'd get a - // type error due to our function return type of type `string` - await fetchUserMail(userId) -} -``` - -## Error handling - -You may use `try / catch` or `switch` to handle exceptions during async execution. - -```res -// For simulation purposes -let authenticate = async () => { - raise(Exn.raiseRangeError("Authentication failed.")) -} - -let checkAuth = async () => { - try { - await authenticate() - } catch { - | Exn.Error(e) => - switch Exn.message(e) { - | Some(msg) => Console.log("JS error thrown: " ++ msg) - | None => Console.log("Some other exception has been thrown") - } - } -} -``` - -Note how we are essentially catching JS errors the same way as described in our [Exception](exception#catch-rescript-exceptions-from-js) section. - -You may unify error and value handling in a single switch as well: - -```res -let authenticate = async () => { - raise(Exn.raiseRangeError("Authentication failed.")) -} - -let checkAuth = async () => { - switch await authenticate() { - | _ => Console.log("ok") - | exception Exn.Error(e) => - switch Exn.message(e) { - | Some(msg) => Console.log("JS error thrown: " ++ msg) - | None => Console.log("Some other exception has been thrown") - } - } -} -``` - -**Important:** When using `await` with a `switch`, always make sure to put the actual await call in the `switch` expression, otherwise your `await` error will not be caught. - -## Piping `await` calls - -You may want to pipe the result of an `await` call right into another function. -This can be done by wrapping your `await` calls in a new `{}` closure. - - - -```res example -@val external fetchUserMail: string => promise = "GlobalAPI.fetchUserMail" - -let fetchData = async () => { - let mail = {await fetchUserMail("1234")}->String.toUpperCase - Console.log(`All upper-cased mail: ${mail}`) -} -``` - -```js -async function fetchData(param) { - var mail = (await GlobalAPI.fetchUserMail("1234")).toUpperCase(); - console.log("All upper-cased mail: " + mail + ""); -} -``` - - - -Note how the original closure was removed in the final JS output. No extra allocations! - -## Pattern matching on `await` calls - -`await` calls are just another kind of expression, so you can use `switch` pattern matching for more complex logic. - - - -```res example -@val external fetchUserMail: string => promise = "GlobalAPI.fetchUserMail" - -let fetchData = async () => { - switch (await fetchUserMail("user1"), await fetchUserMail("user2")) { - | (user1Mail, user2Mail) => { - Console.log("user 1 mail: " ++ user1Mail) - Console.log("user 2 mail: " ++ user2Mail) - } - - | exception JsError(err) => Console.log2("Some error occurred", err) - } -} -``` - -```js -async function fetchData(param) { - var val; - var val$1; - try { - val = await GlobalAPI.fetchUserMail("user1"); - val$1 = await GlobalAPI.fetchUserMail("user2"); - } - catch (raw_err){ - var err = Caml_js_exceptions.internalToOCamlException(raw_err); - if (err.RE_EXN_ID === "JsError") { - console.log("Some error occurred", err._1); - return ; - } - throw err; - } - console.log("user 1 mail: " + val); - console.log("user 2 mail: " + val$1); -} -``` - - - -## `await` multiple promises - -We can utilize the `Promise` module to handle multiple promises. E.g. let's use `Promise.all` to wait for multiple promises before continuing the program: - -```res -let pauseReturn = (value, timeout) => { - Promise.make((resolve, _reject) => { - setTimeout(() => { - resolve(value) - }, timeout)->ignore - }) -} - -let logMultipleValues = async () => { - let promise1 = pauseReturn("value1", 2000) - let promise2 = pauseReturn("value2", 1200) - let promise3 = pauseReturn("value3", 500) - - let all = await Promise.all([promise1, promise2, promise3]) - - switch all { - | [v1, v2, v3] => Console.log(`All values: ${v1}, ${v2}, ${v3}`) - | _ => Console.log("this should never happen") - } -} -``` - -## JS Interop with `async` functions - -`async` / `await` practically works with any function that returns a `promise<'a>` value. Map your `promise` returning function via an `external`, and use it in an `async` function as usual. - -Here's a full example of using the MDN `fetch` API, using `async` / `await` to simulate a login: - -```res -// A generic Response type for typing our fetch requests -module Response = { - type t<'data> - @send external json: t<'data> => promise<'data> = "json" -} - -// A binding to our globally available `fetch` function. `fetch` is a -// standardized function to retrieve data from the network that is available in -// all modern browsers. -@val @scope("globalThis") -external fetch: ( - string, - 'params, -) => promise, "error": Nullable.t}>> = - "fetch" - -// We now use our asynchronous `fetch` function to simulate a login. -// Note how we use `await` with regular functions returning a `promise`. -let login = async (email: string, password: string) => { - let body = { - "email": email, - "password": password, - } - - let params = { - "method": "POST", - "headers": { - "Content-Type": "application/json", - }, - "body": Json.stringifyAny(body), - } - - try { - let response = await fetch("https://reqres.in/api/login", params) - let data = await response->Response.json - - switch Nullable.toOption(data["error"]) { - | Some(msg) => Error(msg) - | None => - switch Nullable.toOption(data["token"]) { - | Some(token) => Ok(token) - | None => Error("Didn't return a token") - } - } - } catch { - | _ => Error("Unexpected network error occurred") - } -} -``` - diff --git a/pages/docs/manual/latest/attribute.mdx b/pages/docs/manual/latest/attribute.mdx deleted file mode 100644 index 6392dccb9..000000000 --- a/pages/docs/manual/latest/attribute.mdx +++ /dev/null @@ -1,90 +0,0 @@ ---- -title: "Attribute (Decorator)" -description: "Annotations in ReScript" -canonical: "/docs/manual/latest/attribute" ---- - -# Attribute (Decorator) - -Like many other languages, ReScript allows annotating a piece of code to express extra functionality. Here's an example: - - - -```res -@inline -let mode = "dev" - -let mode2 = mode -``` -```js -var mode2 = "dev"; -``` - - - -The `@inline` annotation tells `mode`'s value to be inlined into its usage sites (see output). We call such annotation "attribute" (or "decorator" in JavaScript). - -An attribute starts with `@` and goes before the item it annotates. In the above example, it's hooked onto the let binding. - -## Usage - -> **Note:** In previous versions (< 8.3) all our interop related attributes started with a `bs.` prefix (`bs.module`, `bs.val`). Our formatter will automatically drop them in newer ReScript versions. - -You can put an attribute almost anywhere. You can even add extra data to them by using them visually like a function call. Here are a few famous attributes (explained in other sections): - - - -```res -@@warning("-27") - - -@unboxed -type a = Name(string) - -@val external message: string = "message" - -type student = { - age: int, - @as("aria-label") ariaLabel: string, -} - -@deprecated -let customDouble = foo => foo * 2 - -@deprecated("Use SomeOther.customTriple instead") -let customTriple = foo => foo * 3 -``` -```js -``` - - - -1. `@@warning("-27")` is a standalone attribute that annotates the entire file. Those attributes start with `@@`. Here, it carries the data `"-27"`. You can find a full list of all available warnings [here](./warning-numbers). -2. `@unboxed` annotates the type definition. -3. `@val` annotates the `external` statement. -4. `@as("aria-label")` annotates the `ariaLabel` record field. -5. `@deprecated` annotates the `customDouble` expression. This shows a warning while compiling telling consumers to not rely on this method long-term. -6. `@deprecated("Use SomeOther.customTriple instead")` annotates the `customTriple` expression with a string to describe the reason for deprecation. - -For a list of all decorators and their usage, please refer to the [Syntax Lookup](/syntax-lookup) page. - -## Extension Point - -There's a second category of attributes, called "extension points" (a remnant term of our early systems): - - - -```res -%raw("var a = 1") -``` -```js -var a = 1 -``` - - - -Extension points are attributes that don't _annotate_ an item; they _are_ the item. Usually they serve as placeholders for the compiler to implicitly substitute them with another item. - -Extension points start with `%`. A standalone extension point (akin to a standalone regular attribute) starts with `%%`. - -For a list of all extension points and their usage, please refer to the [Syntax Lookup](/syntax-lookup) page. \ No newline at end of file diff --git a/pages/docs/manual/latest/bind-to-global-js-values.mdx b/pages/docs/manual/latest/bind-to-global-js-values.mdx deleted file mode 100644 index c08d45a1e..000000000 --- a/pages/docs/manual/latest/bind-to-global-js-values.mdx +++ /dev/null @@ -1,145 +0,0 @@ ---- -title: "Bind to Global JS Values" -description: "JS interop with global JS values in ReScript" -canonical: "/docs/manual/latest/bind-to-global-js-values" ---- - -# Bind to Global JS Values - -**First**, make sure the value you'd like to model doesn't already exist in our [provided API](api/core). - -Some JS values, like `setTimeout`, live in the global scope. You can bind to them like so: - - - -```res example -@val external setTimeout: (unit => unit, int) => float = "setTimeout" -@val external clearTimeout: float => unit = "clearTimeout" -``` -```js -// Empty output -``` - - - -(We already provide `setTimeout`, `clearTimeout` and others in the [Core API](api/core) module). - -This binds to the JavaScript [`setTimeout`](https://developer.mozilla.org/en-US/docs/Web/API/WindowOrworkerGlobalScope/setTimeout) methods and the corresponding `clearTimeout`. The `external`'s type annotation specifies that `setTimeout`: - -- Takes a function that accepts `unit` and returns `unit` (which on the JS side turns into a function that accepts nothing and returns nothing aka `undefined`), -- and an integer that specifies the duration before calling said function, -- returns a number that is the timeout's ID. This number might be big, so we're modeling it as a float rather than the 32-bit int. - -### Tips & Tricks - -**The above isn't ideal**. See how `setTimeout` returns a `float` and `clearTimeout` accepts one. There's no guarantee that you're passing the float created by `setTimeout` into `clearTimeout`! For all we know, someone might pass it `Math.random()` into the latter. - -We're in a language with a great type system now! Let's leverage a popular feature to solve this problem: abstract types. - - - -```res example -type timerId -@val external setTimeout: (unit => unit, int) => timerId = "setTimeout" -@val external clearTimeout: timerId => unit = "clearTimeout" - -let id = setTimeout(() => Console.log("hello"), 100) -clearTimeout(id) -``` -```js -var id = setTimeout(function (param) { - console.log("hello"); -}, 100); - -clearTimeout(id); -``` - - - -Clearly, `timerId` is a type that can only be created by `setTimeout`! Now we've guaranteed that `clearTimeout` _will_ be passed a valid ID. Whether it's a number under the hood is now a mere implementation detail. - -Since `external`s are inlined, we end up with JS output as readable as hand-written JS. - -## Global Modules - -If you want to bind to a value inside a global module, e.g. `Math.random`, attach a `scope` to your `val` external: - - - -```res example -@scope("Math") @val external random: unit => float = "random" -let someNumber = random() -``` -```js -var someNumber = Math.random(); -``` - - - -you can bind to an arbitrarily deep object by passing a tuple to `scope`: - - - -```res example -@val @scope(("window", "location", "ancestorOrigins")) -external length: int = "length" -``` -```js -// Empty output -``` - - - -This binds to `window.location.ancestorOrigins.length`. - -## Special Global Values - -Global values like `__filename` and `__DEV__` don't always exist; you can't even model them as an `option`, since the mere act of referring to them in ReScript (then compiled into JS) would trigger the usual `Uncaught ReferenceError: __filename is not defined` error in e.g. the browser environment. - -For these troublesome global values, ReScript provides a special approach: `%external(a_single_identifier)`. - - - -```res example -switch %external(__DEV__) { -| Some(_) => Console.log("dev mode") -| None => Console.log("production mode") -} -``` -```js -var match = typeof __DEV__ === "undefined" ? undefined : __DEV__; - -if (match !== undefined) { - console.log("dev mode"); -} else { - console.log("production mode"); -} -``` - - - -That first line's `typeof` check won't trigger a JS ReferenceError. - -Another example: - - - -```res example -switch %external(__filename) { -| Some(f) => Console.log(f) -| None => Console.log("non-node environment") -}; -``` -```js -var match = typeof (__filename) === "undefined" ? undefined : (__filename); - -if (match !== undefined) { - console.log(match); -} else { - console.log("non-node environment"); -} -``` - - - - diff --git a/pages/docs/manual/latest/bind-to-js-function.mdx b/pages/docs/manual/latest/bind-to-js-function.mdx deleted file mode 100644 index 58449fa16..000000000 --- a/pages/docs/manual/latest/bind-to-js-function.mdx +++ /dev/null @@ -1,490 +0,0 @@ ---- -title: "Bind to JS Function" -description: "JS interop with functions in ReScript" -canonical: "/docs/manual/latest/bind-to-js-function" ---- - -# Function - -Binding a JS function is like binding any other value: - - - -```res example -// Import nodejs' path.dirname -@module("path") external dirname: string => string = "dirname" -let root = dirname("/User/github") // returns "User" -``` -```js -var Path = require("path"); -var root = Path.dirname("/User/github"); -``` - - - -We also expose a few special features, described below. - -## Labeled Arguments - -ReScript has [labeled arguments](function.md#labeled-arguments) (that can also be optional). These work on an `external` too! You'd use them to _fix_ a JS function's unclear usage. Assuming we're modeling this: - -```js -// MyGame.js - -function draw(x, y, border) { - // suppose `border` is optional and defaults to false -} -draw(10, 20) -draw(20, 20, true) -``` - -It'd be nice if on ReScript's side, we can bind & call `draw` while labeling things a bit: - - - -```res example -@module("MyGame") -external draw: (~x: int, ~y: int, ~border: bool=?) => unit = "draw" - -draw(~x=10, ~y=20, ~border=true) -draw(~x=10, ~y=20) -``` -```js -var MyGame = require("MyGame"); - -MyGame.draw(10, 20, true); -MyGame.draw(10, 20, undefined); -``` - - - -We've compiled to the same function, but now the usage is much clearer on the ReScript side thanks to labels! - -Note that you can freely reorder the labels on the ReScript side; they'll always correctly appear in their declaration order in the JavaScript output: - - - -```res example -@module("MyGame") -external draw: (~x: int, ~y: int, ~border: bool=?) => unit = "draw" - -draw(~x=10, ~y=20) -draw(~y=20, ~x=10) -``` -```js -var MyGame = require("MyGame"); - -MyGame.draw(10, 20, undefined); -MyGame.draw(10, 20, undefined); -``` - - - -## Object Method - -Functions attached to a JS objects (other than JS modules) require a special way of binding to them, using `send`: - - - -```res example -type document // abstract type for a document object -@send external getElementById: (document, string) => Dom.element = "getElementById" -@val external doc: document = "document" - -let el = getElementById(doc, "myId") -``` -```js -var el = document.getElementById("myId"); -``` - - - -In a `send`, the object is always the first argument. Actual arguments of the method follow (this is a bit what modern OOP objects are really). - -### Chaining - -Ever used `foo().bar().baz()` chaining ("fluent api") in JS OOP? We can model that in ReScript too, through the [pipe operator](pipe.md). - -### Nested function call - -`@send` can also accept a `@scope(("itemOne","itemTwo"))` to access a function on a nested property. - - -```res example -type stripe - -@module("stripe") @new -external make: string => stripe = "default" - -type createSession = {} - -type sessionResult - -@send -@scope(("checkout", "sessions")) -external createCheckoutSession: (stripe, createSession) => - Promise.t = "create" - -let stripe = make("sk_...") -let session = stripe->createCheckoutSession({}) -``` -```js -import Stripe from "stripe"; - -var stripe = new Stripe("sk_..."); -var session = stripe.checkout.sessions.create({}); -``` - - -## Variadic Function Arguments - -You might have JS functions that take an arbitrary amount of arguments. ReScript supports modeling those, under the condition that the arbitrary arguments part is homogenous (aka of the same type). If so, add `variadic` to your `external`. - - - -```res example -@module("path") @variadic -external join: array => string = "join" - -let v = join(["a", "b"]) -``` -```js -var Path = require("path"); -var v = Path.join("a", "b"); -``` - - - -`module` will be explained in [Import from/Export to JS](import-from-export-to-js.md). - -## Modeling Polymorphic Function - -Apart from the above special-case, JS functions in general are often arbitrarily overloaded in terms of argument types and number. How would you bind to those? - -### Trick 1: Multiple `external`s - -If you can exhaustively enumerate the many forms an overloaded JS function can take, simply bind to each differently: - - - -```res example -@module("MyGame") external drawCat: unit => unit = "draw" -@module("MyGame") external drawDog: (~giveName: string) => unit = "draw" -@module("MyGame") external draw: (string, ~useRandomAnimal: bool) => unit = "draw" -``` -```js -// Empty output -``` - - - -Note how all three externals bind to the same JS function, `draw`. - -### Trick 2: Polymorphic Variant + `unwrap` - -If you have the irresistible urge of saying "if only this JS function argument was a variant instead of informally being either `string` or `int`", then good news: we do provide such `external` features through annotating a parameter as a polymorphic variant! Assuming you have the following JS function you'd like to bind to: - -```js -function padLeft(value, padding) { - if (typeof padding === "number") { - return Array(padding + 1).join(" ") + value; - } - if (typeof padding === "string") { - return padding + value; - } - throw new Error(`Expected string or number, got '${padding}'.`); -} -``` - -Here, `padding` is really conceptually a variant. Let's model it as such. - - - -```res example -@val -external padLeft: ( - string, - @unwrap [ - | #Str(string) - | #Int(int) - ]) - => string = "padLeft" -padLeft("Hello World", #Int(4)) -padLeft("Hello World", #Str("Message from ReScript: ")) -``` -```js -padLeft("Hello World", 4); -padLeft("Hello World", "Message from ReScript: "); -``` - - - -Obviously, the JS side couldn't have an argument that's a polymorphic variant! But here, we're just piggy backing on poly variants' type checking and syntax. The secret is the `@unwrap` annotation on the type. It strips the variant constructors and compile to just the payload's value. See the output. - -## Constrain Arguments Better - -Consider the Node `fs.readFileSync`'s second argument. It can take a string, but really only a defined set: `"ascii"`, `"utf8"`, etc. You can still bind it as a string, but we can use poly variants + `string` to ensure that our usage's more correct: - - - -```res example -@module("fs") -external readFileSync: ( - ~name: string, - @string [ - | #utf8 - | @as("ascii") #useAscii - ], -) => string = "readFileSync" - -readFileSync(~name="xx.txt", #useAscii) -``` -```js -var Fs = require("fs"); -Fs.readFileSync("xx.txt", "ascii"); -``` - - - -- Attaching `@string` to the whole poly variant type makes its constructor compile to a string of the same name. -- Attaching a `@as("bla")` to a constructor lets you customize the final string. - -And now, passing something like `"myOwnUnicode"` or other variant constructor names to `readFileSync` would correctly error. - -Aside from string, you can also compile an argument to an int, using `int` instead of `string` in a similar way: - - - -```res example -@val -external testIntType: ( - @int [ - | #onClosed - | @as(20) #onOpen - | #inBinary - ]) - => int = "testIntType" -testIntType(#inBinary) -``` -```js -testIntType(21); -``` - - - -`onClosed` compiles to `0`, `onOpen` to `20` and `inBinary` to **`21`**. - -## Unknown for type safety - -It is best practice to inspect data received from untrusted external functions to ensure it contains what you expect. This helps avoid run-time crashes and unexpected behavior. If you're certain about what an external function returns, simply assert the return value as `string` or `array` or whatever you want it to be. Otherwise use `unknown`. The ReScript type system will prevent you from using an `unknown` until you first inspect it and "convert" it using JSON parsing utilities or similar tools. - -Consider the example below of two external functions that access the value of a property on a JavaScript object. `getPropertyUnsafe` returns an `'a`, which means "anything you want it to be." ReScript allows you to use this value as a `string` or `array` or any other type. Quite convenient! But if the property is missing or contains something unexpected, your code might break. You can make the binding more safe by changing `'a` to `string` or `option<'a>`, but this doesn't completely eliminate the problem. - -The `getPropertySafe` function returns an `unknown`, which could be `null` or a `string` or anything else. But ReScript prevents you from using this value inappropriately until it has been safely parsed. - -```res example -@get_index external getPropertyUnsafe: ({..}, string) => 'a = "" -@get_index external getPropertySafe: ({..}, string) => unknown = "" - -let person = {"name": "Bob", "age": 12} - -let greeting1 = "Hello, " ++ getPropertyUnsafe(person, "name") // works (this time!) -// let greeting2 = "Hello, " ++ getPropertySafe(person, "name") // syntax error -``` - -## Special-case: Event Listeners - -One last trick with polymorphic variants: - - - -```res example -type readline - -@send -external on: ( - readline, - @string [ - | #close(unit => unit) - | #line(string => unit) - ] - ) - => readline = "on" - -let register = rl => - rl - ->on(#close(event => ())) - ->on(#line(line => Console.log(line))); -``` -```js -function register(rl) { - return rl - .on("close", function($$event) {}) - .on("line", function(line) { - console.log(line); - }); -} -``` - - - - - -## Fixed Arguments - -Sometimes it's convenient to bind to a function using an `external`, while passing predetermined argument values to the JS function: - - - -```res example -@val -external processOnExit: ( - @as("exit") _, - int => unit -) => unit = "process.on" - -processOnExit(exitCode => - Console.log("error code: " ++ Int.toString(exitCode)) -); -``` -```js -process.on("exit", function (exitCode) { - console.log("error code: " + exitCode.toString()); -}); -``` - - - -The `@as("exit")` and the placeholder `_` argument together indicates that you want the first argument to compile to the string `"exit"`. You can also use any JSON literal with `as`: `` @as(json`true`) ``, `` @as(json`{"name": "John"}`) ``, etc. - -## Ignore arguments - -You can also explicitly "hide" `external` function parameters in the JS output, which may be useful if you want to add type constraints to other parameters without impacting the JS side: - - - -```res -@val external doSomething: (@ignore 'a, 'a) => unit = "doSomething" - -doSomething("this only shows up in ReScript code", "test") -``` - -```js -doSomething("test"); -``` - - - -**Note:** It's a pretty niche feature, mostly used to map to polymorphic JS APIs. - -## Modeling `this`-based Callbacks - -Many JS libraries have callbacks which rely on this (the source), for example: - -```js -x.onload = function(v) { - console.log(this.response + v) -} -``` - -Here, `this` would point to `x` (actually, it depends on how `onload` is called, but we digress). It's not correct to declare `x.onload` of type `(. unit) -> unit`. Instead, we introduced a special attribute, `this`, which allows us to type `x` as so: - - - -```res example -type x -@val external x: x = "x" -@set external setOnload: (x, @this ((x, int) => unit)) => unit = "onload" -@get external resp: x => int = "response" -setOnload(x, @this (o, v) => Console.log(resp(o) + v)) -``` -```js -x.onload = function (v) { - var o = this; - console.log((o.response + v) | 0); -}; -``` - - - -`@this` reserves the first parameter for the `this` value, and for arity of 0, there is no need for a redundant `unit` type. - -## Function Nullable Return Value Wrapping - -For JS functions that return a value that can also be `undefined` or `null`, we provide `@return(...)`. To automatically convert that value to an `option` type (recall that ReScript `option` type's `None` value only compiles to `undefined` and not `null`). - - - -```res example -type element -type dom - -@send @return(nullable) -external getElementById: (dom, string) => option = "getElementById" - -let test = dom => { - let elem = dom->(getElementById("haha")) - switch (elem) { - | None => 1 - | Some(_ui) => 2 - } -} -``` -```js -function test(dom) { - var elem = dom.getElementById("haha"); - if (elem == null) { - return 1; - } else { - console.log(elem); - return 2; - } -} - -``` - - - -`return(nullable)` attribute will automatically convert `null` and `undefined` to `option` type. - -Currently 4 directives are supported: `null_to_opt`, `undefined_to_opt`, `nullable` and `identity`. - - - -`identity` will make sure that compiler will do nothing about the returned value. It is rarely used, but introduced here for debugging purpose. - -## Tagged template functions - -**Since 11.1** - -**Experimental** You can easily bind to [JS tagged template functions](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Template_literals#tagged_templates). -Tag functions in JS expect as input an array of strings and variadic parameters for the arguments of the interpolation. -To bind to those functions in ReScript, the binding signature must have two arrays as arguments, -the first one being an array of strings and the second can be an array of anything. -You add the `@taggedTemplate` annotation and you're good to go! - - - -```res example -// see https://bun.sh/docs/runtime/shell -type result = {exitCode: int} -@module("bun") @taggedTemplate -external sh: (array, array) => promise = "$" - -let filename = "index.res" -let result = await sh`ls ${filename}` -``` - -```js -import * as $$Bun from "bun"; -var filename = "index.res"; -var result = await $$Bun.$`ls ${filename}`; -``` - - - -Notice that it gets compiled to tagged template literals in JS, which allows -to use JS tools that only work on the literals and not by calling directly the tag function. - -There are plenty of useful JS tools you can bind to, like [`gql`](https://github.com/apollographql/graphql-tag), -[`sql`](https://github.com/porsager/postgres), [`css`](https://github.com/mayank99/ecsstatic) and a lot others! diff --git a/pages/docs/manual/latest/bind-to-js-object.mdx b/pages/docs/manual/latest/bind-to-js-object.mdx deleted file mode 100644 index 09a21c226..000000000 --- a/pages/docs/manual/latest/bind-to-js-object.mdx +++ /dev/null @@ -1,197 +0,0 @@ ---- -title: "Bind to JS Object" -description: "Interop with JS objects in ReScript" -canonical: "/docs/manual/latest/bind-to-js-object" ---- - -# Bind to JS Object - -JavaScript objects are a combination of several use-cases: - -- As a "record" or "struct" in other languages (like ReScript and C). -- As a hash map. -- As a class. -- As a module to import/export. - -ReScript cleanly separates the binding methods for JS object based on these 4 use-cases. This page documents the first three. Binding to JS module objects is described in the [Import from/Export to JS](import-from-export-to-js.md) section. - - - -## Bind to Record-like JS Objects - -### Bind Using ReScript Record - -If your JavaScript object has fixed fields, then it's conceptually like a ReScript record. Since a ReScript record compiles to a clean JavaScript object, you can definitely type a JS object as a ReScript record! - - - -```res example -type person = { - name: string, - friends: array, - age: int, -} - -@module("MySchool") external john: person = "john" - -let johnName = john.name -``` -```js -var MySchool = require("MySchool"); - -var johnName = MySchool.john.name; -``` - - - -External is documented [here](external.md). `@module` is documented [here](import-from-export-to-js.md). - -If you want or need to use different field names on the ReScript and the JavaScript side, you can use the `@as` decorator: - - - -```res example -type action = { - @as("type") type_: string -} - -let action = {type_: "ADD_USER"} -``` -```js -var action = { - type: "ADD_USER" -}; -``` - - - -This is useful to map to JavaScript attribute names that cannot be expressed in ReScript (such as keywords). - -It is also possible to map a ReScript record to a JavaScript array by passing indices to the `@as` decorator: - - - -```res -type t = { - @as("0") foo: int, - @as("1") bar: string, -} - -let value = {foo: 7, bar: "baz"} -``` - -```js -var value = [ - 7, - "baz" -]; -``` - - - - -### Bind Using ReScript Object - -Alternatively, you can use [ReScript object](object.md) to model a JS object too: - - - -```res example -type person = { - "name": string, - "friends": array, - "age": int, -} - -@module("MySchool") external john: person = "john" - -let johnName = john["name"] -``` -```js -var MySchool = require("MySchool"); - -var johnName = MySchool.john.name; -``` - - - -### Bind Using Special Getter and Setter Attributes - -Alternatively, you can use `get` and `set` to bind to individual fields of a JS object: - - - -```res example -type textarea -@set external setName: (textarea, string) => unit = "name" -@get external getName: textarea => string = "name" -``` -```js -``` - - - -You can also use `get_index` and `set_index` to access a dynamic property or an index: - - - -```res example -type t -@new external create: int => t = "Int32Array" -@get_index external get: (t, int) => int = "" -@set_index external set: (t, int, int) => unit = "" - -let i32arr = create(3) -i32arr->set(0, 42) -Console.log(i32arr->get(0)) -``` -```js -var i32arr = new Int32Array(3); -i32arr[0] = 42; -console.log(i32arr[0]); -``` - - - -## Bind to Hash Map-like JS Object - -If your JavaScript object: - -- might or might not add/remove keys -- contains only values that are of the same type - -Then it's not really an object, it's a hash map. Use [Dict](api/core/dict), which contains operations like `get`, `set`, etc. and cleanly compiles to a JavaScript object still. - -## Bind to a JS Object That's a Class - -Use `new` to emulate e.g. `new Date()`: - - - -```res example -type t -@new external createDate: unit => t = "Date" - -let date = createDate() -``` -```js -var date = new Date(); -``` - - - -You can chain `new` and `module` if the JS module you're importing is itself a class: - - - -```res example -type t -@new @module external book: unit => t = "Book" -let myBook = book() -``` -```js -var Book = require("Book"); -var myBook = new Book(); -``` - - diff --git a/pages/docs/manual/latest/browser-support-polyfills.mdx b/pages/docs/manual/latest/browser-support-polyfills.mdx deleted file mode 100644 index c1de7a491..000000000 --- a/pages/docs/manual/latest/browser-support-polyfills.mdx +++ /dev/null @@ -1,18 +0,0 @@ ---- -title: "Browser Support & Polyfills" -description: "Note on browser support in ReScript" -canonical: "/docs/manual/latest/browser-support-polyfills" ---- - -# Browser Support & Polyfills - -ReScript compiles to JavaScript **ES5**, with the exception of optionally allowing to compile to ES6's module import & export. - -For [old browsers](https://caniuse.com/#search=typed%20array), you also need to polyfill TypedArray. The following standard library functions require it: - -- `Int64.float_of_bits` -- `Int64.bits_of_float` -- `Int32.float_of_bits` -- `Int32.bits_of_float` - -If you don't use these functions, you're fine. Otherwise, it'll be a runtime failure. diff --git a/pages/docs/manual/latest/build-configuration-schema.mdx b/pages/docs/manual/latest/build-configuration-schema.mdx deleted file mode 100644 index 21e098dfb..000000000 --- a/pages/docs/manual/latest/build-configuration-schema.mdx +++ /dev/null @@ -1,23 +0,0 @@ ---- -title: "Configuration Schema" -metaTitle: "Build System Configuration Schema" -description: "Schema exploration widget for the ReScript configuration file" -canonical: "/docs/manual/latest/build-configuration-schema" ---- - -import dynamic from "next/dynamic"; - -export const Docson = dynamic( - () => - import("src/components/Docson").then((comp) => { - return comp.make; - }), - { - ssr: false, - loading: () =>
Loading...
, - } -); - -export default function BuildConfigurationSchemaPage() { - return ; -} diff --git a/pages/docs/manual/latest/build-configuration.mdx b/pages/docs/manual/latest/build-configuration.mdx deleted file mode 100644 index 3c548a36a..000000000 --- a/pages/docs/manual/latest/build-configuration.mdx +++ /dev/null @@ -1,245 +0,0 @@ ---- -title: "Configuration" -metaTitle: "Build System Configuration" -description: "Details about the configuration of the ReScript build system (rescript.json)" -canonical: "/docs/manual/latest/build-configuration" ---- - -# Configuration - -`rescript.json` (or `bsconfig.json` in versions prior ReScript 11) is the single, mandatory build meta file needed for `rescript`. - -**The complete configuration schema is [here](./build-configuration-schema)**. We'll _non-exhaustively_ highlight the important parts in prose below. - -## name, namespace - -`name` is the name of the library, used as its "namespace". You can activate namespacing through `"namespace": true` in your `rescript.json`. Namespacing is almost **mandatory**; we haven't turned it on by default yet to preserve backward-compatibility. - -**Explanation**: by default, your files, once used as a third-party dependency, are available globally to the consumer. E.g. if you have a `Util.res` and the consumer also has a file of the same name, they will clash. Turning on `namespace` avoids this by wrapping all your own project's files into an extra module layer; instead of a global `Util` module, the consumer will see you as `MyProject.Util`. **The namespacing affects your consumers, not yourself**. - -Aka, in ReScript, "namespace" is just a fancy term for an auto-generated module that wraps all your project's files (efficiently and correctly, of course!) for third-party consumption. - -We don't do folder-level namespacing for your own project; all your own file names must be unique. This is a constraint that enables several features such as fast search and easier project reorganization. - -**Note**: the `rescript.json` `name` should be the same as the `package.json` `name`, to avoid confusing corner-cases. However, this means that you can't use a camelCased names such as `MyProject`, since `package.json` and npm forbid you to do so (some file systems are case-insensitive). To have the namespace/module as `MyProject`, write `"name": "my-project"`. ReScript will turn that into the camelCased name correctly. - -**Note on custom namespacing**: if for some reason, you need a namespace that is different from what your `name` will produce, you can directly send a string to the `namespace` option. For example, if your package is a binding named `bs-some-thing`, you can use `"namespace": "some-thing"` to get `SomeThing` namespace instead of `BsSomeThing`. - -## sources - -Your source files need to be specified explicitly (we don't want to accidentally drill down into some unrelated directories). Examples: - -```json -{ - "sources": ["src", "examples"] -} -``` - -```json -{ - "sources": { - "dir": "src", - "subdirs": ["page"] - } -} -``` - -```json -{ - "sources": [ - "examples", - { - "dir": "src", - "subdirs": true // recursively builds every subdirectory - } - ] -} -``` - -You can mark your directories as dev-only (for e.g. tests). These won't be built and exposed to third-parties, or even to other "dev" directories in the same project: - -```json -{ - "sources": { - "dir": "test", - "type": "dev" - } -} -``` - -You can also explicitly allow which modules can be seen from outside. This feature is especially useful for library authors who want to have a single entry point for their users. -Here, the file `src/MyMainModule.res` is exposed to outside consumers, while all other files are private. - -```json -{ - "sources": { - "dir": "src", - "public": ["MyMainModule"] - } -} -``` - -## bs-dependencies, bs-dev-dependencies - -List of ReScript dependencies. Just like `package.json`'s dependencies, they'll be searched in `node_modules`. - -Note that only sources marked with `"type":"dev"` will be able to resolve modules from `bs-dev-dependencies`. - -## pinned-dependencies - -**Since 8.4**: List of pinned dependencies. A pinned dependency will always be rebuilt whenever you build a toplevel package (e.g. your main app) with `rescript`. - -This is useful for working on multiple independent ReScript packages simultaneously. More usage details can be found in our dedicated [pinned dependencies](./build-pinned-dependencies) page. - -## external-stdlib - -**Since 9.0**: This setting allows depending on an externally built stdlib package (instead of a locally built stdlib runtime). Useful for shipping packages that are only consumed in JS or TS without any dependencies to the ReScript development toolchain. - -More details can be found on our [external stdlib](./build-external-stdlib) page. - -## js-post-build - -Hook that's invoked every time a file is recompiled. Good for JS build system interop, but please use it **sparingly**. Calling your custom command for every recompiled file slows down your build and worsens the building experience for even third-party users of your lib. - -Example: - -```json -{ - "js-post-build": { - "cmd": "/path/to/node ../../postProcessTheFile.js" - } -} -``` - -Note that the path resolution for the command (`node` in this case) is done so: - -- `/myCommand` is resolved into `/myCommand` -- `package/myCommand` is resolved into `node_modules/package/myCommand` -- `./myCommand` is resolved into `myProjectRoot/myCommand` -- `myCommand` is just called as `myCommand`, aka a globally available executable. But note that ReScript doesn't read into your shell's environment, so if you put e.g. `node`, it won't find it unless you specify an absolute path. Alternatively, add `#!/usr/local/bin/node` to the top of your script to directly call it without prepending `node`. - -The command itself is called from inside `lib/bs`. - -## package-specs - -Output to either CommonJS (the default) or JavaScript module. Example: - -```json -{ - "package-specs": { - "module": "commonjs", - "in-source": true - } -} -``` - -- `"module": "commonjs"` generates output as CommonJS format. -- `"module": "esmodule"` generates output as JavaScript module format. Will be default value in next major. -- `"in-source": true` generates output alongside source files. If you omit it, it'll generate the artifacts into `lib/js`. The output directory is not configurable otherwise. - -This configuration only applies to you, when you develop the project. When the project is used as a third-party library, the consumer's own `rescript.json` `package-specs` overrides the configuration here, logically. - -## suffix - -**Since 11.0**: The suffix can now be freely chosen. However, we still suggest you stick to the convention and use -one of the following: - -- `".js` -- `".mjs"` -- `".cjs"` -- `".res.js"` -- `".res.mjs"` -- `".res.cjs"` - -### Design Decisions - -Generating JS files with the `.res.js` suffix means that, on the JS side, you can do `const myReScriptFile = require('./TheFile.res.js')`. The benefits: - -- It's immediately clear that we're dealing with a generated JS file here. -- It avoids clashes with a potential `TheFile.js` file in the same folder. -- It avoids the need of using a build system loader for ReScript files. This + in-source build means integrating a ReScript project into your pure JS codebase **basically doesn't touch anything in your build pipeline at all**. - -## uncurried - -**Since 11.0**: While we strongly encourage all users to use uncurried mode, it is still possible to opt out. Just set `"uncurried"` to `false` to get the old behavior back: - -```json -{ - "uncurried": false -} -``` - -More details can be found in the [blogpost about "Uncurried Mode"](/blog/uncurried-mode). - -## warnings - -Selectively turn on/off certain warnings and/or turn them into hard errors. Example: - -```json -{ - "warnings": { - "number": "-44-102", - "error": "+5" - } -} -``` - -Turn off warning `44` and `102` (polymorphic comparison). Turn warning `5` (partial application whose result has function type and is ignored) into a hard error. - -The warning numbers are shown in the build output when they're triggered. See [Warning Numbers](./warning-numbers) for the complete list. - -## bsc-flags - -Extra flags to pass to the compiler. For advanced usages. - -- `-open ABC` opens the module `ABC` for each file in the project. `ABC` can either be a dependency, namespaced project or local module of the current project. - -## gentypeconfig - -To enable genType, set `"gentypeconfig"` at top level in the project's `rescript.json`. - -```json -{ - "gentypeconfig": { - "module": "esmodule", - "moduleResolution": "node", - "generatedFileExtension": ".gen.tsx", - "debug": { - "all": false, - "basic": false - } - } -} -``` - -`generatedFileExtension`: File extension used for genType generated files (defaults to `".gen.tsx"`) - -`module`: Module format used for the generated `*.gen.tsx` files (supports `"esmodule"` and `"commonjs"`) - -`moduleResolution`: Module resolution strategy used in genType outputs. This may be required for compatibility with TypeScript projects. Specify the value as the same in `tsconfig.json`. - -- `"node"`(default): Drop extensions in import paths. -- `"node16"`: Use TS output's extension. This provides compatibility with projects using `"moduleResolution": "node16"` and ES Modules. -- `"bundler"`: Use TS input's extension. This provides compatibility with projects using `"moduleResolution": "bundler"` and ES Modules. This also requires TS v5.0+ and `compilerOptions.allowImportingTsExtensions` to `true` - -`debug`: Enable debug logs. - -### Deprecated options - -`language`: the `language` setting is not required from compiler v10.1. - -`shims`: Required only if one needs to export certain basic ReScript data types to JS when one cannot modify the sources to add annotations (e.g. exporting ReScript lists), and if the types are not first-classed in genType. - -## Environment Variables - -We heavily disrecommend the usage of environment variables, but for certain cases, they're justified. - -### Error Output Coloring: `NINJA_ANSI_FORCED` - -This is mostly for other programmatic usage of `rescript` where outputting colors is not desired. - -When `NINJA_ANSI_FORCED` is set to `1`: `rescript` produces color. -When `NINJA_ANSI_FORCED` is set to `0`: `rescript` doesn't produce color. -When `NINJA_ANSI_FORCED` is not set: `rescript` might or might not produce color, depending on a smart detection of where it's outputted. - -> Note that the underlying compiler will always be passed `-color always`. See more details in [this issue](https://github.com/rescript-lang/rescript/issues/2984#issuecomment-410669163). diff --git a/pages/docs/manual/latest/build-external-stdlib.mdx b/pages/docs/manual/latest/build-external-stdlib.mdx deleted file mode 100644 index ae796932c..000000000 --- a/pages/docs/manual/latest/build-external-stdlib.mdx +++ /dev/null @@ -1,59 +0,0 @@ ---- -title: "External Stdlib" -metaTitle: "External Stdlib" -description: "Configuring an external ReScript stdlib package" -canonical: "/docs/manual/latest/build-external-stdlib" ---- - -# External Stdlib - -**Since 9.0** - -Your ReScript project depends on the `rescript` package as a [`devDependency`](https://docs.npmjs.com/specifying-dependencies-and-devdependencies-in-a-package-json-file), which includes our compiler, build system and runtime like `Belt`. However, you had to move it to `dependency` in `package.json` if you publish your code: -- To Docker or other low-storage deployment devices. -- For pure JS/TS consumers who probably won't install `rescript` in their own project. - -In these cases, the size or mere presence of `rescript` can be troublesome, since it includes not just our necessary runtime like `Belt`, but also our compiler and build system. - -To solve that, we now publish our runtime as a standalone package at [`@rescript/std`](https://www.npmjs.com/package/@rescript/std), whose versions mirror `rescript`'s. Now you can keep `rescript` as a `devDependency` and have only `@rescript/std` as your runtime `dependency`. - -**This is an advanced feature**. Please only use it in the aforementioned scenarios. If you already use a JS bundler with dead code elimination, you might not need this feature. - -## Configuration - -Say you want to publish a JS-only ReScript 9.0 library. Install the packages like this: - -```sh -npm install rescript@11.0.1 --save-dev -npm install @rescript/std@11.0.1 -``` - -Then add this to `rescript.json`: - -```json -{ - // ... - "external-stdlib" : "@rescript/std" -} -``` - -Now the compiled JS code will import using the path defined by `external-stdlib`. Check the JS output tab: - - - -```res -Array.forEach([1, 2, 3], num => Console.log(num)) -``` - -```js -// Note the require path starting with "@rescript/std". -var Belt_Array = require("@rescript/std/lib/js/belt_Array.js"); - -Belt_Array.forEach([1, 2, 3], function (num) { - console.log(num); -}); -``` - - - -**Make sure the version number of `rescript` and `@rescript/std` match in your `package.json`** to avoid running into runtime issues due to mismatching stdlib assumptions. diff --git a/pages/docs/manual/latest/build-overview.mdx b/pages/docs/manual/latest/build-overview.mdx deleted file mode 100644 index 845d34351..000000000 --- a/pages/docs/manual/latest/build-overview.mdx +++ /dev/null @@ -1,83 +0,0 @@ ---- -title: "Overview" -metaTitle: "Build System Overview" -description: "Documentation about the ReScript build system and its toolchain" -canonical: "/docs/manual/latest/build-overview" ---- - -# Build System Overview - -ReScript comes with a build system, [`rescript`](https://www.npmjs.com/package/rescript), that's fast, lean and used as the authoritative build system of the community. - -Every ReScript project needs a build description file, `rescript.json`. - -## Options - -See `rescript help`: - -``` -❯ rescript help -Usage: rescript - -`rescript` is equivalent to `rescript build` - -Options: - -v, -version display version number - -h, -help display help - -Subcommands: - build - clean - format - convert - dump - help - -Run `rescript -h` for subcommand help. Examples: - rescript build -h - rescript format -h -``` - -## Build Project - -Each build will create build artifacts from your project's source files. - -**To build a project (including its dependencies / pinned-dependencies)**, run: - -```sh -rescript -``` - -Which is an alias for `rescript build`. - -To keep a build watcher, run: - -```sh -rescript -w -``` - -Any new file change will be picked up and the build will re-run. - -**Note**: third-party libraries (in `node_modules`, or via `pinned-dependencies`) aren't watched, as doing so may exceed the node.js watcher count limit. - -**Note 2**: In case you want to set up a project in a JS-monorepo-esque approach (`npm` and `yarn` workspaces) where changes in your sub packages should be noticed by the build, you will need to define pinned dependencies in your main project's `rescript.json`. More details [here](./build-pinned-dependencies). - -## Clean Project - -If you ever get into a stale build for edge-case reasons, use: - -```sh -rescript clean -``` - -## Compile with stricter errors in CI - -**Since 11.1** - -You may want to compile your project with stricter rules for production, than when developing. With the `-warn-error` build flag, this can easily be done, for instance in a continuous integration script. E.g.: - -```sh -rescript -warn-error +110 -``` - -Here, warning number 110, which is triggered when a [`%todo`](/syntax-lookup#todo) has been found, gets promoted to an error. The full list of warning numbers can be found [here](/docs/manual/latest/warning-numbers). diff --git a/pages/docs/manual/latest/build-performance.mdx b/pages/docs/manual/latest/build-performance.mdx deleted file mode 100644 index f1bf8e6c5..000000000 --- a/pages/docs/manual/latest/build-performance.mdx +++ /dev/null @@ -1,85 +0,0 @@ ---- -title: "Performance" -metaTitle: "Build Performance" -description: "ReScript build performance and measuring tools" -canonical: "/docs/manual/latest/build-performance" ---- - -# Build Performance - -ReScript considers performance at install time, build time and run time as a serious feature; it's one of those things you don't notice until you realize it's missing. - -## Profile Your Build - -Sometime your build can be slow due to some confused infra setups. We provide an interactive visualization of your build's performance via `bstracing`: - -```sh -./node_modules/.bin/bstracing -``` - -Run the above command at your ReScript project's root; it'll spit out a JSON file you can drag and drop into `chrome://tracing`. - -import Image from "src/components/Image"; - - - -## Under the Hood - -ReScript itself uses a build system under the hood, called [Ninja](https://ninja-build.org). Ninja is like Make, but cross-platform, minimal, focuses on perf and destined to be more of a low-level building block than a full-blown build system. In this regard, Ninja's a great implementation detail for `rescript`. - -ReScript reads into `rescript.json` and generates the Ninja build file in `lib/bs`. The file contains the low-level compiler commands, namespacing rules, intermediate artifacts generation & others. It then runs `ninja` for the actual build. - -## The JS Wrapper - -`rescript` itself is a Node.js wrapper which takes care of some miscellaneous tasks, plus the watcher. The lower-level, watcher-less, fast native `rescript` is called `rescript.exe`. It's located at `node_modules/rescript/{your-platform}/rescript.exe`. - -If you don't need the watcher, you can run said `rescript.exe`. This side-steps Node.js' long startup time, which can be in the order of `100ms`. Our editor plugin finds and uses this native `rescript.exe` for better performance. - -## Numbers - -Raw `rescript.exe` build on a small project should be around `70ms`. This doubles when you use the JS `rescript` wrapper which comes with a watcher, which is practically faster since you don't manually run the build at every change (though you should opt for the raw `rescript.exe` for programmatic usage, e.g. inserting rescript into your existing JS build pipeline). - -No-op build (when no file's changed) should be around `15ms`. Incremental rebuild (described soon) of a single file in a project is around `70ms` too. - -Cleaning the artifacts should be instantaneous. - -### Extreme Test - -We've stress-tested `rescript.exe` on a big project of 10,000 files (2 directories, 5000 files each, first 5000 no dependencies, last 5000 10 dependencies on files from the former directory) using https://github.com/rescript-lang/build-benchmark, on a Retina Macbook Pro Early 2015 (3.1 GHz Intel Core i7). - - - -- No-op build of 10k files: `800ms` (the minimum amount of time required to check the mtimes of 10k files). -- Clean build: \<3 minutes. -- Incremental build: depends on the number of the dependents of the file. No dependent means `1s`. - -### Stability - -`rescript` is a file-based build system. We don't do in-memory build, even if that speeds up the build a lot. In-memory builds risk memory leaks, out-of-memory errors, corrupt halfway build and others. Our watcher mode stays open for days or months with no leak. - -The watcher is also just a thin file watcher that calls `rescript.exe`. We don't like babysitting daemon processes. - -## Incrementality & Correctness - -ReScript doesn't take whole seconds to run every time. The bulk of the build performance comes from incremental build, aka re-building a previously built project when a few files changed. - -In short, thanks to our compiler and the build system's architecture, we're able to **only build what's needed**. E.g. if `MyFile.res` isn't changed, then it's not recompiled. You can roughly emulate such incrementalism in languages like JavaScript, but the degree of correctness is unfortunately low. For example, if you rename or move a JS file, then the watcher might get confused and not pick up the "new" file or fail to clean things up correctly, resulting in you needing to clean your build and restart anew, which defeats the purpose. - -Say goodbye to stale build from your JavaScript ecosystem! - -## Speed Up Incremental Build - -ReScript uses the concept of interface files (`.resi`) (or, equivalently, [module signatures](module.md#signatures)). Exposing only what you need naturally speeds up incremental builds. E.g. if you change a `.res` file whose corresponding `.resi` file doesn't expose the changed part, then you've reduced the amount of dependent files you have to rebuild. - -## Programmatic Usage - -Unfortunately, JS build systems are usually the bottleneck for building a JS project nowadays. Having parts of the build blazingly fast doesn't matter much if the rest of the build takes seconds or literally minutes. Here are a few suggestions: - -- Convert more files into ReScript =). Fewer files going through fewer parts of the JS pipeline helps a ton. -- Careful with bringing in more dependencies: libraries, syntax transforms (e.g. the unofficially supported PPX), build step loaders, etc. The bulk of these dragging down the editing & building experience might out-weight the API benefits they provide. - -## Hot Reloading - -Hot reloading refers to maintaining a dev server and listening to file changes in a way that allows the server to pipe some delta changes right into the currently running browser page. This provides a relatively fast iteration workflow while working in specific frameworks. - -However, hot reloading is fragile by nature, and counts on the occasional inconsistencies (bad state, bad eval, etc.) and the heavy devserver setup/config being less of a hassle than the benefits it provides. We err on the side of caution and stability in general, and decided not to provide a built-in hot reloading _yet_. **Note**: you can still use the hot reloading facility provided by your JS build pipeline. diff --git a/pages/docs/manual/latest/build-pinned-dependencies.mdx b/pages/docs/manual/latest/build-pinned-dependencies.mdx deleted file mode 100644 index ebab8d179..000000000 --- a/pages/docs/manual/latest/build-pinned-dependencies.mdx +++ /dev/null @@ -1,110 +0,0 @@ ---- -title: "Pinned Dependencies" -metaTitle: "Pinned Dependencies" -description: "Handling multiple packages within one ReScript project with pinned dependencies" -canonical: "/docs/manual/latest/build-pinned-dependencies" ---- - -# Pinned Dependencies - -Usually we'd recommend to use ReScript in a single-codebase style by using one `rescript.json` file for your whole codebase. - -There are scenarios where you still want to connect and build multiple independent ReScript packages for one main project though (`npm` workspaces-like "monorepos"). This is where `pinned-dependencies` come into play. - -## Package Types - -Before we go into detail, let's first explain all the different package types recognized by the build system: - -- Toplevel (this is usually the final app you are building, which has dependencies to other packages) -- Pinned dependencies (these are your local packages that should always rebuild when you build your toplevel, those should be listed in `bs-dependencies` and `pinned-dependencies`) -- Normal dependencies (these are packages that are consumed from npm and listed via `bs-dependencies`) - -Whenever a package is being built (`rescript build`), the build system will build the toplevel package with its pinned-dependencies. So any changes made in a pinned dependency will automatically be reflected in the final app. - -## Build System Package Rules - -The build system respects the following rules for each package type: - -**Toplevel** -- Warnings reported -- Warn-error respected -- Builds dev dependencies -- Builds pinned dependencies -- Runs custom rules -- Package-specs like JavaScript module or CommonJS overrides all its dependencies - -**Pinned dependencies** -- Warnings reported -- Warn-error respected -- Ignores pinned dependencies -- Builds dev dependencies -- Runs custom rules - -**Normal dependencies** -- Warnings, warn-error ignored -- Ignores dev directories -- Ignores pinned dependencies -- Ignores custom generator rules - -So with that knowledge in mind, let's dive into some more concrete examples to see our pinned dependencies in action. - -## Examples - -### Yarn workspaces - -Let's assume we have a codebase like this: - -``` -myproject/ - app/ - - src/App.res - - rescript.json - common/ - - src/Header.res - - rescript.json - myplugin/ - - src/MyPlugin.res - - rescript.json - package.json -``` - -Our `package.json` file within our codebase root would look like this: - -```json -{ - "name": "myproject", - "private": true, - "workspaces": { - "packages": [ - "app", - "common", - "myplugin" - ] - } -} -``` - - -Our `app` folder would be our toplevel package, consuming our `common` and `myplugin` packages as `pinned-dependencies`. The configuration for `app/rescript.json` looks like this: - -```json -{ - "name": "app", - "version": "1.0.0", - "sources": { - "dir" : "src", - "subdirs" : true - }, - /* ... */ - "bs-dependencies": [ - "common", - "myplugin" - ], - "pinned-dependencies": ["common", "myplugin"], - /* ... */ -} -``` - -Now, whenever we are running `rescript build` within our `app` package, the compiler would always rebuild any changes within its pinned dependencies as well. - -**Important:** ReScript will not rebuild any `pinned-dependencies` in watch mode! This is due to the complexity of file watching, so you'd need to set up your own file-watcher process that runs `rescript build` on specific file changes. diff --git a/pages/docs/manual/latest/control-flow.mdx b/pages/docs/manual/latest/control-flow.mdx deleted file mode 100644 index 0609725a6..000000000 --- a/pages/docs/manual/latest/control-flow.mdx +++ /dev/null @@ -1,211 +0,0 @@ ---- -title: "If-Else & Loops" -description: "If, else, ternary, for, and while" -canonical: "/docs/manual/latest/control-flow" ---- - -# If-Else & Loops - -ReScript supports `if`, `else`, ternary expression (`a ? b : c`), `for` and `while`. - -ReScript also supports our famous pattern matching, which will be covered in [its own section](pattern-matching-destructuring.md) - -## If-Else & Ternary - -Unlike its JavaScript counterpart, ReScript's `if` is an expression; they evaluate to their body's content: - - - -```res -let message = if isMorning { - "Good morning!" -} else { - "Hello!" -} -``` -```js -var message = isMorning ? "Good morning!" : "Hello!"; -``` - - - -**Note:** an `if-else` expression without the final `else` branch implicitly gives `()` (aka the `unit` type). So this: - - - -```res -if showMenu { - displayMenu() -} -``` -```js -if (showMenu) { - displayMenu(); -} -``` - - - -is basically the same as: - - - -```res -if showMenu { - displayMenu() -} else { - () -} -``` -```js -if (showMenu) { - displayMenu() -} -``` - - - -Here's another way to look at it. This is clearly wrong: - -```res -let result = if showMenu { - 1 + 2 -} -``` - -It'll give a type error, saying basically that the implicit `else` branch has the type `unit` while the `if` branch has type `int`. Intuitively, this makes sense: what would `result`'s value be, if `showMenu` was `false`? - -We also have ternary sugar, but **we encourage you to prefer if-else when possible**. - - - -```res -let message = isMorning ? "Good morning!" : "Hello!" -``` -```js -var message = isMorning ? "Good morning!" : "Hello!"; -``` - - - -**`if-else` and ternary are much less used** in ReScript than in other languages; [Pattern-matching](pattern-matching-destructuring.md) kills a whole category of code that previously required conditionals. - -## For Loops - -For loops iterate from a starting value up to (and including) the ending value. - - - -```res -for i in startValueInclusive to endValueInclusive { - Console.log(i) -} -``` -```js -for(var i = startValueInclusive; i <= endValueInclusive; ++i){ - console.log(i); -} -``` - - - - - -```res example -// prints: 1 2 3, one per line -for x in 1 to 3 { - Console.log(x) -} -``` -```js -for(var x = 1; x <= 3; ++x){ - console.log(x); -} -``` - - - -You can make the `for` loop count in the opposite direction by using `downto`. - - - -```res -for i in startValueInclusive downto endValueInclusive { - Console.log(i) -} -``` -```js -for(var i = startValueInclusive; i >= endValueInclusive; --i){ - console.log(i); -} -``` - - - - - -```res example -// prints: 3 2 1, one per line -for x in 3 downto 1 { - Console.log(x) -} -``` -```js -for(var x = 3; x >= 1; --x){ - console.log(x); -} -``` - - - -## While Loops - -While loops execute its body code block while its condition is true. - - - -```res -while testCondition { - // body here -} -``` -```js -while (testCondition) { - // body here -} -``` - - - -### Tips & Tricks - -There's no loop-breaking `break` keyword (nor early `return` from functions, for that matter) in ReScript. However, we can break out of a while loop easily through using a [mutable binding](mutation.md). - - - -```res example -let break = ref(false) - -while !break.contents { - if Math.random() > 0.3 { - break := true - } else { - Console.log("Still running") - } -} -``` -```js -var $$break = { - contents: false -}; - -while(!$$break.contents) { - if (Math.random() > 0.3) { - $$break.contents = true; - } else { - console.log("Still running"); - } -}; -``` - - diff --git a/pages/docs/manual/latest/converting-from-js.mdx b/pages/docs/manual/latest/converting-from-js.mdx deleted file mode 100644 index 5f44b3a19..000000000 --- a/pages/docs/manual/latest/converting-from-js.mdx +++ /dev/null @@ -1,303 +0,0 @@ ---- -title: "Converting from JS" -description: "How to convert to ReScript with an existing JS codebase" -canonical: "/docs/manual/latest/converting-from-js" ---- - -# Converting from JS - -ReScript offers a unique project conversion methodology which: -- Ensures minimal disruption to your teammates (very important!). -- Remove the typical friction of verifying conversion's correctness and performance guarantees. -- Doesn't force you to search for pre-made binding libraries made by others. **ReScript doesn't need the equivalent of TypeScript's `DefinitelyTyped`**. - -## Step 1: Install ReScript - -Run `npm install rescript` on your project, then imitate our [New Project](installation#new-project) workflow by adding a `rescript.json` at the root. Then start `npx rescript -w`. - -## Step 2: Copy Paste the Entire JS File - -Let's work on converting a file called `src/main.js`. - -```js -const school = require('school'); - -const defaultId = 10; - -function queryResult(usePayload, payload) { - if (usePayload) { - return payload.student; - } else { - return school.getStudentById(defaultId); - } -} -``` - -First, copy the entire file content over to a new file called `src/Main.res` by using our [`%%raw` JS embedding trick](embed-raw-javascript): - - - -```res example -%%raw(` -const school = require('school'); - -const defaultId = 10; - -function queryResult(usePayload, payload) { - if (usePayload) { - return payload.student; - } else { - return school.getStudentById(defaultId); - } -} -`) -``` -```js -// Generated by ReScript, PLEASE EDIT WITH CARE -'use strict'; - -const school = require('school'); - -const defaultId = 10; - -function queryResult(usePayload, payload) { - if (usePayload) { - return payload.student; - } else { - return school.getStudentById(defaultId); - } -} - -/* Not a pure module */ -``` - - - -Add this file to `rescript.json`: - -```json - "sources": { - "dir" : "src", - "subdirs" : true - }, -``` - -Open an editor tab for `src/Main.res.js`. Do a command-line `diff -u src/main.js src/Main.res.js`. Aside from whitespaces, you should see only minimal, trivial differences. You're already a third of the way done! - -**Always make sure** that at each step, you keep the ReScript output `.res.js` file open to compare against the existing JavaScript file. Our compilation output is very close to your hand-written JavaScript; you can simply eye the difference to catch conversion bugs! - -## Step 3: Extract Parts into Idiomatic ReScript - -Let's turn the `defaultId` variable into a ReScript let-binding: - - - -```res example -let defaultId = 10 - -%%raw(` -const school = require('school'); - -function queryResult(usePayload, payload) { - if (usePayload) { - return payload.student; - } else { - return school.getStudentById(defaultId); - } -} -`) -``` -```js -// Generated by ReScript, PLEASE EDIT WITH CARE -'use strict'; - -const school = require('school'); - -function queryResult(usePayload, payload) { - if usePayload { - return payload.student - } else { - return school.getStudentById(defaultId) - } -} - -var defaultId = 10; - -exports.defaultId = defaultId; -/* Not a pure module */ -``` - - - -Check the output. Diff it. Code still works. Moving on! Extract the function: - - - -```res -%%raw(` -const school = require('school'); -`) - -let defaultId = 10 - -let queryResult = (usePayload, payload) => { - if usePayload { - payload.student - } else { - school.getStudentById(defaultId) - } -} -``` -```js -``` - - - -Format the code: `./node_modules/.bin/rescript format src/Main.res`. - -We have a type error: "The record field student can't be found". That's fine! **Always ensure your code is syntactically valid first**. Fixing type errors comes later. - -## Step 4: Add `external`s, Fix Types - -The previous type error is caused by `payload`'s record declaration (which supposedly contains the field `student`) not being found. Since we're trying to convert as quickly as possible, let's use our [object](object) feature to avoid needing type declaration ceremonies: - - - -```res -%%raw(` -const school = require('school'); -`) - -let defaultId = 10 - -let queryResult = (usePayload, payload) => { - if usePayload { - payload["student"] - } else { - school["getStudentById"](defaultId) - } -} -``` -```js -``` - - - -Now this triggers the next type error, that `school` isn't found. Let's use [`external`](external) to bind to that module: - - - -```res example -@module external school: 'whatever = "school" - -let defaultId = 10 - -let queryResult = (usePayload, payload) => { - if usePayload { - payload["student"] - } else { - school["getStudentById"](defaultId) - } -} -``` -```js -// Generated by ReScript, PLEASE EDIT WITH CARE -'use strict'; - -var School = require("school"); - -function queryResult(usePayload, payload) { - if (usePayload) { - return payload.student; - } else { - return School.getStudentById(10); - } -} - -var defaultId = 10; - -exports.defaultId = defaultId; -exports.queryResult = queryResult; -/* school Not a pure module */ -``` - - - -We hurrily typed `school` as a polymorphic `'whatever` and let its type be inferred by its usage below. The inference is technically correct, but within the context of bringing it a value from JavaScript, slightly dangerous. This is just the interop trick we've shown in the [`external`](external) page. - -Anyway, the file passes the type checker again. Check the `.res.js` output, diff with the original `.js`; we've now converted a file over to ReScript! - -Now, you can delete the original, hand-written `main.js` file, and grep the files importing `main.js` and change them to importing `Main.res.js`. - -## (Optional) Step 5: Cleanup - -If you prefer more advanced, rigidly typed `payload` and `school`, feel free to do so: - - - -```res example -type school -type student -type payload = { - student: student -} - -@module external school: school = "school" -@send external getStudentById: (school, int) => student = "getStudentById" - -let defaultId = 10 - -let queryResult = (usePayload, payload) => { - if usePayload { - payload.student - } else { - school->getStudentById(defaultId) - } -} -``` -```js -// Generated by ReScript, PLEASE EDIT WITH CARE -'use strict'; - -var School = require("school"); - -function queryResult(usePayload, payload) { - if (usePayload) { - return payload.student; - } else { - return School.getStudentById(10); - } -} - -var defaultId = 10; - -exports.defaultId = defaultId; -exports.queryResult = queryResult; -/* school Not a pure module */ -``` - - - -We've: -- introduced an opaque types for `school` and `student` to prevent misuse of their values -- typed the payload as a record with only the `student` field -- typed `getStudentById` as the sole method of `student` - -Check that the `.res.js` output didn't change. How rigidly to type your JavaScript code is up to you; we recommend not typing them too elaborately; it's sometime an endless chase, and produces diminishing returns, especially considering that the elaborate-ness might turn off your potential teammates. - -## Tips & Tricks - -In the same vein of idea, **resist the urge to write your own wrapper functions for the JS code you're converting**. Use [`external`s](external), which are guaranteed to be erased in the output. And avoid trying to take the occasion to convert JS data structures into ReScript-specific data structures like variant or list. **This isn't the time for that**. - -The moment you produce extra conversion code in the output, your skeptical teammate's mental model might switch from "I recognize this output" to "this conversion might be introducing more problems than it solves. Why are we testing ReScript again?". Then you've lost. - -## Conclusion - -- Paste the JS code into a new ReScript file as embedded raw JS code. -- Compile and keep the output file open. Check and diff against original JS file. Free regression tests. -- Always make sure your file is syntactically valid. Don't worry about fixing types before that. -- (Ab)use [object](object.md) accesses to quickly convert things over. -- Optionally clean up the types for robustness. -- Don't go overboard and turn off your boss and fellow teammates. -- Proudly display that you've conserved the semantics and performance characteristics during the conversion by showing your teammates the eerily familiar output. -- Get promoted for introducing a new technology the safer, mature way. diff --git a/pages/docs/manual/latest/editor-plugins.mdx b/pages/docs/manual/latest/editor-plugins.mdx deleted file mode 100644 index 41393093f..000000000 --- a/pages/docs/manual/latest/editor-plugins.mdx +++ /dev/null @@ -1,20 +0,0 @@ ---- -title: "Editor Plugins" -description: "List of ReScript editor plugins" -canonical: "/docs/manual/latest/editor-plugins" ---- - -# Editor Plugins - -- [VSCode](https://marketplace.visualstudio.com/items?itemName=chenglou92.rescript-vscode) -- [Sublime Text](https://github.com/rescript-lang/rescript-sublime) -- [Vim/Neovim](https://github.com/rescript-lang/vim-rescript) - -## Community Supported - -We don't officially support these; use them at your own risk! - -- [Neovim Tree-sitter](https://github.com/nkrkv/nvim-treesitter-rescript) -- [IDEA](https://github.com/reasonml-editor/reasonml-idea-plugin) -- [Emacs](https://github.com/jjlee/rescript-mode) -- [Zed](https://github.com/humaans/rescript-zed) diff --git a/pages/docs/manual/latest/embed-raw-javascript.mdx b/pages/docs/manual/latest/embed-raw-javascript.mdx deleted file mode 100644 index a1c965c6c..000000000 --- a/pages/docs/manual/latest/embed-raw-javascript.mdx +++ /dev/null @@ -1,110 +0,0 @@ ---- -title: "Embed Raw JavaScript" -description: "Utility syntax to for raw JS usage in ReScript" -canonical: "/docs/manual/latest/embed-raw-javascript" ---- - -# Embed Raw JavaScript - -## Paste Raw JS Code - -First thing first. If you're ever stuck learning ReScript, remember that you can always just paste raw JavaScript code into our source file: - - - -```res example -%%raw(` -// look ma, regular JavaScript! -var message = "hello"; -function greet(m) { - console.log(m) -} -`) -``` -```js -// look ma, regular JavaScript! -var message = "hello"; -function greet(m) { - console.log(m) -} -``` - - - -The `%%raw` special ReScript call takes your code string and pastes it as-is into the output. **You've now technically written your first ReScript file!** - -(The back tick syntax is a multiline string, similar to JavaScript's. Except for us, no escaping is needed inside the string. More on string in a later section.) - -While `%%raw` lets you embed top-level raw JS code, `%raw` lets you embed expression-level JS code: - - - -```res example -let add = %raw(` - function(a, b) { - console.log("hello from raw JavaScript!"); - return a + b - } -`) - -Console.log(add(1, 2)) -``` -```js -var add = function(a, b) { - console.log("hello from raw JavaScript!"); - return a + b -}; - -console.log(add(1, 2)); -``` - - - -The above code: -- declared a ReScript variable `add`, -- with the raw JavaScript value of a function declaration, -- then called that function in ReScript. - -If your boss is ever worried that your teammates can't adopt ReScript, just let them keep writing JavaScript inside ReScript files =). - -## Debugger - -You can also drop a `%debugger` expression in a body: - - - -```res example -let f = (x, y) => { - %debugger - x + y -} -``` -```js -function f(x, y) { - debugger; - return x + y | 0; -} -``` - - - -Output: - -```js -function f(x, y) { - debugger; // JavaScript developer tools will set an breakpoint and stop here - x + y; -} -``` - -## Tips & Tricks - -Embedding raw JS snippets isn't the best way to experience ReScript, though it's also highly useful if you're just starting out. As a matter of fact, the first few ReScript projects were converted through: - -- pasting raw JS snippets inside a file -- examining the JS output (identical to the old hand-written JS) -- gradually extract a few values and functions and making sure the output still looks OK - -At the end, we get a fully safe, converted ReScript file whose JS output is clean enough that we can confidently assert that no new bug has been introduced during the conversion process. - -We have a small guide on this iteration [here](converting-from-js.md). Feel free to peruse it later. diff --git a/pages/docs/manual/latest/equality-comparison.mdx b/pages/docs/manual/latest/equality-comparison.mdx deleted file mode 100644 index f862190a4..000000000 --- a/pages/docs/manual/latest/equality-comparison.mdx +++ /dev/null @@ -1,125 +0,0 @@ ---- -title: "Equality and Comparison" -description: "Handling equality and comparison checks" -canonical: "/docs/manual/latest/equality-comparison" ---- - -# Equality and Comparison - -ReScript has shallow equality `===`, deep equality `==`, and comparison operators `>`, `>=`, `<`, and `<=`. - -## Shallow equality -The shallow equality operator `===` compares two values and either compiles to `===` or a `bool` if the equality is known to the compiler. -It behaves the same as the strict equality operator `===` in JavaScript. - -Using `===` will never add a runtime cost. - - - -```res -let t1 = 1 === 1 // true -let t2 = "foo" === "foo" // true -let t3 = { "foo": "bar" } === { "foo": "bar"} // false - -let doStringsMatch = (s1: string, s2: string) => s1 === s2 -``` -```js -var t1 = true; -var t2 = "foo" === "foo"; -var t3 = ({ foo: "bar" }) === ({ foo: "bar" }); - -function doStringsMatch(s1, s2) { - return s1 === s2; -} -``` - - - -## Deep equality -ReScript has the deep equality operator `==` to check deep equality of two items, which is very different from the loose equality operator like `==` in JavaScript. - -When using `==` in ReScript it will never compile to `==` in JavaScript, -it will either compile to `===`, a runtime call to an internal function that deeply compares the equality, or a `bool` if the equality is known to the compiler. - - - -```res -let t1 = 1 == 1 // true -let t2 = "foo" == "foo" // true -let t3 = { "foo": "bar" } == { "foo": "bar"} // true - -let doStringsMatch = (s1: string, s2: string) => s1 == s2 -``` -```js -import * as Caml_obj from "./stdlib/caml_obj.js"; - -var t1 = true; -var t2 = true; -var t3 = Caml_obj.equal({ foo: "bar" }, { foo: "bar" }); - -function doStringsMatch(s1, s2) { - return s1 === s2; -} -``` - - -`==` will compile to `===` (or a `bool` if the compiler can determine equality) when: - -- Comparing `string`, `char`, `int`, `float`, `bool`, or `unit` -- Comparing variants or polymorphic variants that do not have constructor values - -`==` will compile to a runtime check for deep equality when: -- Comparing `array`, `tuple`, `list`, `object`, `record`, or regular expression `Re.t` -- Comparing variants or polymorphic variants that have constructor values - -> When using `==` pay close attention to the JavaScript output if you're not sure what `==` will compile to. - -## Comparison -ReScript has operators for comparing values that compile to the the same operator in JS, a runtime check using an internal function, or a `bool` if the equality is known to the compiler, - -| operator | comparison | -| --- | ----------- | -| `>` | greater than | -| `>=` | greater than or equal | -| `<` | less than | -| `<=` | less than or equal | - -Comparison can be done on any type. - -An operator will compile to the same operator (or a `bool` if the compiler can determine equality) when: -- Comparing `int`, `float`, `string`, `char`, `bool` - -An operator will compile to a runtime check for deep equality when: -- Comparing `array`, `tuple`, `list`, `object`, `record`, or regular expression (`Re.t`) -- Comparing variants or polymorphic variants - - - -```res -let compareInt = (a: int, b: int) => a > b -let t1 = 1 > 10 -let compareArray = (a: array, b: array) => a > b -let compareOptions = (a: option, b: option) => a < b -``` -```js -import * as Caml_obj from "./stdlib/caml_obj.js"; - -function compareInt(a, b) { - return a > b; -} - -var t1 = false; - -var compareArray = Caml_obj.greaterthan; - -var compareOptions = Caml_obj.lessthan; -``` - - -## Performance of runtime equality checks -The runtime equality check ReScript uses is quite fast and should be adequate for almost all use cases. -For small objects it can be 2x times faster than alternative deep compare functions such as Lodash's [`_.isEqual`](https://lodash.com/docs/4.17.15#isEqual). - -For larger objects instead of using `==` you could manually use a faster alternative such as [fast-deep-compare](https://www.npmjs.com/package/fast-deep-equal), or write a custom comparator function. - -[This repo](https://github.com/jderochervlk/rescript-perf) has benchmarks comparing results of different libraries compared to ReScript's built-in equality function. \ No newline at end of file diff --git a/pages/docs/manual/latest/exception.mdx b/pages/docs/manual/latest/exception.mdx deleted file mode 100644 index f68c2d1d7..000000000 --- a/pages/docs/manual/latest/exception.mdx +++ /dev/null @@ -1,498 +0,0 @@ ---- -title: "Exception" -description: "Exceptions and exception handling in ReScript" -canonical: "/docs/manual/latest/exception" ---- - -# Exception - -Exceptions are just a special kind of variant, thrown in **exceptional** cases (don't abuse them!). Consider using the [`option`](null-undefined-option.mdx) or [`result`](api/core/result) type for recoverable errors. - -You can create your own exceptions like you'd make a variant (exceptions need to be capitalized too). - - - -```res example -exception InputClosed(string) -// later on -raise(InputClosed("The stream has closed!")) -``` -```js -import * as Caml_exceptions from "./stdlib/caml_exceptions.js"; - -var InputClosed = /* @__PURE__ */Caml_exceptions.create("Playground.InputClosed"); - -throw { - RE_EXN_ID: InputClosed, - _1: "The stream has closed!", - Error: new Error() - }; -``` - - - -## Built-in Exceptions - -ReScript has some built-in exceptions: - -### `Not_found` - - - -```res prelude -let getItem = (item: int) => - if (item === 3) { - // return the found item here - 1 - } else { - raise(Not_found) - } - -let result = - try { - getItem(2) - } catch { - | Not_found => 0 // Default value if getItem throws - } -``` -```js -import * as Caml_js_exceptions from "./stdlib/caml_js_exceptions.js"; - -function getItem(item) { - if (item === 3) { - return 1; - } - throw { - RE_EXN_ID: "Not_found", - Error: new Error() - }; -} - -var result; - -try { - result = getItem(2); -} -catch (raw_exn){ - var exn = Caml_js_exceptions.internalToOCamlException(raw_exn); - if (exn.RE_EXN_ID === "Not_found") { - result = 0; - } else { - throw exn; - } -} -``` - - - -Note that the above is just for demonstration purposes; in reality, you'd return an `option` directly from `getItem` and avoid the `try` altogether. - -You can directly match on exceptions _while_ getting another return value from a function: - - - -```res prelude -switch list{1, 2, 3}->List.getExn(4) { -| item => Console.log(item) -| exception Not_found => Console.log("No such item found!") -} -``` -```js -import * as Core__List from "./stdlib/core__List.js"; -import * as Caml_js_exceptions from "./stdlib/caml_js_exceptions.js"; - -var exit = 0; - -var item; - -try { - item = Core__List.getExn({ - hd: 1, - tl: { - hd: 2, - tl: { - hd: 3, - tl: /* [] */0 - } - } - }, 4); - exit = 1; -} -catch (raw_exn){ - var exn = Caml_js_exceptions.internalToOCamlException(raw_exn); - if (exn.RE_EXN_ID === "Not_found") { - console.log("No such item found!"); - } else { - throw exn; - } -} - -if (exit === 1) { - console.log(item); -} -``` - - - -### `Invalid_argument` - -Used to check if argument is valid. This exception takes a string. - - -```res example -let divide = (a, b) => - if b == 0 { - raise(Invalid_argument("Denominator is zero")) - } else { - a / b - } - -// catch error -try divide(2, 0)->Console.log catch { -| Invalid_argument(msg) => Console.log(msg) // Denominator is zero -} -``` - -```js -import * as Caml_int32 from "./stdlib/caml_int32.js"; -import * as Caml_js_exceptions from "./stdlib/caml_js_exceptions.js"; - -function divide(a, b) { - if (b === 0) { - throw { - RE_EXN_ID: "Invalid_argument", - _1: "Denominator is zero", - Error: new Error() - }; - } - return Caml_int32.div(a, b); -} - -try { - console.log(divide(2, 0)); -} -catch (raw_msg){ - var msg = Caml_js_exceptions.internalToOCamlException(raw_msg); - if (msg.RE_EXN_ID === "Invalid_argument") { - console.log(msg._1); - } else { - throw msg; - } -} -``` - - - -### `Assert_failure` - -Raise when you use `assert(condition)` and `condition` is false. The arguments -are the location of the `assert` in the source code (file name, line number, column number). - - - -```res example -let decodeUser = (json: JSON.t) => - switch json { - | Object(userDict) => - switch (userDict->Dict.get("name"), userDict->Dict.get("age")) { - | (Some(String(name)), Some(Number(age))) => (name, age->Float.toInt) - | _ => assert(false) - } - | _ => assert(false) - } - - -try decodeUser(%raw("{}"))->Console.log catch { -| Assert_failure(loc) => Console.log(loc) // ("filename", line, col) -} -``` - -```js -mport * as Caml_js_exceptions from "./stdlib/caml_js_exceptions.js"; - -function decodeUser(json) { - if (!Array.isArray(json) && (json === null || typeof json !== "object") && typeof json !== "number" && typeof json !== "string" && typeof json !== "boolean") { - throw { - RE_EXN_ID: "Assert_failure", - _1: [ - "playground.res", - 8, - 9 - ], - Error: new Error() - }; - } - if (typeof json === "object" && !Array.isArray(json)) { - var match = json["name"]; - var match$1 = json["age"]; - if (match !== undefined && !(!Array.isArray(match) && (match === null || typeof match !== "object") && typeof match !== "number" && typeof match !== "string" && typeof match !== "boolean") && typeof match === "string" && match$1 !== undefined && !(!Array.isArray(match$1) && (match$1 === null || typeof match$1 !== "object") && typeof match$1 !== "number" && typeof match$1 !== "string" && typeof match$1 !== "boolean") && typeof match$1 === "number") { - return [ - match, - match$1 | 0 - ]; - } - throw { - RE_EXN_ID: "Assert_failure", - _1: [ - "playground.res", - 6, - 11 - ], - Error: new Error() - }; - } - throw { - RE_EXN_ID: "Assert_failure", - _1: [ - "playground.res", - 8, - 9 - ], - Error: new Error() - }; -} - -try { - console.log(decodeUser({})); -} -catch (raw_loc){ - var loc = Caml_js_exceptions.internalToOCamlException(raw_loc); - if (loc.RE_EXN_ID === "Assert_failure") { - console.log(loc._1); - } else { - throw loc; - } -} -``` - - - -### `Failure` - -Exception raised to signal that the given arguments do not make sense. This -exception takes a string as an argument. - - - -```res example -let isValidEmail = email => { - let hasAtSign = String.includes(email, "@") - let hasDot = String.includes(email, ".") - if !(hasAtSign && hasDot) { - raise(Failure("Invalid email address")) - } else { - true - } -} - - -let isValid = try isValidEmail("rescript.org") catch { -| Failure(msg) => { - Console.error(msg) - false - } -} -``` - -```js -import * as Caml_js_exceptions from "./stdlib/caml_js_exceptions.js"; - -function isValidEmail(email) { - var hasAtSign = email.includes("@"); - var hasDot = email.includes("."); - if (hasAtSign && hasDot) { - return true; - } - throw { - RE_EXN_ID: "Failure", - _1: "Invalid email address", - Error: new Error() - }; -} - -var isValid; - -try { - isValid = isValidEmail("rescript.org"); -} -catch (raw_msg){ - var msg = Caml_js_exceptions.internalToOCamlException(raw_msg); - if (msg.RE_EXN_ID === "Failure") { - console.error(msg._1); - isValid = false; - } else { - throw msg; - } -} -``` - - - -### `Division_by_zero` - -Exception raised by integer division and remainder operations when their second argument is zero. - - - -```res example -// ReScript raise `Division_by_zero` if the denominator is zero -let result = try Some(10 / 0) catch { -| Division_by_zero => None -} - -Console.log(result) // None -``` - -```js -import * as Caml_int32 from "./stdlib/caml_int32.js"; -import * as Caml_js_exceptions from "./stdlib/caml_js_exceptions.js"; - -var result; - -try { - result = Caml_int32.div(10, 0); -} -catch (raw_exn){ - var exn = Caml_js_exceptions.internalToOCamlException(raw_exn); - if (exn.RE_EXN_ID === "Division_by_zero") { - result = undefined; - } else { - throw exn; - } -} - -console.log(result); -``` - - - -## Catching JS Exceptions - -To distinguish between JavaScript exceptions and ReScript exceptions, ReScript namespaces JS exceptions under the `Exn.Error(payload)` variant. To catch an exception thrown from the JS side: - - -Throw an exception from JS: - -```js -// Example.js - -exports.someJsFunctionThatThrows = () => { - throw new Error("A Glitch in the Matrix!"); -} -``` - -Then catch it from ReScript: - -```res -// import the method in Example.js -@module("./Example") -external someJsFunctionThatThrows: () => unit = "someJsFunctionThatThrows" - -try { - // call the external method - someJSFunctionThatThrows() -} catch { -| Exn.Error(obj) => - switch Exn.message(obj) { - | Some(m) => Console.log("Caught a JS exception! Message: " ++ m) - | None => () - } -} -``` - -The `obj` here is of type `Exn.t`, intentionally opaque to disallow illegal operations. To operate on `obj`, do like the code above by using the standard library's [`Exn`](api/js/exn) module's helpers. - -## Raise a JS Exception - -`raise(MyException)` raises a ReScript exception. To raise a JavaScript exception (whatever your purpose is), use `Exn.raiseError`: - - - -```res example -let myTest = () => { - Exn.raiseError("Hello!") -} -``` -```js -var Js_exn = require("./stdlib/js_exn.js"); - -function myTest() { - return Js_exn.raiseError("Hello!"); -} -``` - - - -Then you can catch it from the JS side: - -```js -// after importing `myTest`... -try { - myTest() -} catch (e) { - console.log(e.message) // "Hello!" -} -``` - -## Catch ReScript Exceptions from JS - -The previous section is less useful than you think; to let your JS code work with your exception-throwing ReScript code, the latter doesn't actually need to throw a JS exception. ReScript exceptions can be used by JS code! - - - -```res example -exception BadArgument({myMessage: string}) - -let myTest = () => { - raise(BadArgument({myMessage: "Oops!"})) -} -``` -```js -var Caml_exceptions = require("./stdlib/caml_exceptions.js"); - -var BadArgument = Caml_exceptions.create("Playground.BadArgument"); - -function myTest() { - throw { - RE_EXN_ID: BadArgument, - myMessage: "Oops!", - Error: new Error() - }; -} -``` - - - -Then, in your JS: - -```js -// after importing `myTest`... -try { - myTest() -} catch (e) { - console.log(e.myMessage) // "Oops!" - console.log(e.Error.stack) // the stack trace -} -``` - -> Note: `RE_EXN_ID` is an internal field for bookkeeping purposes. Don't use it on the JS side. Use the other fields. - -The above `BadArgument` exception takes an inline record type. We special-case compile the exception as `{RE_EXN_ID, myMessage, Error}` for good ergonomics. If the exception instead took ordinary positional arguments, l like the standard library's `Invalid_argument("Oops!")`, which takes a single argument, the argument is compiled to JS as the field `_1` instead. A second positional argument would compile to `_2`, etc. - -## Tips & Tricks - -When you have ordinary variants, you often don't **need** exceptions. For example, instead of throwing when `item` can't be found in a collection, try to return an `option` (`None` in this case) instead. - -### Catch Both ReScript and JS Exceptions in the Same `catch` Clause - -```res -try { - someOtherJSFunctionThatThrows() -} catch { -| Not_found => ... // catch a ReScript exception -| Invalid_argument(_) => ... // catch a second ReScript exception -| Exn.Error(obj) => ... // catch the JS exception -} -``` - -This technically works, but hopefully you don't ever have to work with such code... diff --git a/pages/docs/manual/latest/extensible-variant.mdx b/pages/docs/manual/latest/extensible-variant.mdx deleted file mode 100644 index d4133d584..000000000 --- a/pages/docs/manual/latest/extensible-variant.mdx +++ /dev/null @@ -1,75 +0,0 @@ ---- -title: "Extensible Variant" -description: "Extensible Variants in ReScript" -canonical: "/docs/manual/latest/extensible-variant" ---- - -# Extensible Variant - -Variant types are usually constrained to a fixed set of constructors. There may be very rare cases where you still want to be able to add constructors to a variant type even after its initial type declaration. For this, we offer extensible variant types. - -## Definition and Usage - - - -```res example -type t = .. - -type t += Other - -type t += - | Point(float, float) - | Line(float, float, float, float) -``` -```js -var Caml_exceptions = require("./stdlib/caml_exceptions.js"); - -var Other = Caml_exceptions.create("Playground.Other"); - -var Point = Caml_exceptions.create("Playground.Point"); - -var Line = Caml_exceptions.create("Playground.Line"); -``` - - - -The `..` in the type declaration above defines an extensible variant `type t`. The `+=` operator is then used to add constructors to the given type. - -**Note:** Don't forget the leading `type` keyword when using the `+=` operator! - -## Pattern Matching Caveats - -Extensible variants are open-ended, so the compiler will not be able to exhaustively pattern match all available cases. You will always need to provide a default `_` case for every `switch` expression. - - - - - -```res -let print = v => - switch v { - | Point(x, y) => Console.log2("Point", (x, y)) - | Line(ax, ay, bx, by) => Console.log2("Line", (ax, ay, bx, by)) - | Other - | _ => Console.log("Other") - } -``` -```js -function print(v) { - if (v.RE_EXN_ID === Point) { - console.log("Point", [v._1, v._2]); - } else if (v.RE_EXN_ID === Line) { - console.log("Line", [v._1, v._2, v._3, v._4]); - } else { - console.log("Other"); - } -} -``` - - - -## Tips & Tricks - -**Fun fact:** In ReScript, [exceptions](./exception) are actually extensible variants under the hood, so `exception UserError(string)` is equivalent to `type exn += UserError(string)`. It's one of the very few use-case where extensible variants make sense. - -We usually recommend sticking with common [variants](./variant) as much as possible to reap the benefits of exhaustive pattern matching. \ No newline at end of file diff --git a/pages/docs/manual/latest/external.mdx b/pages/docs/manual/latest/external.mdx deleted file mode 100644 index d722646b9..000000000 --- a/pages/docs/manual/latest/external.mdx +++ /dev/null @@ -1,91 +0,0 @@ ---- -title: "External (Bind to Any JS Library)" -description: "The external keyword" -canonical: "/docs/manual/latest/external" ---- - -# External (Bind to Any JS Library) - -`external` is the primary ReScript features for bringing in and using JavaScript values. - -`external` is like a let binding, but: -- The right side of `=` isn't a value; it's the name of the JS value you're referring to. -- The type for the binding is mandatory, since we need to know what the type of that JS value is. -- Can only exist at the top level of a file or module. - - - -```res example -@val external setTimeout: (unit => unit, int) => float = "setTimeout" -``` -```js -// Empty output -``` - - - -There are several kinds of `external`s, differentiated and/or augmented through the [attribute](attribute.md) they carry. This page deals with the general, shared mechanism behind most `external`s. The different `external`s are documented in their respective pages later. A few notable ones: - -- `@val`, `@scope`: [bind to global JS values](bind-to-global-js-values). -- `@module`: [bind to JS imported/exported values](import-from-export-to-js). -- `@send`: [bind to JS methods](bind-to-js-function). - -You can also use our [Syntax Lookup](/syntax-lookup) tool to find them. - -Related: see also our [list of external decorators](interop-cheatsheet#list-of-decorators). - -## Usage - -Once declared, you can use an `external` as a normal value, just like a let binding. - -## Tips & Tricks - -`external` + ReScript objects are a wonderful combination for quick prototyping. Check the JS output tab: - - - -```res example -// The type of document is just some random type 'a -// that we won't bother to specify -@val external document: 'a = "document" - -// call a method -document["addEventListener"]("mouseup", _event => { - Console.log("clicked!") -}) - -// get a property -let loc = document["location"] - -// set a property -document["location"]["href"] = "rescript-lang.org" -``` -```js -document.addEventListener("mouseup", function(_event) { - console.log("clicked!"); -}); - -var loc = document.location; - -document.location.href = "rescript-lang.org"; -``` - - - -We've specified `document`'s type as `'a`, aka a placeholder type that's polymorphic. Any value can be passed there, so you're not getting much type safety (except the inferences at various call sites). However, this is excellent for quickly getting started using a JavaScript library in ReScript **without needing the equivalent of a repository of typed bindings** like TypeScript's `DefinitelyTyped` repo. - -However, if you want to more rigidly bind to the JavaScript library you want, keep reading the next few interop pages. - -## Performance & Output Readability - -`external`s declarations are inlined into their callers during compilation, **and completely disappear from the JS output**. This means any time you use one, you can be sure that you're not incurring extra JavaScript \<-> ReScript conversion cost. - -Additionally, no extra ReScript-specific runtime is better for output readability. - -> **Note:** do also use `external`s and the `@blabla` attributes in the interface files. Otherwise the inlining won't happen. - -## Design Decisions - -ReScript takes interoperating with existing code very seriously. Our type system has very strong guarantees. However, such strong feature also means that, without a great interop system, it'd be very hard to gradually convert a codebase over to ReScript. Fortunately, our interop are comprehensive and cooperate very well with most existing JavaScript code. - -The combination of a sound type system + great interop means that we get the benefits of a traditional gradual type system regarding incremental codebase coverage & conversion, without the downside of such gradual type system: complex features to support existing patterns, slow analysis, diminishing return in terms of type coverage, etc. diff --git a/pages/docs/manual/latest/faq.mdx b/pages/docs/manual/latest/faq.mdx deleted file mode 100644 index b99590aed..000000000 --- a/pages/docs/manual/latest/faq.mdx +++ /dev/null @@ -1,66 +0,0 @@ ---- -title: "FAQ" -description: "Frequently asked questions about ReScript and its ecosystem" -canonical: "/docs/manual/latest/faq" ---- - -# Frequently Asked Questions - -**What's the goal of this project?** - -We aim to provide the best typed language experience for the JavaScript platform. - - -**What’s the relationship with BuckleScript?** - -BuckleScript is ReScript's old branding, with a sharper focus on proper JS support and familiarity which we previously couldn't achieve to the degree we wanted, due to us needing to cater to various different crowds. - -**What’s ReScript's relationship with OCaml?** - -We reuse and adjust the excellent type system and lots of other high quality components from OCaml for JS experience. -Additionally, ReScript provides its own syntax, build system, IDE, backend, JS interop, extra language features, etc. - -The ReScript toolchain is developed using OCaml, however, the version of ReScript is decoupled against the version of OCaml, -ReScript compiler should build against any reasonable modern version of OCaml compiler. - -For the majority of ReScript users, they don't need to learn OCaml or use OCaml toolchain to be productive in ReScript. - - - -**What’s the relationship with Reason?** - -See [here](/blog/bucklescript-is-rebranding). Reason is a syntax layer for OCaml that BuckleScript also adopted. The current ReScript compiler also supports the old Reason syntax v3.6 for backward compatibility. We will support it for a long time to make sure existing users do not get breaking changes. - -**I come from Reason/OCaml. Will ReScript keep supporting X?** - -Please see our [blog post](/blog/a-note-on-bucklescripts-future-commitments) on this matter. - -**Where can I see the docs in old Reason/OCaml syntax?** - -Switch the doc version to `v8.0.0` in the sidebar on the left! - -**Will ReScript support native compilation eventually?** - -Our focus is a solid JS story right now. In the future, if there’s strong demand, we might consider it. - -**What’s the current state of ReScript?** - -Currently, we're actively working on the editor support. - -**When will we get the `async/await` keywords?** - -async/await has arrived in ReScript 10.1! - -**Why create a new syntax?** - -The existing Reason syntax is owned by a different team with a different vision. Reason aims to be 100% compatible with OCaml syntax and to support all versions of OCaml. In the last few years, we've drawn the conclusion that it’s very hard to deliver such goal without sacrificing user experience. The other reason is that we feel it’s better to have the same vision as a team so that we can make more coherent decisions. - -**Who is behind the project?** - -The ReScript team (Hongbo, Cheng, Cristiano, Maxim, Patrick, Ricky). - -**We have a new forum; will we also have our own Discord?** - -Not now. We've found that too much important information get casually passed in Discord then lost within the noise. We prefer folks to communicate on the [forum](https://forum.rescript-lang.org). This is nicer to the less active members. - -The team doesn't use the old Discord anymore. We encourage you to move your questions to the forum instead. diff --git a/pages/docs/manual/latest/function.mdx b/pages/docs/manual/latest/function.mdx deleted file mode 100644 index 70005e323..000000000 --- a/pages/docs/manual/latest/function.mdx +++ /dev/null @@ -1,620 +0,0 @@ ---- -title: "Function" -description: "Function syntax in ReScript" -canonical: "/docs/manual/latest/function" ---- - -# Function - -_Cheat sheet for the full function syntax at the end_. - -ReScript functions are declared with an arrow and return an expression, just like JS functions. They compile to clean JS functions too. - - - -```res prelude -let greet = (name) => "Hello " ++ name -``` -```js -function greet(name) { - return "Hello " + name; -} -``` - - - -This declares a function and assigns to it the name `greet`, which you can call like so: - - - -```res example -greet("world!") // "Hello world!" -``` -```js -greet("world!"); -``` - - - -Multi-arguments functions have arguments separated by comma: - - - -```res example -let add = (x, y, z) => x + y + z -add(1, 2, 3) // 6 -``` -```js -function add(x, y, z) { - return (x + y | 0) + z | 0; -} -``` - - - -For longer functions, you'd surround the body with a block: - - - -```res example -let greetMore = (name) => { - let part1 = "Hello" - part1 ++ " " ++ name -} -``` -```js -function greetMore(name) { - return "Hello " + name; -} -``` - - - -If your function has no argument, just write `let greetMore = () => {...}`. - -## Labeled Arguments - -Multi-arguments functions, especially those whose arguments are of the same type, can be confusing to call. - - - -```res -let addCoordinates = (x, y) => { - // use x and y here -} -// ... -addCoordinates(5, 6) // which is x, which is y? -``` -```js -function addCoordinates(x, y) { - // use x and y here -} - -addCoordinates(5, 6); -``` - - - -You can attach labels to an argument by prefixing the name with the `~` symbol: - - - -```res -let addCoordinates = (~x, ~y) => { - // use x and y here -} -// ... -addCoordinates(~x=5, ~y=6) -``` -```js -function addCoordinates(x, y) { - // use x and y here -} - -addCoordinates(5, 6); -``` - - - -You can provide the arguments in **any order**: - - - -```res -addCoordinates(~y=6, ~x=5) -``` -```js -addCoordinates(5, 6); -``` - - - -The `~x` part in the declaration means the function accepts an argument labeled `x` and can refer to it in the function body by the same name. You can also refer to the arguments inside the function body by a different name for conciseness: - - - -```res -let drawCircle = (~radius as r, ~color as c) => { - setColor(c) - startAt(r, r) - // ... -} - -drawCircle(~radius=10, ~color="red") -``` -```js -function drawCircle(r, c) { - setColor(c); - return startAt(r, r); -} - -drawCircle(10, "red"); -``` - - - -As a matter of fact, `(~radius)` is just a shorthand for `(~radius as radius)`. - -Here's the syntax for typing the arguments: - - - -```res -let drawCircle = (~radius as r: int, ~color as c: string) => { - // code here -} -``` -```js -function drawCircle(r, c) { - // code here -} -``` - - - -## Optional Labeled Arguments - -Labeled function arguments can be made optional during declaration. You can then omit them when calling the function. - - - -```res -// radius can be omitted -let drawCircle = (~color, ~radius=?) => { - setColor(color) - switch radius { - | None => startAt(1, 1) - | Some(r_) => startAt(r_, r_) - } -} -``` -```js -var Caml_option = require("./stdlib/caml_option.js"); - -function drawCircle(color, radius) { - setColor(color); - if (radius === undefined) { - return startAt(1, 1); - } - var r_ = Caml_option.valFromOption(radius); - return startAt(r_, r_); -} -``` - - - -When given in this syntax, `radius` is **wrapped** in the standard library's `option` type, defaulting to `None`. If provided, it'll be wrapped with a `Some`. So `radius`'s type value is `None | Some(int)` here. - -More on `option` type [here](null-undefined-option.md). - -### Signatures and Type Annotations - -Functions with optional labeled arguments can be confusing when it comes to signature and type annotations. Indeed, the type of an optional labeled argument looks different depending on whether you're calling the function, or working inside the function body. Outside the function, a raw value is either passed in (`int`, for example), or left off entirely. Inside the function, the parameter is always there, but its value is an option (`option`). This means that the type signature is different, depending on whether you're writing out the function type, or the parameter type annotation. The first being a raw value, and the second being an option. - -If we get back to our previous example and both add a signature and type annotations to its argument, we get this: - - - -```res -let drawCircle: (~color: color, ~radius: int=?) => unit = - (~color: color, ~radius: option=?) => { - setColor(color) - switch radius { - | None => startAt(1, 1) - | Some(r_) => startAt(r_, r_) - } - } -``` -```js -function drawCircle(color, radius) { - setColor(color); - if (radius !== undefined) { - return startAt(radius, radius); - } else { - return startAt(1, 1); - } -} -``` - - - -The first line is the function's signature, we would define it like that in an interface file (see [Signatures](module.md#signatures)). The function's signature describes the types that the **outside world** interacts with, hence the type `int` for `radius` because it indeed expects an `int` when called. - -In the second line, we annotate the arguments to help us remember the types of the arguments when we use them **inside** the function's body, here indeed `radius` will be an `option` inside the function. - -So if you happen to struggle when writing the signature of a function with optional labeled arguments, try to remember this! - -### Explicitly Passed Optional - -Sometimes, you might want to forward a value to a function without knowing whether the value is `None` or `Some(a)`. Naively, you'd do: - - - -```res -let result = - switch payloadRadius { - | None => drawCircle(~color) - | Some(r) => drawCircle(~color, ~radius=r) - } -``` -```js -var r = payloadRadius; - -var result = r !== undefined - ? drawCircle(color, Caml_option.valFromOption(r)) - : drawCircle(color); -``` - - - -This quickly gets tedious. We provide a shortcut: - - - -```res -let result = drawCircle(~color, ~radius=?payloadRadius) -``` -```js -var result = drawCircle(1, undefined); -``` - - - -This means "I understand `radius` is optional, and that when I pass it a value it needs to be an `int`, but I don't know whether the value I'm passing is `None` or `Some(val)`, so I'll pass you the whole `option` wrapper". - -### Optional with Default Value - -Optional labeled arguments can also be provided a default value. In this case, they aren't wrapped in an `option` type. - - - -```res -let drawCircle = (~radius=1, ~color) => { - setColor(color) - startAt(radius, radius) -} -``` -```js -function drawCircle(radiusOpt, color) { - var radius = radiusOpt !== undefined ? radiusOpt : 1; - setColor(color); - return startAt(radius, radius); -} -``` - - - -## Recursive Functions - -ReScript chooses the sane default of preventing a function to be called recursively within itself. To make a function recursive, add the `rec` keyword after the `let`: - - - -```res example -let rec neverTerminate = () => neverTerminate() -``` -```js -function neverTerminate(_param) { - while(true) { - _param = undefined; - continue ; - }; -} -``` - - - -A simple recursive function may look like this: - - - -```res example -// Recursively check every item on the list until one equals the `item` -// argument. If a match is found, return `true`, otherwise return `false` -let rec listHas = (list, item) => - switch list { - | list{} => false - | list{a, ...rest} => a === item || listHas(rest, item) - } -``` -```js -function listHas(_list, item) { - while(true) { - var list = _list; - if (!list) { - return false; - } - if (list.hd === item) { - return true; - } - _list = list.tl; - continue ; - }; -} -``` - - - -Recursively calling a function is bad for performance and the call stack. However, ReScript intelligently compiles [tail recursion](https://stackoverflow.com/questions/33923/what-is-tail-recursion) into a fast JavaScript loop. Try checking the JS output of the above code! - -### Mutually Recursive Functions - -Mutually recursive functions start like a single recursive function using the -`rec` keyword, and then are chained together with `and`: - - - -```res example -let rec callSecond = () => callFirst() -and callFirst = () => callSecond() -``` -```js -function callSecond(_param) { - while(true) { - _param = undefined; - continue ; - }; -} - -function callFirst(_param) { - while(true) { - _param = undefined; - continue ; - }; -} -``` - - - -## Partial Application - -**Since 11.0** - -To partially apply a function, use the explicit `...` syntax. - - -```res -let add = (a, b) => a + b -let addFive = add(5, ...) -``` - -```js -function add(a, b) { - return a + b | 0; -} - -function addFive(extra) { - return 5 + extra | 0; -} -``` - - -## Async/Await - -Just as in JS, an async function can be declared by adding `async` before the definition, and `await` can be used in the body of such functions. -The output looks like idiomatic JS: - - - -```res example -let getUserName = async (userId) => userId - -let greetUser = async (userId) => { - let name = await getUserName(userId) - "Hello " ++ name ++ "!" -} -``` -```js -async function greetUser(userId) { - var name = await getUserName(userId); - return "Hello " + name + "!"; -} -``` - - -The return type of `getUser` is inferred to be `promise`. -Similarly, `await getUserName(userId)` returns a `string` when the function returns `promise`. -Using `await` outside of an `async` function (including in a non-async callback to an async function) is an error. - -### Ergonomic error handling - -Error handling is done by simply using `try`/`catch`, or a switch with an `exception` case, just as in functions that are not async. -Both JS exceptions and exceptions defined in ReScript can be caught. The compiler takes care of packaging JS exceptions into the builtin `JsError` exception: - - - -```res example -exception SomeReScriptException - -let somethingThatMightThrow = async () => raise(SomeReScriptException) - -let someAsyncFn = async () => { - switch await somethingThatMightThrow() { - | data => Some(data) - | exception JsError(_) => None - | exception SomeReScriptException => None - } -} -``` -```js -var SomeReScriptException = /* @__PURE__ */Caml_exceptions.create("Example.SomeReScriptException"); - -async function someAsyncFn(param) { - var data; - try { - data = await somethingThatMightThrow(undefined); - } - catch (raw_exn){ - var exn = Caml_js_exceptions.internalToOCamlException(raw_exn); - if (exn.RE_EXN_ID === "JsError") { - return ; - } - if (exn.RE_EXN_ID === SomeReScriptException) { - return ; - } - throw exn; - } - return data; -} -``` - - - - -## The ignore() Function - -Occasionally you may want to ignore the return value of a function. ReScript provides an `ignore()` function that discards the value of its argument and returns `()`: - - - -```res -mySideEffect()->Promise.catch(handleError)->ignore - -setTimeout(myFunc, 1000)->ignore -``` - -```js -$$Promise.$$catch(mySideEffect(), function (prim) { - return handleError(prim); -}); - -setTimeout(function (prim) { - myFunc(); -}, 1000); -``` - - - -## Tips & Tricks - -Cheat sheet for the function syntaxes: - -### Declaration - -```res -// anonymous function -(x, y) => 1 -// bind to a name -let add = (x, y) => 1 - -// labeled -let add = (~first as x, ~second as y) => x + y -// with punning sugar -let add = (~first, ~second) => first + second - -// labeled with default value -let add = (~first as x=1, ~second as y=2) => x + y -// with punning -let add = (~first=1, ~second=2) => first + second - -// optional -let add = (~first as x=?, ~second as y=?) => switch x {...} -// with punning -let add = (~first=?, ~second=?) => switch first {...} -``` - -#### With Type Annotation - -```res -// anonymous function -(x: int, y: int): int => 1 -// bind to a name -let add = (x: int, y: int): int => 1 - -// labeled -let add = (~first as x: int, ~second as y: int) : int => x + y -// with punning sugar -let add = (~first: int, ~second: int) : int => first + second - -// labeled with default value -let add = (~first as x: int=1, ~second as y: int=2) : int => x + y -// with punning sugar -let add = (~first: int=1, ~second: int=2) : int => first + second - -// optional -let add = (~first as x: option=?, ~second as y: option=?) : int => switch x {...} -// with punning sugar -// note that the caller would pass an `int`, not `option` -// Inside the function, `first` and `second` are `option`. -let add = (~first: option=?, ~second: option=?) : int => switch first {...} -``` - -### Application - -```res -add(x, y) - -// labeled -add(~first=1, ~second=2) -// with punning sugar -add(~first, ~second) - -// application with default value. Same as normal application -add(~first=1, ~second=2) - -// explicit optional application -add(~first=?Some(1), ~second=?Some(2)) -// with punning -add(~first?, ~second?) -``` - -#### With Type Annotation - -```res -// labeled -add(~first=1: int, ~second=2: int) -// with punning sugar -add(~first: int, ~second: int) - -// application with default value. Same as normal application -add(~first=1: int, ~second=2: int) - -// explicit optional application -add(~first=?Some(1): option, ~second=?Some(2): option) -// no punning sugar when you want to type annotate -``` - -### Standalone Type Signature - -```res -// first arg type, second arg type, return type -type add = (int, int) => int - -// labeled -type add = (~first: int, ~second: int) => int - -// labeled -type add = (~first: int=?, ~second: int=?, unit) => int -``` - -#### In Interface Files - -To annotate a function from the implementation file (`.res`) in your interface file (`.resi`): - -```res sig -let add: (int, int) => int -``` - -The type annotation part is the same as the previous section on With Type Annotation. - -**Don't** confuse `let add: myType` with `type add = myType`. When used in `.resi` interface files, the former exports the binding `add` while annotating it as type `myType`. The latter exports the type `add`, whose value is the type `myType`. diff --git a/pages/docs/manual/latest/generate-converters-accessors.mdx b/pages/docs/manual/latest/generate-converters-accessors.mdx deleted file mode 100644 index 27defa849..000000000 --- a/pages/docs/manual/latest/generate-converters-accessors.mdx +++ /dev/null @@ -1,114 +0,0 @@ ---- -title: "Generate Converters & Helpers" -description: "All about the @deriving decorator, and how to generate code from types" -canonical: "/docs/manual/latest/generate-converters-accessors" ---- - -# Generate Converters & Helpers - -**Note**: if you're looking for: -- `@deriving(jsConverter)` for records -- `@deriving({jsConverter: newType})` for records -- `@deriving(abstract)` for records -- `@deriving(jsConverter)` for plain and polymorphic variants - -These particular ones are no longer needed. Select a doc version lower than `9.0` in the sidebar to see their old docs. - - - -When using ReScript, you will sometimes come into situations where you want to - -- Automatically generate functions that convert between ReScript's internal and JS runtime values (e.g. variants). -- Convert a record type into an abstract type with generated creation, accessor and method functions. -- Generate some other helper functions, such as functions from record attribute names. - -You can use the `@deriving` decorator for different code generation scenarios. All different options and configurations will be discussed on this page. - -**Note:** Please be aware that extensive use of code generation might make it harder to understand your programs (since the code being generated is not visible in the source code, and you just need to know what kind of functions / values a decorator generates). - -## Generate Functions & Plain Values for Variants - -Use `@deriving(accessors)` on a variant type to create accessor functions for its constructors. - - - -```res -@deriving(accessors) -type action = - | Click - | Submit(string) - | Cancel; -``` - -```js -function submit(param_0) { - return /* Submit */[param_0]; -} - -var click = /* Click */0; - -var cancel = /* Cancel */1; - -exports.click = click; -exports.submit = submit; -exports.cancel = cancel; -``` - - - -Variants constructors with payloads generate functions, payload-less constructors generate plain integers (the internal representation of variants). - -**Note**: -- The generated accessors are lower-cased. -- You can now use these helpers on the JavaScript side! But don't rely on their actual values please. - -### Usage - -```res -let s = submit("hello"); /* gives Submit("hello") */ -``` - -This is useful: - -- When you're passing the accessor function as a higher-order function (which plain variant constructors aren't). -- When you'd like the JS side to use these values & functions opaquely and pass you back a variant constructor (since JS has no such thing). - -Please note that in case you just want to _pipe a payload into a constructor_, you don't need to generate functions for that. Use the `->` syntax instead, e.g. `"test"->Submit`. - -## Generate Field Accessors for Records - -Use `@deriving(accessors)` on a record type to create accessors for its record field names. - - - - -```res -@deriving(accessors) -type pet = {name: string} - -let pets = [{name: "bob"}, {name: "bob2"}] - -pets - ->Array.map(name) - ->Array.joinWith("&") - ->Console.log -``` - -```js -function name(param) { - return param.name; -} - -var pets = [ - { - name: "bob" - }, - { - name: "bob2" - } -]; - -console.log(Belt_Array.map(pets, name).join("&")); -``` - - diff --git a/pages/docs/manual/latest/import-export.mdx b/pages/docs/manual/latest/import-export.mdx deleted file mode 100644 index dea3feccc..000000000 --- a/pages/docs/manual/latest/import-export.mdx +++ /dev/null @@ -1,38 +0,0 @@ ---- -title: "Import & Export" -description: "Importing / exporting in ReScript modules" -canonical: "/docs/manual/latest/import-export" ---- - -# Import & Export - -## Import a Module/File - -Unlike JavaScript, ReScript doesn't have or need import statements: - - - -```res -// Inside School.res -let studentMessage = Student.message -``` -```js -var Student = require("./Student.res.js"); -var studentMessage = Student.message -``` - - - -The above code refers to the `message` binding in the file `Student.res`. Every ReScript file is also a module, so accessing another file's content is the same as accessing another module's content! - -A ReScript project's file names need to be unique. - -## Export Stuff - -By default, every file's type declaration, binding and module is exported, aka publicly usable by another file. **This also means those values, once compiled into JS, are immediately usable by your JS code**. - -To only export a few selected things, use a `.resi` [interface file](module.md#signatures). - -## Work with JavaScript Import & Export - -To see how to import JS modules and export stuff for JS consumption, see the JavaScript Interop section's [Import from/Export to JS](import-from-export-to-js.md). diff --git a/pages/docs/manual/latest/import-from-export-to-js.mdx b/pages/docs/manual/latest/import-from-export-to-js.mdx deleted file mode 100644 index 6328666c1..000000000 --- a/pages/docs/manual/latest/import-from-export-to-js.mdx +++ /dev/null @@ -1,244 +0,0 @@ ---- -title: "Import from / Export to JS" -description: "Importing / exporting JS module content in ReScript" -canonical: "/docs/manual/latest/import-from-export-to-js" ---- - -# Import from/Export to JS - -You've seen how ReScript's idiomatic [Import & Export](import-export.md) works. This section describes how we work with importing stuff from JavaScript and exporting stuff for JavaScript consumption. - -If you're looking for react-specific interop guidance, check out the [React JS Interop guide](../../react/latest/import-export-reactjs.mdx). - -**Note**: due to JS ecosystem's module compatibility issues, our advice of keeping your ReScript file's compiled JS output open in a tab applies here **more than ever**, as you don't want to subtly output the wrong JS module import/export code, on top of having to deal with Babel/Webpack/Jest/Node's CommonJS \<-> JavaScript module compatibility shims. - -In short: **make sure your bindings below output what you'd have manually written in JS**. - -## Output Format - -We support 2 JavaScript import/export formats: - -- JavaScript module: `import * from 'MyReScriptFile'` and `export let ...`. -- CommonJS: `require('myFile')` and `module.exports = ...`. - -The format is [configurable in via `rescript.json`](build-configuration.md#package-specs). - -## Import From JavaScript - -### Import a JavaScript Module's Named Export - -Use the `module` [external](external.md): - - - -```res example -// Import nodejs' path.dirname -@module("path") external dirname: string => string = "dirname" -let root = dirname("/User/github") // returns "User" -``` -```js -import * as Path from "path"; -var root = Path.dirname("/User/github"); -``` -```js -var Path = require("path"); -var root = Path.dirname("/User/github"); -``` - - - -Here's what the `external` does: - -- `@module("path")`: pass the name of the JS module; in this case, `"path"`. The string can be anything: `"./src/myJsFile"`, `"@myNpmNamespace/myLib"`, etc. -- `external`: the general keyword for declaring a value that exists on the JS side. -- `dirname`: the binding name you'll use on the ReScript side. -- `string => string`: the type signature of `dirname`. Mandatory for `external`s. -- `= "dirname"`: the name of the variable inside the `path` JS module. There's repetition in writing the first and second `dirname`, because sometime the binding name you want to use on the ReScript side is different than the variable name the JS module exported. - -### Import a JavaScript Module As a Single Value - -By omitting the string argument to `module`, you bind to the whole JS module: - - - -```res example -@module external leftPad: (string, int) => string = "./leftPad" -let paddedResult = leftPad("hi", 5) -``` -```js -import * as LeftPad from "./leftPad"; -var paddedResult = LeftPad("hi", 5); -``` -```js -var LeftPad = require("./leftPad"); -var paddedResult = LeftPad("hi", 5); -``` - - - -Depending on whether you're compiling ReScript to JavaScript module or CommonJS, **this feature will generate subtly different code**. Please check both output tabs to see the difference. The JavaScript module output here would be wrong! - -### Import an `default` Export - -Use the value `default` on the right hand side: - - - -```res example -@module("./student") external studentName: string = "default" -Console.log(studentName) -``` -```js -import Student from "./student"; -var studentName = Student; -``` - - - -### Use Import Attributes - -**Since 11.1** - -[Import attributes](https://github.com/tc39/proposal-import-attributes) can be used in ReScript, as long as ReScript is configured to output JavaScript module. You do that by passing configuration to the `@module` attribute: - - -```rescript -@module({from: "./myJson.json", with: {type_: "json", \"some-exotic-identifier": "someValue"}}) -external myJson: JSON.t = "default" - -Console.log(myJson) -``` - -```javascript -import MyJsonJson from "./myJson.json" with {"type": "json", "some-exotic-identifier": "someValue"}; - -var myJson = MyJsonJson; - -console.log(myJson); -``` - - -This above imports the local `./myJson.json` file, adding import attributes. - -This is how it works: -1. Instead of passing a string or tuple to `@module`, pass a record. -2. This record should have a `from` key. The value of that is where you want the module to be imported from (just like the regular string to `@module` is). -3. It should also have a `with` key, with another record where you put all the import attributes you want emitted. - -Notice `\"some-exotic-identifier"` - you'll need to escape any key that's not a valid ReScript record key. -Also notice `type_`. Since `type` is a reserved keyword in ReScript, you can use `type_` instead. It will be output as `type` in the JavaScript code. - -## Dynamic Import -Leveraging JavaScript's [dynamic `import`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/import) to reduce bundle size and lazy load code as needed is easy in ReScript. It's also a little bit more convenient than in regular JavaScript because you don't need to keep track of file paths manually with ReScript's module system. - -### Dynamically Importing Parts of a Module -Use the `import` function to dynamically import a specific part of a module. Put whatever `let` binding you want to import in there, and you'll get a `promise` back resolving to that specific binding. - -Let's look at an example. Imagine the following file `MathUtils.res`: - -```rescript -let add = (a, b) => a + b -let sub = (a, b) => a - b -``` - -Now let's dynamically import the add function in another module, e.g. `App.res`: - - -```rescript -// App.res -let main = async () => { - let add = await import(MathUtils.add) - let onePlusOne = add(1, 1) - - Console.log(onePlusOne) -} -``` -```javascript -async function main() { - var add = await import("./MathUtils.mjs").then(function(m) { - return m.add; - }); - - var onePlusOne = add(1, 1); - console.log(onePlusOne); -} -``` - - -### Dynamically Importing an Entire Module -The syntax for importing a whole module looks a little different, since we are operating on the module syntax level; instead of using `import`, you may simply `await` the module itself: - -```rescript -// App.res -let main = async () => { - module Utils = await MathUtils - - let twoPlusTwo = Utils.add(2, 2) - Console.log(twoPlusTwo) -} -``` -```javascript -async function main() { - var Utils = await import("./MathUtils.mjs"); - - var twoPlusTwo = Utils.add(2, 2); - console.log(twoPlusTwo); -} -``` - - -## Export To JavaScript - -### Export a Named Value - -As mentioned in ReScript's idiomatic [Import & Export](import-export.md), every let binding and module is exported by default to other ReScript modules (unless you use a `.resi` [interface file](module#signatures)). If you open up the compiled JS file, you'll see that these values can also directly be used by a _JavaScript_ file too. - -### Export a `default` Value - -If your JS project uses JavaScript module, you're likely exporting & importing some default values: - -```js -// student.js -export default name = "Al"; -``` - -```js -// teacher.js -import studentName from 'student.js'; -``` - -A JavaScript default export is really just syntax sugar for a named export implicitly called `default` (now you know!). So to export a default value from ReScript, you can just do: - - - -```res example -// ReScriptStudent.res -let default = "Bob" -``` -```js -var $$default = "Bob"; - -exports.$$default = $$default; -exports.default = $$default; -// informal transpiler-compatible marker of a default export compiled from JavaScript module -exports.__esModule = true; -``` -```js -var $$default = "Bob"; - -export { - $$default, - $$default as default, -} -``` - - - -You can then import this default export as usual on the JS side: - -```js -// teacher2.js -import studentName from 'ReScriptStudent.js'; -``` - -If your JavaScript's default import is transpiled by Babel/Webpack/Jest into CommonJS `require`s, we've taken care of that too! See the CommonJS output tab for `__esModule`. diff --git a/pages/docs/manual/latest/inlining-constants.mdx b/pages/docs/manual/latest/inlining-constants.mdx deleted file mode 100644 index 3f8c40a08..000000000 --- a/pages/docs/manual/latest/inlining-constants.mdx +++ /dev/null @@ -1,78 +0,0 @@ ---- -title: "Inlining Constants" -description: "Inlining constants" -canonical: "/docs/manual/latest/inlining-constants" ---- - -# Inlining Constants - -Sometimes, in the JavaScript output, you might want a certain value to be forcefully inlined. For example: - -```js -if (process.env.mode === 'development') { - console.log("Dev-only code here!") -} -``` - -The reason is that your JavaScript bundler (e.g. Webpack) might turn that into: - -```js -if ('production' === 'development') { - console.log("Dev-only code here!") -} -``` - -Then your subsequent Uglifyjs optimization would remove that entire `if` block. This is how projects like ReactJS provide a development mode code with plenty of dev warnings, while ensuring that the uglified (minified) production code is free of those expensive blocks. - -So, in ReScript, producing that example `if (process.env.mode === 'development')` output is important. This first try doesn't work: - - - -```res example -@val external process: 'a = "process" - -let mode = "development" - -if (process["env"]["mode"] === mode) { - Console.log("Dev-only code here!") -} -``` -```js -var mode = "development"; - -if (process.env.mode === mode) { - console.log("Dev-only code here!"); -} -``` - - - -The JS output shows `if (process.env.mode === mode)`, which isn't what we wanted. To inline `mode`'s value, use `@inline`: - - - -```res example -@val external process: 'a = "process" - -@inline -let mode = "development" - -if (process["env"]["mode"] === mode) { - Console.log("Dev-only code here!") -} -``` -```js -if (process.env.mode === "development") { - console.log("Dev-only code here!"); -} -``` - - - -Now your resulting JS code can pass through Webpack and Uglifyjs like the rest of your JavaScript code, and that whole `console.log` can be removed. - -The inlining currently only works for **string, float and boolean**. - -## Tips & Tricks - -This is **not** an optimization. This is an edge-case feature for folks who absolutely need particular values inlined for a JavaScript post-processing step, like conditional compilation. Beside the difference in code that the conditional compilation might end up outputting, there's no performance difference between inlining and not inlining simple values in the eyes of a JavaScript engine. diff --git a/pages/docs/manual/latest/installation.mdx b/pages/docs/manual/latest/installation.mdx deleted file mode 100644 index 86f9a935d..000000000 --- a/pages/docs/manual/latest/installation.mdx +++ /dev/null @@ -1,153 +0,0 @@ ---- -title: "Installation" -description: "ReScript installation and setup instructions" -canonical: "/docs/manual/latest/installation" ---- - -# Installation - -## Notes - -With the instructions below, our new standard library [ReScript Core](https://github.com/rescript-lang/rescript-core) will be included by default. (In ReScript 11, it comes as a separate npm package `@rescript/core`. In future versions, it will be included in the `rescript` npm package itself.) - -## Prerequisites - -- [Node.js](https://nodejs.org/) version >= 14 -- One of the following package managers: - - [npm](https://docs.npmjs.com/cli/) (comes with Node.js) - - [yarn](https://yarnpkg.com/) (yarn versions >1 need to set `nodeLinker: node-modules` in `.yarnrc.yml`) - - [pnpm](https://pnpm.io/) - - [bun](https://bun.sh/) - -## New Project - -The fastest and easiest way to spin up a new ReScript project is with the [create-rescript-app](https://github.com/rescript-lang/create-rescript-app) project generator. You can start it with any of the aforementioned package managers or `npx`. - - - -```sh example -npm create rescript-app@latest -``` -```sh -npx create-rescript-app -``` -```sh -yarn create rescript-app -``` -```sh -pnpm create rescript-app -``` -```sh -bun create rescript-app -``` - - - -- Follow the steps of the setup. -- Trigger a ReScript build: - ```sh - npm run res:build - ``` -- If you selected the "basic" template, simply run it with: - ```sh - node src/Demo.res.js - ``` - -That compiles your ReScript into JavaScript, then uses Node.js to run said JavaScript. - -**When taking your first steps with ReScript, we recommend you use our unique workflow of keeping a tab open for the generated JS file** (`.res.js`/`.res.mjs`), so that you can learn how ReScript transforms into JavaScript. Not many languages output clean JavaScript code you can inspect and learn from! With our [VS Code extension](https://marketplace.visualstudio.com/items?itemName=chenglou92.rescript-vscode), use the command "ReScript: Open the compiled JS file for this implementation file" to open the generated JS file for the currently active ReScript source file. - -During development, instead of running `npm run res:build` each time to compile, use `npm run res:dev` to start a watcher that recompiles automatically after file changes. - -## Integrate Into an Existing JS Project - -If you already have a JavaScript project into which you'd like to add ReScript you can do that in the following ways: - -### Quick Setup - -In the root directory of your project, execute: - - -```sh -npm create rescript-app@latest -``` -```sh -npx create-rescript-app -``` -```sh -yarn create rescript-app -``` -```sh -pnpm create rescript-app -``` -```sh -bun create rescript-app -``` - - - -`create-rescript-app` will tell you that a `package.json` file has been detected and ask you if it should install ReScript into your project. Just follow the steps accordingly. - -### Manual Setup -- Install ReScript locally: - - - ```sh - npm install rescript @rescript/core - ``` - ```sh - yarn add rescript @rescript/core - ``` - ```sh - pnpm install rescript @rescript/core - ``` - ```sh - bun install rescript @rescript/core - ``` - - -- Create a ReScript build configuration file (called `rescript.json`) at the root: - ```json - { - "name": "your-project-name", - "sources": [ - { - "dir": "src", // update this to wherever you're putting ReScript files - "subdirs": true - } - ], - "package-specs": [ - { - "module": "esmodule", - "in-source": true - } - ], - "suffix": ".res.js", - "bs-dependencies": [ - "@rescript/core" - ], - "bsc-flags": [ - "-open RescriptCore" - ] - } - ``` - See [Build Configuration](build-configuration) for more details on `rescript.json`. -- Add convenience `npm` scripts to `package.json`: - ```json - "scripts": { - "res:build": "rescript", - "res:dev": "rescript -w" - } - ``` - -Since ReScript compiles to clean readable JS files, the rest of your existing toolchain (e.g. Babel and Webpack) should just work! - -Helpful guides: - -- [Converting from JS](converting-from-js). -- [Shared Data Types](shared-data-types). -- [Import from/Export to JS](import-from-export-to-js). - -### Integrate with a ReactJS Project - -To start a [rescript-react](/docs/react/latest/introduction) app, or to integrate ReScript into an existing ReactJS app, follow the instructions [here](/docs/react/latest/installation). diff --git a/pages/docs/manual/latest/interop-cheatsheet.mdx b/pages/docs/manual/latest/interop-cheatsheet.mdx deleted file mode 100644 index 41c4b195e..000000000 --- a/pages/docs/manual/latest/interop-cheatsheet.mdx +++ /dev/null @@ -1,274 +0,0 @@ ---- -title: "Interop Cheatsheet" -description: "Cheatsheet for various interop scenarios in ReScript" -canonical: "/docs/manual/latest/interop-cheatsheet" ---- - -# Interop Cheatsheet - -This is a glossary with examples. All the features are described by later pages. - -## List of Decorators - -> **Note:** In ReScript < 8.3, all our attributes started with the `bs.` prefix. This is no longer needed and our formatter automatically removes them in newer ReScript versions. - - - -### Attributes - -- `@as`: [here](attribute#usage), [here](bind-to-js-function#fixed-arguments), [here](bind-to-js-function#constrain-arguments-better) and [here](generate-converters-accessors#usage-3) -- [`@deriving`](generate-converters-accessors#generate-functions--plain-values-for-variants) -- [`@get`](bind-to-js-object#bind-using-special-bs-getters--setters) -- [`@get_index`](bind-to-js-object#bind-using-special-bs-getters--setters) - -- [`@inline`](inlining-constants) -- [`@int`](bind-to-js-function#constrain-arguments-better) - -- [`@module`](import-from-export-to-js#import-a-javascript-modules-content) -- [`@new`](bind-to-js-object#bind-to-a-js-object-thats-a-class) -- [`@optional`](generate-converters-accessors#optional-labels) -- [`@return`](bind-to-js-function#function-nullable-return-value-wrapping) -- `@send`: [here](bind-to-js-function#object-method) and [here](pipe#js-method-chaining) -- [`@scope`](bind-to-global-js-values#global-modules) -- [`@set`](bind-to-js-object#bind-using-special-bs-getters--setters) -- [`@set_index`](bind-to-js-object#bind-using-special-bs-getters--setters) -- [`@variadic`](bind-to-js-function#variadic-function-arguments) -- [`@string`](bind-to-js-function#constrain-arguments-better) -- [`@this`](bind-to-js-function#modeling-this-based-callbacks) -- [`@uncurry`](bind-to-js-function#extra-solution) -- [`@unwrap`](bind-to-js-function#trick-2-polymorphic-variant--bsunwrap) -- [`@val`](bind-to-global-js-values#global-modules) -- [`@taggedTemplate`](bind-to-js-function#tagged_template-functions) - -- [`@deprecated`](attribute#usage) -- [`genType`](https://github.com/reason-association/genType) -- [`@JSX`](jsx) -- `@react.component`: [here](/docs/react/latest/introduction) and [here](https://github.com/reasonml/reason-react) -- [`@warning`](attribute#usage) -- [`@unboxed`](variant#untagged-variants) - -### Extension Points - -- [`%debugger`](embed-raw-javascript#debugger) -- [`%external`](bind-to-global-js-values#special-global-values) -- [`%raw`](embed-raw-javascript#paste-raw-js-code) -- [`%re`](primitive-types#regular-expression) -- [`%todo`](/syntax-lookup#todo) - -## Raw JS - - - -```res example -let add = %raw("(a, b) => a + b") -%%raw("const a = 1") -``` -```js -var add = ((a, b) => a + b); -const a = 1 -``` - - - -## Global Value - - - -```res example -@val external setTimeout: (unit => unit, int) => float = "setTimeout" -``` -```js -// Empty output -``` - - - -## Global Module's Value - - - -```res example -@val @scope("Math") -external random: unit => float = "random" - -let someNumber = random() - -@val @scope(("window", "location", "ancestorOrigins")) -external length: int = "length" -``` -```js -var someNumber = Math.random(); -``` - - - -## Nullable - - - -```res example -let a = Some(5) // compiles to 5 -let b = None // compiles to undefined -``` -```js -var a = 5; -var b; -``` - - - -Handling a value that can be `undefined` and `null`, by ditching the `option` type and using `Nullable.t`: - - - -```res example -let jsNull = Nullable.null -let jsUndefined = Nullable.undefined -let result1: Nullable.t = Nullable.make("hello") -let result2: Nullable.t = Nullable.fromOption(Some(10)) -let result3: option = Nullable.toOption(Nullable.make(10)) -``` -```js -import * as Caml_option from "./stdlib/caml_option.js"; -import * as Core__Nullable from "./stdlib/core__Nullable.js"; - -var result2 = Core__Nullable.fromOption(10); - -var jsNull = null; - -var jsUndefined; - -var result1 = "hello"; - -var result3 = Caml_option.nullable_to_opt(10); -``` - - - -## JS Object - -- [Bind to a JS object as a ReScript record](bind-to-js-object#bind-to-record-like-js-objects). -- [Bind to a JS object that acts like a hash map](bind-to-js-object#bind-to-hash-map-like-js-object). -- [Bind to a JS object that's a class](bind-to-js-object#bind-to-a-js-object-thats-a-class). - -## Function - -### Object Method & Chaining - - - -```res example -@send external map: (array<'a>, 'a => 'b) => array<'b> = "map" -@send external filter: (array<'a>, 'a => 'b) => array<'b> = "filter" -[1, 2, 3] - ->map(a => a + 1) - ->filter(a => mod(a, 2) == 0) - ->Console.log -``` -```js -console.log( - [1, 2, 3] - .map(function (a) { - return (a + 1) | 0; - }) - .filter(function (a) { - return a % 2 === 0; - }) -); -``` - - - -### Variadic Arguments - - - -```res example -@module("path") @variadic -external join: array => string = "join" -``` -```js -// Empty output -``` - - - -### Tagged template functions - - - -```res example -// see https://bun.sh/docs/runtime/shell -type result = {exitCode: int} -@module("bun") @taggedTemplate -external sh: (array, array) => promise = "$" - -let filename = "index.res" -let result = await sh`ls ${filename}` -``` - -```js -import * as $$Bun from "bun"; -var filename = "index.res"; -var result = await $$Bun.$`ls ${filename}`; -``` - - - -### Polymorphic Function - - - -```res example -@module("Drawing") external drawCat: unit => unit = "draw" -@module("Drawing") external drawDog: (~giveName: string) => unit = "draw" -``` -```js -// Empty output -``` - - - - - -```res example -@val -external padLeft: ( - string, - @unwrap [ - | #Str(string) - | #Int(int) - ]) - => string = "padLeft" - -padLeft("Hello World", #Int(4)) -padLeft("Hello World", #Str("Message from ReScript: ")) -``` -```js -padLeft("Hello World", 4); -padLeft("Hello World", "Message from ReScript: "); -``` - - - -## JS Module Interop - -[See here](import-from-export-to-js.md) - -## Dangerous Type Cast - -Final escape hatch converter. Do not abuse. - - - -```res example -external convertToFloat: int => float = "%identity" -let age = 10 -let gpa = 2.1 +. convertToFloat(age) -``` -```js -var age = 10; -var gpa = 2.1 + 10; -``` - - diff --git a/pages/docs/manual/latest/interop-with-js-build-systems.mdx b/pages/docs/manual/latest/interop-with-js-build-systems.mdx deleted file mode 100644 index e043504b9..000000000 --- a/pages/docs/manual/latest/interop-with-js-build-systems.mdx +++ /dev/null @@ -1,57 +0,0 @@ ---- -title: "Interop with JS Build Systems" -description: "Documentation on how to interact with existing JS build systems" -canonical: "/docs/manual/latest/interop-with-js-build-systems" ---- - -# Interop with JS Build Systems - -If you come from JS, chances are that you already have a build system in your existing project. Here's an overview of the role `rescript` would play in your build pipeline, if you want to introduce some ReScript code. - -> **Please** try not to wrap `rescript` into your own incremental build framework. ReScript's compilation is very hard to get right, and you'll inevitably run into stale or badly performing builds (therefore erasing much of our value proposition) if you create your own meta layer on top. - -## Popular JS Build Systems - -The JS ecosystem uses a few build systems: [browserify](http://browserify.org/), [rollup](https://github.com/rollup/rollup), [webpack](https://webpack.js.org/), etc. The latter's probably the most popular of the three (as of 2019 =P). These build systems do both the compilation and the linking (aka, bundling many files into one or few files). - -`rescript` only takes care of the compilation step; it maps one `.res`/`.resi` file into one JS output file. As such, in theory, no build system integration is needed from our side. From e.g. the webpack watcher's perspective, the JS files ReScript generates are almost equivalent to your hand-written JS files. We also recommend **that you initially check in those ReScript-generated JS files**, as this workflow means: - -- You can introduce ReScript silently into your codebase without disturbing existing infra. -- You have a **visual** diff of the performance & correctness of your JS file when you update the `.res` files and the JS artifacts change. -- You can let teammates hot-patch the JS files in emergency situations, without needing to first start learning ReScript. -- You can remove ReScript completely from your codebase and things will still work (in case your company decides to stop using us for whatever reason). - -For what it's worth, you can also turn `rescript` into an automated step in your build pipeline, e.g. into a Webpack loader; but such approach is error-prone and therefore discouraged. - -### Tips & Tricks - -You can make ReScript JS files look even more idiomatic through the in-source + bs suffix config in `rescript.json`: - -```json -{ - "package-specs": { - "module": "commonjs", // or whatever module system your project uses - "in-source": true - }, - "suffix": ".res.js" -} -``` - -This will: - -- Generate the JS files alongside your ReScript source files. -- Use the file extension `.res.js`, so that you can require these files on the JS side through `require('./MyFile.res.js')`, without needing a loader. - -## Use Loaders on ReScript Side - -"What if my build system uses a CSS/png/whatever loader and I'd like to use it in ReScript?" - -Loaders are indeed troublesome; in the meantime, please use e.g. `%raw("require('./myStyles.css')")` at the top of your file. This just uses [`raw`](embed-raw-javascript.md) to compile the snippet into an actual JS require. - -## Getting Project's Dependencies - -`rescript` generates one `MyFile.d` file per `MyFile` source file; you'll find them in `lib/bs`. These are human readable, machine-friendly list of the dependencies of said `MyFile`. You can read into them for your purpose (though mind the IO overhead). Use these files instead of creating your own dependency graph; we did the hard work of tracking the dependencies as best as possible (including inner modules, `open`s, module names overlap, etc). - -## Run Script Per File Built - -See [js-post-build](build-configuration#js-post-build). Though please use it sparingly; if you hook up a node.js script after each file built, you'll incur the node startup time per file! diff --git a/pages/docs/manual/latest/introduction.mdx b/pages/docs/manual/latest/introduction.mdx deleted file mode 100644 index 9fa73cdca..000000000 --- a/pages/docs/manual/latest/introduction.mdx +++ /dev/null @@ -1,80 +0,0 @@ ---- -title: "Introduction" -description: "The hows and whys of ReScript" -canonical: "/docs/manual/latest/introduction" ---- - -# ReScript - -Ever wanted a language like JavaScript, but without the warts, with a great type system, and with a lean build toolchain that doesn't waste your time? - -ReScript looks like JS, acts like JS, and compiles to the highest quality of clean, readable and performant JS, directly runnable in browsers and Node. - -**This means you can pick up ReScript and access the vast JavaScript ecosystem and tooling as if you've known ReScript for a long time!** - -**ReScript is the language for folks who don't necessarily love JavaScript, but who still acknowledge its importance**. - -## Difference vs TypeScript - -We respect TypeScript very much and think that it's a positive force in the JavaScript ecosystem. ReScript shares some of the same goals as TypeScript, but is different enough regarding some important nuances: - -- TypeScript's (admittedly noble) goal is to cover the entire JavaScript feature set and more. **ReScript covers only a curated subset of JavaScript**. For example, we emphasize plain data + functions over classes, clean [pattern matching](pattern-matching-destructuring.md) over fragile `if`s and virtual dispatches, [proper data modeling](variant.md) over string abuse, etc. JavaScript supersets will only grow larger over time; ReScript doesn't. \* - -- Consequently, TypeScript's type system is necessarily complex, pitfalls-ridden, potentially requires tweaking, sometimes slow, and requires quite a bit of noisy annotations that often feel like manual bookkeeping rather than clear documentation. In contrast, ReScript's type system: - - - Is deliberately curated to be a simple subset most folks will have an easier time to use. - - Has **no** pitfalls, aka the type system is "sound" (the types will always be correct). E.g. If a type isn't marked as nullable, its value will never lie and let through some `undefined` value silently. **ReScript code has no null/undefined errors**. - - Is the same for everyone. No knobs, no bikeshedding opportunity. - - Runs extremely fast precisely thanks to its simplicity and curation. It's one of the fastest compiler & build system toolchains for JavaScript development. - - **Doesn't need type annotations**. Annotate as much or as little as you'd like. The types are inferred by the language (and, again, are guaranteed correct). - -- Migrating to TypeScript is done "breadth-first," whereas migrating to ReScript is done "depth-first." You can convert your codebase to TypeScript by "turning it on" for all files and annotate here and there; but how much type safety did you gain? How do you measure it? Type errors can still slip in and out of the converted pieces. For ReScript, our interop features draw clear boundaries: there's pure ReScript code, and there's JS interop code. Every piece of converted ReScript code is 100% clean. You'd convert file by file and each conversion increases your safety monotonically. - -\* When you absolutely need to write or interoperate with free-for-all JavaScript, we expose enough escape hatches for you. - -## Other Highlights - -Aside from the aforementioned simple, robust and fast type system, ReScript presents a few more advantages. - -### Faster than JavaScript - -JavaScript's been aggressively optimized by talented engineers over a long span. Unfortunately, even for seasoned JS devs, it can be hard to know how to properly leverage JS's performance. ReScript's type system and compiler naturally guides you toward writing code that's very often performant by default, with good leverage of various Just-In-Time optimizations (hidden classes, inline caching, avoiding deopts, etc). - -A widespread adage to write fast JavaScript code is to write as if there's a type system (in order to trigger JS engines' good optimization heuristics); ReScript gives you a real one and generates code that's friendly to optimizations by default. - -### High Quality Dead Code Elimination - -The JavaScript ecosystem is very reliant on dependencies. Shipping the final product inevitably drags in a huge amount of code, lots of which the project doesn't actually use. These regions of dead code impact loading, parsing and interpretation speed. ReScript provides powerful dead code elimination at all levels: - -- Function- and module-level code elimination is facilitated by the well-engineered type system and purity analysis. -- At the global level, ReScript generates code that is naturally friendly to dead code elimination done by bundling tools such as [Rollup](https://github.com/rollup/rollup) and [Closure Compiler](https://developers.google.com/closure/compiler/), after its own sophisticated elimination pass. -- The same applies for ReScript's own tiny runtime (which is written in ReScript itself). - -### Tiny JS Output - -A `Hello world` ReScript program generates **20 bytes** of JS code. Additionally, the standard library pieces you require in are only included when needed. - -### Fast Iteration Loop - -ReScript's build time is **one or two orders of magnitude** faster than alternatives. In its watcher mode, the build system usually finishes before you switch screen from the editor to the terminal tab (two digits of milliseconds). A fast iteration cycle reduces the need of keeping one's mental state around longer; this in turn allows one to stay in the flow longer and more often. - -### Readable Output & Great Interop - -Unreadable JavaScript code generated from other compiled-to-js languages makes it so that it could be, practically speaking: - -- Hard to debug (cryptic stack trace, mangled variable names) -- Hard to learn from (non-straightforward mapping of concepts from one language to another) -- Hard to profile for performance (unclear what runtime performance cost there is) -- Hard to integrate with existing hand-written JS code - -ReScript's JS output is very readable. This is especially important while learning, where users might want to understand how the code's compiled, and to audit for bugs. - -This characteristic, combined with a fully-featured JS interop system, allows ReScript code to be inserted into an existing JavaScript codebase almost unnoticed. - -### Preservation of Code Structure - -ReScript maps one source file to one JavaScript output file. This eases the integration of existing tools such as bundlers and test runners. You can even start writing a single file without much change to your build setup. Each file's code structure is approximately preserved, too. - -## Conclusion - -We hope the above gave you enough of an idea of ReScript and its differentiators. Feel free to [try it online](/try) to get a feel! diff --git a/pages/docs/manual/latest/json.mdx b/pages/docs/manual/latest/json.mdx deleted file mode 100644 index cc6d5cec0..000000000 --- a/pages/docs/manual/latest/json.mdx +++ /dev/null @@ -1,82 +0,0 @@ ---- -title: "JSON" -description: "Interacting with JSON in ReScript" -canonical: "/docs/manual/latest/json" ---- - -# JSON - -## Parse - -Bind to JavaScript's `JSON.parse` and type the return value as the type you're expecting: - - - -```res example -// declare the shape of the json you're binding to -type data = {names: array} - -// bind to JS' JSON.parse -@scope("JSON") @val -external parseIntoMyData: string => data = "parse" - -let result = parseIntoMyData(`{"names": ["Luke", "Christine"]}`) -let name1 = result.names[0] -``` -```js -var result = JSON.parse("{\"names\": [\"Luke\", \"Christine\"]}"); -var name1 = result.names[0]; -``` - - - -Where `data` can be any type you assume the JSON is. As you can see, this compiles to a straightforward `JSON.parse` call. As with regular JS, this is convenient, but has no guarantee that e.g. the data is correctly shaped, or even syntactically valid. Slightly dangerous. - -## Stringify - -Use [`JSON.stringify`](api/core/json#value-stringify) if your data is of type `JSON.t` or [`JSON.stringifyAny`](api/core/json#value-stringifyAny) if it is not. - - - -```res example -Console.log(JSON.stringifyAny(["Amy", "Joe"])) -``` -```js -console.log(JSON.stringify([ - "Amy", - "Joe" -])); -``` - - - -## Import a JSON file - -Use the `@module` attribute to import JSON files directly. - - - -```res example -@module external studentNames: JSON.t = "./students.json" -Console.log(studentNames) -``` -```js -import * as StudentsJson from "./students.json"; - -var studentNames = StudentsJson; - -console.log(studentNames); -``` -```js -var StudentsJson = require("./students.json"); - -var studentNames = StudentsJson; - -console.log(studentNames); -``` - - - -## Advanced - -Thanks to untagged variants, JSON can be encoded and decoded idiomatically. Check it out on [the variants page](variant#decoding-and-encoding-json-idiomatically). diff --git a/pages/docs/manual/latest/jsx.mdx b/pages/docs/manual/latest/jsx.mdx deleted file mode 100644 index 6973a9731..000000000 --- a/pages/docs/manual/latest/jsx.mdx +++ /dev/null @@ -1,383 +0,0 @@ ---- -title: "JSX" -description: "JSX syntax in ReScript and React" -canonical: "/docs/manual/latest/jsx" ---- - -# JSX - -Would you like some HTML syntax in your ReScript? If not, quickly skip over this section and pretend you didn't see anything! - -ReScript supports the JSX syntax, with some slight differences compared to the one in [ReactJS](https://facebook.github.io/react/docs/introducing-jsx.html). ReScript JSX isn't tied to ReactJS; they translate to normal function calls: - -**Note** for [ReScriptReact](https://rescript-lang.org/docs/react/latest/introduction) readers: this isn't what ReScriptReact turns JSX into, in the end. See Usage section for more info. - -## Capitalized - - - -```res - -``` -```js -React.createElement(MyComponent, { - name: "ReScript", -}); -``` - - - -becomes - - - -```res -MyComponent.createElement(~name="ReScript", ~children=list{}, ()) -``` -```js -React.createElement(MyComponent, { - name: "ReScript", -}); -``` - - - -## Uncapitalized - - - -```res -
child1 child2
-``` -```js -React.createElement("div", { - onClick: handler -}, child1, child2); -``` - -
- -becomes - - - -```res -div(~onClick=handler, ~children=list{child1, child2}, ()) -``` -```js -React.createElement("div", { - onClick: handler -}, child1, child2); -``` - - - -## Fragment - - - -```res -<> child1 child2 -``` -```js -React.createElement(React.Fragment, undefined, child1, child2); -``` - - - -becomes - - - -```res -list{child1, child2} -``` -```js -React.createElement(React.Fragment, undefined, child1, child2); -``` - - - -### Children - - - -```res - child1 child2 -``` -```js -React.createElement(MyComponent, { children: null }, child1, child2); -``` - - - -This is the syntax for passing a list of two items, `child1` and `child2`, to the children position. It transforms to a list containing `child1` and `child2`: - - - -```res -MyComponent.createElement(~children=list{child1, child2}, ()) -``` -```js -React.createElement(MyComponent.make, MyComponent.makeProps(null, undefined), child1, child2); -``` - - - -**Note** again that this isn't the transform for ReScriptReact; ReScriptReact turns the final list into an array. But the idea still applies. - -So naturally, ` myChild ` is transformed to `MyComponent.createElement(~children=list{myChild}, ())`. I.e. whatever you do, the arguments passed to the children position will be wrapped in a list. - -## Usage - -See [ReScriptReact Elements & JSX](https://rescript-lang.org/docs/react/latest/elements-and-jsx) for an example application of JSX, which transforms the above calls into a ReScriptReact-specific call. - -Here's a JSX tag that shows most of the features. - - - -```res - -
{React.string("hello")}
-
-``` -```js -React.createElement(MyComponent, { - children: React.createElement("div", undefined, "hello"), - booleanAttribute: true, - stringAttribute: "string", - intAttribute: 1, - forcedOptional: "hello", - onClick: handleClick -}); -``` - -
- -## Departures From JS JSX - -- Attributes and children don't mandate `{}`, but we show them anyway for ease of learning. Once you format your file, some of them go away and some turn into parentheses. -- Props spread is supported, but there are some restrictions (see below). -- Punning! -- Props and tag names have to follow ReScript's restrictions on identifiers at the exception of hyphens for lowercase tags ([see below](#hyphens-in-tag-names)). - -### Spread Props - -**Since 10.1** - -JSX props spread is supported now, but in a stricter way than in JS. - - - -```res - -``` -```js -React.createElement(Comp, { - a: "a", - b: "b" -}); -``` - - - -Multiple spreads are not allowed: - - - -```res - -``` - - - -The spread must be at the first position, followed by other props: - - - -```res - -``` - - - -### Punning - -"Punning" refers to the syntax shorthand for when a label and a value are the same. For example, in JavaScript, instead of doing `return {name: name}`, you can do `return {name}`. - -JSX supports punning. `` is just a shorthand for ``. The formatter will help you format to the punned syntax whenever possible. This is convenient in the cases where there are lots of props to pass down: - - - -```res - -``` -```js -React.createElement(MyComponent, { - isLoading: true, - text: text, - onClick: onClick -}); -``` - - - -Consequently, a JSX component can cram in a few more props before reaching for extra libraries solutions that avoids props passing. - -**Note** that this is a departure from ReactJS JSX, which does **not** have punning. ReactJS' `` desugars to ``, in order to conform to DOM's idioms and for backward compatibility. - -### Hyphens in tag names - -**Since 11.1** - -JSX now supports lowercase tags with hyphens in their name. This allows to bind -to web components. - -Note though that props names can't have hyphens, you should use `@as` to bind to -such props in your custom `JsxDOM.domProps` type ([see generic JSX transform](#generic-jsx-transform-jsx-beyond-react-experimental)). - - - -```res - -``` -```js -React.createElement("model-viewer", { - "touch-actions": "pan-y", - src: src -}); -``` - - - -## Generic JSX transform: JSX beyond React (experimental) - -**Since 11.1** - -While ReScript comes with first class support for JSX in React, it's also possible to have ReScript delegate JSX to other frameworks. You do that by configuring a _generic JSX transform_. - -This is what you need to do to use a generic JSX transform: -1. Make sure you have a ReScript module that [implements the functions and types necessary for the JSX transform](#implementing-a-generic-jsx-transform-module). -2. Configure `rescript.json` to delegated JSX to that module. - -That's it really. We'll expand on each point below. - -### Configuration -You configure a generic JSX transform by putting any module name in the `module` config of JSX in `rescript.json`. This can be _any valid module name_. Example part from `rescript.json`: - -```json -"jsx": { - "module": "Preact" - }, -``` - -This will now put the `Preact` module in control of the generated JSX calls. The `Preact` module can be defined by anyone - locally in your project, or by a package. As long a it's available in the global scope. The JSX transform will delegate any JSX related code to `Preact`. - -#### What about `@react.component` for components? - -`@react.component` will still be available, and so is a generic `@jsx.component` notation. Both work the same way. - -### Usage Example -Here's a quick usage example (the actual definition of `Preact.res` comes below): - -First, configure `rescript.json`: -```json -"jsx": { - "module": "Preact" - }, -``` - -Now you can build Preact components: -```rescript -// Name.res -@jsx.component // or @react.component if you want -let make = (~name) => Preact.string(`Hello ${name}!`) -``` - -And you can use them just like normal with JSX: -```rescript -let name = -``` - -#### File level configuration -You can configure what JSX transform is used at the file level via `@@jsxConfig`, just like before. Like: -```rescript -@@jsxConfig({module_: "Preact"}) -``` - -This can be convenient if you're mixing different JSX frameworks in the same project. - -### Implementing a generic JSX transform module -Below is a full list of everything you need in a generic JSX transform module, including code comments to clarify. It's an example implementation of a `Preact` transform, so when doing this for other frameworks you'd of course adapt what you import from, and so on. - -> You can easily copy-paste-and-adapt this to your needs if you're creating bindings to a JSX framework. Most often, all you'll need to change is what the `@module("") external` points to, so the runtime calls point to the correct JS module. - -```rescript -// Preact.res -/* Below is a number of aliases to the common `Jsx` module */ -type element = Jsx.element - -type component<'props> = Jsx.component<'props> - -type componentLike<'props, 'return> = Jsx.componentLike<'props, 'return> - -@module("preact") -external jsx: (component<'props>, 'props) => element = "jsx" - -@module("preact") -external jsxKeyed: (component<'props>, 'props, ~key: string=?, @ignore unit) => element = "jsx" - -@module("preact") -external jsxs: (component<'props>, 'props) => element = "jsxs" - -@module("preact") -external jsxsKeyed: (component<'props>, 'props, ~key: string=?, @ignore unit) => element = "jsxs" - -/* These identity functions and static values below are optional, but lets -you move things easily to the `element` type. The only required thing to -define though is `array`, which the JSX transform will output. */ -external array: array => element = "%identity" -@val external null: element = "null" - -external float: float => element = "%identity" -external int: int => element = "%identity" -external string: string => element = "%identity" - -/* These are needed for Fragment (<> ) support */ -type fragmentProps = {children?: element} - -@module("preact") external jsxFragment: component = "Fragment" - -/* The Elements module is the equivalent to the ReactDOM module in React. This holds things relevant to _lowercase_ JSX elements. */ -module Elements = { - /* Here you can control what props lowercase JSX elements should have. - A base that the React JSX transform uses is provided via JsxDOM.domProps, - but you can make this anything. The editor tooling will support - autocompletion etc for your specific type. */ - type props = JsxDOM.domProps - - @module("preact") - external jsx: (string, props) => Jsx.element = "jsx" - - @module("preact") - external div: (string, props) => Jsx.element = "jsx" - - @module("preact") - external jsxKeyed: (string, props, ~key: string=?, @ignore unit) => Jsx.element = "jsx" - - @module("preact") - external jsxs: (string, props) => Jsx.element = "jsxs" - - @module("preact") - external jsxsKeyed: (string, props, ~key: string=?, @ignore unit) => Jsx.element = "jsxs" - - external someElement: element => option = "%identity" -} -``` - -As you can see, most of the things you'll want to implement will be copy paste from the above. But do note that **everything needs to be there unless explicitly noted** or the transform will fail at compile time. \ No newline at end of file diff --git a/pages/docs/manual/latest/lazy-values.mdx b/pages/docs/manual/latest/lazy-values.mdx deleted file mode 100644 index a8743f8a2..000000000 --- a/pages/docs/manual/latest/lazy-values.mdx +++ /dev/null @@ -1,126 +0,0 @@ ---- -title: "Lazy Value" -description: "Data type for deferred computation in ReScript" -canonical: "/docs/manual/latest/lazy-values" ---- - -# Lazy Value - -If you have some expensive computations you'd like to **defer and cache** subsequently, you can wrap it with `lazy`: - - - -```res prelude -@module("node:fs") -external readdirSync: string => array = "readdirSync" - -// Read the directory, only once -let expensiveFilesRead = lazy({ - Console.log("Reading dir") - readdirSync("./pages") -}) -``` -```js -var Fs = require("fs"); - -var expensiveFilesRead = { - LAZY_DONE: false, - VAL: (function () { - console.log("Reading dir"); - return Fs.readdirSync("./pages"); - }) -}; -``` - - - -Check the JS Output tab: that `expensiveFilesRead`'s code isn't executed yet, even though you declared it! You can carry it around without fearing that it'll run the directory read. - -**Note**: a lazy value is **not** a [shared data type](shared-data-types.md). Don't rely on its runtime representation in your JavaScript code. - -## Execute The Lazy Computation - -To actually run the lazy value's computation, use `Lazy.force` from the globally available `Lazy` module: - - - -```res example -// First call. The computation happens -Console.log(Lazy.force(expensiveFilesRead)) // logs "Reading dir" and the directory content - -// Second call. Will just return the already calculated result -Console.log(Lazy.force(expensiveFilesRead)) // logs the directory content -``` -```js -console.log(CamlinternalLazy.force(expensiveFilesRead)); - -console.log(CamlinternalLazy.force(expensiveFilesRead)); -``` - - - -The first time `Lazy.force` is called, the expensive computation happens and the result is **cached**. The second time, the cached value is directly used. - -**You can't re-trigger the computation after the first `force` call**. Make sure you only use a lazy value with computations whose results don't change (e.g. an expensive server request whose response is always the same). - -Instead of using `Lazy.force`, you can also use [pattern matching](pattern-matching-destructuring.md) to trigger the computation: - - - -```res example -switch expensiveFilesRead { -| lazy(result) => Console.log(result) -} -``` -```js -var result = CamlinternalLazy.force(expensiveFilesRead); -``` - - - -Since pattern matching also works on a `let` binding, you can also do: - - - -```res example -let lazy(result) = expensiveFilesRead -Console.log(result) -``` -```js -var result = CamlinternalLazy.force(expensiveFilesRead); -console.log(result); -``` - - - -## Exception Handling - -For completeness' sake, our files read example might raise an exception because of `readdirSync`. Here's how you'd handle it: - - - -```res example -let result = try { - Lazy.force(expensiveFilesRead) -} catch { -| Not_found => [] // empty array of files -} -``` -```js -var result; - -try { - result = CamlinternalLazy.force(expensiveFilesRead); -} catch (raw_exn) { - var exn = Caml_js_exceptions.internalToOCamlException(raw_exn); - if (exn.RE_EXN_ID === "Not_found") { - result = []; - } else { - throw exn; - } -} -``` - - - -Though you should probably handle the exception inside the lazy computation itself. diff --git a/pages/docs/manual/latest/let-binding.mdx b/pages/docs/manual/latest/let-binding.mdx deleted file mode 100644 index 9372c9fae..000000000 --- a/pages/docs/manual/latest/let-binding.mdx +++ /dev/null @@ -1,185 +0,0 @@ ---- -title: "Let Binding" -description: "Let binding syntax for binding to values in ReScript" -canonical: "/docs/manual/latest/let-binding" ---- - -# Let Binding - -A "let binding", in other languages, might be called a "variable declaration". `let` _binds_ values to names. They can be seen and referenced by code that comes _after_ them. - - - -```res example -let greeting = "hello!" -let score = 10 -let newScore = 10 + score -``` -```js -var greeting = "hello!"; -var score = 10; -var newScore = 20; -``` - - - -## Block Scope - -Bindings can be scoped through `{}`. - - - -```res example -let message = { - let part1 = "hello" - let part2 = "world" - part1 ++ " " ++ part2 -} -// `part1` and `part2` not accessible here! -``` -```js -var message = "hello world"; -``` - - - -The value of the last line of a scope is implicitly returned. - -### Design Decisions - -ReScript's `if`, `while` and functions all use the same block scoping mechanism. The code below works **not** because of some special "if scope"; but simply because it's the same scope syntax and feature you just saw: - - - -```res -if displayGreeting { - let message = "Enjoying the docs so far?" - Console.log(message) -} -// `message` not accessible here! -``` -```js -if (displayGreeting) { - console.log("Enjoying the docs so far?"); -} -``` - - - -## Bindings Are Immutable - -Let bindings are "immutable", aka "cannot change". This helps our type system deduce and optimize much more than other languages (and in turn, help you more). - -## Binding Shadowing - -The above restriction might sound unpractical at first. How would you change a value then? Usually, 2 ways: - -The first is to realize that many times, what you want isn't to mutate a variable's value. For example, this JavaScript pattern: - -```js -var result = 0; -result = calculate(result); -result = calculateSomeMore(result); -``` - -...is really just to comment on intermediate steps. You didn't need to mutate `result` at all! You could have just written this JS: - -```js -var result1 = 0; -var result2 = calculate(result1); -var result3 = calculateSomeMore(result2); -``` - -In ReScript, this obviously works too: - - - -```res -let result1 = 0 -let result2 = calculate(result1) -let result3 = calculateSomeMore(result2) -``` -```js -var result1 = 0; -var result2 = calculate(0); -var result3 = calculateSomeMore(result2); -``` - - - -Additionally, reusing the same let binding name overshadows the previous bindings with the same name. So you can write this too: - - - -```res -let result = 0 -let result = calculate(result) -let result = calculateSomeMore(result) -``` -```js -var result = calculate(0); -var result$1 = calculateSomeMore(result); -``` - - - -(Though for the sake of clarity, we don't recommend this). - -As a matter of fact, even this is valid code: - - - -```res example -let result = "hello" -Console.log(result) // prints "hello" -let result = 1 -Console.log(result) // prints 1 -``` -```js -var result = 1; -console.log("hello"); -console.log(1); -``` - - - -The binding you refer to is whatever's the closest upward. No mutation here! -If you need _real_ mutation, e.g. passing a value around, have it modified by many pieces of code, we provide a slightly heavier [mutation feature](mutation.md). - -## Private let bindings - -Private let bindings are introduced in the release [7.2](https://rescript-lang.org/blog/bucklescript-release-7-2). - -In the module system, everything is public by default, -the only way to hide some values is by providing a separate signature to -list public fields and their types: - -```res -module A: { - let b: int -} = { - let a = 3 - let b = 4 -} -``` -`%%private` gives you an option to mark private fields directly - -```res -module A = { - %%private(let a = 3) - let b = 4 -} -``` - -`%%private` also applies to file level modules, so in some cases, -users do not need to provide a separate interface file just to hide some particular values. - -Note interface files are still recommended as a general best practice since they give you better -separate compilation units and also they're better for documentation. - -Still, `%%private` is useful in the following scenarios: - -- **Code generators.** Some code generators want to hide some values but it is sometimes very hard or time consuming for code generators to synthesize the types for public fields. - -- **Quick prototyping.** During prototyping, we still want to hide some values, but the interface file is not stable yet. `%%private` provides you such convenience. - diff --git a/pages/docs/manual/latest/libraries.mdx b/pages/docs/manual/latest/libraries.mdx deleted file mode 100644 index 302f80fb1..000000000 --- a/pages/docs/manual/latest/libraries.mdx +++ /dev/null @@ -1,27 +0,0 @@ ---- -title: "Libraries & Publishing" -description: "Install & publish ReScript packages" -canonical: "/docs/manual/latest/libraries" ---- - -# Libraries & Publishing - -ReScript libraries are just like JavaScript libraries: published & hosted on [NPM](http://npmjs.com). You can reuse your `npm`, `yarn` and `package.json`-related tools to manage them! - -## Tips & Tricks - -### Publish - -We recommend you to check in your compiled JavaScript output, for its [various benefits](interop-with-js-build-systems.md#popular-js-build-systems). If not, then at least consider publishing the JavaScript output by un-ignoring them in your [npmignore](https://docs.npmjs.com/cli/v7/using-npm/developers#keeping-files-out-of-your-package). This way, your published ReScript package comes with plain JavaScript files that JS users can consume. If your project's good, JS users might not even realize that they've installed a library written in ReScript! - -In case your library is only consumed by JS users, you may want to check out our [external stdlib](./build-external-stdlib) configuration as well. - -### Find Libraries - -Search `rescript`-related packages on NPM, or use our [Package Index](/packages). - -If you can't find what you're looking for, remember that **you don't need a wrapper** to use a JS library: - -- Most JS data types, such as array and objects, [map over cleanly to ReScript and vice-versa](shared-data-types.md). -- You also have access to the familiar [Core API](api/core). -- You can use a JavaScript library without needing to install dedicated binding libraries. Check the [`external`](external) page. diff --git a/pages/docs/manual/latest/migrate-to-v11.mdx b/pages/docs/manual/latest/migrate-to-v11.mdx deleted file mode 100644 index 81faf323b..000000000 --- a/pages/docs/manual/latest/migrate-to-v11.mdx +++ /dev/null @@ -1,131 +0,0 @@ ---- -title: "Migrate to v11" -description: "Instructions on upgrading to ReScript 11" -canonical: "/docs/manual/latest/migrate-to-v11" ---- - -# Migrate to ReScript 11 - -## Foreword - -The ReScript community is proud to introduce ReScript V11 which comes with a ton of new features but also removes a lot of bulk. -A migration to it can be very straightforward, but it can also take some time, depending on your code style or what dependencies you use. - -Please have a look at the full [set of breaking changes](#list-of-all-breaking-changes) below to be able to decide whether this is a task you want to undertake. There is also the possibilty to [opt-out of uncurried mode](#minimal-migration) for now, which is probably the most fundamental change of this release. That and other new and notable features are discussed in the following blogposts: - -- [Better interop with customizable variants](/blog/improving-interop) -- [Enhanced Ergonomics for Record Types](/blog/enhanced-ergonomics-for-record-types) -- [First-class Dynamic Import Support](/blog/first-class-dynamic-import-support) -- [Uncurried Mode](/blog/uncurried-mode) - -## Recommended Migration - -### Uncurried Mode - -For uncurried mode to take effect in ReScript 11 there is nothing to configure, it is activated by default. - -### Adapt suffix - -ReScript 11 now allows having arbitrary suffixes in the generated JavaScript files. However, it is still recommended to stick to using `.res.js`, `.res.mjs` or `.res.cjs`. For more information, read the Build System Configuration about [suffixes](/docs/manual/latest/build-configuration#suffix). - -### rescript.json - -The old configuration filename `bsconfig.json` is deprecated. Rename `bsconfig.json` to `rescript.json` to get rid of the deprecation warning. - -### ReScript Core standard library - -[ReScript Core](https://github.com/rescript-lang/rescript-core) is ReScript's new standard library. It replaces the complete `Js` module as well as some of the more frequently used modules from `Belt` and is recommended to use with uncurried mode. - -It will be integrated into the compiler in a future version. In ReScript 11, it still needs to be installed manually: - -```console -$ npm install @rescript/core -``` - -Then add `@rescript/core` to your `rescript.json`'s dependencies: - -```diff - { - "bs-dependencies": [ -+ "@rescript/core" - ] - } -``` - -Open it so it's available in the global scope. - -```diff - { - "bsc-flags": [ -+ "-open RescriptCore", - ] - } -``` - -One major change to be aware of is that array access now returns an `option`. - -```res -let firstItem = myArray[0] // Some("hello") -``` - -If you would like to not use an `option`, you can use [`Array.getUnsafe`](api/core/array#value-getUnsafe). - -For a detailed explanation on migration to ReScript Core, please refer to its [migration guide](https://github.com/rescript-lang/rescript-core#migration). A semi-automated script is available as well. - -See ReScript Core API docs [here](api/core). - -### Removed bindings - -Many Node bindings have been removed from the compiler. Please use [rescript-nodejs](https://github.com/TheSpyder/rescript-nodejs) instead or write your own local bindings. - -## Minimal Migration - -This guide describes the things to do at least to migrate to ReScript 11. - -### Disable uncurried mode - -If you use currying extensively and don't want to bother with adapting your code, or have dependencies that just don't work with uncurried mode yet, just set it to false in your `rescript.json`. - -```json -{ - "uncurried": false -} -``` - -For more information, read the Build System Configuration about [uncurried](/docs/manual/latest/build-configuration#uncurried). - -## List of all breaking changes - -Below is an excerpt from the compiler changelog about all the breaking changes of ReScript 11. - -### Language and Compiler - -- Add smart printer for pipe chains. https://github.com/rescript-lang/rescript/pull/6411 (the formatter will reformat existing code in certain cases) -- Parse `assert` as a regular function. `assert` is no longer a unary expression. Example: before `assert 1 == 2` is parsed as `(assert 1) == 2`, now it is parsed as `assert(1 == 2)`. https://github.com/rescript-lang/rescript/pull/6180 -- Remove support for the legacy Reason syntax. Existing Reason code can be converted to ReScript syntax using ReScript 9 as follows: - - `npx rescript@9 convert ` -- Curried after uncurried is not fused anymore: `(. x) => y => 3` is not equivalent to `(. x, y) => 3` anymore. It's instead equivalent to `(. x) => { y => 3 }`. - Also, `(. int) => string => bool` is not equivalen to `(. int, string) => bool` anymore. - These are only breaking changes for unformatted code. -- Exponentiation operator `**` is now right-associative. `2. ** 3. ** 2.` now compile to `Math.pow(2, Math.pow(3, 2))` and not anymore `Math.pow(Math.pow(2, 3), 2)`. Parentheses can be used to change precedence. -- Stop mangling object field names. If you had objects with field names containing "\__" or leading "_", they won't be mangled in the compiled JavaScript and represented as it is without changes. https://github.com/rescript-lang/rescript/pull/6354 -- `$$default` is no longer exported from the generated JavaScript when using default exports. https://github.com/rescript-lang/rescript/pull/6328 -- `-bs-super-errors` flag has been deprecated along with Super_errors. https://github.com/rescript-lang/rescript/pull/6243 -- Remove unsafe `` j`$(a)$(b)` `` interpolation deprecated in compiler version 10 https://github.com/rescript-lang/rescript/pull/6068 -- `@deriving(jsConverter)` not supported anymore for variant types https://github.com/rescript-lang/rescript/pull/6088 -- New representation for variants, where the tag is a string instead of a number. https://github.com/rescript-lang/rescript/pull/6088 - -### Compiler Libraries - -- Fixed name collision between the newly defined Js.Json.t and the variant constructor in the existing Js.Json.kind type. To address this, the usage of the existing Js.Json.kind type can be updated to Js.Json.Kind.t. https://github.com/rescript-lang/rescript/pull/6317 -- Remove rudimentary node bindings and undocumented `%node` extension. https://github.com/rescript-lang/rescript/pull/6285 -- `@rescript/react` >= 0.12.0-alpha.2 is now required because of the React.fragment's children type fix. https://github.com/rescript-lang/rescript/pull/6238 -- Remove deprecated module `Printexc` - -### Build System and Tools - -- Update watcher rules to recompile only on config and `*.res`/`*.resi`/`*.ml`/`.mli` file changes. Solves the issue of unnecessary recompiles on `.css`, `.ts`, and other unrelated file changes. https://github.com/rescript-lang/rescript/pull/6420 -- Made pinned dependencies transitive: if _a_ is a pinned dependency of _b_ and _b_ is a pinned dependency of _c_, then _a_ is implicitly a pinned dependency of _c_. This change is only breaking if your build process assumes non-transitivity. -- Remove obsolete built-in project templates and the "rescript init" functionality. This is replaced by [create-rescript-app](https://github.com/rescript-lang/create-rescript-app) which is maintained separately. -- Do not attempt to build ReScript from source on npm postinstall for platforms without prebuilt binaries anymore. -- GenType: removed support for `@genType.as` for records and variants which has become unnecessary. Use the language's `@as` instead to channge the runtime representation without requiring any runtime conversion during FFI. https://github.com/rescript-lang/rescript/pull/6099 https://github.com/rescript-lang/rescript/pull/6101 diff --git a/pages/docs/manual/latest/module-functions.mdx b/pages/docs/manual/latest/module-functions.mdx deleted file mode 100644 index 04c6ff458..000000000 --- a/pages/docs/manual/latest/module-functions.mdx +++ /dev/null @@ -1,311 +0,0 @@ ---- -title: "Module Functions" -description: "Module Functions in ReScript" -canonical: "/docs/manual/latest/module-functions" ---- - -# Module Functions - -Module functions can be used to create modules based on types, values, or functions from other modules. -This is a powerful tool that can be used to create abstractions and reusable code that might not be possible with functions, or might have a runtime cost if done with functions. - -This is an advanced part of ReScript and you can generally get by with normal values and functions. - -## Quick example -Next.js has a `useParams` hook that returns an unknown type, -and it's up to the developer in TypeScript to add a type annotation for the parameters returned by the hook. -```TS -const params = useParams<{ tag: string; item: string }>() -``` - -In ReScript we can create a module function that will return a typed response for the `useParams` hook. - -```res example -module Next = { - // define our module function - module MakeParams = (Params: { type t }) => { - @module("next/navigation") - external useParams: unit => Params.t = "useParams" - /* You can use values from the function parameter, such as Params.t */ - } -} - -module Component: { - @react.component - let make: unit => Jsx.element -} = { - // Create a module that matches the module type expected by Next.MakeParams - module P = { - type t = { - tag: string, - item: string, - } - } - - // Create a new module using the Params module we created and the Next.MakeParams module function - module Params = Next.MakeParams(P) - - @react.component - let make = () => { - // Use the functions, values, or types created by the module function - let params = Params.useParams() -
-

- {React.string("Tag: " ++ params.tag /* params is fully typed! */)} -

-

{React.string("Item: " ++ params.item)}

-
- } -} -``` -```js -// Generated by ReScript, PLEASE EDIT WITH CARE - -import * as $$Navigation from "next/navigation"; -import * as JsxRuntime from "react/jsx-runtime"; - -function MakeParams(Params) { - return {}; -} - -var Next = { - MakeParams: MakeParams -}; - -function Playground$Component(props) { - var params = $$Navigation.useParams(); - return JsxRuntime.jsxs("div", { - children: [ - JsxRuntime.jsx("p", { - children: "Tag: " + params.tag - }), - JsxRuntime.jsx("p", { - children: "Item: " + params.item - }) - ] - }); -} - -var Component = { - make: Playground$Component -}; - -export { - Next , - Component , -} -/* next/navigation Not a pure module */ - -``` - - -## Sharing a type with an external binding -This becomes incredibly useful when you need to have types that are unique to a project but shared across multiple components. -Let's say you want to create a library with a `getEnv` function to load in environment variables found in `import.meta.env`. -```res -@val external env: 'a = "import.meta.env" - -let getEnv = () => { - env -} -``` -It's not possible to define types for this that will work for every project, so we just set it as 'a and the consumer of our library can define the return type. -```res -type t = {"LOG_LEVEL": string} - -let values: t = getEnv() -``` -This isn't great and it doesn't take advantage of ReScript's type system and ability to use types without type definitions, and it can't be easily shared across our application. - -We can instead create a module function that can return a module that has contains a `getEnv` function that has a typed response. -```res -module MakeEnv = ( - E: { - type t - }, -) => { - @val external env: E.t = "import.meta.env" - - let getEnv = () => { - env - } -} -``` -And now consumers of our library can define the types and create a custom version of the hook for their application. -Notice that in the JavaScript output that the `import.meta.env` is used directly and doesn't require any function calls or runtime overhead. - - -```res -module Env = MakeEnv({ - type t = {"LOG_LEVEL": string} -}) - -let values = Env.getEnv() -``` -```js -var Env = { - getEnv: getEnv -}; - -var values = import.meta.env; -``` - - -## Shared functions -You might want to share functions across modules, like a way to log a value or render it in React. -Here's an example of module function that takes in a type and a transform to string function. - -```res -module MakeDataModule = ( - T: { - type t - let toString: t => string - }, -) => { - type t = T.t - let log = a => Console.log("The value is " ++ T.toString(a)) - - module Render = { - @react.component - let make = (~value) => value->T.toString->React.string - } -} -``` -You can now take a module with a type of `t` and a `toString` function and create a new module that has the `log` function and the `Render` component. - -```res -module Person = { - type t = { firstName: string, lastName: string } - let toString = person => person.firstName ++ person.lastName -} - -module PersonData = MakeDataModule(Person) -``` - -```js -// Notice that none of the JS output references the MakeDataModule function - -function toString(person) { - return person.firstName + person.lastName; -} - -var Person = { - toString: toString -}; - -function log(a) { - console.log("The value is " + toString(a)); -} - -function Person$MakeDataModule$Render(props) { - return toString(props.value); -} - -var Render = { - make: Person$MakeDataModule$Render -}; - -var PersonData = { - log: log, - Render: Render -}; -``` - - -Now the `PersonData` module has the functions from the `MakeDataModule`. - -```res -@react.component -let make = (~person) => { - let handleClick = _ => PersonData.log(person) -
- {React.string("Hello ")} - - -
-} -``` -```js -function Person$1(props) { - var person = props.person; - var handleClick = function (param) { - log(person); - }; - return JsxRuntime.jsxs("div", { - children: [ - "Hello ", - JsxRuntime.jsx(Person$MakeDataModule$Render, { - value: person - }), - JsxRuntime.jsx("button", { - children: "Log value to console", - onClick: handleClick - }) - ] - }); -} -``` -
- -## Dependency injection -Module functions can be used for dependency injection. -Here's an example of injecting in some config values into a set of functions to access a database. - -```res -module type DbConfig = { - let host: string - let database: string - let username: string - let password: string -} - -module MakeDbConnection = (Config: DbConfig) => { - type client = { - write: string => unit, - read: string => string, - } - @module("database.js") - external makeClient: (string, string, string, string) => client = "makeClient" - - let client = makeClient(Config.host, Config.database, Config.username, Config.password) -} - -module Db = MakeDbConnection({ - let host = "localhost" - let database = "mydb" - let username = "root" - let password = "password" -}) - -let updateDb = Db.client.write("new value") -``` -```js -// Generated by ReScript, PLEASE EDIT WITH CARE - -import * as DatabaseJs from "database.js"; - -function MakeDbConnection(Config) { - var client = DatabaseJs.makeClient(Config.host, Config.database, Config.username, Config.password); - return { - client: client - }; -} - -var client = DatabaseJs.makeClient("localhost", "mydb", "root", "password"); - -var Db = { - client: client -}; - -var updateDb = client.write("new value"); - -export { - MakeDbConnection , - Db , - updateDb , -} -/* client Not a pure module */ -``` - \ No newline at end of file diff --git a/pages/docs/manual/latest/module.mdx b/pages/docs/manual/latest/module.mdx deleted file mode 100644 index 018785b5d..000000000 --- a/pages/docs/manual/latest/module.mdx +++ /dev/null @@ -1,569 +0,0 @@ ---- -title: "Module" -description: "ReScript modules, module signatures and interface files" -canonical: "/docs/manual/latest/module" ---- - -# Module - -## Basics - -**Modules are like mini files**! They can contain type definitions, `let` -bindings, nested modules, etc. - -### Creation - -To create a module, use the `module` keyword. The module name must start with a -**capital letter**. Whatever you could place in a `.res` file, you may place -inside a module definition's `{}` block. - - - -```res example -module School = { - type profession = Teacher | Director - - let person1 = Teacher - let getProfession = (person) => - switch person { - | Teacher => "A teacher" - | Director => "A director" - } -} -``` -```js -function getProfession(person) { - if (person) { - return "A director"; - } else { - return "A teacher"; - } -} - -var School = { - person1: /* Teacher */0, - getProfession: getProfession -}; -``` - - - -A module's contents (including types!) can be accessed much like a record's, -using the `.` notation. This demonstrates modules' utility for namespacing. - - - -```res -let anotherPerson: School.profession = School.Teacher -Console.log(School.getProfession(anotherPerson)) /* "A teacher" */ -``` -```js -var anotherPerson = /* Teacher */0; -console.log("A teacher"); -``` - - - -Nested modules work too. - - - -```res example -module MyModule = { - module NestedModule = { - let message = "hello" - } -} - -let message = MyModule.NestedModule.message -``` -```js -var NestedModule = { - message: message -}; - -var MyModule = { - NestedModule: NestedModule -}; - -var message = MyModule.NestedModule.message; -``` - - - -### `open`ing a module - -Constantly referring to a value/type in a module can be tedious. Instead, we can "open" a module and refer to its contents without always prepending them with the -module's name. Instead of writing: - - - -```res -let p = School.getProfession(School.person1) -``` -```js -var p = School.getProfession(School.person1); -``` - - - -We can write: - - - -```res -open School -let p = getProfession(person1) -``` -```js -var p = School.getProfession(School.person1); -``` - - - -The content of `School` module are made visible (**not** copied into the file, but simply made visible!) in scope. `profession`, `getProfession` and `person1` will thus correctly be found. - -**Use `open` this sparingly, it's convenient, but makes it hard to know where some values come from**. You should usually use `open` in a local scope: - - - -```res -let p = { - open School - getProfession(person1) -} -/* School's content isn't visible here anymore */ -``` -```js -var p = School.getProfession(School.person1); -``` - - - -### Use `open!` to ignore shadow warnings - -There are situations where `open` will cause a warning due to existing identifiers (bindings, types) being redefined. Use `open!` to explicitly tell the compiler that this is desired behavior. - -```res -let map = (arr, value) => { - value -} - -// opening Array would shadow our previously defined `map` -// `open!` will explicitly turn off the automatic warning -open! Array -let arr = map([1,2,3], (a) => { a + 1}) -``` - -**Note:** Same as with `open`, don't overuse `open!` statements if not necessary. Use (sub)modules to prevent shadowing issues. - -### Destructuring modules - -**Since 9.0.2** - -As an alternative to `open`ing a module, you can also destructure a module's functions and values into separate let bindings (similarly on how we'd destructure an object in JavaScript). - - - -```res -module User = { - let user1 = "Anna" - let user2 = "Franz" -} - -// Destructure by name -let {user1, user2} = module(User) - -// Destructure with different alias -let {user1: anna, user2: franz} = module(User) -``` - -```js -var user1 = "Anna"; - -var user2 = "Franz"; - -var User = { - user1: user1, - user2: user2 -}; -``` - - - -**Note:** You can't extract types with module destructuring — use a type alias instead (`type user = User.myUserType`). - -### Extending modules - -Using `include` in a module statically "spreads" a module's content into a new one, thus often fulfill the role of "inheritance" or "mixin". - -**Note**: this is equivalent to a compiler-level copy paste. **We heavily discourage `include`**. Use it as last resort! - - - -```res example -module BaseComponent = { - let defaultGreeting = "Hello" - let getAudience = (~excited) => excited ? "world!" : "world" -} - -module ActualComponent = { - /* the content is copied over */ - include BaseComponent - /* overrides BaseComponent.defaultGreeting */ - let defaultGreeting = "Hey" - let render = () => defaultGreeting ++ " " ++ getAudience(~excited=true) -} -``` -```js -function getAudience(excited) { - if (excited) { - return "world!"; - } else { - return "world"; - } -} - -var BaseComponent = { - defaultGreeting: "Hello", - getAudience: getAudience -}; - -var defaultGreeting = "Hey"; - -function render(param) { - return "Hey world!"; -} - -var ActualComponent = { - getAudience: getAudience, - defaultGreeting: defaultGreeting, - render: render -}; -``` - - - -**Note**: `open` and `include` are very different! The former brings a module's content into your current scope, so that you don't have to refer to a value by prefixing it with the module's name every time. The latter **copies over** the definition of a module statically, then also do an `open`. - -### Every `.res` file is a module - -Every ReScript file is itself compiled to a module of the same name as the file name, capitalized. The file `React.res` implicitly forms a module `React`, which can be seen by other source files. - -**Note**: ReScript file names should, by convention, be capitalized so that their casing matches their module name. Uncapitalized file names are not invalid, but will be implicitly transformed into a capitalized module name. I.e. `file.res` will be compiled into the module `File`. To simplify and minimize the disconnect here, the convention is therefore to capitalize file names. - -## Signatures - -A module's type is called a "signature", and can be written explicitly. If a -module is like a `.res` (implementation) file, then a module's signature is like -a `.resi` (interface) file. - -### Creation - -To create a signature, use the `module type` keyword. The signature name must start with a -**capital letter**. Whatever you could place in a `.resi` file, you may place -inside a signature definition's `{}` block. - - - -```res example -/* Picking up previous section's example */ -module type EstablishmentType = { - type profession - let getProfession: profession => string -} -``` -```js -// Empty output -``` - - - -A signature defines the list of requirements that a module must satisfy in order -for that module to match the signature. Those requirements are of the form: - -- `let x: int` requires a `let` binding named `x`, of type `int`. -- `type t = someType` requires a type field `t` to be equal to `someType`. -- `type t` requires a type field `t`, but without imposing any requirements on the actual, concrete type of `t`. We'd use `t` in other entries in the signature to describe relationships, e.g. `let makePair: t => (t, t)` but we cannot, for example, assume that `t` is an `int`. This gives us great, enforced abstraction abilities. - -To illustrate the various kinds of type entries, consider the above signature -`EstablishmentType` which requires that a module: - -- Declare a type named `profession`. -- Must include a function that takes in a value of the type `profession` and returns a string. - -**Note**: - -Modules of the type `EstablishmentType` can contain more fields than the -signature declares, just like the module `School` in the previous section (if we -choose to assign it the type `EstablishmentType`. Otherwise, `School` exposes -every field). This effectively makes the `person1` field an enforced -implementation detail! Outsiders can't access it, since it's not present in the -signature; the signature **constrained** what others can access. - -The type `EstablishmentType.profession` is **abstract**: it doesn't have a -concrete type; it's saying "I don't care what the actual type is, but it's used -as input to `getProfession`". This is useful to fit many modules under the same -interface: - - - -```res -module Company: EstablishmentType = { - type profession = CEO | Designer | Engineer | ... - - let getProfession = (person) => ... - let person1 = ... - let person2 = ... -} -``` -```js -function getProfession(person) { - ... -} - -var person1 = ... - -var person2 = ... - -var Company = { - getProfession: getProfession, - person1: person1, - person2: person2 -}; -``` - - - -It's also useful to hide the underlying type as an implementation detail others -can't rely on. If you ask what the type of `Company.profession` is, instead of -exposing the variant, it'll only tell you "it's `Company.profession`". - -### Extending module signatures - -Like modules themselves, module signatures can also be extended by other module signatures using `include`. Again, **heavily discouraged**: - - - -```res example -module type BaseComponent = { - let defaultGreeting: string - let getAudience: (~excited: bool) => string -} - -module type ActualComponent = { - /* the BaseComponent signature is copied over */ - include BaseComponent - let render: unit => string -} -``` -```js -// Empty output -``` - - - -**Note**: `BaseComponent` is a module **type**, not an actual module itself! - -If you do not have a defined module type, you can extract it from an actual module -using `include (module type of ActualModuleName)`. For example, we can extend the -`List` module from the standard library, which does not define a module -type. - - - -```res example -module type MyList = { - include (module type of List) - let myListFun: list<'a> => list<'a> -} -``` -```js -// Empty output -``` - - - -### Every `.resi` file is a signature - -Similar to how a `React.res` file implicitly defines a module `React`, a file -`React.resi` implicitly defines a signature for `React`. If `React.resi` isn't -provided, the signature of `React.res` defaults to exposing all the fields of the -module. Because they don't contain implementation files, `.resi` files are used -in the ecosystem to also document the public API of their corresponding modules. - - - -```res example -/* file React.res (implementation. Compiles to module React) */ -type state = int -let render = (str) => str -``` -```js -function render(str) { - return str; -} -``` - - - -```res sig -/* file React.resi (interface. Compiles to the signature of React.res) */ -type state = int -let render: string => string -``` - -## Module Functions - -Modules can be passed to functions! It would be the equivalent of passing a file -as a first-class item. However, modules are at a different "layer" of the -language than other common concepts, so we can't pass them to *regular* -functions. Instead, we pass them to special functions called module functions. - -The syntax for defining and using module functions is very much like the syntax -for defining and using regular functions. The primary differences are: - -- Module functions use the `module` keyword instead of `let`. -- Module functions take modules as arguments and return a module. -- Module functions *require* annotating arguments. -- Module functions must start with a capital letter (just like modules/signatures). - -Here's an example `MakeSet` module function, that takes in a module of the type -`Comparable` and returns a new set that can contain such comparable items. - - - -```res prelude -module type Comparable = { - type t - let equal: (t, t) => bool -} - -module MakeSet = (Item: Comparable) => { - // let's use a list as our naive backing data structure - type backingType = list - let empty = list{} - let add = (currentSet: backingType, newItem: Item.t): backingType => - // if item exists - if currentSet->List.some(x => Item.equal(x, newItem)) { - currentSet // return the same (immutable) set (a list really) - } else { - list{ - newItem, - ...currentSet // prepend to the set and return it - } - } -} -``` -```js -var List = require("./stdlib/list.js"); - -function MakeSet(Item) { - var add = function(currentSet, newItem) { - if ( - List.exists(function(x) { - return Item.equal(x, newItem); - }, currentSet) - ) { - return currentSet; - } else { - return { - hd: newItem, - tl: currentSet, - }; - } - }; - return { - empty: /* [] */ 0, - add: add, - }; -} -``` - - - -Module functions can be applied using function application syntax. In this case, we're -creating a set, whose items are pairs of integers. - - - -```res example -module IntPair = { - type t = (int, int) - let equal = ((x1: int, y1: int), (x2, y2)) => x1 == x2 && y1 == y2 - let create = (x, y) => (x, y) -} - -/* IntPair abides by the Comparable signature required by MakeSet */ -module SetOfIntPairs = MakeSet(IntPair) -``` -```js -function equal(param, param$1) { - if (param[0] === param$1[0]) { - return param[1] === param$1[1]; - } else { - return false; - } -} - -function create(x, y) { - return [x, y]; -} - -var IntPair = { - equal: equal, - create: create, -}; - -var SetOfIntPairs = { - empty: /* [] */ 0, - add: add, -}; -``` - - - -### Module functions types - -Like with module types, module function types also act to constrain and hide what we may -assume about module functions. The syntax for module function types are consistent with those -for function types, but with types capitalized to represent the signatures of -modules the module functions accepts as arguments and return values. In the -previous example, we're exposing the backing type of a set; by giving `MakeSet` -a module function signature, we can hide the underlying data structure! - - - -```res -module type Comparable = ... - -module type MakeSetType = (Item: Comparable) => { - type backingType - let empty: backingType - let add: (backingType, Item.t) => backingType -} - -module MakeSet: MakeSetType = (Item: Comparable) => { - ... -} -``` -```js -// Empty output -``` - - - -## Exotic Module Filenames - -**Since 8.3** - -It is possible to use non-conventional characters in your filenames (which is sometimes needed for specific JS frameworks). Here are some examples: - -- `src/Button.ios.res` -- `pages/[id].res` - -Please note that modules with an exotic filename will not be accessible from other ReScript modules. - -## Tips & Tricks - -Modules and module functions are at a different "layer" of language than the rest (functions, let bindings, data structures, etc.). For example, you can't easily pass them into a tuple or record. Use them judiciously, if ever! Lots of times, just a record or a function is enough. diff --git a/pages/docs/manual/latest/mutation.mdx b/pages/docs/manual/latest/mutation.mdx deleted file mode 100644 index 750613a4a..000000000 --- a/pages/docs/manual/latest/mutation.mdx +++ /dev/null @@ -1,75 +0,0 @@ ---- -title: "Mutation" -description: "Imperative and mutative programming capabilities in ReScript" -canonical: "/docs/manual/latest/mutation" ---- - -# Mutation - -ReScript has great traditional imperative & mutative programming capabilities. You should use these features sparingly, but sometimes they allow your code to be more performant and written in a more familiar pattern. - -## Mutate Let-binding - -Let-bindings are immutable, but you can wrap it with a `ref`, exposed as a record with a single mutable field in the standard library: - - - -```res prelude -let myValue = ref(5) -``` -```js -var myValue = { - contents: 5 -}; -``` - - - -## Usage - -You can get the actual value of a `ref` box through accessing its `contents` field: - - - -```res example -let five = myValue.contents // 5 -``` -```js -var five = myValue.contents; -``` - - - -Assign a new value to `myValue` like so: - - - -```res example -myValue.contents = 6 -``` -```js -myValue.contents = 6; -``` - - - -We provide a syntax sugar for this: - - - -```res example -myValue := 6 -``` -```js -myValue.contents = 6; -``` - - - -Note that the previous binding `five` stays `5`, since it got the underlying item on the `ref` box, not the `ref` itself. - -**Note**: you might see in the JS output tabs above that `ref` allocates an object. Worry not; local, non-exported `ref`s allocations are optimized away. - -## Tip & Tricks - -Before reaching for `ref`, know that you can achieve lightweight, local "mutations" through [overriding let bindings](let-binding.md#binding-shadowing). diff --git a/pages/docs/manual/latest/newcomer-examples.mdx b/pages/docs/manual/latest/newcomer-examples.mdx deleted file mode 100644 index 87d4168e4..000000000 --- a/pages/docs/manual/latest/newcomer-examples.mdx +++ /dev/null @@ -1,134 +0,0 @@ ---- -title: "Newcomer Examples" -description: "Quick examples for users new to ReScript" -canonical: "/docs/manual/latest/newcomer-examples" ---- - -# Newcomer Examples - - - -An example is worth a thousand words. - -This section is dedicated to newcomers trying to figure out general idioms & conventions. If you're a beginner who's got a good idea for an example, please suggest an edit! - -## Use the [`option` type](null-undefined-option.md) - - - -```res example -let possiblyNullValue1 = None -let possiblyNullValue2 = Some("Hello") - -switch possiblyNullValue2 { -| None => Console.log("Nothing to see here.") -| Some(message) => Console.log(message) -} -``` -```js -var possiblyNullValue1; -var possiblyNullValue2 = "Hello"; - -if (possiblyNullValue2 !== undefined) { - console.log(possiblyNullValue2); -} else { - console.log("Nothing to see here."); -} - -``` - - - -## Create a Parametrized Type - - - -```res example -type universityStudent = {gpa: float} - -type response<'studentType> = { - status: int, - student: 'studentType, -} -``` -```js -// Empty output -``` - - - -## Creating a JS Object - - - -```res example -let student1 = { - "name": "John", - "age": 30, -} -``` -```js -var student1 = { - name: "John", - age: 30, -}; -``` - - - -Or using [record](record.md): - - - -```res example -type payload = { - name: string, - age: int, -} - -let student1 = { - name: "John", - age: 30, -} -``` -```js -var student1 = { - name: "John", - age: 30, -}; -``` - - - -## Modeling a JS Module with Default Export - -See [here](import-from-export-to-js.md#import-a-javascript-module-itself-es6-module-format). - -## Checking for JS nullable types using the `option` type - -For a function whose argument is passed a JavaScript value that's potentially `null` or `undefined`, it's idiomatic to convert it to an `option`. The conversion is done through the helper functions in ReScript's [`Nullable`](api/core/nullable#value-toOption) module. In this case, `toOption`: - - - -```res example -let greetByName = (possiblyNullName) => { - let optionName = Nullable.toOption(possiblyNullName) - switch optionName { - | None => "Hi" - | Some(name) => "Hello " ++ name - } -} -``` -```js -function greetByName(possiblyNullName) { - if (possiblyNullName == null) { - return "Hi"; - } else { - return "Hello " + possiblyNullName; - } -} -``` - - - -This check compiles to `possiblyNullName == null` in JS, so checks for the presence of `null` or `undefined`. diff --git a/pages/docs/manual/latest/null-undefined-option.mdx b/pages/docs/manual/latest/null-undefined-option.mdx deleted file mode 100644 index f49153ce2..000000000 --- a/pages/docs/manual/latest/null-undefined-option.mdx +++ /dev/null @@ -1,203 +0,0 @@ ---- -title: "Null, Undefined and Option" -description: "JS interop with nullable and optional values in ReScript" -canonical: "/docs/manual/latest/null-undefined-option" ---- - -# Null, Undefined and Option - -ReScript itself doesn't have the notion of `null` or `undefined`. This is a _great_ thing, as it wipes out an entire category of bugs. No more `undefined is not a function`, and `cannot access someAttribute of undefined`! - -However, the **concept** of a potentially nonexistent value is still useful, and safely exists in our language. - -We represent the existence and nonexistence of a value by wrapping it with the `option` type. Here's its definition from the standard library: - - - -```res example -type option<'a> = None | Some('a) -``` -```js -// Empty output -``` - - - -It means "a value of type option is either None (representing nothing) or that actual value wrapped in a Some". - -**Note** how the `option` type is just a regular [variant](variant.md). - -## Example - -Here's a normal value: - - - -```res example -let licenseNumber = 5 -``` -```js -var licenseNumber = 5; -``` - - - -To represent the concept of "maybe null", you'd turn this into an `option` type by wrapping it. For the sake of a more illustrative example, we'll put a condition around it: - - - -```res -let licenseNumber = - if personHasACar { - Some(5) - } else { - None - } -``` -```js -var licenseNumber = personHasACar ? 5 : undefined; -``` - - - -Later on, when another piece of code receives such value, it'd be forced to handle both cases through [pattern matching](pattern-matching-destructuring.md): - - - -```res -switch licenseNumber { -| None => - Console.log("The person doesn't have a car") -| Some(number) => - Console.log("The person's license number is " ++ Int.toString(number)) -} -``` -```js -var number = licenseNumber; - -if (number !== undefined) { - console.log("The person's license number is " + number.toString()); -} else { - console.log("The person doesn't have a car"); -} -``` - - - -By turning your ordinary number into an `option` type, and by forcing you to handle the `None` case, the language effectively removed the possibility for you to mishandle, or forget to handle, a conceptual `null` value! **A pure ReScript program doesn't have null errors**. - -## Interoperate with JavaScript `undefined` and `null` - -The `option` type is common enough that we special-case it when compiling to JavaScript: - - - -```res example -let x = Some(5) -``` -```js -var x = 5; -``` - - - -simply compiles down to `5`, and - - - -```res example -let x = None -``` -```js -var x; -``` - - - -compiles to `undefined`! If you've got e.g. a string in JavaScript that you know might be `undefined`, type it as `option` and you're done! Likewise, you can send a `Some(5)` or `None` to the JS side and expect it to be interpreted correctly =) - -### Caveat 1 - -The option-to-undefined translation isn't perfect, because on our side, `option` values can be composed: - - - -```res example -let x = Some(Some(Some(5))) -``` -```js -var x = 5; -``` - - - -This still compiles to `5`, but this gets troublesome: - - - -```res example -let x = Some(None) -``` -```js -var Caml_option = require("./stdlib/caml_option.js"); - -var x = Caml_option.some(undefined); -``` - -(See output tab). - - - -What's this `Caml_option.some` thing? Why can't this compile to `undefined`? Long story short, when dealing with a polymorphic `option` type (aka `option<'a>`, for any `'a`), many operations become tricky if we don't mark the value with some special annotation. If this doesn't make sense, don't worry; just remember the following rule: - -- **Never, EVER, pass a nested `option` value (e.g. `Some(Some(Some(5)))`) into the JS side.** -- **Never, EVER, annotate a value coming from JS as `option<'a>`. Always give the concrete, non-polymorphic type.** - -### Caveat 2 - -Unfortunately, lots of times, your JavaScript value might be _both_ `null` or `undefined`. In that case, you unfortunately can't type such value as e.g. `option`, since our `option` type only checks for `undefined` and not `null` when dealing with a `None`. - -#### Solution: More Sophisticated `undefined` & `null` Interop - -To solve this, we provide access to more elaborate `null` and `undefined` helpers through the [`Nullable`](api/core/nullable) module. This somewhat works like an `option` type, but is different from it. - -#### Examples - -To create a JS `null`, use the value `Nullable.null`. To create a JS `undefined`, use `Nullable.undefined` (you can naturally use `None` too, but that's not the point here; the `Nullable.*` helpers wouldn't work with it). - -If you're receiving, for example, a JS string that can be `null` and `undefined`, type it as: - - - -```res example -@module("MyConstant") external myId: Nullable.t = "myId" -``` -```js -// Empty output -``` - - - -To create such a nullable string from our side (presumably to pass it to the JS side, for interop purpose), do: - - - -```res example -@module("MyIdValidator") external validate: Nullable.t => bool = "validate" -let personId: Nullable.t = Nullable.make("abc123") - -let result = validate(personId) -``` -```js -var MyIdValidator = require("MyIdValidator"); -var personId = "abc123"; -var result = MyIdValidator.validate(personId); -``` - - - -The `return` part "wraps" a string into a nullable string, to make the type system understand and track the fact that, as you pass this value around, it's not just a string, but a string that can be `null` or `undefined`. - -#### Convert to/from `option` - -`Nullable.fromOption` converts from a `option` to `Nullable.t`. `Nullable.toOption` does the opposite. diff --git a/pages/docs/manual/latest/object.mdx b/pages/docs/manual/latest/object.mdx deleted file mode 100644 index 41adf4485..000000000 --- a/pages/docs/manual/latest/object.mdx +++ /dev/null @@ -1,195 +0,0 @@ ---- -title: "Object" -description: "Interoping with JS objects in ReScript" -canonical: "/docs/manual/latest/object" ---- - -# Object - -ReScript objects are like [records](record.md), but: - -- No type declaration needed. -- Structural and more polymorphic, [unlike records](record.md#record-types-are-found-by-field-name). -- Doesn't support updates unless the object comes from the JS side. -- Doesn't support [pattern matching](pattern-matching-destructuring). - - - -Although ReScript records compile to clean JavaScript objects, ReScript objects are a better candidate for emulating/binding to JS objects, as you'll see. - -## Type Declaration - -**Optional**, unlike for records. The type of an object is inferred from the value, so you never really need to write down its type definition. Nevertheless, here's its type declaration syntax: - - - -```res prelude -type person = { - "age": int, - "name": string -}; -``` -```js -// Empty output -``` - - - -Visually similar to record type's syntax, with the field names quoted. - - - -## Creation - -To create a new object: - - - -```res example -let me = { - "age": 5, - "name": "Big ReScript" -} -``` -```js -var me = { - "age": 5, - "name": "Big ReScript" -}; -``` - - - -**Note**: as said above, unlike for record, this `me` value does **not** try to find a conforming type declaration with the field `"age"` and `"name"`; rather, the type of `me` is inferred as `{"age": int, "name": string}`. This is convenient, but also means this code passes type checking without errors: - - - -```res -type person = { - "age": int -}; - -let me = { - "age": "hello!" // age is a string. No error. -} -``` -```js -var me = { - "age": "hello!" -}; -``` - - - -Since the type checker doesn't try to match `me` with the type `person`. If you ever want to force an object value to be of a predeclared object type, just annotate the value: - -```res -let me: person = { - "age": "hello!" -} -``` - -Now the type system will error properly. - -## Access - - - -```res -let age = me["age"] -``` -```js -var age = me["age"]; -``` - - - -## Update - -Disallowed unless the object is a binding that comes from the JavaScript side. In that case, use `=` - - - -```res example -type student = { - @set "age": int, - @set "name": string, -} -@module("MyJSFile") external student1: student = "student1" - -student1["name"] = "Mary" -``` -```js -var MyJSFile = require("MyJSFile"); -MyJSFile.student1.name = "Mary"; -``` - - - -## Combine Types - -You can spread one object type definition into another using `...`: - - - -```res example -type point2d = { - "x": float, - "y": float, -} -type point3d = { - ...point2d, - "z": float, -} - -let myPoint: point3d = { - "x": 1.0, - "y": 2.0, - "z": 3.0, -} -``` -```js -var myPoint = { - x: 1.0, - y: 2.0, - z: 3.0 -}; -``` - - - -This only works with object types, not object values! - -## Tips & Tricks - -Since objects don't require type declarations, and since ReScript infers all the types for you, you get to very quickly and easily (and dangerously) bind to any JavaScript API. Check the JS output tab: - - - -```res example -// The type of document is just some random type 'a -// that we won't bother to specify -@val external document: 'a = "document" - -// call a method -document["addEventListener"]("mouseup", _event => { - Console.log("clicked!") -}) - -// get a property -let loc = document["location"] - -// set a property -document["location"]["href"] = "rescript-lang.org" -``` -```js -document.addEventListener("mouseup", function(_event) { - console.log("clicked!"); -}); -var loc = document.location; -document.location.href = "rescript-lang.org"; -``` - - - -The `external` feature and the usage of this trick are also documented in the [external](external#tips--tricks) section later. It's an excellent way to start writing some ReScript code without worrying about whether bindings to a particular library exists. diff --git a/pages/docs/manual/latest/overview.mdx b/pages/docs/manual/latest/overview.mdx deleted file mode 100644 index 9ffd6bbc4..000000000 --- a/pages/docs/manual/latest/overview.mdx +++ /dev/null @@ -1,255 +0,0 @@ ---- -title: "Overview" -metaTitle: "Language Features Overview" -description: "A quick overview on ReScript's syntax" -canonical: "/docs/manual/latest/overview" ---- - -# Overview - -## Comparison to JS - -### Semicolon - -| JavaScript | ReScript | -| ------------------------------------ | -------------------- | -| Rules enforced by linter/formatter | No semicolon needed! | - -### Comments - -| JavaScript | ReScript | -| -------------------- | -------------------------------- | -| `// Line comment` | Same | -| `/* Comment */` | Same | -| `/** Doc Comment */` | `/** Before Types/Values */` | -| | `/*** Standalone Doc Comment */` | - -### Variable - -| JavaScript | ReScript | -| ----------------------- | ------------------------------ | -| `const x = 5;` | `let x = 5` | -| `var x = y;` | No equivalent (thankfully) | -| `let x = 5; x = x + 1;` | `let x = ref(5); x := x.contents + 1` | - -### String & Character - -| JavaScript | ReScript | -| ----------------------------- | --------------------- | -| `"Hello world!"` | Same | -| `'Hello world!'` | Strings must use `"` | -| `"hello " + "world"` | `"hello " ++ "world"` | -| `` `hello ${message}` `` | Same | -| `` sql`select ${fnName};` `` | Same | - -### Boolean - -| JavaScript | ReScript | -| ----------------------------------------------------- | ---------------------------------------------- | -| `true`, `false` | Same | -| `!true` | Same | -| ||, `&&`, `<=`, `>=`, `<`, `>` | Same | -| `a === b`, `a !== b` | Same | -| No deep equality (recursive compare) | `a == b`, `a != b` | -| `a == b` | No equality with implicit casting (thankfully) | - -### Number - -| JavaScript | ReScript | -| ----------- | ------------ | -| `3` | Same \* | -| `3.1415` | Same | -| `3 + 4` | Same | -| `3.0 + 4.5` | `3.0 +. 4.5` | -| `5 % 3` | `mod(5, 3)` | - -\* JS has no distinction between integer and float. - -### Object/Record - -| JavaScript | ReScript | -| ------------------- | --------------------------------------- | -| no types | `type point = {x: int, mutable y: int}` | -| `{x: 30, y: 20}` | Same | -| `point.x` | Same | -| `point.y = 30;` | Same | -| `{...point, x: 30}` | Same | - -### Array - -| JavaScript | ReScript | -| --------------------- | ---------------------------------- | -| `[1, 2, 3]` | Same | -| `myArray[1] = 10` | Same | -| `[1, "Bob", true]` | `(1, "Bob", true)` \* | - -\* ReScript does not have heterogenous arrays. Use tuples or [Untagged Variants](variant#untagged-variants) instead. - -### Null - -| JavaScript | ReScript | -| ------------------- | --------- | -| `null`, `undefined` | `None` \* | - -\* Again, only a spiritual equivalent; we don't have nulls, nor null bugs! But we do have an `option` type for when you actually need nullability. - -### Function - -| JavaScript | ReScript | -| ------------------------------- | ---------------------------- | -| `arg => retVal` | Same | -| `function named(arg) {...}` | `let named = (arg) => {...}` | -| `const f = function(arg) {...}` | `let f = (arg) => {...}` | -| `add(4, add(5, 6))` | Same | - -### Async Function / Await - -| JavaScript | ReScript | -| ---------------------------------------- | ---------------------------------- | -| `async (arg) => {...}` | Same | -| `async function named(arg) {...}` | `let named = async (arg) => {...}` | -| `await somePromise` | Same | -| `async (arg): Promise => {...}` | `async (arg): string => {...}` (note the return type)| - -### Blocks - - - - - - - - - - - - - - -
JavaScriptReScript
- ``` - const myFun = (x, y) => { - const doubleX = x + x; - const doubleY = y + y; - return doubleX + doubleY; - }; - ``` - - ``` - let myFun = (x, y) => { - let doubleX = x + x - let doubleY = y + y - doubleX + doubleY - } - ``` -
- -### If-else - -| JavaScript | ReScript | -| --------------------- | ------------------------------------------------------------------- | -| `if (a) {b} else {c}` | `if a {b} else {c}` \* | -| `a ? b : c` | Same | -| `switch` | `switch` but [super-powered pattern matching!](pattern-matching-destructuring.md) | - -\* Our conditionals are always expressions! You can write `let result = if a {"hello"} else {"bye"}` - -### Destructuring - -| JavaScript | ReScript | -| ----------------------------- | --------------------------------------------- | -| `const {a, b} = data` | `let {a, b} = data` | -| `const [a, b] = data` | `let [a, b] = data` \* | -| `const {a: aa, b: bb} = data` | `let {a: aa, b: bb} = data` | - -\* Gives good compiler warning that `data` might not be of length 2. - -### Loop - -| JavaScript | ReScript | -| ------------------------------------- | ------------------------------ | -| `for (let i = 0; i <= 10; i++) {...}` | `for i in 0 to 10 {...}` | -| `for (let i = 10; i >= 0; i--) {...}` | `for i in 10 downto 0 {...}` | -| `while (true) {...}` | `while true {...}` | - -### JSX - -| JavaScript | ReScript | -| ----------------------------------------- | -------------------------- | -| `` | Same | -| `` | `` \* | -| `` | `` | -| No children spread | `...children` | - -\* Argument punning! - -### Exception - -| JavaScript | ReScript | -| ----------------------------------------------- | ------------------------------------------ | -| `throw new SomeError(...)` | `raise(SomeError(...))` | -| `try {a} catch (err) {...} finally {...}` | `try a catch { \| SomeError(err) => ...}` \* | - -\* No finally. - -### Blocks - -The last expression of a block delimited by `{}` implicitly returns (including function body). In JavaScript, this can only be simulated via an immediately-invoked function expression (since function bodies have their own local scope). - - - - - - - - - - - - - - -
JavaScriptReScript
- ``` - let result = (function() { - const x = 23; - const y = 34; - return x + y; - })(); - ``` - - ``` - let result = { - let x = 23 - let y = 34 - x + y - } - ``` -
- -## Common Features' JS Output - -Feature | Example | JavaScript Output ---------------------------------|--------------------------------------|---------------------- -String | `"Hello"` | `"Hello"` -String Interpolation | `` `Hello ${message}` `` | `"Hello " + message` -Character (disrecommended) | `'x'` | `120` (char code) -Integer | `23`, `-23` | `23`, `-23` -Float | `23.0`, `-23.0` | `23.0`, `-23.0` -Integer Addition | `23 + 1` | `23 + 1` -Float Addition | `23.0 +. 1.0` | `23.0 + 1.0` -Integer Division/Multiplication | `2 / 23 * 1` | `2 / 23 * 1` -Float Division/Multiplication | `2.0 /. 23.0 *. 1.0` | `2.0 / 23.0 * 1.0` -Float Exponentiation | `2.0 ** 3.0` | `Math.pow(2.0, 3.0)` -String Concatenation | `"Hello " ++ "World"` | `"Hello " + "World"` -Comparison | `>`, `<`, `>=`, `<=` | `>`, `<`, `>=`, `<=` -Boolean operation | `!`, `&&`, || | `!`, `&&`, || -Shallow and deep Equality | `===`, `==` | `===`, `==` -List (disrecommended) | `list{1, 2, 3}` | `{hd: 1, tl: {hd: 2, tl: {hd: 3, tl: 0}}}` -List Prepend | `list{a1, a2, ...oldList}` | `{hd: a1, tl: {hd: a2, tl: theRest}}` -Array | `[1, 2, 3]` | `[1, 2, 3]` -Record | `type t = {b: int}; let a = {b: 10}` | `var a = {b: 10}` -Multiline Comment | `/* Comment here */` | Not in output -Single line Comment | `// Comment here` | Not in output - -_Note that this is a cleaned-up comparison table; a few examples' JavaScript output are slightly different in reality._ diff --git a/pages/docs/manual/latest/pattern-matching-destructuring.mdx b/pages/docs/manual/latest/pattern-matching-destructuring.mdx deleted file mode 100644 index 2aa70d210..000000000 --- a/pages/docs/manual/latest/pattern-matching-destructuring.mdx +++ /dev/null @@ -1,864 +0,0 @@ ---- -title: "Pattern Matching / Destructuring" -description: "Pattern matching and destructuring complex data structures in ReScript" -canonical: "/docs/manual/latest/pattern-matching-destructuring" ---- - -# Pattern Matching / Destructuring - -One of ReScript's **best** feature is our pattern matching. Pattern matching combines 3 brilliant features into one: - -- Destructuring. -- `switch` based on shape of data. -- Exhaustiveness check. - -We'll dive into each aspect below. - -## Destructuring - -Even JavaScript has destructuring, which is "opening up" a data structure to extract the parts we want and assign variable names to them: - - - -```res example -let coordinates = (10, 20, 30) -let (x, _, _) = coordinates -Console.log(x) // 10 -``` -```js -var coordinates = [10, 20, 30]; -var x = 10; -console.log(10); -``` - - - -Destructuring works with most built-in data structures: - - - -```res -// Record -type student = {name: string, age: int} -let student1 = {name: "John", age: 10} -let {name} = student1 // "John" assigned to `name` - -// Variant -type result = - | Success(string) -let myResult = Success("You did it!") -let Success(message) = myResult // "You did it!" assigned to `message` -``` -```js -var student1 = { - name: "John", - age: 10 -}; -var name = "John"; - -var myResult = /* Success */{ - _0: "You did it!" -}; -var message = "You did it!" - -var myArray = [1, 2, 3]; -if (myArray.length !== 2) { - throw { - RE_EXN_ID: "Match_failure", - _1: [ - "playground.res", - 14, - 4 - ], - Error: new Error() - }; -} -var item1 = myArray[0]; -var item2 = myArray[1]; - -var myList = { - hd: 1, - tl: { - hd: 2, - tl: { - hd: 3, - tl: /* [] */0 - } - } -}; -// ... -``` - - - -You can also use destructuring anywhere you'd usually put a binding: - - - -```res example -type result = - | Success(string) -let displayMessage = (Success(m)) => { - // we've directly extracted the success message - // string by destructuring the parameter - Console.log(m) -} -displayMessage(Success("You did it!")) -``` -```js -function displayMessage(m) { - console.log(m._0); -} - -displayMessage(/* Success */{ - _0: "You did it!" -}); -``` - - - -For a record, you can rename the field while destructuring: - - - -```res -let {name: n} = student1 // "John" assigned to `n` -``` -```js -var n = "John"; -``` - - - -You _can_ in theory destructure array and list at the top level too: - -```res -let myArray = [1, 2, 3] -let [item1, item2, _] = myArray -// 1 assigned to `item1`, 2 assigned to `item2`, 3rd item ignored - -let myList = list{1, 2, 3} -let list{head, ...tail} = myList -// 1 assigned to `head`, `list{2, 3}` assigned to tail -``` - -But the array example is **highly disrecommended** (use tuple instead) and the list example will error on you. They're only there for completeness' sake. As you'll see below, the proper way of using destructuring array and list is using `switch`. - -## `switch` Based on Shape of Data - -While the destructuring aspect of pattern matching is nice, it doesn't really change the way you think about structuring your code. One paradigm-changing way of thinking about your code is to execute some code based on the shape of the data. - -Consider a variant: - - - -```res prelude -type payload = - | BadResult(int) - | GoodResult(string) - | NoResult -``` -```js -// Empty output -``` - - - -We'd like to handle each of the 3 cases differently. For example, print a success message if the value is `GoodResult(...)`, do something else when the value is `NoResult`, etc. - -In other languages, you'd end up with a series of if-elses that are hard to read and error-prone. In ReScript, you can instead use the supercharged `switch` pattern matching facility to destructure the value while calling the right code based on what you destructured: - - - -```res example -let data = GoodResult("Product shipped!") -switch data { -| GoodResult(theMessage) => - Console.log("Success! " ++ theMessage) -| BadResult(errorCode) => - Console.log("Something's wrong. The error code is: " ++ Int.toString(errorCode)) -| NoResult => - Console.log("Bah.") -} -``` -```js -var data = { - TAG: "GoodResult", - _0: "Product shipped!" -}; - -if (typeof data !== "object") { - console.log("Bah."); -} else if (data.TAG === "BadResult") { - console.log("Something's wrong. The error code is: " + "Product shipped!".toString()); -} else { - console.log("Success! Product shipped!"); -} -``` - - - -In this case, `message` will have the value `"Success! Product shipped!"`. - -Suddenly, your if-elses that messily checks some structure of the value got turned into a clean, compiler-verified, linear list of code to execute based on exactly the shape of the value. - -### Complex Examples - -Here's a real-world scenario that'd be a headache to code in other languages. Given this data structure: - - - -```res prelude -type status = Vacations(int) | Sabbatical(int) | Sick | Present -type reportCard = {passing: bool, gpa: float} -type student = {name: string, status: status, reportCard: reportCard} -type person = - | Teacher({name: string, age: int}) - | Student(student) -``` -```js -// Empty output -``` - - - -Imagine this requirement: - -- Informally greet a person who's a teacher and if his name is Mary or Joe. -- Greet other teachers formally. -- If the person's a student, congratulate him/her score if they passed the semester. -- If the student has a gpa of 0 and is on vacations or sabbatical, display a different message. -- A catch-all message for a student. - -ReScript can do this easily! - - - -```res prelude -let person1 = Teacher({name: "Jane", age: 35}) - -let message = switch person1 { -| Teacher({name: "Mary" | "Joe"}) => - `Hey, still going to the party on Saturday?` -| Teacher({name}) => - // this is matched only if `name` isn't "Mary" or "Joe" - `Hello ${name}.` -| Student({name, reportCard: {passing: true, gpa}}) => - `Congrats ${name}, nice GPA of ${Float.toString(gpa)} you got there!` -| Student({ - reportCard: {gpa: 0.0}, - status: Vacations(daysLeft) | Sabbatical(daysLeft) - }) => - `Come back in ${Int.toString(daysLeft)} days!` -| Student({status: Sick}) => - `How are you feeling?` -| Student({name}) => - `Good luck next semester ${name}!` -} -``` -```js -var person1 = { - TAG: "Teacher", - name: "Jane", - age: 35 -}; - -var message; - -if (person1.TAG === "Teacher") { - message = "Hello Jane."; -} else { - var match = "Jane"; - var match$1 = match.status; - var name = match.name; - var match$2 = match.reportCard; - if (match$2.passing) { - message = "Congrats " + name + ", nice GPA of " + match$2.gpa.toString() + " you got there!"; - } else { - var exit = 0; - if (typeof match$1 !== "object") { - message = match$1 === "Sick" ? "How are you feeling?" : "Good luck next semester " + name + "!"; - } else { - exit = 1; - } - if (exit === 1) { - message = match.reportCard.gpa !== 0.0 ? "Good luck next semester " + name + "!" : "Come back in " + match$1._0.toString() + " days!"; - } - - } -} -``` - - - -**Note** how we've: -- drilled deep down into the value concisely -- using a **nested pattern check** `"Mary" | "Joe"` and `Vacations | Sabbatical` -- while extracting the `daysLeft` number from the latter case -- and assigned the greeting to the binding `message`. - -Here's another example of pattern matching, this time on an inline tuple. - - - -```res -type animal = Dog | Cat | Bird -let categoryId = switch (isBig, myAnimal) { -| (true, Dog) => 1 -| (true, Cat) => 2 -| (true, Bird) => 3 -| (false, Dog | Cat) => 4 -| (false, Bird) => 5 -} -``` -```js -var categoryId = isBig ? (myAnimal + 1) | 0 : myAnimal >= 2 ? 5 : 4; -``` - - - -**Note** how pattern matching on a tuple is equivalent to a 2D table: - -isBig \ myAnimal | Dog | Cat | Bird ------------------|-----|-----|------ -true | 1 | 2 | 3 -false | 4 | 4 | 5 - -### Fall-Through Patterns - -The nested pattern check, demonstrated in the earlier `person` example, also works at the top level of a `switch`: - - - -```res prelude -let myStatus = Vacations(10) - -switch myStatus { -| Vacations(days) -| Sabbatical(days) => Console.log(`Come back in ${Int.toString(days)} days!`) -| Sick -| Present => Console.log("Hey! How are you?") -} -``` -```js -var myStatus = { - TAG: /* Vacations */0, - _0: 10 -}; - -if (typeof myStatus === "number") { - console.log("Hey! How are you?"); -} else { - console.log("Come back in " + (10).toString() + " days!"); -} -``` - - - -Having multiple cases fall into the same handling can clean up certain types of logic. - -### Ignore Part of a Value - -If you have a value like `Teacher(payload)` where you just want to pattern match on the `Teacher` part and ignore the `payload` completely, you can use the `_` wildcard like this: - - - -```res example -switch person1 { -| Teacher(_) => Console.log("Hi teacher") -| Student(_) => Console.log("Hey student") -} -``` -```js -if (person1.TAG === "Teacher") { - console.log("Hi teacher"); -} else { - console.log("Hey student"); -} -``` - - - -`_` also works at the top level of the `switch`, serving as a catch-all condition: - - - -```res example -switch myStatus { -| Vacations(_) => Console.log("Have fun!") -| _ => Console.log("Ok.") -} -``` -```js -if (typeof myStatus !== "object" || myStatus.TAG !== "Vacations") { - console.log("Ok."); -} else { - console.log("Have fun!"); -} -``` - - - -**Do not** abuse a top-level catch-all condition. Instead, prefer writing out all the cases: - - - -```res example -switch myStatus { -| Vacations(_) => Console.log("Have fun!") -| Sabbatical(_) | Sick | Present => Console.log("Ok.") -} -``` -```js -if (typeof myStatus !== "object" || myStatus.TAG !== "Vacations") { - console.log("Ok."); -} else { - console.log("Have fun!"); -} -``` - - - -Slightly more verbose, but a one-time writing effort. This helps when you add a new variant case e.g. `Quarantined` to the `status` type and need to update the places that pattern match on it. A top-level wildcard here would have accidentally and silently continued working, potentially causing bugs. - -### If Clause - -Sometime, you want to check more than the shape of a value. You want to also run some arbitrary check on it. You might be tempted to write this: - - - -```res example -switch person1 { -| Teacher(_) => () // do nothing -| Student({reportCard: {gpa}}) => - if gpa < 0.5 { - Console.log("What's happening") - } else { - Console.log("Heyo") - } -} -``` -```js -if (person1.TAG !== "Teacher") { - if ("Jane".reportCard.gpa < 0.5) { - console.log("What's happening"); - } else { - console.log("Heyo"); - } -} -``` - - - -`switch` patterns support a shortcut for the arbitrary `if` check, to keep your pattern linear-looking: - - - -```res example -switch person1 { -| Teacher(_) => () // do nothing -| Student({reportCard: {gpa}}) if gpa < 0.5 => - Console.log("What's happening") -| Student(_) => - // fall-through, catch-all case - Console.log("Heyo") -} -``` -```js -if (person1.TAG) { - if (person1.reportCard.gpa < 0.5) { - console.log("What's happening"); - } else { - console.log("Heyo"); - } -} -``` - - - -**Note:** Rescript versions < 9.0 had a `when` clause, not an `if` clause.  Rescript 9.0 changed `when` to `if`.  (`when` may still work, but is deprecated.) - -### Match on Exceptions - -If the function throws an exception (covered later), you can also match on _that_, in addition to the function's normally returned values. - - - -```res -switch List.find(i => i === theItem, myItems) { -| item => Console.log(item) -| exception Not_found => Console.log("No such item found!") -} -``` -```js -var exit = 0; - -var item; - -try { - item = List.find(function(i) { - return i === theItem; - }, myItems); - exit = 1; -} -catch (raw_exn){ - var exn = Caml_js_exceptions.internalToOCamlException(raw_exn); - if (exn.RE_EXN_ID === "Not_found") { - console.log("No such item found!"); - } else { - throw exn; - } -} - -if (exit === 1) { - console.log(item); -} -``` - - - -### Match on Array - - - -```res example -let students = ["Jane", "Harvey", "Patrick"] -switch students { -| [] => Console.log("There are no students") -| [student1] => - Console.log("There's a single student here: " ++ student1) -| manyStudents => - // display the array of names - Console.log2("The students are: ", manyStudents) -} -``` -```js -var students = ["Jane", "Harvey", "Patrick"]; - -var len = students.length; - -if (len !== 1) { - if (len !== 0) { - console.log("The students are: ", students); - } else { - console.log("There are no students"); - } -} else { - var student1 = students[0]; - console.log("There's a single student here: " + student1); -} -``` - - - -### Match on List - -Pattern matching on list is similar to array, but with the extra feature of extracting the tail of a list (all elements except the first one): - - - -```res example -let rec printStudents = (students) => { - switch students { - | list{} => () // done - | list{student} => Console.log("Last student: " ++ student) - | list{student1, ...otherStudents} => - Console.log(student1) - printStudents(otherStudents) - } -} -printStudents(list{"Jane", "Harvey", "Patrick"}) -``` -```js -function printStudents(_students) { - while(true) { - var students = _students; - if (!students) { - return; - } - var otherStudents = students.tl; - var student = students.hd; - if (otherStudents) { - console.log(student); - _students = otherStudents; - continue; - } - console.log("Last student: " + student); - return; - }; -} - -printStudents({ - hd: "Jane", - tl: { - hd: "Harvey", - tl: { - hd: "Patrick", - tl: /* [] */0 - } - } -}); -``` - - - - -### Small Pitfall - -**Note**: you can only pass literals (i.e. concrete values) as a pattern, not let-binding names or other things. The following doesn't work as expected: - - - -```res example -let coordinates = (10, 20, 30) -let centerY = 20 -switch coordinates { -| (x, _centerY, _) => Console.log(x) -} -``` -```js -var coordinates = [10, 20, 30]; -var centerY = 20; - -console.log(10); -``` - - - -A first time ReScript user might accidentally write that code, assuming that it's matching on `coordinates` when the second value is of the same value as `centerY`. In reality, this is interpreted as matching on coordinates and assigning the second value of the tuple to the name `centerY`, which isn't what's intended. - -## Exhaustiveness Check - -As if the above features aren't enough, ReScript also provides arguably the most important pattern matching feature: **compile-time check of missing patterns**. - -Let's revisit one of the above examples: - - - -```res -let message = switch person1 { -| Teacher({name: "Mary" | "Joe"}) => - `Hey, still going to the party on Saturday?` -| Student({name, reportCard: {passing: true, gpa}}) => - `Congrats ${name}, nice GPA of ${Float.toString(gpa)} you got there!` -| Student({ - reportCard: {gpa: 0.0}, - status: Vacations(daysLeft) | Sabbatical(daysLeft) - }) => - `Come back in ${Int.toString(daysLeft)} days!` -| Student({status: Sick}) => - `How are you feeling?` -| Student({name}) => - `Good luck next semester ${name}!` -} -``` -```js -if (person1.TAG) { - var match$1 = person1.status; - var name = person1.name; - var match$2 = person1.reportCard; - if (match$2.passing) { - "Congrats " + name + ", nice GPA of " + match$2.gpa.toString() + " you got there!"; - } else if (typeof match$1 === "number") { - if (match$1 !== 0) { - "Good luck next semester " + name + "!"; - } else { - "How are you feeling?"; - } - } else if (person1.reportCard.gpa !== 0.0) { - "Good luck next semester " + name + "!"; - } else { - "Come back in " + match$1._0.toString() + " days!"; - } -} else { - switch (person1.name) { - case "Joe": - case "Mary": - break; - default: - throw { - RE_EXN_ID: "Match_failure", - _1: [ - "playground.res", - 13, - 0 - ], - Error: new Error() - }; - } -} -``` - - - -Did you see what we removed? This time, we've omitted the handling of the case where `person1` is `Teacher({name})` when `name` isn't Mary or Joe. - -Failing to handle every scenario of a value likely constitutes the majority of program bugs out there. This happens very often when you refactor a piece of code someone else wrote. Fortunately for ReScript, the compiler will tell you so: - -``` -Warning 8: this pattern-matching is not exhaustive. -Here is an example of a value that is not matched: -Some({name: ""}) -``` - -**BAM**! You've just erased an entire category of important bugs before you even ran the code. In fact, this is how most of nullable values is handled: - - - -```res example -let myNullableValue = Some(5) - -switch myNullableValue { -| Some(_v) => Console.log("value is present") -| None => Console.log("value is absent") -} -``` -```js -var myNullableValue = 5; - -if (myNullableValue !== undefined) { - console.log("value is present"); -} else { - console.log("value is absent"); -} -``` - - - -If you don't handle the `None` case, the compiler warns. No more `undefined` bugs in your code! - -## Conclusion & Tips & Tricks - -Hopefully you can see how pattern matching is a game changer for writing correct code, through the concise destructuring syntax, the proper conditions handling of `switch`, and the static exhaustiveness check. - -Below is some advice: - -Avoid using the wildcard `_` unnecessarily. Using the wildcard `_` will bypass the compiler's exhaustiveness check. Consequently, the compiler will not be able to notify you of probable errors when you add a new case to a variant. Try only using `_` against infinite possibilities, e.g. string, int, etc. - -Use the `if` clause sparingly. - -**Flatten your pattern-match whenever you can**. This is a real bug remover. Here's a series of examples, from worst to best: - - - -```res example -let optionBoolToBool = opt => { - if opt == None { - false - } else if opt === Some(true) { - true - } else { - false - } -} -``` -```js -function optionBoolToBool(opt) { - if (opt === undefined) { - return false; - } else { - return opt === true; - } -} -``` - - - -Now that's just silly =). Let's turn it into pattern-matching: - - - -```res example -let optionBoolToBool = opt => { - switch opt { - | None => false - | Some(a) => a ? true : false - } -} -``` -```js -function optionBoolToBool(opt) { - if (opt !== undefined && opt) { - return true; - } else { - return false; - } -} -``` - - - -Slightly better, but still nested. Pattern-matching allows you to do this: - - - -```res example -let optionBoolToBool = opt => { - switch opt { - | None => false - | Some(true) => true - | Some(false) => false - } -} -``` -```js -function optionBoolToBool(opt) { - if (opt !== undefined && opt) { - return true; - } else { - return false; - } -} -``` - - - -Much more linear-looking! Now, you might be tempted to do this: - - - -```res example -let optionBoolToBool = opt => { - switch opt { - | Some(true) => true - | _ => false - } -} -``` -```js -function optionBoolToBool(opt) { - if (opt !== undefined && opt) { - return true; - } else { - return false; - } -} -``` - - - -Which is much more concise, but kills the exhaustiveness check mentioned above; refrain from using that. This is the best: - - - -```res example -let optionBoolToBool = opt => { - switch opt { - | Some(trueOrFalse) => trueOrFalse - | None => false - } -} -``` -```js -function optionBoolToBool(opt) { - if (opt !== undefined) { - return opt; - } else { - return false; - } -} -``` - - - -Pretty darn hard to make a mistake in this code at this point! Whenever you'd like to use an if-else with many branches, prefer pattern matching instead. It's more concise and [performant](variant#design-decisions) too. diff --git a/pages/docs/manual/latest/pipe.mdx b/pages/docs/manual/latest/pipe.mdx deleted file mode 100644 index 32c45277a..000000000 --- a/pages/docs/manual/latest/pipe.mdx +++ /dev/null @@ -1,244 +0,0 @@ ---- -title: "Pipe" -description: "The Pipe operator (->)" -canonical: "/docs/manual/latest/pipe" ---- - -# Pipe - -ReScript provides a tiny but surprisingly useful operator `->`, called the "pipe", that allows you to "flip" your code inside-out. `a(b)` becomes `b->a`. It's a simple piece of syntax that doesn't have any runtime cost. - -Why would you use it? Imagine you have the following: - - - -```res -validateAge(getAge(parseData(person))) -``` -```js -validateAge(getAge(parseData(person))); -``` - - - -This is slightly hard to read, since you need to read the code from the innermost part, to the outer parts. Use pipe to streamline it: - - - -```res -person - ->parseData - ->getAge - ->validateAge -``` -```js -validateAge(getAge(parseData(person))); -``` - - - -Basically, `parseData(person)` is transformed into `person->parseData`, and `getAge(person->parseData)` is transformed into `person->parseData->getAge`, etc. - -**This works when the function takes more than one argument too**. - - - -```res -a(one, two, three) -``` -```js -a(one, two, three); -``` - - - -is the same as - - - -```res -one->a(two, three) -``` -```js -a(one, two, three); -``` - - - -This also works with labeled arguments. - -Pipes are used to emulate object-oriented programming. For example, `myStudent.getName` in other languages like Java would be `myStudent->getName` in ReScript (equivalent to `getName(myStudent)`). This allows us to have the readability of OOP without the downside of dragging in a huge class system just to call a function on a piece of data. - -## Tips & Tricks - -Do **not** abuse pipes; they're a means to an end. Inexperienced engineers sometimes shape a library's API to take advantage of the pipe. This is backwards. - -## JS Method Chaining - -_This section requires understanding of [our binding API](bind-to-js-function.md#object-method)_. - -JavaScript's APIs are often attached to objects, and are often chainable, like so: - -```js -const result = [1, 2, 3].map(a => a + 1).filter(a => a % 2 === 0); - -asyncRequest() - .setWaitDuration(4000) - .send(); -``` - -Assuming we don't need the chaining behavior above, we'd bind to each case of this using [`@send`](/syntax-lookup#send-decorator) from the aforementioned binding API page: - - - -```res prelude -type request -@val external asyncRequest: unit => request = "asyncRequest" -@send external setWaitDuration: (request, int) => request = "setWaitDuration" -@send external send: request => unit = "send" -``` -```js -// Empty output -``` - - - -You'd use them like this: - - - -```res example -let result = Array.filter( - Array.map([1, 2, 3], a => a + 1), - a => mod(a, 2) == 0 -) - -send(setWaitDuration(asyncRequest(), 4000)) -``` -```js -var result = [1, 2, 3].map(function(a) { - return a + 1 | 0; -}).filter(function(a) { - return a % 2 === 0; -}); - -asyncRequest().setWaitDuration(4000).send(); -``` - - - -This looks much worse than the JS counterpart! Clean it up visually with pipe: - - - -```res example -let result = [1, 2, 3] - ->Array.map(a => a + 1) - ->Array.filter(a => mod(a, 2) == 0) - -asyncRequest()->setWaitDuration(4000)->send -``` -```js -var result = [1, 2, 3].map(function(a) { - return a + 1 | 0; -}).filter(function(a) { - return a % 2 === 0; -}); - -asyncRequest().setWaitDuration(4000).send(); -``` - - - -## Pipe Into Variants - -You can pipe into a variant's constructor as if it was a function: - - - -```res -let result = name->preprocess->Some -``` -```js -var result = preprocess(name); -``` - - - -We turn this into: - - - -```res -let result = Some(preprocess(name)) -``` -```js -var result = preprocess(name); -``` - - - -**Note** that using a variant constructor as a function wouldn't work anywhere else beside here. - -## Pipe Placeholders - -A placeholder is written as an underscore and it tells ReScript that you want to fill in an argument of a function later. These two have equivalent meaning: - -```res -let addTo7 = (x) => add3(3, x, 4) -let addTo7 = add3(3, _, 4) -``` - -Sometimes you don't want to pipe the value you have into the first position. In these cases you can mark a placeholder value to show which argument you would like to pipe into. - -Let's say you have a function `namePerson`, which takes a `person` then a `name` argument. If you are transforming a person then pipe will work as-is: - - - -```res -makePerson(~age=47) - ->namePerson("Jane") -``` -```js -namePerson(makePerson(47), "Jane"); -``` - - - -If you have a name that you want to apply to a person object, you can use a placeholder: - - - -```res -getName(input) - ->namePerson(personDetails, _) -``` -```js -var __x = getName(input); -namePerson(personDetails, __x); -``` - - - -This allows you to pipe into any positional argument. It also works for named arguments: - - - -```res -getName(input) - ->namePerson(~person=personDetails, ~name=_) -``` -```js -var __x = getName(input); -namePerson(personDetails, __x); -``` - - - -## Triangle Pipe (Deprecated) - -You might see usages of another pipe, `|>`, in some codebases. These are deprecated. - -Unlike `->` pipe, the `|>` pipe puts the subject as the last (not first) argument of the function. `a |> f(b)` turns into `f(b, a)`. - -For a more thorough discussion on the rationale and differences between the two operators, please refer to the [Data-first and Data-last comparison by Javier Chávarri](https://www.javierchavarri.com/data-first-and-data-last-a-comparison/) diff --git a/pages/docs/manual/latest/polymorphic-variant.mdx b/pages/docs/manual/latest/polymorphic-variant.mdx deleted file mode 100644 index 084d4cf59..000000000 --- a/pages/docs/manual/latest/polymorphic-variant.mdx +++ /dev/null @@ -1,433 +0,0 @@ ---- -title: "Polymorphic Variant" -description: "The Polymorphic Variant data structure in ReScript" -canonical: "/docs/manual/latest/polymorphic-variant" ---- - -# Polymorphic Variant - -Polymorphic variants (or poly variant) are a cousin of [variant](variant). With these differences: - -- They start with a `#` and the constructor name doesn't need to be capitalized. -- They don't require an explicit type definition. The type is inferred from usage. -- Values of different poly variant types can share the constructors they have in common (aka, poly variants are "structurally" typed, as opposed to ["nominally" typed](variant#variant-types-are-found-by-field-name)). - -They're a convenient and useful alternative to regular variants, but should **not** be abused. See the drawbacks at the end of this page. - -## Creation - -We provide 3 syntaxes for a poly variant's constructor: - - - -```res -let myColor = #red -let myLabel = #"aria-hidden" -let myNumber = #7 -``` - -```js -var myColor = "red"; -var myLabel = "aria-hidden"; -var myNumber = 7; -``` - - - -**Take a look at the output**. Poly variants are _great_ for JavaScript interop. For example, you can use it to model JavaScript string and number enums like TypeScript, but without confusing their accidental usage with regular strings and numbers. - -`myColor` uses the common syntax. The second and third syntaxes are to support expressing strings and numbers more conveniently. We allow the second one because otherwise it'd be invalid syntax since symbols like `-` and others are usually reserved. - -## Type Declaration - -Although **optional**, you can still pre-declare a poly variant type: - -```res -// Note the surrounding square brackets, and # for constructors -type color = [#red | #green | #blue] -``` - -These types can also be inlined, unlike for regular variant: - - - -```res -let render = (myColor: [#red | #green | #blue]) => { - switch myColor { - | #blue => Console.log("Hello blue!") - | #red - | #green => Console.log("Hello other colors") - } -} -``` - -```js -function render(myColor) { - if (myColor === "green" || myColor === "red") { - console.log("Hello other colors"); - } else { - console.log("Hello blue!"); - } -} -``` - - - -**Note**: because a poly variant value's type definition is **inferred** and not searched in the scope, the following snippet won't error: - - - -```res -type color = [#red | #green | #blue] - -let render = myColor => { - switch myColor { - | #blue => Console.log("Hello blue!") - | #green => Console.log("Hello green!") - // works! - | #yellow => Console.log("Hello yellow!") - } -} -``` - -```js -function render(myColor) { - if (myColor === "yellow") { - console.log("Hello yellow!"); - } else if (myColor === "green") { - console.log("Hello green!"); - } else { - console.log("Hello blue!"); - } -} -``` - - - -That `myColor` parameter's type is inferred to be `#red`, `#green` or `#yellow`, and is unrelated to the `color` type. If you intended `myColor` to be of type `color`, annotate it as `myColor: color` in any of the places. - -## Constructor Arguments - -This is similar to a regular variant's [constructor arguments](variant#constructor-arguments): - - - -```res -type account = [ - | #Anonymous - | #Instagram(string) - | #Facebook(string, int) -] - -let me: account = #Instagram("Jenny") -let him: account = #Facebook("Josh", 26) -``` - -```js -var me = { - NAME: "Instagram", - VAL: "Jenny" -}; - -var him = { - NAME: "Facebook", - VAL: [ - "Josh", - 26 - ] -}; -``` - - - -### Combine Types and Pattern Match - -You can use poly variant types within other poly variant types to create a sum of all constructors: - - - -```res -type red = [#Ruby | #Redwood | #Rust] -type blue = [#Sapphire | #Neon | #Navy] - -// Contains all constructors of red and blue. -// Also adds #Papayawhip -type color = [red | blue | #Papayawhip] - -let myColor: color = #Ruby -``` - -```js -var myColor = "Ruby"; -``` - - - -There's also some special [pattern matching](./pattern-matching-destructuring) syntax to match on constructors defined in a specific poly variant type: - - - -```res -// Continuing the previous example above... - -switch myColor { -| #...blue => Console.log("This blue-ish") -| #...red => Console.log("This red-ish") -| other => Console.log2("Other color than red and blue: ", other) -} -``` - -```js -var other = myColor; - -if (other === "Neon" || other === "Navy" || other === "Sapphire") { - console.log("This is blue-ish"); -} else if (other === "Rust" || other === "Ruby" || other === "Redwood") { - console.log("This is red-ish"); -} else { - console.log("Other color than red and blue: ", other); -} -``` - - - -This is a shorter version of: - -```res -switch myColor { -| #Sapphire | #Neon | #Navy => Console.log("This is blue-ish") -| #Ruby | #Redwood | #Rust => Console.log("This is red-ish") -| other => Console.log2("Other color than red and blue: ", other) -} -``` - -## Structural Sharing - -Since poly variants value don't have a source of truth for their type, you can write such code: - - - -```res -type preferredColors = [#white | #blue] - -let myColor: preferredColors = #blue - -let displayColor = v => { - switch v { - | #red => "Hello red" - | #green => "Hello green" - | #white => "Hey white!" - | #blue => "Hey blue!" - } -} - -Console.log(displayColor(myColor)) -``` - -```js -var myColor = "blue"; - -function displayColor(v) { - if (v === "white") { - return "Hey white!"; - } else if (v === "red") { - return "Hello red"; - } else if (v === "green") { - return "Hello green"; - } else { - return "Hey blue!"; - } -} - -console.log(displayColor("blue")); -``` - - - -With a regular variant, the line `displayColor(myColor)` would fail, since it'd complain that the type of `myColor` doesn't match the type of `v`. No problem with poly variant. - -## JavaScript Output - -Poly variants are great for JavaScript interop! You can share their values to JS code, or model incoming JS values as poly variants. - -- `#red` and `#"I am red 😃"` compile to JavaScipt `"red"` and `"I am red 😃"`. -- `#1` compiles to JavaScript `1`. -- Poly variant constructor with 1 argument, like `Instagram("Jenny")` compile to a straightforward `{NAME: "Instagram", VAL: "Jenny"}`. 2 or more arguments like `#Facebook("Josh", 26)` compile to a similar object, but with `VAL` being an array of the arguments. - -### Bind to Functions - -For example, let's assume we want to bind to `Intl.NumberFormat` and want to make sure that our users only pass valid locales, we could define an external binding like this: - - - -```res -type t - -@scope("Intl") @val -external makeNumberFormat: ([#"de-DE" | #"en-GB" | #"en-US"]) => t = "NumberFormat" - -let intl = makeNumberFormat(#"de-DE") -``` - -```js -var intl = Intl.NumberFormat("de-DE"); -``` - - - -The JS output is identical to handwritten JS, but we also get to enjoy type errors if we accidentally write `makeNumberFormat(#"de-DR")`. - -More advanced usage examples for poly variant interop can be found in [Bind to JS Function](bind-to-js-function#constrain-arguments-better). - -### Bind to String Enums - -Let's assume we have a TypeScript module that expresses following enum export: - -```js -// direction.js -enum Direction { - Up = "UP", - Down = "DOWN", - Left = "LEFT", - Right = "RIGHT", -} - -export const myDirection = Direction.Up -``` - -For this particular example, we can also inline poly variant type definitions to design the type for the imported `myDirection` value: - - - - -```res -type direction = [ #UP | #DOWN | #LEFT | #RIGHT ] -@module("./direction.js") external myDirection: direction = "myDirection" -``` - -```js -var DirectionJs = require("./direction.js"); - -var myDirection = DirectionJs.myDirection; -``` - - - -Again: since we were using poly variants, the JS Output is practically zero-cost and doesn't add any extra code! - -## Extra Constraints on Types - -The previous poly variant type annotations we've looked at are the regular "closed" kind. However, there's a way to express "I want at least these constructors" (lower bound) and "I want at most these constructors" (upper bound): - -```res -// Only #Red allowed. Closed. -let basic: [#Red] = #Red - -// May contain #Red, or any other value. Open -// here, foreground will actually be inferred as [> #Red | #Green] -let foreground: [> #Red] = #Green - -// The value must be, at most, one of #Red or #Blue -// Only #Red and #Blue are valid values -let background: [< #Red | #Blue] = #Red -``` - -**Note:** We added this info for educational purposes. In most cases you will not want to use any of this stuff, since it makes your APIs pretty unreadable / hard to use. - -### Closed `[` - -This is the simplest poly variant definition, and also the most practical one. Like a common variant type, this one defines an exact set of constructors. - -```res -type rgb = [ #Red | #Green | #Blue ] - -let color: rgb = #Green -``` - -In the example above, `color` will only allow one of the three constructors that are defined in the `rgb` type. This is usually the way how poly variants should be defined. - -In case you want to define a type that is extensible, you'll need to use the lower / upper bound syntax. - -### Lower Bound `[>` - -A lower bound defines the minimum set of constructors a poly variant type is aware of. It is also considered an "open poly variant type", because it doesn't restrict any additional values. - -Here is an example on how to make a minimum set of `basicBlueTones` extensible for a new `color` type: - -```res -type basicBlueTone<'a> = [> #Blue | #DeepBlue | #LightBlue ] as 'a -type color = basicBlueTone<[#Blue | #DeepBlue | #LightBlue | #Purple]> - -let color: color = #Purple - -// This will fail due to missing minimum constructors: -type notWorking = basicBlueTone<[#Purple]> -``` - -Here, the compiler will enforce the user to define `#Blue | #DeepBlue | #LightBlue` as the minimum set of constructors when trying to extend `basicBlueTone<'a>`. - -**Note:** Since we want to define an extensible poly variant, we need to provide a type placeholder `<'a>`, and also add `as 'a` after the poly variant declaration, which essentially means: "Given type `'a` is constraint to the minimum set of constructors (`#Blue | #DeepBlue | #LightBlue`) defined in `basicBlueTone`". - -### Upper Bound `[<` - -The upper bound works in the opposite way than a lower bound: the extending type may only use constructors that are stated in the upper bound constraint. - -Here another example, but with red colors: - -```res -type validRed<'a> = [< #Fire | #Crimson | #Ash] as 'a -type myReds = validRed<[#Ash]> - -// This will fail due to unlisted constructor not defined by the lower bound -type notWorking = validRed<[#Purple]> -``` - -## Coercion - -You can convert a poly variant to a `string` or `int` at no cost: - - - -```res -type company = [#Apple | #Facebook] -let theCompany: company = #Apple - -let message = "Hello " ++ (theCompany :> string) -``` - -```js -var theCompany = "Apple"; -var message = "Hello " + theCompany; -``` - - - -**Note**: for the coercion to work, the poly variant type needs to be closed; you'd need to annotate it, since otherwise, `theCompany` would be inferred as `[> #Apple]`. - -## Tips & Tricks - -### Variant vs Polymorphic Variant - -One might think that polymorphic variants are superior to regular [variants](./variant). As always, there are trade-offs: - -- Due to their "structural" nature, poly variant's type errors might be more confusing. If you accidentally write `#blur` instead of `#blue`, ReScript will still error but can't indicate the correct source as easily. Regular variants' source of truth is the type definition, so the error can't go wrong. -- It's also harder to refactor poly variants. Consider this: - ```res - let myFruit = #Apple - let mySecondFruit = #Apple - let myCompany = #Apple - ``` - Refactoring the first one to `#Orange` doesn't mean we should refactor the third one. Therefore, the editor plugin can't touch the second one either. Regular variant doesn't have such problem, as these 2 values presumably come from different variant type definitions. - -- You might lose some nice pattern match checks from the compiler: - ```res - let myColor = #red - - switch myColor { - | #red => Console.log("Hello red!") - | #blue => Console.log("Hello blue!") - } - ``` - Because there's no poly variant definition, it's hard to know whether the `#blue` case can be safely removed. - -In most scenarios, we'd recommend to use regular variants over polymorphic variants, especially when you are writing plain ReScript code. In case you want to write zero-cost interop bindings or generate clean JS output, poly variants are oftentimes a better option. diff --git a/pages/docs/manual/latest/primitive-types.mdx b/pages/docs/manual/latest/primitive-types.mdx deleted file mode 100644 index 7c5843bb7..000000000 --- a/pages/docs/manual/latest/primitive-types.mdx +++ /dev/null @@ -1,272 +0,0 @@ ---- -title: "Primitive Types" -description: "Primitive Data Types in ReScript" -canonical: "/docs/manual/latest/primitive-types" ---- - -# Primitive Types - -ReScript comes with the familiar primitive types like `string`, `int`, `float`, etc. - - - -## String - -ReScript `string`s are delimited using **double** quotes (single quotes are reserved for the character type below). - - - -```res example -let greeting = "Hello world!" -let multilineGreeting = "Hello - world!" -``` -```js -var greeting = "Hello world!"; -var multilineGreeting = "Hello\n world!"; -``` - - - -To concatenate strings, use `++`: - - - -```res example -let greetings = "Hello " ++ "world!" -``` -```js -var greetings = "Hello world!"; -``` - - - -### String Interpolation - -There's a special syntax for string that allows - -- multiline string just like before -- no special character escaping -- Interpolation - - - -```res example -let name = "Joe" - -let greeting = `Hello -World -👋 -${name} -` -``` -```js -var name = "Joe"; - -var greeting = "Hello\nWorld\n👋\n" + name + "\n"; -``` - - - -This is just like JavaScript's backtick string interpolation, except without needing to escape special characters. - -### Usage - -See the familiar `String` API in the [API docs](api/core/string). Since a ReScript string maps to a JavaScript string, you can mix & match the string operations in all standard libraries. - -### Tips & Tricks - -**You have a good type system now!** In an untyped language, you'd often overload the meaning of string by using it as: - -- a unique id: `var BLUE_COLOR = "blue"` -- an identifier into a data structure: `var BLUE = "blue" var RED = "red" var colors = [BLUE, RED]` -- the name of an object field: `person["age"] = 24` -- an enum: `if (audio.canPlayType() === 'probably') {...}` [(ಠ_ಠ)](https://developer.mozilla.org/en-US/docs/Web/API/HTMLMediaElement/canPlayType#Return_value) -- other crazy patterns you'll soon find horrible, after getting used to ReScript's alternatives. - -The more you overload the poor string type, the less the type system (or a teammate) can help you! ReScript provides concise, fast and maintainable types & data structures alternatives to the use-cases above (e.g. variants, in a later section). - -## Char - -ReScript has a type for a string with a single letter: - - - -```res example -let firstLetterOfAlphabet = 'a' -``` -```js -var firstLetterOfAlphabet = /* "a" */97; -``` - - - -**Note**: Char doesn't support Unicode or UTF-8 and is therefore not recommended. - -To convert a String to a Char, use `String.get("a", 0)`. To convert a Char to a String, use `String.make(1, 'a')`. - -## Regular Expression - -ReScript regular expressions compile cleanly to their JavaScript counterpart: - - - -```res example -let r = %re("/b/g") -``` -```js -var r = /b/g; -``` - - - -A regular expression like the above has the type `Re.t`. The [RegExp](api/core/regexp) module contains the regular expression helpers you have seen in JS. - -## Boolean - -A ReScript boolean has the type `bool` and can be either `true` or `false`. Common operations: - -- `&&`: logical and. -- `||`: logical or. -- `!`: logical not. -- `<=`, `>=`, `<`, `>` -- `==`: structural equal, compares data structures deeply. `(1, 2) == (1, 2)` is `true`. Convenient, but use with caution. -- `===`: referential equal, compares shallowly. `(1, 2) === (1, 2)` is `false`. `let myTuple = (1, 2); myTuple === myTuple` is `true`. -- `!=`: structural unequal. -- `!==`: referential unequal. - -ReScript's `true/false` compiles into a JavaScript `true/false`. - -## Integers - -32-bits, truncated when necessary. We provide the usual operations on them: `+`, `-`, `*`, `/`, etc. See [Int](api/core/int) for helper functions. - -**Be careful when you bind to JavaScript numbers!** Since ReScript integers have a much smaller range than JavaScript numbers, data might get lost when dealing with large numbers. In those cases it’s much safer to bind the numbers as **float**. Be extra mindful of this when binding to JavaScript Dates and their epoch time. - -To improve readability, you may place underscores in the middle of numeric literals such as `1_000_000`. Note that underscores can be placed anywhere within a number, not just every three digits. - -## Floats - -Float requires other operators: `+.`, `-.`, `*.`, `/.`, etc. Like `0.5 +. 0.6`. See [Float](api/core/float) for helper functions. - -As with integers, you may use underscores within literals to improve readability. - -### Int-to-Float Coercion - -`int` values can be coerced to `float` with the `:>` (type coercion) operator. - - - -```res example -let result = (1 :> float) +. 2. -``` -```js -var result = 1 + 2; -``` - - - -## Big Integers (experimental) - -**Since 11.1** - -For values which are too large to be represented by Int or Float, there is the `bigint` primitive type. -We provide the usual operations on them: `+`, `-`, `*`, `/`, etc. See [BigInt](api/core/bigint) for helper functions. - -A `bigint` number is denoted by a trailing `n` like so: `42n`. - -As `bigint` is a different data type than `int`, it's necessary to open the corresponding module to overload the operators. - - - -```res example -open! BigInt - -let a = 9007199254740991n + 9007199254740991n -let b = 2n ** 2n -``` -```js -var a = 9007199254740991n + 9007199254740991n; - -var p = 2n ** 2n; -``` - - - -It also supports all the bitwise operations, except unsigned shift right (`>>>`), which is not supported by JS itself for `bigint`s. - - - -```res example -open! BigInt - -let a = land(1n, 1n) -let b = lor(1n, 1n) -let c = lxor(1n, 1n) -let d = lnot(1n) -let e = lsl(1n, 1n) -let f = asr(1n, 1n) -``` -```js -var Js_bigint = require("./stdlib/js_bigint.js"); - -var a = 1n & 1n; - -var b = 1n | 1n; - -var c = 1n ^ 1n; - -var d = Js_bigint.lnot(1n); - -var e = (1n << 1n); - -var f = (1n >> 1n); -``` - - - -It can also be pattern-matched. - - - -```res example -let bigintValue = 1n - -switch bigintValue { -| 1n => Console.log("Small bigint") -| 100n => Console.log("Larger bigint") -| _ => Console.log("Other bigint") -} -``` -```js -if (1n !== 1n) { - if (1n !== 100n) { - console.log("Other bigint"); - } else { - console.log("Larger bigint"); - } -} else { - console.log("Small bigint"); -} - -var bigintValue = 1n; -``` - - - - -## Unit - -The `unit` type indicates the absence of a specific value. It has only a single value, `()`, which acts as a placeholder when no other value exists or is needed. It compiles to JavaScript's `undefined` and resembles the `void` type in languages such as C++. What's the point of such a type? - -Consider the `Math.random` function. Its type signature is `unit => float`, which means it receives a `unit` as input and calculates a random `float` as output. You use the function like this - `let x = Math.random()`. Notice `()` as the first and only function argument. - -Imagine a simplified `Console.log` function that prints a message. Its type signature is `string => unit` and you'd use it like this `Console.log("Hello!")`. It takes a string as input, prints it, and then returns nothing useful. When `unit` is the output of a function it means the function performs some kind of side-effect. - -## Unknown - -The `unknown` type represents values with contents that are a mystery or are not 100% guaranteed to be what you think they are. It provides type-safety when interacting with data received from an untrusted source. For example, suppose an external function is supposed to return a `string`. It might. But if the documentation is not accurate or the code has bugs, the function could return `null`, an `array`, or something else you weren't expecting. - -The ReScript type system helps you avoid run-time crashes and unpredicatable behavior by preventing you from using `unknown` in places that expect a `string` or `int` or some other type. The ReScript core libraries also provide utility functions to help you inspect `unknown` values and access their contents. In some cases you may need a JSON parsing library to convert `unknown` values to types you can safely use. - -Consider using `unknown` when receiving data from [external JavaScript functions](/docs/manual/latest/bind-to-js-function) diff --git a/pages/docs/manual/latest/project-structure.mdx b/pages/docs/manual/latest/project-structure.mdx deleted file mode 100644 index 93e77f0f0..000000000 --- a/pages/docs/manual/latest/project-structure.mdx +++ /dev/null @@ -1,57 +0,0 @@ ---- -title: "Project Structure" -description: "Notes on project structure and other rough ReScript guidelines" -canonical: "/docs/manual/latest/project-structure" ---- - -# Project Structure - -These are the existing, non-codified community practices that are currently propagated through informal agreement. We might remove some of them at one point, and enforce some others. Right now, they're just recommendations for ease of newcomers. - -## File Casing - -Capitalized file names (aka first letter upper-cased). - -**Justification**: Module names can only be capitalized. Newcomers often ask how a file maps to a module, and why `draw.res` maps to the module `Draw`, and sometimes try to refer to a module through uncapitalized identifiers. Using `Draw.res` makes this mapping more straightforward. It also helps certain file names that'd be awkward in uncapitalized form: `uRI.res`. - -## Ignore `.merlin` File - -This is generated by the build system and you should not have to manually edit it. Don't check it into the repo. - -**Justification**: `.merlin` is for editor tooling. The file contains absolute paths, which are also not cross-platform (e.g. Windows paths are different). - -## Folders - -Try not to have too many nested folders. Keep your project flat, and have fewer files (reminder: you can use nested modules). - -**Justification**: The file system is a _tree_, but your code's dependencies are a _graph_. Because of that, any file & folder organization is usually imperfect. While it's still valuable to group related files together in a folder, the time wasted debating & getting decision paralysis over these far outweight their benefits. We'll always recommend you to Get Work Done instead of debating about these issues. - -## Third-party Dependencies - -Keep them to a minimum. - -**Justification**: A compiled, statically typed language cannot model its dependencies easily by muddling along like in a dynamic language, especially when we're still piggy-backing on NPM/Yarn (to reduce learning overhead in the medium-term). Keeping dependencies simple & lean helps reduce possibility of conflicts (e.g. two diamond dependencies, or clashing interfaces). - -## Documentation - -Have them. Spend more effort making them great (examples, pitfalls) and professional rather than _just_ fancy-looking. Do use examples, and avoid using names such as `foo` and `bar`. There's always more concrete names (it's an example, no need to be abstract/generalized just yet. The API docs will do this plentily). For blog posts, don't repeat the docs themselves, describe the _transition_ from old to new, and why (e.g. "it was a component, now it's a function, because ..."). - -**Justification**: It's hard for newcomers to distinguish between a simple/decent library and one that's fancy-looking. For the sake of the community, don't try too hard to one-up each other's libraries. Do spread the words, but use your judgement too. - -## PPX & Other Meta-tools - -Keep them to a minimum. PPX, unless used in renown cases (printer, accessors and serializer/deserializer generation), can cause big learning churn for newcomers; on top of the syntax, semantics, types, build tool & FFI that they already have to learn, learning per-library custom transformations of the code is an extra step. More invasive macros makes the code itself less semantically meaningful too, since the essence would be hiding somewhere else. - -## Paradigm - -Don't abuse overly fancy features. Do leave some breathing room for future APIs but don't over-architect things. - -**Justification**: Simple code helps newcomers understand and potentially contribute to your code. Contributing is the best way for them to learn. The extra help you receive might also surpass the gain of using a slightly more clever language trick. But do try new language tricks in some of more casual projects! You might discover new ways of architecting code. - -## Publishing - -If it's a wrapper for a JS library, don't publish the JS artifacts. If it's a legit library, publish the artifacts in lib/js if you think JS consumers might use it. This is especially the case when you gradually convert a JS lib to ReScript while not breaking existing JS consumers. - -Do put the keywords `"rescript"` in your package.json `keywords` field. This allows us to find the library much more easily for future purposes. - -**Justification**: Be nice to JS consumers of your library. They're your future ReScripters. diff --git a/pages/docs/manual/latest/promise.mdx b/pages/docs/manual/latest/promise.mdx deleted file mode 100644 index 16a2b891e..000000000 --- a/pages/docs/manual/latest/promise.mdx +++ /dev/null @@ -1,182 +0,0 @@ ---- -title: "Promises" -description: "JS Promise handling in ReScript" -canonical: "/docs/manual/latest/promise" ---- - -# Promise - -> **Note:** Starting from ReScript 10.1 and above, we recommend using [async / await](./async-await) when interacting with Promises. - -## `promise` type - -**Since 10.1** - -In ReScript, every JS promise is represented with the globally available `promise<'a>` type. For ReScript versions < 10.1, use its original alias `Js.Promise.t<'a>` instead. - -Here's a usage example in a function signature: - -```resi -// User.resi file - -type user = {name: string} - -let fetchUser: string => promise -``` - -To work with promise values (instead of using `async` / `await`) you may want to use the built-in `Promise` module. - -## Promise - -A builtin module to create, chain and manipulate promises. - -### Creating a promise - -```res -let p1 = Promise.make((resolve, reject) => { - // We use uncurried functions for resolve / reject - // for cleaner JS output without unintended curry calls - resolve("hello world") -}) - -let p2 = Promise.resolve("some value") - -// You can only reject `exn` values for streamlined catch handling -exception MyOwnError(string) -let p3 = Promise.reject(MyOwnError("some rejection")) -``` - -### Access the contents and transform a promise - -```res -let logAsyncMessage = () => { - open Promise - Promise.resolve("hello world") - ->then(msg => { - // then callbacks require the result to be resolved explicitly - resolve("Message: " ++ msg) - }) - ->then(msg => { - Console.log(msg) - - // Even if there is no result, we need to use resolve() to return a promise - resolve() - }) - ->ignore // Requires ignoring due to unhandled return value -} -``` - -For comparison, the `async` / `await` version of the same code would look like this: - -```res -let logAsyncMessage = async () => { - let msg = await Promise.resolve("hello world") - Console.log(`Message: ${msg}`) -} -``` - -Needless to say, the async / await version offers better ergonomics and less opportunities to run into type issues. - -### Handling Rejected Promises - -You can handle a rejected promise using the [`Promise.catch()`](./api/core/promise#value-catch) method, which allows you to catch and manage errors effectively. - -### Run multiple promises in parallel - -In case you want to launch multiple promises in parallel, use `Promise.all`: - - - - -```res -@val -external fetchMessage: string => promise = "global.fetchMessage" - -let logAsyncMessage = async () => { - let messages = await Promise.all([fetchMessage("message1"), fetchMessage("message2")]) - - Console.log(messages->Array.joinWith(", ")) -} -``` - -```js -async function logAsyncMessage(param) { - var messages = await Promise.all([ - global.fetchMessage("message1"), - global.fetchMessage("message2") - ]); - console.log(messages.join(", ")); -} - -export { - logAsyncMessage , -} -``` - - - -## Js.Promise module (legacy - do not use) - -> **Note:** The `Js.Promise` bindings are following the outdated data-last convention from a few years ago. We kept those APIs for backwards compatibility. Either use [`Promise`](api/core/promise) or a third-party promise binding instead. - -ReScript has built-in support for [JavaScript promises](api/js/promise). The 3 functions you generally need are: - -- `Js.Promise.resolve: 'a => Js.Promise.t<'a>` -- `Js.Promise.then_: ('a => Js.Promise.t<'b>, Js.Promise.t<'a>) => Js.Promise.t<'b>` -- `Js.Promise.catch: (Js.Promise.error => Js.Promise.t<'a>, Js.Promise.t<'a>) => Js.Promise.t<'a>` - -Additionally, here's the type signature for creating a promise on the ReScript side: - -```res -Js.Promise.make: ( - ( - ~resolve: (. 'a) => unit, - ~reject: (. exn) => unit - ) => unit -) => Js.Promise.t<'a> -``` - -This type signature means that `make` takes a callback that takes 2 named arguments, `resolve` and `reject`. Both arguments are themselves [uncurried callbacks]( -function.md#uncurried-function) (with a dot). `make` returns the created promise. - -### Usage - -Using the [pipe operator](pipe.md): - - - -```res example -let myPromise = Js.Promise.make((~resolve, ~reject) => resolve(. 2)) - -myPromise->Js.Promise.then_(value => { - Console.log(value) - Js.Promise.resolve(value + 2) -}, _)->Js.Promise.then_(value => { - Console.log(value) - Js.Promise.resolve(value + 3) -}, _)->Js.Promise.catch(err => { - Console.log2("Failure!!", err) - Js.Promise.resolve(-2) -}, _) -``` -```js -var myPromise = new Promise(function (resolve, reject) { - return resolve(2); -}); - -myPromise - .then(function (value) { - console.log(value); - return Promise.resolve((value + 2) | 0); - }) - .then(function (value) { - console.log(value); - return Promise.resolve((value + 3) | 0); - }) - .catch(function (err) { - console.log("Failure!!", err); - return Promise.resolve(-2); - }); -``` - - diff --git a/pages/docs/manual/latest/record.mdx b/pages/docs/manual/latest/record.mdx deleted file mode 100644 index b251d7aa4..000000000 --- a/pages/docs/manual/latest/record.mdx +++ /dev/null @@ -1,557 +0,0 @@ ---- -title: "Record" -description: "Record types in ReScript" -canonical: "/docs/manual/latest/record" ---- - -# Record - -Records are like JavaScript objects but: - -- are immutable by default -- have fixed fields (not extensible) - -## Type Declaration - -A record needs a mandatory type declaration: - - - -```res prelude -type person = { - age: int, - name: string, -} -``` -```js -// Empty output -``` - - - -## Creation - -To create a `person` record (declared above): - - - -```res prelude -let me = { - age: 5, - name: "Big ReScript" -} -``` -```js -var me = { - age: 5, - name: "Big ReScript" -}; -``` - - - -When you create a new record value, ReScript tries to find a record type declaration that conforms to the shape of the value. So the `me` value here is inferred as of type `person`. - -The type is found by looking above the `me` value. **Note**: if the type instead resides in another file or module, you need to explicitly indicate which file or module it is: - - - -```res example -// School.res -type person = {age: int, name: string} -``` -```js -// Empty output -``` - - - - - -```res -// Example.res - -let me: School.person = {age: 20, name: "Big ReScript"} -/* or */ -let me2 = {School.age: 20, name: "Big ReScript"} -``` -```js -var me = { - age: 20, - name: "Big ReScript" -}; -var me2 = { - age: 20, - name: "Big ReScript" -}; -``` - - - -In both `me` and `me2` the record definition from `School` is found. The first one, `me` with the regular type annotation, is preferred. - -## Access - -Use the familiar dot notation: - - - -```res example -let name = me.name -``` -```js -var name = "Big ReScript"; -``` - - - -## Immutable Update - -New records can be created from old records with the `...` spread operator. The original record isn't mutated. - - - -```res example -let meNextYear = {...me, age: me.age + 1} -``` -```js -var meNextYear = { - age: 21, - name: "Big ReScript" -}; -``` - - - -**Note**: spread cannot add new fields to the record value, as a record's shape is fixed by its type. - -## Mutable Update - -Record fields can optionally be mutable. This allows you to efficiently update those fields in-place with the `=` operator. - - - -```res example -type person = { - name: string, - mutable age: int -} - -let baby = {name: "Baby ReScript", age: 5} -baby.age = baby.age + 1 // `baby.age` is now 6. Happy birthday! -``` -```js -var baby = { - name: "Baby ReScript", - age: 5 -}; - -baby.age = baby.age + 1 | 0; -``` - - - -Fields not marked with `mutable` in the type declaration cannot be mutated. - -## JavaScript Output - -ReScript records compile to straightforward JavaScript objects; see the various JS output tabs above. - -## Optional Record Fields -ReScript [`v10`](/blog/release-10-0-0#experimental-optional-record-fields) introduced optional record fields. This means that you can define fields that can be omitted when creating the record. It looks like this: - - - -```res example -type person = { - age: int, - name?: string -} -``` -```js -// Empty output -``` - - - -Notice how `name` has a suffixed `?`. That means that the field itself is _optional_. - -### Creation -You can omit any optional fields when creating a record. Not setting an optional field will default the field's value to `None`: - - - -```res example -type person = { - age: int, - name?: string -} - -let me = { - age: 5, - name: "Big ReScript" -} - -let friend = { - age: 7 -} -``` -```js -var me = { - age: 5, - name: "Big ReScript" -}; - -var friend = { - age: 7 -}; -``` - - - -This has consequences for pattern matching, which we'll expand a bit on soon. - -## Immutable Update -Updating an optional field via an immutable update above lets you set that field value without needing to care whether it's optional or not. - - - -```res example -type person = { - age: int, - name?: string -} - -let me = { - age: 123, - name: "Hello" -} - -let withoutName = { - ...me, - name: "New Name" -} -``` -```js -import * as Caml_obj from "./stdlib/caml_obj.js"; - -var me = { - age: 123, - name: "Hello" -}; - -var newrecord = Caml_obj.obj_dup(me); - -newrecord.name = "New Name"; - -var withoutName = newrecord; -``` - - - - -However, if you want to set the field to an optional value, you prefix that value with `?`: - - - -```res example -type person = { - age: int, - name?: string -} - -let me = { - age: 123, - name: "Hello" -} - -let maybeName = Some("My Name") - -let withoutName = { - ...me, - name: ?maybeName -} -``` -```js -import * as Caml_obj from "./stdlib/caml_obj.js"; - -var me = { - age: 123, - name: "Hello" -}; - -var maybeName = "My Name"; - -var newrecord = Caml_obj.obj_dup(me); - -newrecord.name = maybeName; - -var withoutName = newrecord; -``` - - - -You can unset an optional field's value via that same mechanism by setting it to `?None`. - -### Pattern Matching on Optional Fields -[Pattern matching](pattern-matching-destructuring), one of ReScript's most important features, has two caveats when you deal with optional fields. - -When matching on the value directly, it's an `option`. Example: - - - -```res -type person = { - age: int, - name?: string, -} - -let me = { - age: 123, - name: "Hello", -} - -let isRescript = switch me.name { -| Some("ReScript") => true -| Some(_) | None => false -} -``` -```js -var isRescript; - -isRescript = "Hello" === "ReScript" ? true : false; - -var me = { - age: 123, - name: "Hello" -}; -``` - - - -But, when matching on the field as part of the general record structure, it's treated as the underlying, non-optional value: - - - -```res -type person = { - age: int, - name?: string, -} - -let me = { - age: 123, - name: "Hello", -} - -let isRescript = switch me { -| {name: "ReScript"} => true -| _ => false -} - -``` -```js -var isRescript; - -isRescript = "Hello" === "ReScript" ? true : false; - -var me = { - age: 123, - name: "Hello" -}; -``` - - - -Sometimes you _do_ want to know whether the field was set or not. You can tell the pattern matching engine about that by prefixing your option match with `?`, like this: - - - -```res -type person = { - age: int, - name?: string, -} - -let me = { - age: 123, - name: "Hello", -} - -let nameWasSet = switch me { -| {name: ?None} => false -| {name: ?Some(_)} => true -} -``` -```js -var nameWasSet = true; - -var me = { - age: 123, - name: "Hello" -}; -``` - - - -## Record Type Spread - -In ReScript v11, you can now spread one or more record types into a new record type. It looks like this: - -```rescript -type a = { - id: string, - name: string, -} - -type b = { - age: int -} - -type c = { - ...a, - ...b, - active: bool -} -``` - -`type c` will now be: - -```rescript -type c = { - id: string, - name: string, - age: int, - active: bool, -} -``` - -Record type spreads act as a 'copy-paste' mechanism for fields from one or more records into a new record. This operation inlines the fields from the spread records directly into the new record definition, while preserving their original properties, such as whether they are optional or mandatory. It's important to note that duplicate field names are not allowed across the records being spread, even if the fields have the same type. - -## Record Type Coercion - -Record type coercion gives us more flexibility when passing around records in our application code. In other words, we can now coerce a record `a` to be treated as a record `b` at the type level, as long as the original record `a` contains the same set of fields in `b`. Here's an example: - -```rescript -type a = { - name: string, - age: int, -} - -type b = { - name: string, - age: int, -} - -let nameFromB = (b: b) => b.name - -let a: a = { - name: "Name", - age: 35, -} - -let name = nameFromB(a :> b) -``` - -Notice how we _coerced_ the value `a` to type `b` using the coercion operator `:>`. This works because they have the same record fields. This is purely at the type level, and does not involve any runtime operations. - -Additionally, we can also coerce records from `a` to `b` whenever `a` is a super-set of `b` (i.e. `a` containing all the fields of `b`, and more). The same example as above, slightly altered: - -```rescript -type a = { - id: string, - name: string, - age: int, - active: bool, -} - -type b = { - name: string, - age: int, -} - -let nameFromB = (b: b) => b.name - -let a: a = { - id: "1", - name: "Name", - age: 35, - active: true, -} - -let name = nameFromB(a :> b) -``` - -Notice how `a` now has more fields than `b`, but we can still coerce `a` to `b` because `b` has a subset of the fields of `a`. - -In combination with [optional record fields](/docs/manual/latest/record#optional-record-fields), one may coerce a mandatory field of an `option` type to an optional field: - -```rescript -type a = { - name: string, - - // mandatory, but explicitly typed as option - age: option, -} - -type b = { - name: string, - // optional field - age?: int, -} - -let nameFromB = (b: b) => b.name - -let a: a = { - name: "Name", - age: Some(35), -} - -let name = nameFromB(a :> b) -``` - -## Tips & Tricks - -### Record Types Are Found By Field Name -With records, you **cannot** say "I'd like this function to take any record type, as long as they have the field `age`". The following **won't work as intended**: - - - -```res -type person = {age: int, name: string} -type monster = {age: int, hasTentacles: bool} - -let getAge = (entity) => entity.age -``` -```js -function getAge(entity) { - return entity.age; -} -``` - - - -Instead, `getAge` will infer that the parameter `entity` must be of type `monster`, the closest record type with the field `age`. The following code's last line fails: - -```res -let kraken = {age: 9999, hasTentacles: true} -let me = {age: 5, name: "Baby ReScript"} - -getAge(kraken) -getAge(me) // type error! -``` - -The type system will complain that `me` is a `person`, and that `getAge` only works on `monster`. If you need such capability, use ReScript objects, described [here](object.md). - -### Optional Fields in Records Can Be Useful for Bindings -Many JavaScript APIs tend to have large configuration objects that can be a bit annoying to model as records, since you previously always needed to specify all record fields when creating a record. - -Optional record fields, introduced in [`v10`](/blog/release-10-0-0#experimental-optional-record-fields), is intended to help with this. Optional fields will let you avoid having to specify all fields, and let you just specify the one's you care about. A significant improvement in ergonomics for bindings and other APIs with for example large configuration objects. - -## Design Decisions - -After reading the constraints in the previous sections, and if you're coming from a dynamic language background, you might be wondering why one would bother with record in the first place instead of straight using object, since the former needs explicit typing and doesn't allow different records with the same field name to be passed to the same function, etc. - -1. The truth is that most of the times in your app, your data's shape is actually fixed, and if it's not, it can potentially be better represented as a combination of variant (introduced next) + record instead. -2. Since a record type is resolved through finding that single explicit type declaration (we call this "nominal typing"), the type error messages end up better than the counterpart ("structural typing", like for tuples). This makes refactoring easier; changing a record type's fields naturally allows the compiler to know that it's still the same record, just misused in some places. Otherwise, under structural typing, it might get hard to tell whether the definition site or the usage site is wrong. - diff --git a/pages/docs/manual/latest/reserved-keywords.mdx b/pages/docs/manual/latest/reserved-keywords.mdx deleted file mode 100644 index 519050c5a..000000000 --- a/pages/docs/manual/latest/reserved-keywords.mdx +++ /dev/null @@ -1,92 +0,0 @@ ---- -title: "Reserved Keywords" -description: "All reserved keywords in ReScript" -canonical: "/docs/manual/latest/reserved-keywords" ---- - -# Reserved Keywords - -> **Note**: Some of these words are reserved purely for backward compatibility. -> -> If you _need_ to use one of these names as binding and/or field name, see [Use Illegal Identifier Names](use-illegal-identifier-names.md). - -- `and` -- `as` -- `assert` - - - - - - -- `constraint` - - - - - - -- `else` - - -- `exception` -- `external` - - -- `false` -- `for` - - - - - -- `if` -- `in` -- `include` - - - - -- `lazy` -- `let` - - -- `module` -- `mutable` - - - - - - - -- `of` -- `open` - - - - - - - -- `rec` - - - - -- `switch` - - - -- `true` -- `try` -- `type` - - - - - - -- `when` -- `while` -- `with` diff --git a/pages/docs/manual/latest/scoped-polymorphic-types.mdx b/pages/docs/manual/latest/scoped-polymorphic-types.mdx deleted file mode 100644 index a693fff74..000000000 --- a/pages/docs/manual/latest/scoped-polymorphic-types.mdx +++ /dev/null @@ -1,100 +0,0 @@ ---- -title: "Scoped Polymorphic Types" -description: "Scoped Polymorphic Types in ReScript" -canonical: "/docs/manual/latest/scoped-polymorphic-types" ---- - -# Scoped Polymorphic Types - -Scoped Polymorphic Types in ReScript are functions with the capability to handle arguments of any type within a specific scope. This feature is particularly valuable when working with JavaScript APIs, as it allows your functions to accommodate diverse data types while preserving ReScript's strong type checking. - -## Definition and Usage - -Scoped polymorphic types in ReScript offer a flexible and type-safe way to handle diverse data types within specific scopes. This documentation provides an example to illustrate their usage in a JavaScript context. - -### Example: Logging API - -Consider a logging example within a JavaScript context that processes various data types: - -```js -const logger = { - log: (data) => { - if (typeof data === "string") { - /* handle string */ - } else if (typeof data === "number") { - /* handle number */ - } else { - /* handle other types */ - } - }, -}; -``` - -In ReScript, we can bind to this function as a record with a scoped polymorphic function type: - -```res prelude -type logger = { log: 'a. 'a => unit } - -@module("jsAPI") external getLogger: unit => logger = "getLogger" -``` - -The `logger` type represents a record with a single field `log`, which is a scoped polymorphic function type `'a. 'a => unit`. The `'a` indicates a type variable that can be any type within the scope of the `log` function. - -Now, we can utilize the function obtained from `getLogger`: - - - -```res example -let myLogger = getLogger() - -myLogger.log("Hello, ReScript!") -myLogger.log(42) -``` - -```js -var myLogger = JsAPI.getLogger(); - -myLogger.log("Hello, ReScript!"); -myLogger.log(42); -``` - - - -In this example, we create an instance of the logger by calling `getLogger()`, and then we can use the `log` function on the `myLogger` object to handle different data types. - -## Limitations of Normal Polymorphic Types - -Let's consider the same logging example in ReScript, but this time using normal polymorphic types: - -```res -type logger<'a> = { log: 'a => unit} - -@module("jsAPI") external getLogger: unit => logger<'a> = "getLogger" -``` - -In this case, the `logger` type is a simple polymorphic function type `'a => unit`. However, when we attempt to use this type in the same way as before, we encounter an issue: - -```res -let myLogger = getLogger() - -myLogger.log("Hello, ReScript!") -myLogger.log(42) // Type error! -``` - -The problem arises because the type inference in ReScript assigns a concrete type to the `logger` function based on the first usage. In this example, after the first call to `myLogger`, the compiler infers the type `logger` for `myLogger`. Consequently, when we attempt to pass an argument of type `number` in the next line, a type error occurs because it conflicts with the inferred type `logger`. - -In contrast, scoped polymorphic types, such as `'a. 'a => unit`, overcome this limitation by allowing type variables within the scope of the function. They ensure that the type of the argument is preserved consistently within that scope, regardless of the specific value used in the first invocation. - -## Limitations of Scoped Polymorphic Types - -Scoped polymorphic types work only when they are directly applied to let-bindings or record fields (as demonstrated in the logger example above). They can neither be applied to function bodies, nor to separate type definitions: - -```res -exception Abort - -let testExn: 'a. unit => 'a = () => raise(Abort) // Works! - -let testExn2 = (): 'a. 'a = raise(Abort) // Syntax error! -type fn = 'a. 'a => unit // Syntax error! -``` - diff --git a/pages/docs/manual/latest/shared-data-types.mdx b/pages/docs/manual/latest/shared-data-types.mdx deleted file mode 100644 index 530713b8d..000000000 --- a/pages/docs/manual/latest/shared-data-types.mdx +++ /dev/null @@ -1,44 +0,0 @@ ---- -title: "Shared Data Types" -description: "Data types that share runtime presentation between JS and ReScript" -canonical: "/docs/manual/latest/shared-data-types" ---- - -# Shared Data Types - -ReScript's built-in values of type `string`, `float`, `array` and a few others have a rather interesting property: they compile to the exact same value in JavaScript! - -This means that if you're passing e.g. a ReScript string to the JavaScript side, the JS side can directly use it as a native JS string. It also means that you can import a JS string and pretend it's a native ReScript string. - -Unlike most compiled-to-js languages, in ReScript, **you don't need to write data converters back and forth for most of our values**! - -**Shared, bidirectionally usable types**: -- String. ReScript strings are JavaScript strings, vice-versa. (Caveat: only our backtick string `` `hello 👋 ${personName}` `` supports unicode and interpolation). -- Float. ReScript floats are JS numbers, vice-versa. -- Array. In addition to the [Array API](api/core/array), we provide our own [Belt.Array](api/belt/array#set) API too. -- Tuple. Compiles to a JS array. You can treat a fixed-sized, heterogenous JS array as ReScript tuple too. -- Boolean. -- Record. Record compiles to JS object. Therefore you can also treat JS objects as records. If they're too dynamic, consider modeling them on the ReScript side as a hashmap/dictionary [`Dict`](api/core/dict) or a ReScript object. -- Object. ReScript objects are JavaScript objects, vice-versa. -- Function. They compile to clean JS functions. -- Module. ReScript files are considered top-level modules, and are compiled to JS files 1 to 1. Nested modules are compiled to JavaScript objects. -- Polymorphic variants. -- Unit. The `unit` type, which has a single value `()`, compiles to `undefined` too. Likewise, you can treat an incoming JS `undefined` as `()` if that's the only value it'll ever be. - -**Types that are slightly different than JS, but that you can still use from JS**: -- Int. **Ints are 32-bits**! Be careful, you can potentially treat them as JS numbers and vice-versa, but if the number's large, then you better treat JS numbers as floats. For example, we bind to `Date` using `float`s. -- Option. The `option` type's `None` value compiles into JS `undefined`. The `Some` value, e.g. `Some(5)`, compiles to `5`. Likewise, you can treat an incoming JS `undefined` as `None`. **JS `null` isn't handled here**. If your JS value can be `null`, use [Nullable](api/core/nullable) helpers. -- Exception. -- Variant. Check the compiled JavaScript output of variant to see its shape. We don't recommend exporting a ReScript variant for pure JS usage, since they're harder to read as plain JS code, but you can do it. -- List, which is just a regular variant. - -**Non-shared types (aka internal types)**: - -- Character. -- Int64. -- Lazy values. -- Everything else. - -Many of these are stable, which means that you can still serialize/deserialize them as-is without manual conversions. But we discourage actively peeking into their structure otherwise. - -These types require manual conversions if you want to export them for JS consumption. For a seamless JS/TypeScript integration experience, you might want to use [genType](https://github.com/cristianoc/gentype) instead of doing conversions by hand. diff --git a/pages/docs/manual/latest/tagged-templates.mdx b/pages/docs/manual/latest/tagged-templates.mdx deleted file mode 100644 index f3dabb194..000000000 --- a/pages/docs/manual/latest/tagged-templates.mdx +++ /dev/null @@ -1,124 +0,0 @@ ---- -title: "Tagged templates" -description: "Using tagged templates in ReScript" -canonical: "/docs/manual/latest/tagged-templates" ---- - -# Tagged templates - -**Since 11.1** - -Tagged templates provide a special form of string interpolation, enabling the creation of template literals -where placeholders aren't restricted to strings. Moreover, the resulting output isn't confined solely to -strings either. You can take a look at the [JS documentation -about tagged templates](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Template_literals#tagged_templates) -to learn more about them. - -## Define a tag function - -Tag functions in ReScript have the following signature: -```res -let myTagFunction : (array, array<'param>) => 'output -``` -As you can see, you can have any type you want both for the placeholder array and for the output. - -Given how string interpolation works, you'll always have the following invariant: -```res -Array.length(strings) == Array.length(placeholder) + 1 -``` - -Let's say you want to interpolate strings with all kind of builtin types and make it work inside React components, -you can define the following tag function: - - - -```res prelude -type params = - | I(int) - | F(float) - | S(string) - | Bool(bool) - -let s = (strings, parameters) => { - let text = Array.reduceWithIndex(parameters, Array.getUnsafe(strings, 0), ( - acc, - param, - i, - ) => { - let s = Array.getUnsafe(strings, i + 1) - let p = switch param { - | I(i) => Int.toString(i) - | F(f) => Float.toString(f) - | S(s) => s - | Bool(true) => "true" - | Bool(false) => "false" - } - acc ++ p ++ s - }) - React.string(text) -} -``` -```js -import * as Core__Array from "./stdlib/core__Array.js"; - -function s(strings, parameters) { - return Core__Array.reduceWithIndex(parameters, strings[0], (function (acc, param, i) { - var s = strings[i + 1 | 0]; - var p; - switch (param.TAG) { - case "I" : - case "F" : - p = param._0.toString(); - break; - case "S" : - p = param._0; - break; - case "Bool" : - p = param._0 ? "true" : "false"; - break; - - } - return acc + p + s; - })); -} -``` - - - -## Write tagged template literals - -Now that you have defined your tag function, you can use it this way: - - - -```res example -module Greetings = { - @react.component - let make = (~name, ~age) => { -
{s`hello ${S(name)} you're ${I(age)} year old!`}
- } -} -``` -```js -function Greetings(props) { - return React.createElement("div", undefined, s([ - "hello ", - " you're ", - " year old!" - ], [ - { - TAG: "S", - _0: props.name - }, - { - TAG: "I", - _0: props.age - } - ])); -} -``` - -
- -Pretty neat, isn't it? As you can see, it looks like any regular template literal but it accepts placeholders that are not strings -and it outputs something that is not a string either, a `React.element` in this case. \ No newline at end of file diff --git a/pages/docs/manual/latest/try.mdx b/pages/docs/manual/latest/try.mdx deleted file mode 100644 index 1f6c913ff..000000000 --- a/pages/docs/manual/latest/try.mdx +++ /dev/null @@ -1,9 +0,0 @@ ---- -title: "Try" -description: "Try ReScript via Command Line" -canonical: "/docs/manual/latest/try" ---- - -## Try Online - -Our [Playground](/try) lets you try ReScript online, and comes with the [ReScript React bindings](/docs/react/latest/introduction) and the new [ReScript Core](https://github.com/rescript-lang/rescript-core) standard library preinstalled. diff --git a/pages/docs/manual/latest/tuple.mdx b/pages/docs/manual/latest/tuple.mdx deleted file mode 100644 index 951584c4f..000000000 --- a/pages/docs/manual/latest/tuple.mdx +++ /dev/null @@ -1,104 +0,0 @@ ---- -title: "Tuple" -description: "Tuple types and values in ReScript" -canonical: "/docs/manual/latest/tuple" ---- - -# Tuple - -Tuples are a ReScript-specific data structure that don't exist in JavaScript. They are: - -- immutable -- ordered -- fix-sized at creation time -- heterogeneous (can contain different types of values) - - - -```res example -let ageAndName = (24, "Lil' ReScript") -let my3dCoordinates = (20.0, 30.5, 100.0) -``` -```js -var ageAndName = [24, "Lil' ReScript"]; -var my3dCoordinates = [20.0, 30.5, 100.0]; -``` - - - -Tuples' types can be used in type annotations as well. Tuple types visually resemble tuples values. - - - -```res prelude -let ageAndName: (int, string) = (24, "Lil' ReScript") -// a tuple type alias -type coord3d = (float, float, float) -let my3dCoordinates: coord3d = (20.0, 30.5, 100.0) -``` -```js -var ageAndName = [24, "Lil' ReScript"]; -var my3dCoordinates = [20.0, 30.5, 100.0]; -``` - - -**Note**: there's no tuple of size 1. You'd just use the value itself. - -## Usage - -To get a specific member of a tuple, destructure it: - - - -```res example -let (_, y, _) = my3dCoordinates // now you've retrieved y -``` -```js -var y = 30.5; -``` - - - -The `_` means you're ignoring the indicated members of the tuple. - -Tuples aren't meant to be updated mutatively. You'd create new ones by destructuring the old ones: - - - -```res example -let coordinates1 = (10, 20, 30) -let (c1x, _, _) = coordinates1 -let coordinates2 = (c1x + 50, 20, 30) -``` -```js -var coordinates1 = [10, 20, 30]; -var c1x = 10; -var coordinates2 = [60, 20, 30]; -``` - - - -## Tips & Tricks - -You'd use tuples in handy situations that pass around multiple values without too much ceremony. For example, to return many values: - - - -```res -let getCenterCoordinates = () => { - let x = doSomeOperationsHere() - let y = doSomeMoreOperationsHere() - (x, y) -} -``` -```js -function getCenterCoordinates(param) { - var x = doSomeOperationsHere(undefined); - var y = doSomeMoreOperationsHere(undefined); - return [x, y]; -} -``` - - - -Try to keep the usage of tuple **local**. For data structures that are long-living and passed around often, prefer a **record**, which has named fields. diff --git a/pages/docs/manual/latest/type.mdx b/pages/docs/manual/latest/type.mdx deleted file mode 100644 index b39bca6d4..000000000 --- a/pages/docs/manual/latest/type.mdx +++ /dev/null @@ -1,278 +0,0 @@ ---- -title: "Type" -description: "Types and type definitions in ReScript" -canonical: "/docs/manual/latest/type" ---- - -# Type - -Types are the highlight of ReScript! They are: -- **Strong**. A type can't change into another type. In JavaScript, your variable's type might change when the code runs (aka at runtime). E.g. a `number` variable might change into a `string` sometimes. This is an anti-feature; it makes the code much harder to understand when reading or debugging. -- **Static**. ReScript types are erased after compilation and don't exist at runtime. Never worry about your types dragging down performance. You don't need type info during runtime; we report all the information (especially all the type errors) during compile time. Catch the bugs earlier! -- **Sound**. This is our biggest differentiator versus many other typed languages that compile to JavaScript. Our type system is guaranteed to **never** be wrong. Most type systems make a guess at the type of a value and show you a type in your editor that's sometime incorrect. We don't do that. We believe that a type system that is sometime incorrect can end up being dangerous due to expectation mismatches. -- **Fast**. Many developers underestimate how much of their project's build time goes into type checking. Our type checker is one of the fastest around. -- **Inferred**. You don't have to write down the types! ReScript can deduce them from their values. Yes, it might seem magical that we can deduce all of your program's types, without incorrectness, without your manual annotation, and do so quickly. Welcome to ReScript =). - -The following sections explore more of our type system. - -## Inference - -This let-binding doesn't contain any written type: - - - -```res example -let score = 10 -let add = (a, b) => a + b -``` -```js -var score = 10; -function add(a, b) { - return a + b | 0; -} -``` - - - -ReScript knows that `score` is an `int`, judging by the value `10`. This is called **inference**. Likewise, it also knows that the `add` function takes 2 `int`s and returns an `int`, judging from the `+` operator, which works on ints. - -## Type Annotation - -But you can also optionally write down the type, aka annotate your value: - - - -```res example -let score: int = 10 -``` -```js -var score = 10; -``` - - - -If the type annotation for `score` doesn't correspond to our inferred type for it, we'll show you an error during compilation time. We **won't** silently assume your type annotation is correct, unlike many other languages. - -You can also wrap any expression in parentheses and annotate it: - - - -```res -let myInt = 5 -let myInt: int = 5 -let myInt = (5: int) + (4: int) -let add = (x: int, y: int) : int => x + y -let drawCircle = (~radius as r: int): circleType => /* code here */ -``` -```js -var myInt = 9; -function add(x, y) { - return x + y | 0; -} -function drawCircle(r) { - /* code here */ -} -``` - - - -Note: in the last line, `(~radius as r: int)` is a labeled argument. More on this in the [function](function.md) page. - -## Type Alias - -You can refer to a type by a different name. They'll be equivalent: - - - -```res example -type scoreType = int -let x: scoreType = 10 -``` -```js -var x = 10; -``` - - - -## Type Parameter (Aka Generic) - -Types can accept parameters, akin to generics in other languages. The parameters' names **need** to start with `'`. - -The use-case of a parameterized type is to kill duplications. Before: - - - -```res example -// this is a tuple of 3 items, explained next -type intCoordinates = (int, int, int) -type floatCoordinates = (float, float, float) - -let a: intCoordinates = (10, 20, 20) -let b: floatCoordinates = (10.5, 20.5, 20.5) -``` -```js -var a = [10, 20, 20]; -var b = [10.5, 20.5, 20.5]; -``` - - - -After: - - - -```res example -type coordinates<'a> = ('a, 'a, 'a) - -let a: coordinates = (10, 20, 20) -let b: coordinates = (10.5, 20.5, 20.5) -``` -```js -var a = [10, 20, 20]; -var b = [10.5, 20.5, 20.5]; -``` - - - -Note that the above codes are just contrived examples for illustration purposes. Since the types are inferred, you could have just written: - - - -```res example -let buddy = (10, 20, 20) -``` -```js -var buddy = [10, 20, 20]; -``` - - - -The type system infers that it's a `(int, int, int)`. Nothing else needed to be written down. - -Type arguments appear in many places. Our `array<'a>` type is such a type that requires a type parameter. - - - -```res example -// inferred as `array` -let greetings = ["hello", "world", "how are you"] -``` -```js -// inferred as `array` -var greetings = ["hello", "world", "how are you"]; -``` - - - -If types didn't accept parameters, the standard library would need to define the types `arrayOfString`, `arrayOfInt`, `arrayOfTuplesOfInt`, etc. That'd be tedious. - -Types can receive many arguments, and be composable. - - - - - -```res example -type result<'a, 'b> = - | Ok('a) - | Error('b) - -type myPayload = {data: string} - -type myPayloadResults<'errorType> = array> - -let payloadResults: myPayloadResults = [ - Ok({data: "hi"}), - Ok({data: "bye"}), - Error("Something wrong happened!") -] -``` -```js -var payloadResults = [ - { - TAG: /* Ok */0, - _0: {data: "hi"} - }, - { - TAG: /* Ok */0, - _0: {data: "bye"} - }, - { - TAG: /* Error */1, - _0: "Something wrong happened!" - } -]; -``` - - - -## Recursive Types - -Just like a function, a type can reference itself within itself using `rec`: - - - -```res example -type rec person = { - name: string, - friends: array -} -``` -```js -// Empty output -``` - - - -## Mutually Recursive Types - -Types can also be _mutually_ recursive through `and`: - - - -```res example -type rec student = {taughtBy: teacher} -and teacher = {students: array} -``` -```js -// Empty output -``` - - - -## Type Escape Hatch - -ReScript's type system is robust and does not allow dangerous, unsafe stuff like implicit type casting, randomly guessing a value's type, etc. However, out of pragmatism, we expose a single escape hatch for you to "lie" to the type system: - - - -```res -external myShadyConversion: myType1 => myType2 = "%identity" -``` -```js -// Empty output -``` - - - -This declaration converts a `myType1` of your choice to `myType2` of your choice. You can use it like so: - - - -```res example -external convertToFloat : int => float = "%identity" -let age = 10 -let gpa = 2.1 +. convertToFloat(age) -``` -```js -var age = 10; -var gpa = 2.1 + 10; -``` - - - -Obviously, do **not** abuse this feature. Use it tastefully when you're working with existing, overly dynamic JS code, for example. - -More on externals [here](external.md). - -**Note**: this particular `external` is the only one that isn't preceded by a `@` [attribute](attribute.md). diff --git a/pages/docs/manual/latest/typescript-integration.mdx b/pages/docs/manual/latest/typescript-integration.mdx deleted file mode 100644 index 193683cec..000000000 --- a/pages/docs/manual/latest/typescript-integration.mdx +++ /dev/null @@ -1,283 +0,0 @@ ---- -title: "TypeScript" -description: "GenType - Interoperability between ReScript and TypeScript" -canonical: "/docs/manual/latest/typescript-integration" ---- - -# ReScript & TypeScript - -The ReScript compiler includes a code generation tool that lets you export ReScript values and types to use in TypeScript, and import TypeScript values and types into ReScript. It is called "genType". - -The implementation of genType performs a type-directed transformation of ReScript programs after compilation. The transformed programs operate on data types idiomatic to TypeScript. - -For example, a ReScript variant (which is represented as custom objects with tags at runtime): - -```res -@genType -type t = | A(int) | B(string) -``` - -is exported to a TypeScript type: - -```ts -type t = { TAG: "A"; _0: number } | { TAG: "B"; _0: string }; -``` - -## A Quick Example - -Let's assume we are working on a TypeScript codebase and we want to integrate a single ReScript function. - -We want to be able to import the function like any other one in our existing TypeScript code, but we also want to preserve all the ReScript types in the TypeScript type system. - -**That's exactly what genType was made for!** - -First we'll set up a function: - -```res -// src/Color.res - -@genType -type color = - | Red - | Blue - -@genType -let printColorMessage = (~color, ~message) => { - let prefix = switch color { - | Red => "\x1b[91m" - | Blue => "\x1b[94m" - } - let reset = "\x1b[0m" - - Console.log(prefix ++ message ++ reset) -} - -``` - -On a successful compile, `genType` will convert `src/Color.res` to a TypeScript file called `src/Color.gen.tsx` which will look something like this: - -```ts -// src/Color.gen.tsx - -/* TypeScript file generated from Color.res by genType. */ - -/* eslint-disable */ -/* tslint:disable */ - -import * as ColorJS from "./Color.res.js"; - -export type color = "Red" | "Blue"; - -export const printColorMessage: ( - color: color -) => void = ColorJS.printColorMessage as any; -``` - -genType automatically maps the `color` variant to TS via a string union type `"Red" | "Blue"`. - -Within our TypeScript application, we can now import and use the function in the following manner: - -```ts -// src/app.ts - -import { printColorMessage } from "./Color.gen.tsx"; - -printColorMessage("Red", "Hello, genType!"); -``` - -## Exporting an entire module - -_Since ReScript `11.0.0`_ modules can be annotated with `@genType` as well. In that case, all types and values of the module will be converted to TS types. Example: - - - -```res example -@genType -module Size = { - type t = - | Small - | Medium - | Large - - let getNum = (size: t) => - switch size { - | Small => 1. - | Medium => 5. - | Large => 10. - } -} -``` - -```ts -import * as MyCompBS__Es6Import from './MyComp.res'; -const MyCompBS: any = MyCompBS__Es6Import; - -export type Size_t = "Small" | "Medium" | "Large"; - -export const Size_getNum: (size:Size_t) => number = MyCompBS.Size.getNum; - -export const Size: { getNum: (size:Size_t) => number } = MyCompBS.Size -``` - - - -## Setup - -Add a `gentypeconfig` section to your `rescript.json` (See [Configuration](/docs/manual/latest/build-configuration#gentypeconfig) for details). - -Every `genType` powered project requires a configuration item `"gentypeconfig"` at top level in the project's `rescript.json`. - -The minimal configuration of genType is following: - -```json -{ - "gentypeconfig": { - "module": "esmodule", - "moduleResolution": "node", - "generatedFileExtension": ".gen.tsx" - } -} -``` - -And don't forget to make sure `allowJs` is set to `true` in the project's `tsconfig.json`: - -```json -{ - "compilerOptions": { - "allowJs": true - } -} -``` - -### TypeScript Module Resolutions - -Make sure to set the same `moduleResolution` value in both `rescript.json` and `tsconfig.json`, so that the output of genType is done with the preferred module resolution. - -For example if the TypeScript project uses JavaScript modules with `Node16` / `NodeNext` module resolution: - -```json -// tsconfig.json -{ - "compilerOptions": { - "moduleResolution": "node16" - } -} -``` - -Then `moduleResolution` in `gentypeconfig` should be same value: - -```json -// rescript.json -{ - "gentypeconfig": { - "moduleResolution": "node16" - } -} -``` - -In case of the TypeScript project using `Bundler` module resolution, `allowImportingTsExtensions` should also be `true`: - -```json -// tsconfig.json -{ - "compilerOptions": { - "moduleResolution": "bundler", - "allowImportingTsExtensions": true - } -} -``` - -```json -// rescript.json -{ - "gentypeconfig": { - "moduleResolution": "bundler" - } -} -``` - -## Testing the Whole Setup - -Open any relevant `*.res` file and add `@genType` annotations to any bindings / values / functions to be used from JavaScript. If an annotated value uses a type, the type must be annotated too. See e.g. [Hooks.res](https://github.com/rescript-lang/rescript/blob/master/tests/gentype_tests/typescript-react-example/src/Hooks.res). - -Save the file and rebuild the project via `npm run res:build` or similar. You should now see a `*.gen.tsx` file with the same name (e.g. `MyComponent.res` -> `MyComponent.gen.tsx`). - -Any values exported from `MyComponent.res` can then be imported from TypeScript. For example: - -```js -import MyComponent from "./components/MyComponent.gen.tsx"; -``` - -## Experimental features - -These features are for experimentation only. They could be changed/removed any time, and not be considered breaking changes. - -- Export object and record types as interfaces. To activate, add `"exportInterfaces": true` to the configuration. The types are also renamed from `name` to `Iname`. - - -## Shims - -A shim is a TS file that provides user-provided definitions for library types. - -Required only if one needs to export certain basic ReScript data types to JS when one cannot modify the sources to add annotations (e.g. exporting ReScript lists), and if the types are not first-classed in genType. - - Example: `Array` with format: `"RescriptModule=JavaScriptModule"` - -Configure your shim files within `"gentypeconfig"` in your [`rescript.json`]: - -```json -{ - "gentypeconfig": { - "shims": { - "Js": "Js", - "ReactEvent": "ReactEvent", - "RescriptPervasives": "RescriptPervasives", - "ReasonReact": "ReactShim" - }, - }, -} -``` - -and add relevant `.shim.ts` files in a directory which is visible by ReScript e.g. - -``` -├── rescript.json -├── src -│ ├── shims -│ │ ├── Js.shim.ts -│ │ ├── ReactEvent.shim.ts -│ │ └── RescriptPervasives.shim.ts -``` - -Here are some examples: - -```ts -// Excerpt from https://github.com/rescript-lang/rescript/blob/master/tests/gentype_tests/typescript-react-example/src/shims/Js.shim.ts -export type Json_t = unknown; -export type t = unknown; -``` - -```ts -// Excerpt from https://github.com/rescript-lang/rescript/tree/master/tests/gentype_tests/typescript-react-example/src/shims/JsxEvent.shim.ts -export type inputFocusEvent = React.FocusEvent; -``` - -More complete example shims can be found [here](https://github.com/rescript-lang/rescript/blob/master/tests/gentype_tests/typescript-react-example/src/shims/). - -## Deprecated features - -Features related to generating runtimes were deprecated since v11 and should no longer be used. - -- **`@genType("alias")`** and **`@genType.as("alias")`** -- **`@genType.opaque`** -- **`@genType.import`** -- TypeScript Shims - -genType does not generate anything runtime-related, and in the near future it generates definition files (`*.d.ts`) directly (See the [roadmap](https://github.com/rescript-lang/rescript/issues/6196)). - -If any runtime code is required for interoperability with JavaScript / TypeScript projects, it can be written by hand, or request a relevant features (e.g. `@deriving`) to the compiler. - -## Limitations - -- **in-source = true**. Currently only supports ReScript projects with [in-source generation](/docs/manual/latest/build-configuration#package-specs) and file suffixes that end on `.js`, like `.res.js` or `.bs.js`. - -- **Limited namespace support**. Currently there's limited [namespace](/docs/manual/latest/build-configuration#name-namespace) support, and only `namespace:true` is possible, not e.g. `namespace:"custom"`. diff --git a/pages/docs/manual/latest/use-illegal-identifier-names.mdx b/pages/docs/manual/latest/use-illegal-identifier-names.mdx deleted file mode 100644 index a1e2eaf58..000000000 --- a/pages/docs/manual/latest/use-illegal-identifier-names.mdx +++ /dev/null @@ -1,51 +0,0 @@ ---- -title: "Use Illegal Identifier Names" -description: "Handling (JS) naming collisions in ReScript" -canonical: "/docs/manual/latest/use-illegal-identifier-names" ---- - -# Use Illegal Identifier Names - -Sometime, for e.g. a let binding or a record field, you might want to use: -- A capitalized name. -- A name that contains illegal characters (e.g. emojis, hyphen, space). -- A name that's one of ReScript's reserved keywords. - -We provide an escape hatch syntax for these cases: - - - -```res example -let \"my-🍎" = 10 - -type element = { - \"aria-label": string -} - -let myElement = { - \"aria-label": "close" -} - -let label = myElement.\"aria-label" - -let calculate = (~\"Props") => { - \"Props" + 1 -} -``` -```js -var my$$unknown$unknown$unknown$unknown = 10; - -var myElement = { - "aria-label": "close" -}; - -var label = myElement["aria-label"]; - -function calculate(Props) { - return Props + 1 | 0; -} -``` - - - -See the output. **Use them only when necessary**, for interop with JavaScript. This is a last-resort feature. If you abuse this, many of the compiler guarantees will go away. diff --git a/pages/docs/manual/latest/variant.mdx b/pages/docs/manual/latest/variant.mdx deleted file mode 100644 index 69e1d7973..000000000 --- a/pages/docs/manual/latest/variant.mdx +++ /dev/null @@ -1,798 +0,0 @@ ---- -title: "Variant" -description: "Variant data structures in ReScript" -canonical: "/docs/manual/latest/variant" ---- - -# Variant - -So far, most of ReScript's data structures might look familiar to you. This section introduces an extremely important, and perhaps unfamiliar, data structure: variant. - -Most data structures in most languages are about "this **and** that". A variant allows us to express "this **or** that". - - - -```res example -type myResponse = - | Yes - | No - | PrettyMuch - -let areYouCrushingIt = Yes -``` -```js -var areYouCrushingIt = "Yes"; -``` - - - -`myResponse` is a variant type with the cases `Yes`, `No` and `PrettyMuch`, which are called "variant constructors" (or "variant tag"). The `|` bar separates each constructor. - -**Note**: a variant's constructors need to be capitalized. - -## Variant Needs an Explicit Definition - -If the variant you're using is in a different file, bring it into scope like you'd do [for a record](record.md#record-needs-an-explicit-definition): - - - -```res example -// Zoo.res -type animal = Dog | Cat | Bird -``` -```js -// Empty output -``` - - - - - -```res -// Example.res -let pet: Zoo.animal = Dog // preferred -// or -let pet2 = Zoo.Dog -``` -```js -var pet = "Dog"; -var pet2 = "Dog"; -``` - - - -## Constructor Arguments - -A variant's constructors can hold extra data separated by comma. - - - -```res prelude -type account = - | None - | Instagram(string) - | Facebook(string, int) -``` -```js -// Empty output -``` - - - -Here, `Instagram` holds a `string`, and `Facebook` holds a `string` and an `int`. Usage: - - - -```res example -let myAccount = Facebook("Josh", 26) -let friendAccount = Instagram("Jenny") -``` -```js -var myAccount = { - TAG: "Facebook", - _0: "Josh", - _1: 26 -}; -var friendAccount = { - TAG: "Instagram", - _0: "Jenny" -}; -``` - - - -### Labeled Variant Payloads (Inline Record) - -If a variant payload has multiple fields, you can use a record-like syntax to label them for better readability: - - - -```res example -type user = - | Number(int) - | Id({name: string, password: string}) - -let me = Id({name: "Joe", password: "123"}) -``` -```js -var me = { - TAG: "Id", - name: "Joe", - password: "123" -}; -``` - - - -This is technically called an "inline record", and only allowed within a variant constructor. You cannot inline a record type declaration anywhere else in ReScript. - -Of course, you can just put a regular record type in a variant too: - - - -```res example -type u = {name: string, password: string} -type user = - | Number(int) - | Id(u) - -let me = Id({name: "Joe", password: "123"}) -``` -```js -var me = { - TAG: "Id", - _0: { - name: "Joe", - password: "123" - } -}; -``` - - - -The output is slightly uglier and less performant than the former. - -## Variant Type Spreads -Just like [with records](record#record-type-spread), it's possible to use type spreads to create new variants from other variants: - -```rescript -type a = One | Two | Three -type b = | ...a | Four | Five -``` - -Type `b` is now: -```rescript -type b = One | Two | Three | Four | Five -``` - -Type spreads act as a 'copy-paste', meaning all constructors are copied as-is from `a` to `b`. Here are the rules for spreads to work: -- You can't overwrite constructors, so the same constructor name can exist in only one place as you spread. This is true even if the constructors are identical. -- All variants and constructors must share the same runtime configuration - `@unboxed`, `@tag`, `@as` and so on. -- You can't spread types in recursive definitions. - -Note that you need a leading `|` if you want to use a spread in the first position of a variant definition. - -### Pattern Matching On Variant - -See the [Pattern Matching/Destructuring](pattern-matching-destructuring) section later. - -## JavaScript Output - -A variant value compiles to 3 possible JavaScript outputs depending on its type declaration: - -- If the variant value is a constructor with no payload, it compiles to a string of the constructor name. Example: `Yes` compiles to `"Yes"`. -- If it's a constructor with a payload, it compiles to an object with the field `TAG` and the field `_0` for the first payload, `_1` for the second payload, etc. The value of `TAG` is the constructor name as string by default, but note that the name of the `TAG` field as well as the string value used for each constructor name [can be customized](#tagged-variants). -- Labeled variant payloads (the inline record trick earlier) compile to an object with the label names instead of `_0`, `_1`, etc. The object will have the `TAG` field as per the previous rule. - -Check the output in these examples: - - - -```res example -type greeting = Hello | Goodbye -let g1 = Hello -let g2 = Goodbye - -type outcome = Good | Error(string) -let o1 = Good -let o2 = Error("oops!") - -type family = Child | Mom(int, string) | Dad (int) -let f1 = Child -let f2 = Mom(30, "Jane") -let f3 = Dad(32) - -type person = Teacher | Student({gpa: float}) -let p1 = Teacher -let p2 = Student({gpa: 99.5}) - -type s = {score: float} -type adventurer = Warrior(s) | Wizard(string) -let a1 = Warrior({score: 10.5}) -let a2 = Wizard("Joe") -``` -```js -var g1 = "Hello"; - -var g2 = "Goodbye"; - -var o1 = "Good"; - -var o2 = { - TAG: "Error", - _0: "oops!" -}; - -var f1 = "Child"; - -var f2 = { - TAG: "Mom", - _0: 30, - _1: "Jane" -}; - -var f3 = { - TAG: "Dad", - _0: 32 -}; - -var p1 = "Teacher"; - -var p2 = { - TAG: "Student", - gpa: 99.5 -}; - -var a1 = { - TAG: "Warrior", - _0: { - score: 10.5 - } -}; - -var a2 = { - TAG: "Wizard", - _0: "Joe" -}; -``` - - - -## Tagged variants - -- The `@tag` attribute lets you customize the discriminator (default: `TAG`). -- `@as` attributes control what each variant case is discriminated on (default: the variant case name as string). - -### Example: Binding to TypeScript enums - -```typescript -// direction.ts -/** Direction of the action. */ -enum Direction { - /** The direction is up. */ - Up = "UP", - - /** The direction is down. */ - Down = "DOWN", - - /** The direction is left. */ - Left = "LEFT", - - /** The direction is right. */ - Right = "RIGHT", -} - -export const myDirection = Direction.Up; -``` - -You can bind to the above enums like so: - -```rescript -/** Direction of the action. */ -type direction = - | /** The direction is up. */ - @as("UP") - Up - - | /** The direction is down. */ - @as("DOWN") - Down - - | /** The direction is left. */ - @as("LEFT") - Left - - | /** The direction is right. */ - @as("RIGHT") - Right - -@module("./direction.js") external myDirection: direction = "myDirection" -``` - -Now, this maps 100% to the TypeScript code, including letting us bring over the documentation strings so we get a nice editor experience. - -### String literals - -The same logic is easily applied to string literals from TypeScript, only here the benefit is even larger, because string literals have the same limitations in TypeScript that polymorphic variants have in ReScript: - -```typescript -// direction.ts -type direction = "UP" | "DOWN" | "LEFT" | "RIGHT"; -``` - -There's no way to attach documentation strings to string literals in TypeScript, and you only get the actual value to interact with. - -### Valid `@as` payloads -Here's a list of everything you can put in the `@as` tag of a variant constructor: -- A string literal: `@as("success")` -- An int: `@as(5)` -- A float: `@as(1.5)` -- True/false: `@as(true)` and `@as(false)` -- Null: `@as(null)` -- Undefined: `@as(undefined)` - -## Untagged variants - -With _untagged variants_ it is possible to mix types together that normally can't be mixed in the ReScript type system, as long as there's a way to discriminate them at runtime. For example, with untagged variants you can represent a heterogenous array: - -```rescript -@unboxed type listItemValue = String(string) | Boolean(bool) | Number(float) - -let myArray = [String("Hello"), Boolean(true), Boolean(false), Number(13.37)] -``` - -Here, each value will be _unboxed_ at runtime. That means that the variant payload will be all that's left, the variant case name wrapping the payload itself will be stripped out and the payload will be all that remains. - -It, therefore, compiles to this JS: - -```javascript -var myArray = ["hello", true, false, 13.37]; -``` - -In the above example, reaching back into the values is as simple as pattern matching on them. - -### Advanced: Unboxing rules -#### No overlap in constructors -A variant can be unboxed if no constructors have overlap in their runtime representation. - -For example, you can't have `String1(string) | String2(string)` in the same unboxed variant, because there's no way for ReScript to know at runtime which of `String1` or `String2` that `string` belongs to, as it could belong to both. -The same goes for two records - even if they have fully different shapes, they're still JavaScript `object` at runtime. - -Don't worry - the compiler will guide you and ensure there's no overlap. - -#### What you can unbox -Here's a list of all possible things you can unbox: -- `string`: `String(string)` -- `float`: `Float(float)`. Note you can only have one of `float` or `int` because JavaScript only has `number` (not actually `int` and `float` like in ReScript) so we can't disambiguate between `float` and `int` at runtime. -- `int`: `Int(int)`. See note above on `float`. -- `bigint`: `BigInt(int)`. **Since 11.1** This is a distinct type from JavaScript's `number` type so you can use it beside either `float` or `int`. -- `bool`: `Boolean(bool)` -- `array<'value>`: `List(array)` -- `('a, 'b, 'c)`: `Tuple((string, int, bool))`. Any size of tuples works, but you can have only one case of array or tuple in a variant. -- `promise<'value>`: `Promise(promise)` -- `Dict.t`: `Object(Dict.t)` -- `Date.t`: `Date(Date.t)`. A JavaScript date. -- `Blob.t`: `Blob(Blob.t)`. A JavaScript blob. -- `File.t`: `File(File.t)`. A JavaScript file. -- `RegExp.t`: `RegExp(RegExp.t)`. A JavaScript regexp instance. - -Again notice that the constructor names can be anything, what matters is what's in the payload. - -> **Under the hood**: Untagged variants uses a combination of JavaScript `typeof` and `instanceof` checks to discern between unboxed constructors at runtime. This means that we could add more things to the list above detailing what can be unboxed, if there are useful enough use cases. - -### Pattern matching on unboxed variants -Pattern matching works the same on unboxed variants as it does on regular variants. In fact, in the perspective of ReScript's type system there's no difference between untagged and tagged variants. You can do virtually the same things with both. That's the beauty of untagged variants - they're just variants to you as a developer. - -Here's an example of pattern matching on an unboxed nullable value that illustrates the above: - -```rescript -module Null = { - @unboxed type t<'a> = Present('a) | @as(null) Null -} - -type userAge = {ageNum: Null.t} - -type rec user = { - name: string, - age: Null.t, - bestFriend: Null.t, -} - -let getBestFriendsAge = user => - switch user.bestFriend { - | Present({age: Present({ageNum: Present(ageNum)})}) => Some(ageNum) - | _ => None - } -``` -No difference to how you'd do with a regular variant. But, the runtime representation is different to a regular variant. - -> Notice how `@as` allows us to say that an untagged variant case should map to a specific underlying _primitive_. `Present` has a type variable, so it can hold any type. And since it's an unboxed type, only the payloads `'a` or `null` will be kept at runtime. That's where the magic comes from. - -### Decoding and encoding JSON idiomatically - -With untagged variants, we have everything we need to define a native JSON type: - -```rescript -@unboxed -type rec json = - | @as(null) Null - | Boolean(bool) - | String(string) - | Number(float) - | Object(Dict.t) - | Array(array) - -let myValidJsonValue = Array([String("Hi"), Number(123.)]) -``` - -Here's an example of how you could write your own JSON decoders easily using the above, leveraging pattern matching: - -```rescript -@unboxed -type rec json = - | @as(null) Null - | Boolean(bool) - | String(string) - | Number(float) - | Object(Dict.t) - | Array(array) - -type rec user = { - name: string, - age: int, - bestFriend: option, -} - -let rec decodeUser = json => - switch json { - | Object(userDict) => - switch ( - userDict->Dict.get("name"), - userDict->Dict.get("age"), - userDict->Dict.get("bestFriend"), - ) { - | (Some(String(name)), Some(Number(age)), Some(maybeBestFriend)) => - Some({ - name, - age: age->Float.toInt, - bestFriend: maybeBestFriend->decodeUser, - }) - | _ => None - } - | _ => None - } - -let decodeUsers = json => - switch json { - | Array(array) => array->Array.map(decodeUser)->Array.keepSome - | _ => [] - } -``` - -Encoding that same structure back into JSON is also easy: - -```rescript -let rec userToJson = user => Object( - Dict.fromArray([ - ("name", String(user.name)), - ("age", Number(user.age->Int.toFloat)), - ( - "bestFriend", - switch user.bestFriend { - | None => Null - | Some(friend) => userToJson(friend) - }, - ), - ]), -) - -let usersToJson = users => Array(users->Array.map(userToJson)) -``` - -This can be extrapolated to many more cases. - -### Advanced: Catch-all Constructors -With untagged variants comes a rather interesting capability - catch-all cases are now possible to encode directly into a variant. - -Let's look at how it works. Imagine you're using a third party API that returns a list of available animals. You could of course model it as a regular `string`, but given that variants can be used as "typed strings", using a variant would give you much more benefit: - - -```rescript -type animal = Dog | Cat | Bird - -type apiResponse = { - animal: animal -} - -let greetAnimal = (animal: animal) => - switch animal { - | Dog => "Wof" - | Cat => "Meow" - | Bird => "Kashiiin" - } -``` -```javascript -``` - - - -This is all fine and good as long as the API returns `"Dog"`, `"Cat"` or `"Bird"` for `animal`. -However, what if the API changes before you have a chance to deploy new code, and can now return `"Turtle"` as well? Your code would break down because the variant `animal` doesn't cover `"Turtle"`. - -So, we'll need to go back to `string`, loosing all of the goodies of using a variant, and then do manual conversion into the `animal` variant from `string`, right? -Well, this used to be the case before, but not anymore! We can leverage untagged variants to bake in handling of unknown values into the variant itself. - -Let's update our type definition first: -```rescript -@unboxed -type animal = Dog | Cat | Bird | UnknownAnimal(string) -``` - -Notice we've added `@unboxed` and the constructor `UnknownAnimal(string)`. Remember how untagged variants work? You remove the constructors and just leave the payloads. This means that the variant above at runtime translates to this (made up) JavaScript type: -``` -type animal = "Dog" | "Cat" | "Bird" | string -``` -So, any string not mapping directly to one of the payloadless constructors will now map to the general `string` case. - -As soon as we've added this, the compiler complains that we now need to handle this additional case in our pattern match as well. Let's fix that: - - -```rescript -@unboxed -type animal = Dog | Cat | Bird | UnknownAnimal(string) - -type apiResponse = { - animal: animal -} - -let greetAnimal = (animal: animal) => - switch animal { - | Dog => "Wof" - | Cat => "Meow" - | Bird => "Kashiiin" - | UnknownAnimal(otherAnimal) => - `I don't know how to greet animal ${otherAnimal}` - } -``` -```javascript -function greetAnimal(animal) { - if (!(animal === "Cat" || animal === "Dog" || animal === "Bird")) { - return "I don't know how to greet animal " + animal; - } - switch (animal) { - case "Dog" : - return "Wof"; - case "Cat" : - return "Meow"; - case "Bird" : - return "Kashiiin"; - - } -} -``` - - -There! Now the external API can change as much as it wants, we'll be forced to write all code that interfaces with `animal` in a safe way that handles all possible cases. All of this baked into the variant definition itself, so no need for labor intensive manual conversion. - -This is useful in any scenario when you use something enum-style that's external and might change. Additionally, it's also useful when something external has a large number of possible values that are known, but where you only care about a subset of them. With a catch-all case you don't need to bind to all of them just because they can happen, you can safely just bind to the ones you care about and let the catch-all case handle the rest. - -## Coercion -In certain situations, variants can be coerced to other variants, or to and from primitives. Coercion is always zero cost. - -### Coercing Variants to Other Variants -You can coerce a variant to another variant if they're identical in runtime representation, and additionally if the variant you're coercing can be represented as the variant you're coercing to. - -Here's an example using [variant type spreads](#variant-type-spreads): -```rescript -type a = One | Two | Three -type b = | ...a | Four | Five - -let one: a = One -let four: b = Four - -// This works because type `b` can always represent type `a` since all of type `a`'s constructors are spread into type `b` -let oneAsTypeB = (one :> b) -``` - -### Coercing Variants to Primitives -Variants that are guaranteed to always be represented by a single primitive at runtime can be coerced to that primitive. - -It works with strings, the default runtime representation of payloadless constructors: -```rescript -// Constructors without payloads are represented as `string` by default -type a = One | Two | Three - -let one: a = One - -// All constructors are strings at runtime, so you can safely coerce it to a string -let oneAsString = (one :> string) -``` - -If you were to configure all of your construtors to be represented as `int` or `float`, you could coerce to those too: -```rescript -type asInt = | @as(1) One | @as(2) Two | @as(3) Three - -let oneInt: asInt = One -let toInt = (oneInt :> int) -``` - -### Advanced: Coercing `string` to Variant -In certain situtations it's possible to coerce a `string` to a variant. This is an advanced technique that you're unlikely to need much, but when you do it's really useful. - -You can coerce a `string` to a variant when: -- Your variant is `@unboxed` -- Your variant has a "catch-all" `string` case - -Let's look at an example: -```rescript -@unboxed -type myEnum = One | Two | Other(string) - -// Other("Other thing") -let asMyEnum = ("Other thing" :> myEnum) - -// One -let asMyEnum = ("One" :> myEnum) -``` - -This works because the variant is unboxed **and** has a catch-all case. So, if you throw a string at this variant that's not representable by the payloadless constructors, like `"One"` or `"Two"`, it'll _always_ end up in `Other(string)`, since that case can represent any `string`. - -## Tips & Tricks - -**Be careful** not to confuse a constructor carrying 2 arguments with a constructor carrying a single tuple argument: - - - -```res example -type account = - | Facebook(string, int) // 2 arguments -type account2 = - | Instagram((string, int)) // 1 argument - happens to be a 2-tuple -``` -```js -// Empty output -``` - - - -### Variants Must Have Constructors - -If you come from an untyped language, you might be tempted to try `type myType = int | string`. This isn't possible in ReScript; you'd have to give each branch a constructor: `type myType = Int(int) | String(string)`. The former looks nice, but causes lots of trouble down the line. - -### Interop with JavaScript - -_This section assumes knowledge about our JavaScript interop. Skip this if you haven't felt the itch to use variants for wrapping JS functions yet_. - -Quite a few JS libraries use functions that can accept many types of arguments. In these cases, it's very tempting to model them as variants. For example, suppose there's a `myLibrary.draw` JS function that takes in either a `number` or a `string`. You might be tempted to bind it like so: - - - -```res example -// reserved for internal usage -@module("myLibrary") external draw : 'a => unit = "draw" - -type animal = - | MyFloat(float) - | MyString(string) - -let betterDraw = (animal) => - switch animal { - | MyFloat(f) => draw(f) - | MyString(s) => draw(s) - } - -betterDraw(MyFloat(1.5)) -``` -```js -var MyLibrary = require("myLibrary"); - -function betterDraw(animal) { - MyLibrary.draw(animal._0); -} - -betterDraw({ - TAG: "MyFloat", - _0: 1.5 - }); -``` - - - -**Try not to do that**, as this generates extra noisy output. Instead, use the `@unboxed` attribute to guide ReScript to generate more efficient code: - - - - -```res example -// reserved for internal usage -@module("myLibrary") external draw : 'a => unit = "draw" - -@unboxed -type animal = - | MyFloat(float) - | MyString(string) - -let betterDraw = (animal) => - switch animal { - | MyFloat(f) => draw(f) - | MyString(s) => draw(s) - } - -betterDraw(MyFloat(1.5)) -``` -```js -var MyLibrary = require("myLibrary"); - -function betterDraw(animal) { - MyLibrary.draw(animal); -} - -MyLibrary.draw(1.5); -``` - - - -Alternatively, define two `external`s that both compile to the same JS call: - - - -```res example -@module("myLibrary") external drawFloat: float => unit = "draw" -@module("myLibrary") external drawString: string => unit = "draw" -``` -```js -// Empty output -``` - - - -ReScript also provides [a few other ways](bind-to-js-function.md#modeling-polymorphic-function) to do this. - -### Variant Types Are Found By Field Name - -Please refer to this [record section](record#tips--tricks). Variants are the same: a function can't accept an arbitrary constructor shared by two different variants. Again, such feature exists; it's called a polymorphic variant. We'll talk about this in the future =). - -## Design Decisions - -Variants, in their many forms (polymorphic variant, open variant, GADT, etc.), are likely _the_ feature of a type system such as ReScript's. The aforementioned `option` variant, for example, obliterates the need for nullable types, a major source of bugs in other languages. Philosophically speaking, a problem is composed of many possible branches/conditions. Mishandling these conditions is the majority of what we call bugs. **A type system doesn't magically eliminate bugs; it points out the unhandled conditions and asks you to cover them**\*. The ability to model "this or that" correctly is crucial. - -For example, some folks wonder how the type system can safely eliminate badly formatted JSON data from propagating into their program. They don't, not by themselves! But if the parser returns the `option` type `None | Some(actualData)`, then you'd have to handle the `None` case explicitly in later call sites. That's all there is. - -Performance-wise, a variant can potentially tremendously speed up your program's logic. Here's a piece of JavaScript: - -```js -let data = 'dog' -if (data === 'dog') { - ... -} else if (data === 'cat') { - ... -} else if (data === 'bird') { - ... -} -``` - -There's a linear amount of branch checking here (`O(n)`). Compare this to using a ReScript variant: - - - -```res example -type animal = Dog | Cat | Bird -let data = Dog -switch data { -| Dog => Console.log("Wof") -| Cat => Console.log("Meow") -| Bird => Console.log("Kashiiin") -} -``` -```js -console.log("Wof"); - -var data = "Dog"; -``` - - - -The compiler sees the variant, then - -1. conceptually turns them into `type animal = "Dog" | "Cat" | "Bird"` -2. compiles `switch` to a constant-time jump table (`O(1)`). diff --git a/pages/docs/manual/latest/warning-numbers.mdx b/pages/docs/manual/latest/warning-numbers.mdx deleted file mode 100644 index 80800170d..000000000 --- a/pages/docs/manual/latest/warning-numbers.mdx +++ /dev/null @@ -1,15 +0,0 @@ ---- -title: "Warning Numbers" -description: "Available compiler warning numbers in ReScript" -canonical: "/docs/manual/latest/warning-numbers" ---- - -import { make as WarningTable } from "src/components/WarningTable.mjs"; - -# Warning Numbers - -You can configure which warnings the ReScript compiler generates -[in the build configuration](/docs/manual/latest/build-configuration#warnings) or -using the [`@warning()`](/syntax-lookup#expression-warning-decorator) or the [`@@warning()`](/syntax-lookup#module-warning-decorator) decorator. - - diff --git a/scripts/extract-indices.mjs b/scripts/extract-indices.mjs index b70665c41..2b5372020 100644 --- a/scripts/extract-indices.mjs +++ b/scripts/extract-indices.mjs @@ -154,11 +154,8 @@ const extractApiIndex = version => { fs.writeFileSync(DOM_INDEX_FILE, JSON.stringify(domIndex), "utf8"); }; -// extractApiIndex("latest"); +// For versions > 10, please use gendocs.res extractApiIndex("v10.0.0"); extractApiIndex("v9.0.0"); extractApiIndex("v8.0.0"); - - -// v8.0.0 api stuff diff --git a/scripts/extract-tocs.mjs b/scripts/extract-tocs.mjs index c83a57bff..9eed38926 100644 --- a/scripts/extract-tocs.mjs +++ b/scripts/extract-tocs.mjs @@ -135,101 +135,18 @@ const createTOC = (result) => { }, {}); }; -const createLatestManualToc = () => { - const MD_DIR = path.join(__dirname, "../pages/docs/manual/latest"); - const SIDEBAR_JSON = path.join( - __dirname, - "../data/sidebar_manual_latest.json" - ); - const TARGET_FILE = path.join( - __dirname, - "../index_data/manual_latest_toc.json" - ); - - const sidebarJson = JSON.parse(fs.readFileSync(SIDEBAR_JSON)); +const createManualToc = (version) => { + const versionNoDot = version.replaceAll(".", ""); + const MD_DIR = path.join(__dirname, `../pages/docs/manual/${version}`); - const FILE_ORDER = Object.values(sidebarJson).reduce((acc, items) => { - return acc.concat(items); - }, []); - - const files = glob.sync(`${MD_DIR}/*.?(js|md?(x))`); - const ordered = orderFiles(files, FILE_ORDER); - - const result = ordered.map((filepath) => processFile(filepath, sidebarJson)); - const toc = createTOC(result); - - fs.writeFileSync(TARGET_FILE, JSON.stringify(toc), "utf8"); -}; - -const createV1000ManualToc = () => { - const MD_DIR = path.join(__dirname, "../pages/docs/manual/v10.0.0"); const SIDEBAR_JSON = path.join( __dirname, - "../data/sidebar_manual_v1000.json" - ); - const TARGET_FILE = path.join( - __dirname, - "../index_data/manual_v1000_toc.json" - ); - - const sidebarJson = JSON.parse(fs.readFileSync(SIDEBAR_JSON)); - - const FILE_ORDER = Object.values(sidebarJson).reduce((acc, items) => { - return acc.concat(items); - }, []); - - const files = glob.sync(`${MD_DIR}/*.?(js|md?(x))`); - const ordered = orderFiles(files, FILE_ORDER); - - const result = ordered.map((filepath) => processFile(filepath, sidebarJson)); - const toc = createTOC(result); - - fs.writeFileSync(TARGET_FILE, JSON.stringify(toc), "utf8"); -}; - -const createReasonCompilerToc = () => { - const MD_DIR = path.join(__dirname, "../pages/docs/reason-compiler/latest"); - const TARGET_FILE = path.join( - __dirname, - "../index_data/reason_compiler_toc.json" + `../data/sidebar_manual_${versionNoDot}.json` ); - const files = glob.sync(`${MD_DIR}/*.md?(x)`); - const result = files.map(processFile); - const toc = createTOC(result); - - fs.writeFileSync(TARGET_FILE, JSON.stringify(toc), "utf8"); -}; - -const createV900ManualToc = () => { - const MD_DIR = path.join(__dirname, "../pages/docs/manual/v9.0.0"); - const SIDEBAR_JSON = path.join(__dirname, "../data/sidebar_manual_v900.json"); const TARGET_FILE = path.join( __dirname, - "../index_data/manual_v900_toc.json" - ); - - const sidebarJson = JSON.parse(fs.readFileSync(SIDEBAR_JSON)); - - const FILE_ORDER = Object.values(sidebarJson).reduce((acc, items) => { - return acc.concat(items); - }, []); - - const files = glob.sync(`${MD_DIR}/*.?(js|md?(x))`); - const ordered = orderFiles(files, FILE_ORDER); - - const result = ordered.map((filepath) => processFile(filepath, sidebarJson)); - const toc = createTOC(result); - - fs.writeFileSync(TARGET_FILE, JSON.stringify(toc), "utf8"); -}; - -const createV800ManualToc = () => { - const MD_DIR = path.join(__dirname, "../pages/docs/manual/v8.0.0"); - const SIDEBAR_JSON = path.join(__dirname, "../data/sidebar_manual_v800.json"); - const TARGET_FILE = path.join( - __dirname, - "../index_data/manual_v800_toc.json" + `../index_data/manual_${versionNoDot}_toc.json` ); const sidebarJson = JSON.parse(fs.readFileSync(SIDEBAR_JSON)); @@ -298,7 +215,7 @@ const createCommunityToc = () => { const debugToc = () => { const MD_DIR = path.join(__dirname, "../pages/docs/manual/latest"); - const files = glob.sync(`${MD_DIR}/introduction.md?(x)`); + const files = glob.sync(`${MD_DIR}/introduction.md?(x)`); const result = files.map(processFile); const toc = createTOC(result); @@ -309,13 +226,9 @@ const debugToc = () => { debugToc(); */ -// main -createLatestManualToc(); -createV1000ManualToc(); -createV900ManualToc(); -createV800ManualToc(); -createReasonCompilerToc(); -createReactToc("latest"); -createReactToc("v0.10.0"); -createReactToc("v0.11.0"); +let manualVersions = ["v12.0.0", "v11.0.0", "v10.0.0", "v9.0.0", "v8.0.0"]; +let reactManualVersions = ["latest", "v0.10.0", "v0.11.0"]; + +manualVersions.forEach(createManualToc); +reactManualVersions.forEach(createReactToc); createCommunityToc(); diff --git a/src/ApiDocs.res b/src/ApiDocs.res index 0326e502e..f631c79f3 100644 --- a/src/ApiDocs.res +++ b/src/ApiDocs.res @@ -164,7 +164,7 @@ module SidebarTree = {
{React.string(node.name ++ " Module")} {switch url { - | Some({version}) => + | Some(url) => let onChange = evt => { open Url ReactEvent.Form.preventDefault(evt) @@ -177,10 +177,7 @@ module SidebarTree = { ("/" ++ (version ++ ("/" ++ Array.join(url.pagepath, "/"))))) router->Next.Router.push(targetUrl) } - let version = switch version { - | Latest | NoVersion => "latest" - | Version(version) => version - } + let version = url->Url.getVersionString let availableVersions = switch node.name { | "Core" => [("latest", "v11.0")] | _ => ApiLayout.allApiVersions @@ -351,10 +348,7 @@ let default = (props: props) => { | _ => React.null } - let version = switch Url.parse(router.asPath).version { - | Latest | NoVersion => "latest" - | Version(v) => v - } + let version = Url.parse(router.asPath)->Url.getVersionString let sidebar = switch props { | Ok({toctree, module_: {items}}) => diff --git a/src/DocsOverview.res b/src/DocsOverview.res index fb83d8568..8677d8d74 100644 --- a/src/DocsOverview.res +++ b/src/DocsOverview.res @@ -20,18 +20,9 @@ let default = (~showVersionSelect=true) => { let router = Next.Router.useRouter() let url = router.route->Url.parse - let version = switch url.version { - | Url.Latest => "latest" - | NoVersion => "latest" - | Version(version) => version - } + let version = url->Url.getVersionString - let languageManual = [ - ("Overview", `/docs/manual/${version}/introduction`), - ("Language Features", `/docs/manual/${version}/overview`), - ("JS Interop", `/docs/manual/${version}/embed-raw-javascript`), - ("Build System", `/docs/manual/${version}/build-overview`), - ] + let languageManual = Constants.languageManual(version) let ecosystem = [ ("Package Index", "/packages"), @@ -55,8 +46,14 @@ let default = (~showVersionSelect=true) => { ("/" ++ (version ++ ("/" ++ Array.join(url.pagepath, "/"))))) router->Next.Router.push(targetUrl) } +
- +
} else { React.null diff --git a/src/common/App.res b/src/common/App.res index 136b7fb0c..cd7d60aa2 100644 --- a/src/common/App.res +++ b/src/common/App.res @@ -115,6 +115,14 @@ let make = (props: props): React.element => { frontmatter}> content + | Version("v11.0.0") => + frontmatter}> + content + + | Version("v12.0.0") => + frontmatter}> + content + | _ => React.null } }} diff --git a/src/common/Constants.res b/src/common/Constants.res index dd9386857..6c49e0213 100644 --- a/src/common/Constants.res +++ b/src/common/Constants.res @@ -1,10 +1,24 @@ +type versionMapping = { + latest: string, + next: string, +} + +type versions = { + @as("VERSION_NEXT") next: string, + @as("VERSION_LATEST") latest: string, +} + +@scope("process") external versions: versions = "env" + // This is used for the version dropdown in the manual layouts let allManualVersions = [ - ("latest", "v11"), + ("latest", versions.latest), ("v10.0.0", "v9.1 - v10.1"), ("v9.0.0", "v8.2 - v9.0"), ("v8.0.0", "v6.0 - v8.2"), ] +let nextVersion = ("next", versions.next) + let allReactVersions = [("latest", "v0.12.0"), ("v0.11.0", "v0.11.0"), ("v0.10.0", "v0.10.0")] // Used for the DocsOverview and collapsible navigation diff --git a/src/common/Url.res b/src/common/Url.res index dfa46b0ed..8013901dd 100644 --- a/src/common/Url.res +++ b/src/common/Url.res @@ -1,5 +1,6 @@ type version = | Latest + | Next | NoVersion | Version(string) @@ -57,13 +58,14 @@ let prettyString = (str: string) => { let parse = (route: string): t => { let fullpath = route->String.split("/")->Array.filter(s => s !== "") let foundVersionIndex = Array.findIndex(fullpath, chunk => { - Re.test(%re(`/latest|v\d+(\.\d+)?(\.\d+)?/`), chunk) + Re.test(%re(`/latest|next|v\d+(\.\d+)?(\.\d+)?/`), chunk) }) let (version, base, pagepath) = if foundVersionIndex == -1 { (NoVersion, fullpath, []) } else { let version = switch fullpath[foundVersionIndex] { + | Some("next") => Next | Some("latest") => Latest | Some(v) => Version(v) | None => NoVersion @@ -77,3 +79,10 @@ let parse = (route: string): t => { {fullpath, base, version, pagepath} } + +let getVersionString = url => + switch url.version { + | Next => "next" + | Latest | NoVersion => "latest" + | Version(version) => version + } diff --git a/src/common/Url.resi b/src/common/Url.resi index 91343406b..57d1de25e 100644 --- a/src/common/Url.resi +++ b/src/common/Url.resi @@ -1,5 +1,6 @@ type version = | Latest + | Next | NoVersion | Version(string) @@ -19,3 +20,5 @@ type breadcrumb = { let parse: string => t let prettyString: string => string + +let getVersionString: t => string diff --git a/src/components/Navigation.res b/src/components/Navigation.res index e3ac14a0a..610b285c1 100644 --- a/src/components/Navigation.res +++ b/src/components/Navigation.res @@ -148,15 +148,10 @@ module DocsSection = { let router = Next.Router.useRouter() let url = router.route->Url.parse - let (version, setVersion) = React.useState(_ => - switch url.version { - | Url.Latest => "latest" - | NoVersion => "latest" - | Version(version) => version - } - ) + let (version, setVersion) = React.useState(_ => url->Url.getVersionString) let languageManual = Constants.languageManual(version) + let documentation = [ { imgSrc: "/static/ic_manual@2x.png", @@ -334,9 +329,16 @@ module DocsSection = {
{switch version { + | "next" => + + {React.string("This docs version is work in progress!")} + | "latest" => {React.string("This is the latest docs version")} diff --git a/src/components/Search.res b/src/components/Search.res index 6a30dbec8..a01f5161e 100644 --- a/src/components/Search.res +++ b/src/components/Search.res @@ -77,10 +77,7 @@ let make = () => { let (state, setState) = React.useState(_ => Inactive) let router = Next.Router.useRouter() - let version = switch Url.parse(router.route).version { - | Version(v) => v - | _ => "latest" - } + let version = Url.parse(router.route)->Url.getVersionString let handleCloseModal = () => { let () = switch ReactDOM.querySelector(".DocSearch-Modal") { diff --git a/src/components/VersionSelect.res b/src/components/VersionSelect.res index a10c20bec..b4bad6266 100644 --- a/src/components/VersionSelect.res +++ b/src/components/VersionSelect.res @@ -1,7 +1,11 @@ @react.component -let make = (~onChange, ~version: string, ~availableVersions: array<(string, string)>) => { +let make = ( + ~onChange, + ~version: string, + ~nextVersion: option<(string, string)>=?, + ~availableVersions: array<(string, string)>, +) => { // array<(version, label)> - let children = Array.map(availableVersions, ((ver, label)) => { }) @@ -10,6 +14,17 @@ let make = (~onChange, ~version: string, ~availableVersions: array<(string, stri name="versionSelection" value=version onChange> + {nextVersion->Option.isSome + ? + : React.null} {React.array(children)} + {switch nextVersion { + | None => React.null + | Some((value, label)) => + <> + + + + }} } diff --git a/src/components/VersionSelect.resi b/src/components/VersionSelect.resi index dddf92b43..18568d4b9 100644 --- a/src/components/VersionSelect.resi +++ b/src/components/VersionSelect.resi @@ -2,5 +2,6 @@ let make: ( ~onChange: ReactEvent.Form.t => unit, ~version: string, + ~nextVersion: (string, string)=?, ~availableVersions: array<(string, string)>, ) => React.element diff --git a/src/layouts/BeltDocsLayout10_0_0.res b/src/layouts/BeltDocsLayout10_0_0.res index ab1301fa9..1a3e94d6a 100644 --- a/src/layouts/BeltDocsLayout10_0_0.res +++ b/src/layouts/BeltDocsLayout10_0_0.res @@ -122,11 +122,7 @@ module Docs = { let url = route->Url.parse - let version = switch url.version { - | Version(version) => version - | NoVersion => "latest" - | Latest => "latest" - } + let version = url->Url.getVersionString let prefix = { open Url diff --git a/src/layouts/BeltDocsLayout8_0_0.res b/src/layouts/BeltDocsLayout8_0_0.res index 32a53b5a7..614adad86 100644 --- a/src/layouts/BeltDocsLayout8_0_0.res +++ b/src/layouts/BeltDocsLayout8_0_0.res @@ -149,11 +149,7 @@ module Docs = { let url = route->Url.parse - let version = switch url.version { - | Version(version) => version - | NoVersion => "latest" - | Latest => "latest" - } + let version = url->Url.getVersionString let prefix = { open Url diff --git a/src/layouts/BeltDocsLayout9_0_0.res b/src/layouts/BeltDocsLayout9_0_0.res index 71df69ca1..c40eb5a4c 100644 --- a/src/layouts/BeltDocsLayout9_0_0.res +++ b/src/layouts/BeltDocsLayout9_0_0.res @@ -122,11 +122,7 @@ module Docs = { let url = route->Url.parse - let version = switch url.version { - | Version(version) => version - | NoVersion => "latest" - | Latest => "latest" - } + let version = url->Url.getVersionString let prefix = { open Url diff --git a/src/layouts/DocsLayout.res b/src/layouts/DocsLayout.res index d66ddc43c..1d50c6b4a 100644 --- a/src/layouts/DocsLayout.res +++ b/src/layouts/DocsLayout.res @@ -54,6 +54,7 @@ let make = ( ~frontmatter=?, ~version: option=?, ~availableVersions: option>=?, + ~nextVersion: option<(string, string)>=?, ~activeToc: option=?, ~categories: array, ~components=MarkdownComponents.default, @@ -102,7 +103,7 @@ let make = ( ("/" ++ (version ++ ("/" ++ Array.join(url.pagepath, "/"))))) router->Next.Router.push(targetUrl) } - + | None => {React.string(version)} } | None => React.null @@ -215,9 +216,7 @@ module Make = (Content: StaticContent) => { Array.push(arr, next)->ignore acc->Dict.set(category, arr) } - | None => - Console.log2("has NO category", next) - () + | None => Console.log2("has NO category", next) } acc }) @@ -240,6 +239,7 @@ module Make = (Content: StaticContent) => { ?frontmatter, ?version, ?availableVersions, + nextVersion: Constants.nextVersion, ?activeToc, categories, ?components, diff --git a/src/layouts/DocsLayout.resi b/src/layouts/DocsLayout.resi index f02706e2a..626611a22 100644 --- a/src/layouts/DocsLayout.resi +++ b/src/layouts/DocsLayout.resi @@ -9,6 +9,7 @@ let make: ( ~frontmatter: JSON.t=?, ~version: string=?, ~availableVersions: array<(string, string)>=?, + ~nextVersion: (string, string)=?, ~activeToc: SidebarLayout.Toc.t=?, ~categories: array, ~components: MarkdownComponents.t=?, diff --git a/src/layouts/DomDocsLayout10_0_0.res b/src/layouts/DomDocsLayout10_0_0.res index a6bd3743b..73bd5a4c0 100644 --- a/src/layouts/DomDocsLayout10_0_0.res +++ b/src/layouts/DomDocsLayout10_0_0.res @@ -58,11 +58,7 @@ module Docs = { let url = route->Url.parse - let version = switch url.version { - | Version(version) => version - | NoVersion => "latest" - | Latest => "latest" - } + let version = url->Url.getVersionString let prefix = { open Url diff --git a/src/layouts/DomDocsLayout8_0_0.res b/src/layouts/DomDocsLayout8_0_0.res index c0155d85f..259dcbb88 100644 --- a/src/layouts/DomDocsLayout8_0_0.res +++ b/src/layouts/DomDocsLayout8_0_0.res @@ -58,11 +58,7 @@ module Docs = { let url = route->Url.parse - let version = switch url.version { - | Version(version) => version - | NoVersion => "latest" - | Latest => "latest" - } + let version = url->Url.getVersionString let prefix = { open Url diff --git a/src/layouts/DomDocsLayout9_0_0.res b/src/layouts/DomDocsLayout9_0_0.res index 75baff1f8..79827dc7d 100644 --- a/src/layouts/DomDocsLayout9_0_0.res +++ b/src/layouts/DomDocsLayout9_0_0.res @@ -58,11 +58,7 @@ module Docs = { let url = route->Url.parse - let version = switch url.version { - | Version(version) => version - | NoVersion => "latest" - | Latest => "latest" - } + let version = url->Url.getVersionString let prefix = { open Url diff --git a/src/layouts/JsDocsLayout10_0_0.res b/src/layouts/JsDocsLayout10_0_0.res index 5ac403eed..d44dbf21c 100644 --- a/src/layouts/JsDocsLayout10_0_0.res +++ b/src/layouts/JsDocsLayout10_0_0.res @@ -166,11 +166,7 @@ module Docs = { let url = route->Url.parse - let version = switch url.version { - | Version(version) => version - | NoVersion => "latest" - | Latest => "latest" - } + let version = url->Url.getVersionString let prefix = { open Url diff --git a/src/layouts/JsDocsLayout8_0_0.res b/src/layouts/JsDocsLayout8_0_0.res index 33521843c..31681f9bc 100644 --- a/src/layouts/JsDocsLayout8_0_0.res +++ b/src/layouts/JsDocsLayout8_0_0.res @@ -172,11 +172,7 @@ module Docs = { let url = route->Url.parse - let version = switch url.version { - | Version(version) => version - | NoVersion => "latest" - | Latest => "latest" - } + let version = url->Url.getVersionString let prefix = { open Url diff --git a/src/layouts/JsDocsLayout9_0_0.res b/src/layouts/JsDocsLayout9_0_0.res index aa95c1ce1..cb766f6e9 100644 --- a/src/layouts/JsDocsLayout9_0_0.res +++ b/src/layouts/JsDocsLayout9_0_0.res @@ -172,11 +172,7 @@ module Docs = { let url = route->Url.parse - let version = switch url.version { - | Version(version) => version - | NoVersion => "latest" - | Latest => "latest" - } + let version = url->Url.getVersionString let prefix = { open Url diff --git a/src/layouts/ManualDocsLayout.res b/src/layouts/ManualDocsLayout.res index fcdeaf6be..d44228a0a 100644 --- a/src/layouts/ManualDocsLayout.res +++ b/src/layouts/ManualDocsLayout.res @@ -1,8 +1,3 @@ -module LatestLayout = DocsLayout.Make({ - // Structure defined by `scripts/extract-tocs.js` - @module("index_data/manual_latest_toc.json") external tocData: SidebarLayout.Toc.raw = "default" -}) - module V800Layout = DocsLayout.Make({ // Structure defined by `scripts/extract-tocs.js` @module("index_data/manual_v800_toc.json") external tocData: SidebarLayout.Toc.raw = "default" @@ -18,38 +13,55 @@ module V1000Layout = DocsLayout.Make({ @module("index_data/manual_v1000_toc.json") external tocData: SidebarLayout.Toc.raw = "default" }) -module Latest = { +module V1100Layout = DocsLayout.Make({ + // Structure defined by `scripts/extract-tocs.js` + @module("index_data/manual_v1100_toc.json") external tocData: SidebarLayout.Toc.raw = "default" +}) + +module V1200Layout = DocsLayout.Make({ + // Structure defined by `scripts/extract-tocs.js` + @module("index_data/manual_v1200_toc.json") external tocData: SidebarLayout.Toc.raw = "default" +}) + +module V1200 = { @react.component let make = (~frontmatter=?, ~components=MarkdownComponents.default, ~children) => { + let title = "Language Manual" let router = Next.Router.useRouter() - let route = router.route - - let url = route->Url.parse - - let version = switch url.version { - | Version(version) => version - | NoVersion => "latest" - | Latest => "latest" - } + let version = router.route->Url.parse->Url.getVersionString let breadcrumbs = list{ - { - open Url - {name: "Docs", href: "/docs/" ++ version} - }, - { - open Url - { - name: "Language Manual", - href: "/docs/manual/" ++ (version ++ "/introduction"), - } - }, + {Url.name: "Docs", href: "/docs/" ++ version}, + {Url.name: "Language Manual", href: "/docs/manual/" ++ (version ++ "/introduction")}, } + + children + + } +} + +module V1100 = { + @react.component + let make = (~frontmatter=?, ~components=MarkdownComponents.default, ~children) => { let title = "Language Manual" - let version = "latest" + let router = Next.Router.useRouter() + let version = router.route->Url.parse->Url.getVersionString + + let breadcrumbs = list{ + {Url.name: "Docs", href: "/docs/" ++ version}, + {Url.name: "Language Manual", href: "/docs/manual/" ++ (version ++ "/introduction")}, + } - children - + } } @@ -71,11 +83,7 @@ module V1000 = { let url = route->Url.parse - let version = switch url.version { - | Version(version) => version - | NoVersion => "latest" - | Latest => "latest" - } + let version = url->Url.getVersionString let breadcrumbs = list{ { @@ -119,11 +127,7 @@ module V900 = { let url = route->Url.parse - let version = switch url.version { - | Version(version) => version - | NoVersion => "latest" - | Latest => "latest" - } + let version = url->Url.getVersionString let breadcrumbs = list{ { @@ -163,11 +167,7 @@ module V800 = { let url = route->Url.parse - let version = switch url.version { - | Version(version) => version - | NoVersion => "latest" - | Latest => "latest" - } + let version = url->Url.getVersionString let breadcrumbs = list{ { @@ -234,3 +234,5 @@ module V800 = { } } + +module Latest = V1100 diff --git a/src/layouts/ReactDocsLayout.res b/src/layouts/ReactDocsLayout.res index 3f8d3bdc1..7beb45d75 100644 --- a/src/layouts/ReactDocsLayout.res +++ b/src/layouts/ReactDocsLayout.res @@ -24,27 +24,12 @@ module Latest = { let make = (~frontmatter=?, ~components=MarkdownComponents.default, ~children) => { let router = Next.Router.useRouter() let route = router.route - let url = route->Url.parse - - let version = switch url.version { - | Version(version) => version - | NoVersion => "latest" - | Latest => "latest" - } + let version = url->Url.getVersionString let breadcrumbs = list{ - { - open Url - {name: "Docs", href: "/docs/latest"} - }, - { - open Url - { - name: "rescript-react", - href: "/docs/react/" ++ (version ++ "/introduction"), - } - }, + {Url.name: "Docs", href: "/docs/latest"}, + {name: "rescript-react", href: "/docs/react/" ++ (version ++ "/introduction")}, } let title = "rescript-react" @@ -69,27 +54,12 @@ module V0110 = { let make = (~frontmatter=?, ~components=MarkdownComponents.default, ~children) => { let router = Next.Router.useRouter() let route = router.route - let url = route->Url.parse - - let version = switch url.version { - | Version(version) => version - | NoVersion => "latest" - | Latest => "latest" - } + let version = url->Url.getVersionString let breadcrumbs = list{ - { - open Url - {name: "Docs", href: "/docs/latest"} - }, - { - open Url - { - name: "rescript-react", - href: "/docs/react/" ++ (version ++ "/introduction"), - } - }, + {Url.name: "Docs", href: "/docs/latest"}, + {name: "rescript-react", href: "/docs/react/" ++ (version ++ "/introduction")}, } let title = "rescript-react" @@ -113,27 +83,12 @@ module V0100 = { let make = (~frontmatter=?, ~components=MarkdownComponents.default, ~children) => { let router = Next.Router.useRouter() let route = router.route - let url = route->Url.parse - - let version = switch url.version { - | Version(version) => version - | NoVersion => "latest" - | Latest => "latest" - } + let version = url->Url.getVersionString let breadcrumbs = list{ - { - open Url - {name: "Docs", href: "/docs/latest"} - }, - { - open Url - { - name: "rescript-react", - href: "/docs/react/" ++ (version ++ "/introduction"), - } - }, + {Url.name: "Docs", href: "/docs/latest"}, + {name: "rescript-react", href: "/docs/react/" ++ (version ++ "/introduction")}, } let title = "rescript-react" diff --git a/src/layouts/ReasonCompilerDocsLayout.res b/src/layouts/ReasonCompilerDocsLayout.res index b9883af68..bea7038fb 100644 --- a/src/layouts/ReasonCompilerDocsLayout.res +++ b/src/layouts/ReasonCompilerDocsLayout.res @@ -14,11 +14,8 @@ module Toc = SidebarLayout.Toc let interopNavs = [ { - open NavItem - { - name: "Overview", - href: "/docs/reason-compiler/latest/interop-overview", - } + NavItem.name: "Overview", + href: "/docs/reason-compiler/latest/interop-overview", }, { name: "Better Data Structures Printing (Debug Mode)", @@ -30,11 +27,8 @@ let interopNavs = [ let advancedNavs = [ { - open NavItem - { - name: "Conditional Compilation", - href: "/docs/reason-compiler/latest/conditional-compilation", - } + NavItem.name: "Conditional Compilation", + href: "/docs/reason-compiler/latest/conditional-compilation", }, { name: "Extended Compiler Options", @@ -51,10 +45,7 @@ let advancedNavs = [ ] let categories = [ - { - open Category - {name: "Interop", items: interopNavs} - }, + {Category.name: "Interop", items: interopNavs}, {name: "Advanced", items: advancedNavs}, ] @@ -77,24 +68,11 @@ let make = (~components=MarkdownComponents.default, ~children) => { let url = route->Url.parse - let version = switch url.version { - | Version(version) => version - | NoVersion => "latest" - | Latest => "latest" - } + let version = url->Url.getVersionString let prefix = list{ - { - open Url - {name: "Docs", href: "/docs/" ++ version} - }, - { - open Url - { - name: "Old Docs", - href: "/docs/reason-compiler/" ++ (version ++ "/interop-overview"), - } - }, + {Url.name: "Docs", href: "/docs/" ++ version}, + {name: "Old Docs", href: "/docs/reason-compiler/" ++ (version ++ "/interop-overview")}, } let breadcrumbs = List.concat(