Skip to content

Commit c4d0ec5

Browse files
authoredJun 1, 2022
Detached event bus (#3)
* external weakmapped eventbus * updated package version * updated dependencies + added engines specs * updated readme
1 parent 48b96c3 commit c4d0ec5

8 files changed

+183
-187
lines changed
 

‎README.md

+3-2
Original file line numberDiff line numberDiff line change
@@ -16,11 +16,12 @@ on variables mutability has started and developed into the current reactive exec
1616

1717
#### Is it ready to use?
1818

19-
At the current stage mutableJS is still in research stage and can (thought to) be used with the `@mutablejs/dom` package.
19+
The package is still pretty experimental, but is already used in some small usable web-apps.\
20+
`@mutablejs/core` is primarily used with the [@mutablejs/dom](https://www.npmjs.com/package/@mutablejs/dom) package, see the [Starter Repository](https://github.com/mutableJS/template-starter-typescript).
2021

2122
#### Accompanying packages?
2223

23-
Currently there is 1 more package: [@mutablejs/dom](https://www.npmjs.com/package/@mutablejs/dom) and 1 more repo: [mutableJS / Demo repo](https://github.com/mutableJS/demo).
24+
Currently there is 1 more package: [@mutablejs/dom](https://www.npmjs.com/package/@mutablejs/dom) and 2 more repos: [mutableJS / TypeScript Starter Template](https://github.com/mutableJS/template-starter-typescript), [mutableJS / Demo Repo](https://github.com/mutableJS/demo).
2425

2526
## Installation
2627

‎package.json

+7-4
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"name": "@mutablejs/core",
3-
"version": "0.1.81",
3+
"version": "0.2.0",
44
"description": "Mutable state, with reactive functions - automatic recalculations on state changes",
55
"keywords": [
66
"mutable state",
@@ -39,9 +39,12 @@
3939
},
4040
"license": "MIT",
4141
"devDependencies": {
42-
"@types/node": "^17.0.31",
43-
"typescript": "^4.6.4",
44-
"vite": "^2.9.8"
42+
"@types/node": "^17.0.38",
43+
"typescript": "^4.7.2",
44+
"vite": "^2.9.9"
45+
},
46+
"engines": {
47+
"node": ">=17.0.0"
4548
},
4649
"dependencies": {}
4750
}

‎src/lib/eventBus.ts

+12-16
Original file line numberDiff line numberDiff line change
@@ -1,22 +1,18 @@
1-
// TBD: Events clean up needed?
1+
import { Mutable } from './types';
22

3-
export type MutationCallback<Value = any> = (newVal: Value, oldVal: Value) => void;
3+
type Callback<T> = (newVal: T, oldVal: T) => void;
44

5-
function eventBus() {
6-
const refs: MutationCallback[] = [];
5+
const eventBus = new WeakMap<object, Callback<any>[]>();
76

8-
const change: MutationCallback = (...data) => {
9-
refs.forEach((fn) => fn(...data));
10-
};
7+
export function listen<T>(mutable: Mutable<T>, callback: Callback<T>) {
8+
const listeners = eventBus.get(mutable) || [];
9+
listeners.push(callback);
1110

12-
const changeHandler = (callback: MutationCallback) => {
13-
refs.push(callback);
14-
};
15-
16-
return {
17-
change,
18-
changeHandler,
19-
};
11+
eventBus.set(mutable, listeners);
2012
}
2113

22-
export default eventBus;
14+
export function emit<T>(mutable: Mutable<T>, newVal: T, oldVal?: T) {
15+
eventBus.get(mutable)?.forEach((fn) => {
16+
fn(newVal, oldVal);
17+
});
18+
}

‎src/lib/mutable.ts

+13-16
Original file line numberDiff line numberDiff line change
@@ -1,46 +1,43 @@
11
import { Mutable } from './types';
22
import isRegularObject from './utils/isRegularObject';
3-
import eventBus from './eventBus';
3+
import { emit } from './eventBus';
44

55
export function mutable<Value extends any>(initialValue?: Value) {
6-
const events = eventBus();
7-
86
const obj = new Proxy(
97
{
108
value:
119
isRegularObject(initialValue) || Array.isArray(initialValue)
12-
? mutableObject(initialValue, (newVal, oldVal) =>
13-
events.change(newVal, oldVal),
14-
)
10+
? mutableObject(initialValue, handleChange)
1511
: initialValue,
1612
},
1713
{
1814
get(target, prop) {
1915
switch (prop) {
2016
case '_mutable':
2117
return true;
22-
case 'onChange':
23-
return (callback: Mutable<Value>['onChange']) => {
24-
events.changeHandler(callback);
25-
};
2618
default:
2719
return target.value;
2820
}
2921
},
30-
set(...[target, , value]) {
31-
const prevValue = target.value;
22+
set(target, prop, value) {
23+
const oldVal = target.value;
3224

33-
if (value !== prevValue) {
25+
if (value !== oldVal) {
3426
target.value = value;
35-
events.change(value, prevValue);
27+
28+
handleChange(value, oldVal);
3629
}
3730

3831
return true;
3932
},
4033
},
41-
);
34+
) as Mutable<Value>;
35+
36+
function handleChange(newVal: Value, oldVal: any) {
37+
emit(obj, newVal, oldVal);
38+
}
4239

43-
return obj as Mutable<Value>;
40+
return obj;
4441
}
4542

4643
function mutableObject<Obj extends object | unknown[]>(

‎src/lib/mutableFn.ts

+3-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import { MaybeMutable, Mutable } from './types';
22
import mutable from './mutable';
33
import isMutable from './isMutable';
4+
import { listen } from './eventBus';
45

56
type MaybeMutableTuple<Input extends [...any[]]> = Input extends [
67
infer First,
@@ -30,7 +31,7 @@ export function mutableFn<Params extends any[], ReturnType>(
3031
const pureParams = [] as unknown as Params;
3132
params.forEach((arg, i) => {
3233
if (isMutable(arg)) {
33-
arg.onChange((newVal) => {
34+
listen(arg, (newVal) => {
3435
pureParams[i] = newVal;
3536

3637
rerun();
@@ -42,7 +43,7 @@ export function mutableFn<Params extends any[], ReturnType>(
4243

4344
Object.entries(arg).forEach(([key, item]) => {
4445
if (isMutable(item)) {
45-
item.onChange((newVal) => {
46+
listen(item, (newVal) => {
4647
pureParams[i][key] = newVal;
4748

4849
rerun();

‎src/lib/processMaybeMutable.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,13 @@
11
import { MaybeMutable } from './types';
22
import isMutable from './isMutable';
3+
import { listen } from './eventBus';
34

45
export function processMaybeMutable<Data extends any>(
56
actionFn: (data: Data) => void,
67
) {
78
return (data: MaybeMutable<Data>) => {
89
if (isMutable(data)) {
9-
data.onChange(actionFn);
10+
listen(data, actionFn);
1011

1112
actionFn(data.value);
1213
} else if (data) {

‎src/lib/types.ts

-3
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,5 @@
1-
import { MutationCallback } from './eventBus';
2-
31
export type Mutable<Value> = {
42
readonly _mutable: true;
5-
readonly onChange: (callback: MutationCallback<Value>) => void;
63
value: Value;
74
};
85

0 commit comments

Comments
 (0)