Skip to content

Commit d314fcf

Browse files
authored
Merge pull request #42 from grammarly/f-deprecate-propexpr
Deprecate property expressions
2 parents 14eadeb + fec56a9 commit d314fcf

File tree

6 files changed

+69
-41
lines changed

6 files changed

+69
-41
lines changed

README.md

+20-18
Original file line numberDiff line numberDiff line change
@@ -358,16 +358,22 @@ console.log(oneC.set(7, bigobj))
358358
Focal also gives you the ability to define these lenses quite conveniently:
359359

360360
```typescript
361-
// create the above lenses by ONLY providing a getter function¹
362-
const abc = Lens.prop((obj: typeof bigobj.one) => obj.a.b.c)
363-
364-
const one = Lens.prop((obj: typeof bigobj) => obj.one)
365-
366-
const two = Lens.prop((obj: typeof bigobj) => obj.two)
367-
368-
// ¹ RESTRICTIONS APPLY! the getter function in this case can only be a simple
369-
// property path access function which consists of a single property access expression
370-
// and has no side effects.
361+
// create the above lenses by providing key names
362+
const abc = Lens.key<typeof bigobj.one>()('a', 'b', 'c')
363+
364+
const one = Lens.key<typeof bigobj>()('one')
365+
366+
const two = Lens.key<typeof bigobj>()('two')
367+
368+
// Note the extra nullary call. It is there for better type inference. To create a type safe
369+
// lens we need to tell the compiler both the type of source and the type of destination property.
370+
// Unlike with Atom.lens, the compiler doesn't yet know the type of the source data, so we have to
371+
// explicitly state it. The compiler then can easily infer the type of the destination property by
372+
// its name.
373+
//
374+
// We need the nullary call because we have two generic type parameters, and want one of them
375+
// explicit and another inferred. So the nullary call is there only to supply the type argument
376+
// for the source data type.
371377
```
372378

373379
The best part about this is that it is completely type safe, and all of the IDE tools (like auto-completion, refactoring, etc.) will just work.
@@ -405,8 +411,8 @@ const obj = Atom.create({
405411
a: 5
406412
})
407413

408-
// create a lens to look at an `a` property of the object
409-
const a = Lens.prop((x: typeof obj) => x.a)
414+
// create a lens to look at the `a` property of the object
415+
const a = Lens.key<typeof obj>()('a')
410416

411417
// create a lensed atom that will hold a value of the `a` property of our object
412418
const lensed = obj.lens(a)
@@ -427,11 +433,7 @@ console.log(obj.get())
427433
Note how the source atom's value has changed when we set a new value to the lensed atom – that's it. There's also a neat shortcut to create lensed atoms:
428434

429435
```typescript
430-
const lensed = obj.lens(x => x.a) // ¹
431-
432-
// ¹ SAME RESTRICTIONS APPLY! just like with Lens.prop, the getter function argument
433-
// to the atom's `lens` method can only be a simple property path access function which
434-
// consists of a single property access expression and has no side effects.
436+
const lensed = obj.lens('a')
435437
```
436438

437439
We don't need to explicitly create a `Lens` – atom's `lens` method already has a couple of overloads to create lensed atoms on the spot. Also note that we didn't need to add the `typeof obj` type annotation here: the compiler already knows the type of data we're operating on (from the type of `obj`, which in this case is `Atom<{ a: number }`) and can infer the type of `x` for us.
@@ -468,7 +470,7 @@ const App = (props: { state: Atom<{ count: number }> }) =>
468470
this is where we take apart the app state and give only a part of it
469471
to the counter component:
470472
*/}
471-
<Counter count={props.state.lens(x => x.count)} />
473+
<Counter count={props.state.lens('count')} />
472474
</div>
473475
```
474476

packages/examples/all/src/player/index.tsx

+4-4
Original file line numberDiff line numberDiff line change
@@ -65,11 +65,11 @@ class App extends React.Component<{ state: Atom<AppState> }, {}> {
6565
return (
6666
<div>
6767
<TimeLine
68-
currentTime={state.lens(x => x.currentTime)}
69-
maxDuration={state.view(x => x.maxDuration)}
68+
currentTime={state.lens('currentTime')}
69+
maxDuration={state.view('maxDuration')}
7070
/>
71-
<Volume volume={state.lens(x => x.volume)} />
72-
<Play status={state.lens(x => x.status)} />
71+
<Volume volume={state.lens('volume')} />
72+
<Play status={state.lens('status')} />
7373
</div>
7474
)
7575
}

packages/examples/all/src/player/model.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,7 @@ export class AudioModel {
3636
this.durationSubscription = Observable
3737
.fromEvent(this.audio, 'canplaythrough')
3838
.subscribe(() =>
39-
atom.lens(x => x.maxDuration).set(this.audio.duration)
39+
atom.lens('maxDuration').set(this.audio.duration)
4040
)
4141

4242
this.timeSubscription = atom
@@ -71,11 +71,11 @@ export class AudioModel {
7171
)
7272

7373
this.volumeSubscription = atom
74-
.lens(x => x.volume)
74+
.lens('volume')
7575
.subscribe(x => this.audio.volume = x / 10)
7676

7777
this.currentTimeSubscription = atom
78-
.lens(x => x.currentTime)
78+
.lens('currentTime')
7979
.subscribe(x => this.audio.currentTime = parseInt(x, 10))
8080
}
8181

packages/focal/src/atom/base.ts

+2
Original file line numberDiff line numberDiff line change
@@ -164,6 +164,8 @@ export interface Atom<T> extends ReadOnlyAtom<T> {
164164
set(newValue: T): void
165165

166166
/**
167+
* DEPRECATED: please use other overloads instead!
168+
*
167169
* Create a lensed atom using a property expression, which specifies
168170
* a path inside the atom value's data structure.
169171
*

packages/focal/src/lens/json.ts

+30-4
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,9 @@ import {
99
setKey,
1010
conservatively,
1111
findIndex,
12-
Option
12+
Option,
13+
DEV_ENV,
14+
warning
1315
} from './../utils'
1416

1517
import { Lens, Prism } from './base'
@@ -151,13 +153,35 @@ export function keyImpl<TObject>(k?: string) {
151153
)
152154
}
153155

156+
let propExprDeprecatedWarnings = 0
157+
158+
function warnPropExprDeprecated(path: string[]) {
159+
// don't warn more than a few times
160+
if (propExprDeprecatedWarnings < 10) {
161+
propExprDeprecatedWarnings++
162+
163+
const propExpr = `x.${path.join('.')}`
164+
const keys = `'${path.join("', '")}'`
165+
166+
warning(
167+
`The property expression overload of Atom.lens and Lens.prop are deprecated and ` +
168+
`will be removed in next versions of Focal. Please use the key name overload for ` +
169+
`Atom.lens and Lens.key instead. ` +
170+
`You can convert your code by changing the calls:
171+
a.lens(x => ${propExpr}) to a.lens(${keys}),
172+
Lens.prop((x: T) => ${propExpr}) to Lens.key<T>()(${keys}).`
173+
)
174+
}
175+
}
176+
154177
export function propImpl<TObject, TProperty>(
155178
getter: PropExpr<TObject, TProperty>
156179
): Lens<TObject, TProperty> {
180+
const path = extractPropertyPath(getter as PropExpr<TObject, TProperty>)
181+
if (DEV_ENV) warnPropExprDeprecated(path)
182+
157183
// @TODO can we optimize this?
158-
return Lens.compose<TObject, TProperty>(
159-
...extractPropertyPath(getter as PropExpr<TObject, TProperty>)
160-
.map(keyImpl()))
184+
return Lens.compose<TObject, TProperty>(...path.map(keyImpl()))
161185
}
162186

163187
export function indexImpl<TItem>(i: number): Prism<TItem[], TItem> {
@@ -216,6 +240,8 @@ declare module './base' {
216240
export let key: typeof keyImpl
217241

218242
/**
243+
* DEPRECATED: please use Lens.key instead!
244+
*
219245
* Create a lens to an object's property. The argument is a property expression, which
220246
* is a limited form of a getter, with following restrictions:
221247
* - should be a pure function

packages/focal/src/react/intrinsic.ts

+10-12
Original file line numberDiff line numberDiff line change
@@ -33,7 +33,7 @@ export type GenericLiftedIntrinsic<T> =
3333
: never
3434

3535
export type LiftedIntrinsicsHTML = {
36-
[K in keyof React.ReactHTML]: GenericLiftedIntrinsic<React.ReactHTML[K]>
36+
readonly [K in keyof React.ReactHTML]: GenericLiftedIntrinsic<React.ReactHTML[K]>
3737
}
3838

3939
export interface LiftedFragmentAttributes extends ObservableReactChildren, React.Attributes {}
@@ -44,13 +44,11 @@ export interface LiftedFragment {
4444
React.ReactElement<LiftWrapperProps<ObservableReactHTMLProps<{}>>>
4545
}
4646

47-
export const enum ExtendedIntrinsics {
48-
fragment = 'Fragment'
47+
interface ExtraLiftedIntrinsics {
48+
readonly Fragment: LiftedFragment
4949
}
5050

51-
export type LiftedIntrinsics = LiftedIntrinsicsHTML & {
52-
[k in ExtendedIntrinsics.fragment]: LiftedFragment
53-
}
51+
export type LiftedIntrinsics = LiftedIntrinsicsHTML & ExtraLiftedIntrinsics
5452

5553
export function createLiftedIntrinsics(): LiftedIntrinsics {
5654
const html: (keyof LiftedIntrinsicsHTML)[] = [
@@ -67,14 +65,14 @@ export function createLiftedIntrinsics(): LiftedIntrinsics {
6765
'var', 'video', 'wbr'
6866
]
6967

70-
const r = {} as any
71-
html.forEach(e => r[e] = liftIntrinsic(e))
68+
const r: {
69+
-readonly [P in keyof LiftedIntrinsics]?: LiftedIntrinsics[P];
70+
} = {}
7271

73-
const fragmentTag: LiftedIntrinsics[ExtendedIntrinsics.fragment] =
74-
(props: LiftedFragmentAttributes) =>
75-
React.createElement(LiftWrapper, { component: React.Fragment, props })
72+
html.forEach(e => r[e] = liftIntrinsic(e))
7673

77-
r[ExtendedIntrinsics.fragment] = fragmentTag
74+
r.Fragment = (props: LiftedFragmentAttributes) =>
75+
React.createElement(LiftWrapper, { component: React.Fragment, props })
7876

7977
return r as LiftedIntrinsics
8078
}

0 commit comments

Comments
 (0)