-
Notifications
You must be signed in to change notification settings - Fork 97
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
14 changed files
with
399 additions
and
146 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,16 @@ | ||
--- | ||
"@patternfly/pfe-core": minor | ||
--- | ||
**Decorators**: Added `@listen`. Use it to attach element event listeners to | ||
class methods. | ||
|
||
```ts | ||
@customElement('custom-input') | ||
class CustomInput extends LitElement { | ||
@property({ type: Boolean }) dirty = false; | ||
@listen('keyup', { once: true }) | ||
protected onKeyup() { | ||
this.dirty = true; | ||
} | ||
} | ||
``` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,28 @@ | ||
--- | ||
"@patternfly/pfe-core": major | ||
--- | ||
**Decorators**: Added `@observes`. Use it to add property change callback by | ||
decorating them with the name of the property to observe | ||
|
||
```ts | ||
@customElement('custom-button') | ||
class CustomButton extends LitElement { | ||
#internals = this.attachInternals(); | ||
|
||
@property({ type: Boolean }) disabled = false; | ||
|
||
@observes('disabled') | ||
protected disabledChanged() { | ||
this.#internals.ariaDisabled = | ||
this.disabled ? 'true' | ||
: this.ariaDisabled ?? 'false'; | ||
} | ||
} | ||
``` | ||
|
||
Breaking change: This commit makes some changes to the internal APIs of the | ||
pre-existing `@observed` observer, most notably it changes the constructor | ||
signature of the `PropertyObserverController` and associated functions. Most | ||
users should not have to make any changes, but if you directly import and use | ||
those functions, check the commit log to see how to update your call sites. | ||
|
85 changes: 46 additions & 39 deletions
85
core/pfe-core/controllers/property-observer-controller.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,52 +1,59 @@ | ||
import type { ReactiveController, ReactiveElement } from 'lit'; | ||
|
||
export const observedController: unique symbol = Symbol('observed properties controller'); | ||
|
||
export type ChangeCallback<T = ReactiveElement> = ( | ||
export type ChangeCallback<T extends ReactiveElement, V = T[keyof T]> = ( | ||
this: T, | ||
old?: T[keyof T], | ||
newV?: T[keyof T], | ||
old?: V, | ||
newV?: V, | ||
) => void; | ||
|
||
export type ChangeCallbackName = `_${string}Changed`; | ||
|
||
export type PropertyObserverHost<T> = T & Record<ChangeCallbackName, ChangeCallback<T>> & { | ||
[observedController]: PropertyObserverController; | ||
}; | ||
|
||
/** This controller holds a cache of observed property values which were set before the element updated */ | ||
export class PropertyObserverController implements ReactiveController { | ||
private static hosts = new WeakMap<HTMLElement, PropertyObserverController>(); | ||
|
||
private values = new Map<string, [methodName: string, values: [unknown, unknown]]>(); | ||
export interface PropertyObserverOptions<T extends ReactiveElement> { | ||
propertyName: string & keyof T; | ||
callback: ChangeCallback<T>; | ||
waitFor?: 'connected' | 'updated' | 'firstUpdated'; | ||
} | ||
|
||
private delete(key: string) { | ||
this.values.delete(key); | ||
} | ||
export class PropertyObserverController< | ||
T extends ReactiveElement | ||
> implements ReactiveController { | ||
private oldVal: T[keyof T]; | ||
|
||
constructor(private host: ReactiveElement) { | ||
if (PropertyObserverController.hosts.get(host)) { | ||
return PropertyObserverController.hosts.get(host) as PropertyObserverController; | ||
} | ||
host.addController(this); | ||
(host as PropertyObserverHost<ReactiveElement>)[observedController] = this; | ||
constructor( | ||
private host: T, | ||
private options: PropertyObserverOptions<T> | ||
) { | ||
this.oldVal = host[options.propertyName]; | ||
} | ||
|
||
/** Set any cached valued accumulated between constructor and connectedCallback */ | ||
hostUpdate(): void { | ||
for (const [key, [methodName, [oldVal, newVal]]] of this.values) { | ||
// @ts-expect-error: be cool, typescript | ||
this.host[methodName as keyof ReactiveElement]?.(oldVal, newVal); | ||
this.delete(key); | ||
async hostUpdate(): Promise<void> { | ||
const { oldVal, options: { waitFor, propertyName, callback } } = this; | ||
if (!callback) { | ||
throw new Error(`no callback for ${propertyName}`); | ||
} | ||
} | ||
|
||
/** Once the element has updated, we no longer need this controller, so we remove it */ | ||
hostUpdated(): void { | ||
this.host.removeController(this); | ||
} | ||
|
||
cache(key: string, methodName: string, ...vals: [unknown, unknown]): void { | ||
this.values.set(key, [methodName, vals]); | ||
const newVal = this.host[propertyName]; | ||
this.oldVal = newVal; | ||
if (newVal !== oldVal) { | ||
switch (waitFor) { | ||
case 'connected': | ||
if (!this.host.isConnected) { | ||
const origConnected = this.host.connectedCallback; | ||
await new Promise<void>(resolve => { | ||
this.host.connectedCallback = function() { | ||
resolve(origConnected?.call(this)); | ||
}; | ||
}); | ||
} | ||
break; | ||
case 'firstUpdated': | ||
if (!this.host.hasUpdated) { | ||
await this.host.updateComplete; | ||
} | ||
break; | ||
case 'updated': | ||
await this.host.updateComplete; | ||
break; | ||
} | ||
} | ||
callback.call(this.host, oldVal as T[keyof T], newVal); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,26 @@ | ||
import type { LitElement } from 'lit'; | ||
|
||
/** | ||
* Listens for a given event on the custom element. | ||
* equivalent to calling `this.addEventListener` in the constructor | ||
* @param type event type e.g. `click` | ||
* @param options event listener options object e.g. `{ passive: true }` | ||
*/ | ||
export function listen(type: string, options?: EventListenerOptions) { | ||
return function( | ||
proto: LitElement, | ||
methodName: string, | ||
): void { | ||
const origConnected = proto.connectedCallback; | ||
const origDisconnected = proto.disconnectedCallback; | ||
const listener = (proto as any)[methodName] as EventListener; | ||
proto.connectedCallback = function() { | ||
origConnected?.call(this); | ||
this.addEventListener(type, listener, options); | ||
}; | ||
proto.disconnectedCallback = function() { | ||
origDisconnected?.call(this); | ||
this.removeEventListener(type, listener, options); | ||
}; | ||
}; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,40 @@ | ||
import type { ReactiveElement } from 'lit'; | ||
|
||
import { | ||
PropertyObserverController, | ||
type ChangeCallback, | ||
type PropertyObserverOptions, | ||
} from '@patternfly/pfe-core/controllers/property-observer-controller.js'; | ||
|
||
/** | ||
* Observes changes on the given property and calls the decorated method | ||
* with the old and new values when it changes. In cases where the decorated method | ||
* needs to access uninitialized class fields, You may need to wait for the element to connect | ||
* before running your effects. In that case, you can optionally specify which | ||
* lifecycle state to wait for. e.g.: | ||
* - `waitFor: 'firstUpdate'` waits until the first update cycle has completed | ||
* - `waitFor: 'updated'` waits until the next update cycle has completed | ||
* - `waitFor: 'connected'` waits until the element connects | ||
* @param propertyName property to react to | ||
* @param [options] options including lifecycle to wait on. | ||
*/ | ||
export function observes<T extends ReactiveElement>( | ||
propertyName: string & keyof T, | ||
options?: Partial<Pick<PropertyObserverOptions<T>, 'waitFor'>>, | ||
) { | ||
return function(proto: T, methodName: string): void { | ||
const callback = proto[methodName as keyof T] as ChangeCallback<T>; | ||
if (typeof callback !== 'function') { | ||
throw new Error('@observes must decorate a class method'); | ||
} | ||
const klass = proto.constructor as typeof ReactiveElement; | ||
klass.addInitializer(instance => { | ||
instance.addController(new PropertyObserverController(instance as T, { | ||
...options, | ||
propertyName, | ||
callback, | ||
})); | ||
}); | ||
}; | ||
} | ||
|
Oops, something went wrong.