partial.lenses
Advanced tools
Comparing version 0.6.0 to 0.7.0
@@ -94,2 +94,7 @@ "use strict"; | ||
}); | ||
L.deleteAll = _ramda2.default.curry(function (lens, data) { | ||
while (L.view(lens, data) !== undefined) { | ||
data = L.delete(lens, data); | ||
}return data; | ||
}); | ||
L.lens = _ramda2.default.lens; | ||
@@ -162,7 +167,17 @@ L.over = _ramda2.default.curry(function (l, x2x, s) { | ||
var i = xs.findIndex(predicate); | ||
if (i < 0) return L.append; | ||
return L.index(i); | ||
return i < 0 ? L.append : i; | ||
}); | ||
}; | ||
L.findWith = function (l) { | ||
for (var _len3 = arguments.length, ls = Array(_len3 > 1 ? _len3 - 1 : 0), _key3 = 1; _key3 < _len3; _key3++) { | ||
ls[_key3 - 1] = arguments[_key3]; | ||
} | ||
var lls = L.apply(undefined, [l].concat(ls)); | ||
return L(L.find(function (x) { | ||
return L.view(lls, x) !== undefined; | ||
}), lls); | ||
}; | ||
L.index = function (i) { | ||
@@ -186,7 +201,3 @@ return _ramda2.default.lens(function (xs) { | ||
L.append = _ramda2.default.lens(function () {}, function (x, xs) { | ||
if (x === undefined) { | ||
return xs; | ||
} else { | ||
if (xs === undefined) return [x];else return xs.concat([x]); | ||
} | ||
return x === undefined ? xs : xs === undefined ? [x] : xs.concat([x]); | ||
}); | ||
@@ -203,2 +214,2 @@ | ||
exports.default = L; | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/partial.lenses.js"],"names":[],"mappings":"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","file":"partial.lenses.js","sourcesContent":["import R from \"ramda\"\n\n//\n\nconst deleteKey = (k, o) => {\n  if (o === undefined || !(k in o))\n    return o\n  let r\n  for (const p in o) {\n    if (p !== k) {\n      if (undefined === r)\n        r = {}\n      r[p] = o[p]\n    }\n  }\n  return r\n}\n\nconst setKey = (k, v, o) => {\n  if (o === undefined)\n    return {[k]: v}\n  if (k in o && R.equals(v, o[k]))\n    return o\n  const r = {[k]: v}\n  for (const p in o)\n    if (p !== k)\n      r[p] = o[p]\n  return r\n}\n\n//\n\nconst dropped = xs => Object.keys(xs).length === 0 ? undefined : xs\n\n//\n\nconst toPartial = transform => x => undefined === x ? x : transform(x)\n\n//\n\nconst conserve = (c0, c1) => R.equals(c0, c1) ? c0 : c1\n\nconst toConserve = f => (y, c0) =>  conserve(c0, f(y, c0))\n\n//\n\nexport const lift = l => {\n  switch (typeof l) {\n  case \"string\": return L.prop(l)\n  case \"number\": return L.index(l)\n  default:       return l\n  }\n}\n\nconst L = (l, ...ls) =>\n  ls.length === 0 ? lift(l) : R.compose(lift(l), ...ls.map(lift))\n\nL.compose = L\nL.delete = R.curry((l, s) => R.set(lift(l), undefined, s))\nL.lens = R.lens\nL.over = R.curry((l, x2x, s) => R.over(lift(l), x2x, s))\nL.set = R.curry((l, x, s) => R.set(lift(l), x, s))\nL.view = R.curry((l, s) => R.view(lift(l), s))\n\nL.choose = x2yL => toFunctor => target => {\n  const l = lift(x2yL(target))\n  return R.map(focus => R.set(l, focus, target), toFunctor(R.view(l, target)))\n}\n\nL.firstOf = (l, ...ls) => L.choose(x => {\n  const lls = [l, ...ls]\n  return lls[Math.max(0, lls.findIndex(l => L.view(l, x) !== undefined))]\n})\n\nL.replace = R.curry((inn, out) =>\n  R.lens(x => R.equals(x, inn) ? out : x,\n         toConserve(y => R.equals(y, out) ? inn : y)))\n\nL.default = L.replace(undefined)\nL.required = inn => L.replace(inn, undefined)\nL.define = v => R.compose(L.required(v), L.default(v))\n\nL.normalize = transform =>\n  R.lens(toPartial(transform), toConserve(toPartial(transform)))\n\nL.prop = k =>\n  R.lens(o => o && o[k],\n         (v, o) => v === undefined ? deleteKey(k, o) : setKey(k, v, o))\n\nL.find = predicate => L.choose(xs => {\n  if (xs === undefined)\n    return L.append\n  const i = xs.findIndex(predicate)\n  if (i < 0)\n    return L.append\n  return L.index(i)\n})\n\nL.index = i => R.lens(xs => xs && xs[i], (x, xs) => {\n  if (x === undefined) {\n    if (xs === undefined)\n      return undefined\n    if (i < xs.length)\n      return dropped(xs.slice(0, i).concat(xs.slice(i+1)))\n    return xs\n  } else {\n    if (xs === undefined)\n      return Array(i).concat([x])\n    if (xs.length <= i)\n      return xs.concat(Array(i - xs.length), [x])\n    if (R.equals(x, xs[i]))\n      return xs\n    return xs.slice(0, i).concat([x], xs.slice(i+1))\n  }\n})\n\nL.append = R.lens(() => {}, (x, xs) => {\n  if (x === undefined) {\n    return xs\n  } else {\n    if (xs === undefined)\n      return [x]\n    else\n      return xs.concat([x])\n  }\n})\n\nL.filter = p => R.lens(xs => xs && xs.filter(p), (ys, xs) =>\n  conserve(xs, dropped(R.concat(ys || [], (xs || []).filter(R.complement(p))))))\n\nexport default L\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/partial.lenses.js"],"names":[],"mappings":"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","file":"partial.lenses.js","sourcesContent":["import R from \"ramda\"\n\n//\n\nconst deleteKey = (k, o) => {\n  if (o === undefined || !(k in o))\n    return o\n  let r\n  for (const p in o) {\n    if (p !== k) {\n      if (undefined === r)\n        r = {}\n      r[p] = o[p]\n    }\n  }\n  return r\n}\n\nconst setKey = (k, v, o) => {\n  if (o === undefined)\n    return {[k]: v}\n  if (k in o && R.equals(v, o[k]))\n    return o\n  const r = {[k]: v}\n  for (const p in o)\n    if (p !== k)\n      r[p] = o[p]\n  return r\n}\n\n//\n\nconst dropped = xs => Object.keys(xs).length === 0 ? undefined : xs\n\n//\n\nconst toPartial = transform => x => undefined === x ? x : transform(x)\n\n//\n\nconst conserve = (c0, c1) => R.equals(c0, c1) ? c0 : c1\n\nconst toConserve = f => (y, c0) =>  conserve(c0, f(y, c0))\n\n//\n\nexport const lift = l => {\n  switch (typeof l) {\n  case \"string\": return L.prop(l)\n  case \"number\": return L.index(l)\n  default:       return l\n  }\n}\n\nconst L = (l, ...ls) =>\n  ls.length === 0 ? lift(l) : R.compose(lift(l), ...ls.map(lift))\n\nL.compose = L\nL.delete = R.curry((l, s) => R.set(lift(l), undefined, s))\nL.deleteAll = R.curry((lens, data) => {\n  while (L.view(lens, data) !== undefined)\n    data = L.delete(lens, data)\n  return data\n})\nL.lens = R.lens\nL.over = R.curry((l, x2x, s) => R.over(lift(l), x2x, s))\nL.set = R.curry((l, x, s) => R.set(lift(l), x, s))\nL.view = R.curry((l, s) => R.view(lift(l), s))\n\nL.choose = x2yL => toFunctor => target => {\n  const l = lift(x2yL(target))\n  return R.map(focus => R.set(l, focus, target), toFunctor(R.view(l, target)))\n}\n\nL.firstOf = (l, ...ls) => L.choose(x => {\n  const lls = [l, ...ls]\n  return lls[Math.max(0, lls.findIndex(l => L.view(l, x) !== undefined))]\n})\n\nL.replace = R.curry((inn, out) =>\n  R.lens(x => R.equals(x, inn) ? out : x,\n         toConserve(y => R.equals(y, out) ? inn : y)))\n\nL.default = L.replace(undefined)\nL.required = inn => L.replace(inn, undefined)\nL.define = v => R.compose(L.required(v), L.default(v))\n\nL.normalize = transform =>\n  R.lens(toPartial(transform), toConserve(toPartial(transform)))\n\nL.prop = k =>\n  R.lens(o => o && o[k],\n         (v, o) => v === undefined ? deleteKey(k, o) : setKey(k, v, o))\n\nL.find = predicate => L.choose(xs => {\n  if (xs === undefined)\n    return L.append\n  const i = xs.findIndex(predicate)\n  return i < 0 ? L.append : i\n})\n\nL.findWith = (l, ...ls) => {\n  const lls = L(l, ...ls)\n  return L(L.find(x => L.view(lls, x) !== undefined), lls)\n}\n\nL.index = i => R.lens(xs => xs && xs[i], (x, xs) => {\n  if (x === undefined) {\n    if (xs === undefined)\n      return undefined\n    if (i < xs.length)\n      return dropped(xs.slice(0, i).concat(xs.slice(i+1)))\n    return xs\n  } else {\n    if (xs === undefined)\n      return Array(i).concat([x])\n    if (xs.length <= i)\n      return xs.concat(Array(i - xs.length), [x])\n    if (R.equals(x, xs[i]))\n      return xs\n    return xs.slice(0, i).concat([x], xs.slice(i+1))\n  }\n})\n\nL.append = R.lens(() => {}, (x, xs) =>\n  x === undefined ? xs : xs === undefined ? [x] : xs.concat([x]))\n\nL.filter = p => R.lens(xs => xs && xs.filter(p), (ys, xs) =>\n  conserve(xs, dropped(R.concat(ys || [], (xs || []).filter(R.complement(p))))))\n\nexport default L\n"]} |
{ | ||
"name": "partial.lenses", | ||
"version": "0.6.0", | ||
"version": "0.7.0", | ||
"description": "Ramda compatible lenses", | ||
@@ -33,9 +33,9 @@ "main": "lib/partial.lenses.js", | ||
"babel-cli": "^6.5.1", | ||
"babel-eslint": "^4.1.8", | ||
"babel-eslint": "^5.0.0", | ||
"babel-plugin-transform-es2015-modules-commonjs": "^6.5.2", | ||
"babel-preset-es2015": "^6.5.0", | ||
"babel-preset-stage-2": "^6.5.0", | ||
"eslint": "^2.0.0", | ||
"eslint": "^2.2.0", | ||
"mocha": "^2.4.5" | ||
} | ||
} |
157
README.md
@@ -1,2 +0,2 @@ | ||
[ [Examples](#examples) | [Reference](#reference) | [Background](#background) ] | ||
[ [Tutorial](#tutorial) | [Reference](#reference) | [Background](#background) ] | ||
@@ -16,5 +16,5 @@ This library provides a collection of [Ramda](http://ramdajs.com/) compatible | ||
[![npm version](https://badge.fury.io/js/partial.lenses.svg)](http://badge.fury.io/js/partial.lenses) | ||
[![npm version](https://badge.fury.io/js/partial.lenses.svg)](http://badge.fury.io/js/partial.lenses) [![](https://david-dm.org/dirty-js/partial.lenses.svg)](https://david-dm.org/dirty-js/partial.lenses) | ||
## Examples | ||
## Tutorial | ||
@@ -28,5 +28,12 @@ Let's work with the following sample JSON object: | ||
First we define a parameterized lens for accessing texts: | ||
First we import libraries | ||
```js | ||
import L from "partial.lenses" | ||
import R from "ramda" | ||
``` | ||
and compose a parameterized lens for accessing texts: | ||
```js | ||
const textIn = language => | ||
@@ -42,5 +49,11 @@ L.compose(L.prop("contents"), | ||
Like with ordinary lenses, we can now use the partial lens to view or query | ||
texts: | ||
Take a moment to read through the above definition line by line. Each line has | ||
a specific purpose. The purpose of the `L.prop(...)` lines is probably obvious. | ||
The other lines we will mention below. | ||
### Querying data | ||
Thanks to the parameterized search part, `L.find(R.whereEq({language}))`, of the | ||
lens composition, we can use it to query texts: | ||
```js | ||
@@ -53,3 +66,4 @@ > L.view(textIn("sv"), data) | ||
If we query a text that does not exist, we get the default: | ||
Partial lenses can deal with missing data. If we use the partial lens to query | ||
a text that does not exist, we get the default: | ||
@@ -61,4 +75,5 @@ ```js | ||
With the partial lens we defined, we get the default even if we query from | ||
`undefined`: | ||
We get this default, rather than undefined, thanks to the last part, | ||
`L.default("")`, of our lens composition. We get the default even if we query | ||
from `undefined`: | ||
@@ -72,2 +87,4 @@ ```js | ||
### Updating data | ||
As with ordinary lenses, we can use the same lens to update texts: | ||
@@ -81,2 +98,4 @@ | ||
### Inserting data | ||
The same partial lens also allows us to insert new texts: | ||
@@ -91,4 +110,8 @@ | ||
Note the position into which the new text was inserted. | ||
Note the position into which the new text was inserted. The array of texts is | ||
kept sorted thanks to the `L.normalize(R.sortBy(R.prop("language")))` part of | ||
our lens. | ||
### Deleting data | ||
Finally, we can use the same partial lens to delete texts: | ||
@@ -101,2 +124,8 @@ | ||
Note that a single text is actually a part of an object. The key to having the | ||
whole object vanish, rather than just the `text` property, is the | ||
`L.default({language})` part of our lens composition. A `L.default(value)` lens | ||
works *symmetrically*. When set with `value`, the result is `undefined`, which | ||
means that the focus of the lens is to be deleted. | ||
If we delete all of the texts, we get the required value: | ||
@@ -110,7 +139,21 @@ | ||
The `contents` property is not removed thanks to the `L.required([])` part of | ||
our lens composition. `L.required` is the dual of `L.default`. `L.default` | ||
replaces undefined values when viewed and `L.required` replaces undefined values | ||
when set. | ||
Note that unless required and default values are explicitly specified as part of | ||
the lens, they will both be undefined. | ||
For clarity, the code snippets in this section avoided some of the shorthands | ||
that this library supports. In particular, | ||
### Exercise | ||
Take out one (or more) `L.required(...)`, `L.normalize(...)` or `L.default(...)` | ||
part(s) from the lens composition and try to predict what happens when you rerun | ||
the examples with the modified lens composition. Verify your reasoning by | ||
actually rerunning the examples. | ||
### Shorthands | ||
For clarity, the previous code snippets avoided some of the shorthands that this | ||
library supports. In particular, | ||
* `L.compose(...)` can be abbreviated as `L(...)`, | ||
@@ -120,2 +163,59 @@ * `L.prop(string)` can be abbreviated as `string`, and | ||
### Systematic decomposition | ||
It is also typical to compose lenses out of short paths following the schema of | ||
the JSON data being manipulated. Reconsider the lens from the start of the | ||
example: | ||
```js | ||
const textIn = language => | ||
L.compose(L.prop("contents"), | ||
L.required([]), | ||
L.normalize(R.sortBy(R.prop("language"))), | ||
L.find(R.whereEq({language})), | ||
L.default({language}), | ||
L.prop("text"), | ||
L.default("")) | ||
``` | ||
Following the structure or schema of the JSON, we could break this into three | ||
separate lenses: | ||
* a lens for accessing the contents of a data object, | ||
* a parameterized lens for querying a content object from contents, and | ||
* a lens for accessing the text of a content object. | ||
Furthermore, we could organize the lenses into an object following the structure | ||
of the JSON: | ||
```js | ||
const M = { | ||
data: { | ||
contents: L("contents", | ||
L.required([]), | ||
L.normalize(R.sortBy(R.prop("language")))) | ||
}, | ||
contents: { | ||
contentIn: language => L(L.find(R.whereEq({language})), | ||
L.default({language})) | ||
}, | ||
content: { | ||
text: L("text", L.default("")) | ||
} | ||
} | ||
``` | ||
Using the above object, we could rewrite the parameterized `textIn` lens as: | ||
```js | ||
const textIn = language => L(M.data.contents, | ||
M.contents.contentIn(language), | ||
M.content.text) | ||
``` | ||
This style of organizing lenses is overkill for our toy example. In a more | ||
realistic case the `data` object would contain many more properties. Also, | ||
rather than composing a lens, like `textIn` above, to access a leaf property | ||
from the root of our object, we might actually compose lenses incrementally as | ||
we inspect the model structure. | ||
## Reference | ||
@@ -170,2 +270,9 @@ | ||
#### L.deleteAll(l, s) | ||
`L.deleteAll(l, s)` deletes all the non `undefined` items targeted by the lens | ||
`l` from `s`. This only makes sense for a lens that | ||
* can potentially focus on more than one item and | ||
* will focus on `undefined` when it doesn't find an item to focus on. | ||
### Lenses | ||
@@ -186,4 +293,12 @@ | ||
determined by the given function that maps the underlying view, which can be | ||
undefined, to a lens. | ||
undefined, to a lens. The lens returned by the given function will be lifted. | ||
Note that the type of `L.choose` is | ||
```haskell | ||
choose :: (Maybe a -> PartialLens b s) -> PartialLens a s -> PartialLens b s | ||
``` | ||
which is very similar to the type of the monadic bind operation. | ||
#### L.filter(predicate) | ||
@@ -211,2 +326,17 @@ | ||
#### L.findWith(l, ...ls) | ||
`L.findWith(l, ...ls)` is defined as | ||
```js | ||
L.findWith = (l, ...ls) => { | ||
const lls = L(l, ...ls) | ||
return L(L.find(x => L.view(lls, x) !== undefined), lls) | ||
} | ||
``` | ||
and basically chooses an index from an array through which the given lens, `L(l, | ||
...ls)`, focuses on a defined item and then returns a lens that focuses on that | ||
item. | ||
#### L.firstOf(l, ...ls) | ||
@@ -222,2 +352,3 @@ | ||
#### L.index(integer) | ||
@@ -224,0 +355,0 @@ |
@@ -60,2 +60,7 @@ import R from "ramda" | ||
L.delete = R.curry((l, s) => R.set(lift(l), undefined, s)) | ||
L.deleteAll = R.curry((lens, data) => { | ||
while (L.view(lens, data) !== undefined) | ||
data = L.delete(lens, data) | ||
return data | ||
}) | ||
L.lens = R.lens | ||
@@ -95,7 +100,10 @@ L.over = R.curry((l, x2x, s) => R.over(lift(l), x2x, s)) | ||
const i = xs.findIndex(predicate) | ||
if (i < 0) | ||
return L.append | ||
return L.index(i) | ||
return i < 0 ? L.append : i | ||
}) | ||
L.findWith = (l, ...ls) => { | ||
const lls = L(l, ...ls) | ||
return L(L.find(x => L.view(lls, x) !== undefined), lls) | ||
} | ||
L.index = i => R.lens(xs => xs && xs[i], (x, xs) => { | ||
@@ -119,12 +127,4 @@ if (x === undefined) { | ||
L.append = R.lens(() => {}, (x, xs) => { | ||
if (x === undefined) { | ||
return xs | ||
} else { | ||
if (xs === undefined) | ||
return [x] | ||
else | ||
return xs.concat([x]) | ||
} | ||
}) | ||
L.append = R.lens(() => {}, (x, xs) => | ||
x === undefined ? xs : xs === undefined ? [x] : xs.concat([x])) | ||
@@ -131,0 +131,0 @@ L.filter = p => R.lens(xs => xs && xs.filter(p), (ys, xs) => |
@@ -115,2 +115,8 @@ import R from "ramda" | ||
describe("L.findWith", () => { | ||
testEq('L.view(L.findWith("x", 1), [{x: ["a"]},{x: ["b","c"]}])', "c") | ||
testEq('L.set(L.findWith("x", 1), "d", [{x: ["a"]},{x: ["b","c"]}])', [{x: ["a"]},{x: ["b","d"]}]) | ||
testEq('L.delete(L.findWith("x", 1), [{x: ["a"]},{x: ["b","c"]}])', [{x: ["a"]},{x: ["b"]}]) | ||
}) | ||
describe("L.filter", () => { | ||
@@ -125,1 +131,5 @@ testEq('L.view(L.filter(R.lt(9)), [3,1,4,1,5,9,2])', []) | ||
}) | ||
describe("L.deleteAll", () => { | ||
testEq('L.deleteAll(L.find(x => x < 2), [3,1,4,1,5,9,2])', [3,4,5,9,2]) | ||
}) |
Sorry, the diff of this file is not supported yet
Major refactor
Supply chain riskPackage has recently undergone a major refactor. It may be unstable or indicate significant internal changes. Use caution when updating to versions that include significant changes.
Found 1 instance in 1 package
55953
398
452
1