@glimmer/tracking
Version:
Glimmer property tracking library
132 lines (117 loc) • 14.1 kB
JavaScript
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.tracked = undefined;
exports.setPropertyDidChange = setPropertyDidChange;
var _env = require("@glimmer/env");
var _validator = require("@glimmer/validator");
/**
* @decorator
*
* Marks a property as tracked.
*
* By default, a component's properties are expected to be static,
* meaning you are not able to update them and have the template update accordingly.
* Marking a property as tracked means that when that property changes,
* a rerender of the component is scheduled so the template is kept up to date.
*
* @example
*
* ```typescript
* import Component from '@glimmer/component';
* import { tracked } from '@glimmer/tracking';
*
* export default class MyComponent extends Component {
* @tracked
* remainingApples = 10
* }
* ```
*
* When something changes the component's `remainingApples` property, the rerender
* will be scheduled.
*
* @example Computed Properties
*
* In the case that you have a getter that depends on other properties, tracked
* properties accessed within the getter will automatically be tracked for you.
* That means when any of those dependent tracked properties is changed, a
* rerender of the component will be scheduled.
*
* In the following example we have two properties,
* `eatenApples`, and `remainingApples`.
*
*
* ```typescript
* import Component from '@glimmer/component';
* import { tracked } from '@glimmer/tracking';
*
* const totalApples = 100;
*
* export default class MyComponent extends Component {
* @tracked
* eatenApples = 0
*
* get remainingApples() {
* return totalApples - this.eatenApples;
* }
*
* increment() {
* this.eatenApples = this.eatenApples + 1;
* }
* }
* ```
*/
var tracked = exports.tracked = function tracked() {
for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {
args[_key] = arguments[_key];
}
var target = args[0],
key = args[1],
descriptor = args[2]; // Error on `@tracked()`, `@tracked(...args)`, and `@tracked get propName()`
if (_env.DEBUG && typeof target === 'string') throwTrackedWithArgumentsError(args);
if (_env.DEBUG && target === undefined) throwTrackedWithEmptyArgumentsError();
if (_env.DEBUG && descriptor && descriptor.get) throwTrackedComputedPropertyError();
if (descriptor) {
return descriptorForField(target, key, descriptor);
} else {
// In TypeScript's implementation, decorators on simple class fields do not
// receive a descriptor, so we define the property on the target directly.
Object.defineProperty(target, key, descriptorForField(target, key));
}
};
function throwTrackedComputedPropertyError() {
throw new Error("The @tracked decorator does not need to be applied to getters. Properties implemented using a getter will recompute automatically when any tracked properties they access change.");
}
function throwTrackedWithArgumentsError(args) {
throw new Error("You attempted to use @tracked with " + (args.length > 1 ? 'arguments' : 'an argument') + " ( @tracked(" + args.map(function (d) {
return "'" + d + "'";
}).join(', ') + ") ), which is no longer necessary nor supported. Dependencies are now automatically tracked, so you can just use " + '`@tracked`' + ".");
}
function throwTrackedWithEmptyArgumentsError() {
throw new Error('You attempted to use @tracked(), which is no longer necessary nor supported. Remove the parentheses and you will be good to go!');
}
function descriptorForField(_target, key, desc) {
if (_env.DEBUG && desc && (desc.value || desc.get || desc.set)) {
throw new Error("You attempted to use @tracked on " + String(key) + ", but that element is not a class field. @tracked is only usable on class fields. Native getters and setters will autotrack add any tracked fields they encounter, so there is no need mark getters and setters with @tracked.");
}
var _trackedData = (0, _validator.trackedData)(key, desc && desc.initializer),
getter = _trackedData.getter,
setter = _trackedData.setter;
return {
enumerable: true,
configurable: true,
get: function get() {
return getter(this);
},
set: function set(newValue) {
setter(this, newValue);
propertyDidChange();
}
};
}
var propertyDidChange = function propertyDidChange() {};
function setPropertyDidChange(cb) {
propertyDidChange = cb;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../packages/@glimmer/tracking/src/tracked.ts"],"names":[],"mappings":"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","sourcesContent":["import { DEBUG } from '@glimmer/env';\nimport { trackedData } from '@glimmer/validator';\n\n/**\n * @decorator\n *\n * Marks a property as tracked.\n *\n * By default, a component's properties are expected to be static,\n * meaning you are not able to update them and have the template update accordingly.\n * Marking a property as tracked means that when that property changes,\n * a rerender of the component is scheduled so the template is kept up to date.\n *\n * @example\n *\n * ```typescript\n * import Component from '@glimmer/component';\n * import { tracked } from '@glimmer/tracking';\n *\n * export default class MyComponent extends Component {\n *    @tracked\n *    remainingApples = 10\n * }\n * ```\n *\n * When something changes the component's `remainingApples` property, the rerender\n * will be scheduled.\n *\n * @example Computed Properties\n *\n * In the case that you have a getter that depends on other properties, tracked\n * properties accessed within the getter will automatically be tracked for you.\n * That means when any of those dependent tracked properties is changed, a\n * rerender of the component will be scheduled.\n *\n * In the following example we have two properties,\n * `eatenApples`, and `remainingApples`.\n *\n *\n * ```typescript\n * import Component from '@glimmer/component';\n * import { tracked } from '@glimmer/tracking';\n *\n * const totalApples = 100;\n *\n * export default class MyComponent extends Component {\n *    @tracked\n *    eatenApples = 0\n *\n *    get remainingApples() {\n *      return totalApples - this.eatenApples;\n *    }\n *\n *    increment() {\n *      this.eatenApples = this.eatenApples + 1;\n *    }\n *  }\n * ```\n */\nexport let tracked: PropertyDecorator = (...args: any[]) => {\n  let [target, key, descriptor] = args;\n\n  // Error on `@tracked()`, `@tracked(...args)`, and `@tracked get propName()`\n  if (DEBUG && typeof target === 'string') throwTrackedWithArgumentsError(args);\n  if (DEBUG && target === undefined) throwTrackedWithEmptyArgumentsError();\n  if (DEBUG && descriptor && descriptor.get) throwTrackedComputedPropertyError();\n\n  if (descriptor) {\n    return descriptorForField(target, key, descriptor);\n  } else {\n    // In TypeScript's implementation, decorators on simple class fields do not\n    // receive a descriptor, so we define the property on the target directly.\n    Object.defineProperty(target, key, descriptorForField(target, key));\n  }\n};\n\nfunction throwTrackedComputedPropertyError() {\n  throw new Error(\n    `The @tracked decorator does not need to be applied to getters. Properties implemented using a getter will recompute automatically when any tracked properties they access change.`\n  );\n}\n\nfunction throwTrackedWithArgumentsError(args: any[]) {\n  throw new Error(\n    `You attempted to use @tracked with ${\n      args.length > 1 ? 'arguments' : 'an argument'\n    } ( @tracked(${args\n      .map((d) => `'${d}'`)\n      .join(\n        ', '\n      )}) ), which is no longer necessary nor supported. Dependencies are now automatically tracked, so you can just use ${'`@tracked`'}.`\n  );\n}\n\nfunction throwTrackedWithEmptyArgumentsError() {\n  throw new Error(\n    'You attempted to use @tracked(), which is no longer necessary nor supported. Remove the parentheses and you will be good to go!'\n  );\n}\n\n/**\n * Whenever a tracked computed property is entered, the current tracker is\n * saved off and a new tracker is replaced.\n *\n * Any tracked properties consumed are added to the current tracker.\n *\n * When a tracked computed property is exited, the tracker's tags are\n * combined and added to the parent tracker.\n *\n * The consequence is that each tracked computed property has a tag\n * that corresponds to the tracked properties consumed inside of\n * itself, including child tracked computed properties.\n */\ntype DecoratorPropertyDescriptor = (PropertyDescriptor & { initializer?: any }) | undefined;\n\nfunction descriptorForField<T extends object, K extends keyof T>(\n  _target: T,\n  key: K,\n  desc?: DecoratorPropertyDescriptor\n): DecoratorPropertyDescriptor {\n  if (DEBUG && desc && (desc.value || desc.get || desc.set)) {\n    throw new Error(\n      `You attempted to use @tracked on ${String(\n        key\n      )}, but that element is not a class field. @tracked is only usable on class fields. Native getters and setters will autotrack add any tracked fields they encounter, so there is no need mark getters and setters with @tracked.`\n    );\n  }\n\n  let { getter, setter } = trackedData<T, K>(key, desc && desc.initializer);\n\n  return {\n    enumerable: true,\n    configurable: true,\n\n    get(this: T): any {\n      return getter(this);\n    },\n\n    set(this: T, newValue: any): void {\n      setter(this, newValue);\n      propertyDidChange();\n    },\n  };\n}\n\nlet propertyDidChange = function () {};\n\nexport function setPropertyDidChange(cb: () => void) {\n  propertyDidChange = cb;\n}\n"],"sourceRoot":""}
;