1
1
import { assert } from '@ember/debug' ;
2
2
import { computed } from '@ember/object' ;
3
3
4
+ /**
5
+ @module @ember -data/model
6
+ */
7
+ import { expectTypeOf } from 'expect-type' ;
8
+
4
9
import { DEBUG } from '@ember-data/env' ;
5
10
import { recordIdentifierFor } from '@ember-data/store' ;
6
11
import { peekCache } from '@ember-data/store/-private' ;
12
+ import type { Value } from '@warp-drive/core-types/json/raw' ;
7
13
8
- import { computedMacroWithOptionalParams } from './util' ;
14
+ import type { Model } from './model' ;
15
+ import type { DataDecorator , DecoratorPropertyDescriptor } from './util' ;
16
+ import { isElementDescriptor } from './util' ;
9
17
10
18
/**
11
- @module @ember -data/model
12
- */
19
+ * Options provided to the attr decorator are
20
+ * supplied to the associated transform. Any
21
+ * key-value pair is valid; however, it is highly
22
+ * recommended to only use statically defined values
23
+ * that could be serialized to JSON.
24
+ *
25
+ * If no transform is provided, the only valid
26
+ * option is `defaultValue`.
27
+ *
28
+ * Examples:
29
+ *
30
+ * ```ts
31
+ * class User extends Model {
32
+ * @attr ('string', { defaultValue: 'Anonymous' }) name;
33
+ * @attr ('date', { defaultValue: () => new Date() }) createdAt;
34
+ * @attr ({ defaultValue: () => ({}) }) preferences;
35
+ * @attr ('boolean') hasVerifiedEmail;
36
+ * @attr address;
37
+ * }
38
+ *
39
+ * @typedoc
40
+ */
41
+ type AttrOptions = {
42
+ /**
43
+ * The default value for this attribute.
44
+ *
45
+ * Default values can be provided as a value or a function that will be
46
+ * executed to generate the default value.
47
+ *
48
+ * Default values *should not* be stateful (object, arrays, etc.) as
49
+ * they will be shared across all instances of the record.
50
+ *
51
+ * @typedoc
52
+ */
53
+ defaultValue ?: string | number | boolean | null | ( ( ) => unknown ) ;
54
+ } ;
13
55
14
56
/**
15
57
`attr` defines an attribute on a [Model](/ember-data/release/classes/Model).
16
58
By default, attributes are passed through as-is, however you can specify an
17
59
optional type to have the value automatically transformed.
18
- Ember Data ships with four basic transform types: `string`, `number`,
60
+ EmberData ships with four basic transform types: `string`, `number`,
19
61
`boolean` and `date`. You can define your own transforms by subclassing
20
62
[Transform](/ember-data/release/classes/Transform).
21
63
@@ -101,7 +143,7 @@ import { computedMacroWithOptionalParams } from './util';
101
143
@param {Object } options a hash of options
102
144
@return {Attribute }
103
145
*/
104
- function attr ( type , options ) {
146
+ function _attr ( type ?: string | AttrOptions , options ?: AttrOptions & object ) {
105
147
if ( typeof type === 'object' ) {
106
148
options = type ;
107
149
type = undefined ;
@@ -118,9 +160,9 @@ function attr(type, options) {
118
160
} ;
119
161
120
162
return computed ( {
121
- get ( key ) {
163
+ get ( this : Model , key : string ) {
122
164
if ( DEBUG ) {
123
- if ( [ 'currentState' ] . indexOf ( key ) !== - 1 ) {
165
+ if ( [ 'currentState' ] . includes ( key ) ) {
124
166
throw new Error (
125
167
`'${ key } ' is a reserved property name on instances of classes extending Model. Please choose a different property name for your attr on ${ this . constructor . toString ( ) } `
126
168
) ;
@@ -131,28 +173,31 @@ function attr(type, options) {
131
173
}
132
174
return peekCache ( this ) . getAttr ( recordIdentifierFor ( this ) , key ) ;
133
175
} ,
134
- set ( key , value ) {
176
+ set ( this : Model , key : string , value : Value ) {
135
177
if ( DEBUG ) {
136
- if ( [ 'currentState' ] . indexOf ( key ) !== - 1 ) {
178
+ if ( [ 'currentState' ] . includes ( key ) ) {
137
179
throw new Error (
138
180
`'${ key } ' is a reserved property name on instances of classes extending Model. Please choose a different property name for your attr on ${ this . constructor . toString ( ) } `
139
181
) ;
140
182
}
141
183
}
184
+ const identifier = recordIdentifierFor ( this ) ;
142
185
assert (
143
- `Attempted to set '${ key } ' on the deleted record ${ recordIdentifierFor ( this ) } ` ,
186
+ `Attempted to set '${ key } ' on the deleted record ${ identifier . type } : ${ identifier . id } ( ${ identifier . lid } ) ` ,
144
187
! this . currentState . isDeleted
145
188
) ;
146
- const identifier = recordIdentifierFor ( this ) ;
147
189
const cache = peekCache ( this ) ;
148
190
149
191
const currentValue = cache . getAttr ( identifier , key ) ;
150
192
if ( currentValue !== value ) {
151
193
cache . setAttr ( identifier , key , value ) ;
152
194
153
195
if ( ! this . isValid ) {
196
+ // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
154
197
const { errors } = this ;
198
+ // eslint-disable-next-line @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access
155
199
if ( errors . get ( key ) ) {
200
+ // eslint-disable-next-line @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access
156
201
errors . remove ( key ) ;
157
202
this . currentState . cleanErrorRequests ( ) ;
158
203
}
@@ -164,4 +209,43 @@ function attr(type, options) {
164
209
} ) . meta ( meta ) ;
165
210
}
166
211
167
- export default computedMacroWithOptionalParams ( attr ) ;
212
+ export function attr ( ) : DataDecorator ;
213
+ export function attr ( type : string ) : DataDecorator ;
214
+ export function attr ( options : AttrOptions ) : DataDecorator ;
215
+ export function attr ( type : string , options ?: AttrOptions & object ) : DataDecorator ;
216
+ export function attr ( target : object , key : string , desc : PropertyDescriptor ) : DecoratorPropertyDescriptor ;
217
+ export function attr (
218
+ type ?: string | AttrOptions | object ,
219
+ options ?: ( AttrOptions & object ) | string ,
220
+ desc ?: PropertyDescriptor
221
+ ) : DataDecorator | DecoratorPropertyDescriptor {
222
+ const args = [ type , options , desc ] ;
223
+ return isElementDescriptor ( args ) ? _attr ( ) ( ...args ) : _attr ( type , options as object ) ;
224
+ }
225
+
226
+ // positive tests
227
+ expectTypeOf ( attr ( { } , 'key' , { } ) ) . toEqualTypeOf < DecoratorPropertyDescriptor > ( ) ;
228
+ expectTypeOf ( attr ( 'string' ) ) . toEqualTypeOf < DataDecorator > ( ) ;
229
+ expectTypeOf ( attr ( { } ) ) . toEqualTypeOf < DataDecorator > ( ) ;
230
+ expectTypeOf ( attr ( 'string' , { } ) ) . toEqualTypeOf < DataDecorator > ( ) ;
231
+ expectTypeOf ( attr ( ) ) . toEqualTypeOf < DataDecorator > ( ) ;
232
+
233
+ expectTypeOf ( attr ( 'string' , { defaultValue : 'hello' } ) ) . toEqualTypeOf < DataDecorator > ( ) ;
234
+ expectTypeOf ( attr ( { defaultValue : 'hello' } ) ) . toEqualTypeOf < DataDecorator > ( ) ;
235
+ expectTypeOf ( attr ( 'string' , { defaultValue : ( ) => { } } ) ) . toEqualTypeOf < DataDecorator > ( ) ;
236
+ expectTypeOf ( attr ( { defaultValue : ( ) => { } } ) ) . toEqualTypeOf < DataDecorator > ( ) ;
237
+
238
+ /* prettier-ignore */
239
+ expectTypeOf (
240
+ // @ts -expect-error
241
+ attr (
242
+ { defaultValue : { } }
243
+ )
244
+ ) . toBeNever ;
245
+ expectTypeOf (
246
+ attr (
247
+ // @ts -expect-error
248
+ 1 ,
249
+ { defaultValue : 'hello' }
250
+ )
251
+ ) . toBeNever ;
0 commit comments