Computed Properties
Singular computations
Entity classes are just normal classes, so any common derived data can just be added as getters to the class itself.
import { Entity } from '@data-client/rest';
class User extends Entity {
id = '';
firstName = '';
lastName = '';
username = '';
email = '';
get fullName() {
return `${this.firstName} ${this.lastName}`;
}
pk() {
return this.id;
}
static key = 'User';
}
If the computations are expensive feel free to add some memoization.
import { Entity } from '@data-client/rest';
import memoize from 'nano-memoize';
class User extends Entity {
truelyExpensiveValue = memoize(() => {
// compute that expensive thing!
});
}
tip
If you simply want to deserialize a field to a more useful form like Temporal.Instant or BigNumber, you can use the declarative static schema.
import { Entity } from '@data-client/rest';
import BigNumber from 'bignumber.js';
class User extends Entity {
id = '';
firstName = '';
lastName = '';
createdAt = Temporal.Instant.fromEpochSeconds(0);
lifetimeBlinkCount = BigNumber(0);
pk() {
return this.id;
}
static key = 'User';
static schema = {
createdAt: Temporal.Instant.from,
lifetimeBlinkCount: BigNumber,
};
}
Global computations
Query can be used for computations of derived data from more than one entity. We generally call these aggregates.
api/User
UsersPage
import { Query, schema } from '@data-client/rest'; import { UserResource, User } from './api/User'; const getUserCount = new Query( new schema.All(User), (entries, { isAdmin } = {}) => { if (isAdmin !== undefined) return entries.filter(user => user.isAdmin === isAdmin).length; return entries.length; }, ); function UsersPage() { useFetch(UserResource.getList); const userCount = useCache(getUserCount); const adminCount = useCache(getUserCount, { isAdmin: true }); if (userCount === undefined) return <div>No users in cache yet</div>; return ( <div> <div>Total users: {userCount}</div> <div>Total admins: {adminCount}</div> </div> ); } render(<UsersPage />);