Skip to content

Commit 7580f31

Browse files
committed
Wrap up readme
1 parent 1e9e744 commit 7580f31

File tree

3 files changed

+357
-2
lines changed

3 files changed

+357
-2
lines changed

README.md

+355
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,357 @@
11
# Primed Model
22
Hassle free TypeScript model management
3+
4+
Dynamically create your instances from JSON objects to take full advantage of OOP. This approach is to be contrasted with factory functions for managing highly structured data
5+
6+
**Note**: This project uses reflect-metadata for storing metadata in classes and properties
7+
8+
## Provided resources
9+
- `Model` <*decorator*> - To register the classes
10+
- `Primed` <*decorator*> - To mark properties to be dynamically instantiated
11+
- `Base` <*class*> - To extend the models, configure their constructor's typing, and give internal functionality
12+
13+
14+
```typescript
15+
@Model
16+
class Person extends Base<Person> {
17+
name: string = ''
18+
middleName: string = ''
19+
lastName: string = ''
20+
21+
get fullName(){
22+
return [this.name, this.middleName, this.lastName].join(' ').trim() || 'Empty Name'
23+
}
24+
25+
@Primed(Person)
26+
parent!: Person
27+
28+
@Primed('Cat', { array: true })
29+
cats!: Cat[]
30+
}
31+
32+
@Model
33+
class Cat extends Base<Cat>{
34+
name: string | null = null
35+
breed: string | null = null
36+
}
37+
```
38+
**Note**: `Cat` is being passed to the `Primed` decorator as a *string* in the `Person` class because TS/JS does not allow classes which haven't been defined to be referenced by value
39+
40+
41+
## Features
42+
43+
- Recursively and dynamically create model instances, automatically instantiated by default, unless `required: false` is specified in the `Primed` options
44+
- Specify properties for classes defined after the point of reference passing a string to `Primed`
45+
- Pass a factory function to `Primed` for custom data types
46+
- Getters are enumerable by default so that they show when iterating over the keys or in the string representation of your class (using `JSON.stringify`)
47+
- Providced `clone` method for copying whole instances
48+
49+
50+
## Example usage
51+
52+
<table>
53+
<tr>
54+
<th> Code </th>
55+
<th> Output </th>
56+
</tr>
57+
<tr>
58+
<td>
59+
60+
```javascript
61+
new Person()
62+
63+
64+
65+
66+
67+
68+
69+
70+
71+
72+
73+
74+
75+
76+
77+
78+
79+
80+
81+
82+
83+
84+
85+
```
86+
87+
</td>
88+
<td>
89+
90+
```javascript
91+
{
92+
name: "",
93+
middleName: "",
94+
lastName: "",
95+
fullName: "Empty Name",
96+
parent: {
97+
name: "",
98+
middleName: "",
99+
lastName: "",
100+
fullName: "Empty Name",
101+
cats: [
102+
{
103+
name: null,
104+
breed: null
105+
}
106+
]
107+
},
108+
cats:[
109+
{
110+
name: null,
111+
breed: null
112+
}
113+
]
114+
}
115+
```
116+
117+
</td>
118+
</tr>
119+
<tr>
120+
<td>
121+
122+
```javascript
123+
new Person({
124+
name: "Alice",
125+
lastName: "Liddell",
126+
cats: [
127+
{
128+
name: "garfield"
129+
}
130+
],
131+
parent: {
132+
name: "Bob",
133+
cats: [
134+
{
135+
name: "Tom"
136+
}
137+
]
138+
}
139+
})
140+
141+
142+
143+
144+
145+
146+
147+
```
148+
149+
</td>
150+
<td>
151+
152+
```javascript
153+
{
154+
name: "Alice",
155+
middleName: "",
156+
lastName: "Liddell",
157+
fullName: "Alice Liddell",
158+
parent: {
159+
name: "Bob",
160+
middleName: "",
161+
lastName: "",
162+
fullName: "Bob",
163+
cats: [
164+
{
165+
name: "Tom",
166+
breed: null
167+
}
168+
]
169+
},
170+
cats: [
171+
{
172+
name: "Garfield",
173+
breed: null
174+
}
175+
]
176+
}
177+
```
178+
179+
</td>
180+
</tr>
181+
</table>
182+
183+
## Custom properties examples
184+
185+
You can define your own functions that can be passed into the `Prime` decorator for custom behavior
186+
187+
188+
**Luxon**'s DateTime
189+
190+
```typescript
191+
import { DateTime } from 'luxon'
192+
193+
function PrimedDateTime(value?: string | DateTime): DateTime {
194+
if(value instanceof DateTime){
195+
return DateTime.fromJSDate(value.toJSDate()) // Copy the value
196+
} else if(typeof value === 'string'){
197+
return DateTime.fromISO(value) // Build from string
198+
} else {
199+
return DateTime.local() // Create default value
200+
}
201+
}
202+
203+
@Model
204+
class Foo extends Base<Foo, FooInput>{
205+
@Primed(PrimedDateTime)
206+
someDateTime!: DateTime
207+
}
208+
```
209+
210+
<details>
211+
<summary> Decimal from decimal.js </summary>
212+
213+
```typescript
214+
import { Decimal } from 'decimal.js'
215+
216+
function PrimedDecimal(value: number | string | Decimal = 0): Decimal {
217+
return new Decimal(value)
218+
}
219+
```
220+
</details>
221+
222+
223+
<details>
224+
<summary> JavaScript's Date </summary>
225+
226+
```typescript
227+
function PrimedDate(value?: string | Date): Date {
228+
if(typeof value === 'undefined'){
229+
return new Date() // Create default value
230+
} else {
231+
return new Date(value) // Build from string or copy existing
232+
}
233+
}
234+
```
235+
236+
</details>
237+
238+
<details>
239+
<summary> string id </summary>
240+
241+
```typescript
242+
function PrimedId(value?: string): string {
243+
return value ? value : "-1"
244+
}
245+
```
246+
247+
</details>
248+
249+
250+
## Noteworthy
251+
- If you're minifying/compressing/uglyfing your JS, you must pass a string to the `Model` decorator with the name or the class. The function name is being relied uppon for initializing properties that depend on it at runtime
252+
```typescript
253+
@Model('Foo')
254+
class Foo extends Base<Foo>{
255+
@Primed(Bar)
256+
bar!: Bar
257+
}
258+
```
259+
- Pass `required: false` to the `Primed` *options* to prevent primed properties from being automatically instantiated
260+
```typescript
261+
@Model
262+
class Foo extends Base<Foo>{
263+
@Primed(Bar, { required: false })
264+
bar!: Bar
265+
}
266+
```
267+
- Pass `array: true` to the `Primed` *options* to automatically instantiate arrays
268+
```typescript
269+
@Model
270+
class Foo extends Base<Foo>{
271+
@Primed(Bar, { array: true })
272+
bar!: Bar[]
273+
}
274+
```
275+
- If the payload type differs from the class's type and you want those typings, you can pass an second interface (which can be a partial of the class) to the `Base` class when extending
276+
```typescript
277+
interface FooInput{
278+
someNumber: number,
279+
someDate: string,
280+
}
281+
282+
@Model
283+
class Foo extends Base<Foo, FooInput>{
284+
someString: string = ''
285+
286+
@Primed(PrimedDecimal)
287+
someNumber!: Decimal
288+
289+
@Primed(PrimedDateTime)
290+
someDate!: DateTime
291+
}
292+
```
293+
- Auto initialization will stop after detecting a circular reference
294+
295+
<table>
296+
<tr>
297+
<th> Code </th>
298+
<th> Output </th>
299+
</tr>
300+
<tr>
301+
<td>
302+
303+
```typescript
304+
@Model
305+
class Alpha extends Base<Alpha> {
306+
@Primed('Bravo')
307+
bravo!: Bravo
308+
}
309+
310+
@Model
311+
class Bravo extends Base<Bravo> {
312+
@Primed('Charlie')
313+
charlie!: Charlie
314+
}
315+
316+
@Model
317+
class Charlie extends Base<Charlie> {
318+
@Primed(Alpha)
319+
alpha!: Alpha
320+
}
321+
322+
new Alpha()
323+
```
324+
325+
</td>
326+
<td>
327+
328+
```javascript
329+
{
330+
bravo: {
331+
charlie: {
332+
alpha: {
333+
bravo: undefined
334+
}
335+
}
336+
}
337+
}
338+
339+
340+
341+
342+
343+
344+
345+
346+
347+
348+
```
349+
350+
</td>
351+
</tr>
352+
</table>
353+
354+
## To do
355+
356+
- Add tests
357+
- Implement change detection mechanism

package-lock.json

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"name": "primed-model",
3-
"version": "0.3.6",
3+
"version": "1.0.0",
44
"description": "TypeScript model management",
55
"main": "dist/app.js",
66
"types": "dist/app.d.ts",

0 commit comments

Comments
 (0)