@iobroker/dm-gui-components 9.0.2 → 9.0.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +18 -5
- package/build/Communication.d.ts +7 -6
- package/build/Communication.js +39 -22
- package/build/Communication.js.map +1 -1
- package/build/DeviceActionButton.d.ts +1 -2
- package/build/DeviceActionButton.js +2 -2
- package/build/DeviceActionButton.js.map +1 -1
- package/build/DeviceCard.d.ts +4 -5
- package/build/DeviceCard.js +34 -41
- package/build/DeviceCard.js.map +1 -1
- package/build/DeviceList.d.ts +6 -2
- package/build/DeviceList.js +26 -8
- package/build/DeviceList.js.map +1 -1
- package/build/DeviceStatus.d.ts +1 -2
- package/build/DeviceStatus.js +2 -2
- package/build/DeviceStatus.js.map +1 -1
- package/build/StateOrObjectHandler.d.ts +6 -2
- package/build/StateOrObjectHandler.js +128 -108
- package/build/StateOrObjectHandler.js.map +1 -1
- package/build/hooks.js +3 -2
- package/build/hooks.js.map +1 -1
- package/build/protocol/DmProtocolV1.js +68 -4
- package/build/protocol/DmProtocolV1.js.map +1 -1
- package/package.json +2 -2
|
@@ -1,3 +1,6 @@
|
|
|
1
|
+
const emptySubscription = {
|
|
2
|
+
unsubscribe: () => { },
|
|
3
|
+
};
|
|
1
4
|
export class StateOrObjectHandler {
|
|
2
5
|
socket;
|
|
3
6
|
objectSubs = new Map();
|
|
@@ -8,134 +11,151 @@ export class StateOrObjectHandler {
|
|
|
8
11
|
async addListener(item, callback) {
|
|
9
12
|
if (item === undefined) {
|
|
10
13
|
callback(undefined);
|
|
11
|
-
return;
|
|
14
|
+
return emptySubscription;
|
|
12
15
|
}
|
|
13
16
|
if (typeof item !== 'object') {
|
|
14
17
|
callback(item);
|
|
15
|
-
return;
|
|
18
|
+
return emptySubscription;
|
|
16
19
|
}
|
|
17
20
|
if ('en' in item) {
|
|
18
21
|
callback(item);
|
|
19
|
-
return;
|
|
22
|
+
return emptySubscription;
|
|
20
23
|
}
|
|
21
24
|
try {
|
|
22
25
|
if ('objectId' in item) {
|
|
23
|
-
|
|
24
|
-
if (!obj) {
|
|
25
|
-
callback(undefined);
|
|
26
|
-
return;
|
|
27
|
-
}
|
|
28
|
-
const parts = item.property.split('.');
|
|
29
|
-
let current = obj;
|
|
30
|
-
for (const part of parts) {
|
|
31
|
-
if (current[part] === undefined) {
|
|
32
|
-
callback(undefined);
|
|
33
|
-
return;
|
|
34
|
-
}
|
|
35
|
-
current = current[part];
|
|
36
|
-
}
|
|
37
|
-
callback(current);
|
|
38
|
-
};
|
|
39
|
-
const existing = this.objectSubs.get(item.objectId);
|
|
40
|
-
if (existing) {
|
|
41
|
-
existing.notifiers.push(notifyValue);
|
|
42
|
-
if (existing.loaded) {
|
|
43
|
-
// Already have the value — notify immediately without re-fetching
|
|
44
|
-
notifyValue(existing.cached);
|
|
45
|
-
}
|
|
46
|
-
// If still loading, notifyValue will be called once the load completes
|
|
47
|
-
return;
|
|
48
|
-
}
|
|
49
|
-
const sub = {
|
|
50
|
-
notifiers: [notifyValue],
|
|
51
|
-
handler: null,
|
|
52
|
-
loaded: false,
|
|
53
|
-
cached: undefined,
|
|
54
|
-
};
|
|
55
|
-
this.objectSubs.set(item.objectId, sub);
|
|
56
|
-
const handler = (_id, obj) => {
|
|
57
|
-
sub.cached = obj;
|
|
58
|
-
for (const n of sub.notifiers) {
|
|
59
|
-
n(obj);
|
|
60
|
-
}
|
|
61
|
-
};
|
|
62
|
-
sub.handler = handler;
|
|
63
|
-
const obj = await this.socket.getObject(item.objectId);
|
|
64
|
-
sub.cached = obj;
|
|
65
|
-
sub.loaded = true;
|
|
66
|
-
// Notify all notifiers (including any added while the fetch was in progress)
|
|
67
|
-
for (const n of sub.notifiers) {
|
|
68
|
-
n(obj);
|
|
69
|
-
}
|
|
70
|
-
await this.socket.subscribeObject(item.objectId, handler);
|
|
71
|
-
return;
|
|
26
|
+
return this.addObjectListener(item.objectId, item.property, callback);
|
|
72
27
|
}
|
|
73
28
|
if ('stateId' in item) {
|
|
74
|
-
|
|
75
|
-
let val = state?.val;
|
|
76
|
-
if (val === undefined || val === null) {
|
|
77
|
-
callback(undefined);
|
|
78
|
-
return;
|
|
79
|
-
}
|
|
80
|
-
if (item.mapping) {
|
|
81
|
-
if (typeof val === 'boolean') {
|
|
82
|
-
val = val.toString();
|
|
83
|
-
}
|
|
84
|
-
callback(item.mapping[val]);
|
|
85
|
-
}
|
|
86
|
-
else {
|
|
87
|
-
callback(val);
|
|
88
|
-
}
|
|
89
|
-
};
|
|
90
|
-
const existing = this.stateSubs.get(item.stateId);
|
|
91
|
-
if (existing) {
|
|
92
|
-
existing.notifiers.push(notifyValue);
|
|
93
|
-
if (existing.loaded) {
|
|
94
|
-
// Already have the value — notify immediately without re-fetching
|
|
95
|
-
notifyValue(existing.cached);
|
|
96
|
-
}
|
|
97
|
-
// If still loading, notifyValue will be called once the load completes
|
|
98
|
-
return;
|
|
99
|
-
}
|
|
100
|
-
const sub = {
|
|
101
|
-
notifiers: [notifyValue],
|
|
102
|
-
handler: null,
|
|
103
|
-
loaded: false,
|
|
104
|
-
cached: undefined,
|
|
105
|
-
};
|
|
106
|
-
this.stateSubs.set(item.stateId, sub);
|
|
107
|
-
const handler = (_id, state) => {
|
|
108
|
-
sub.cached = state;
|
|
109
|
-
for (const n of sub.notifiers) {
|
|
110
|
-
n(state);
|
|
111
|
-
}
|
|
112
|
-
};
|
|
113
|
-
sub.handler = handler;
|
|
114
|
-
const state = await this.socket.getState(item.stateId);
|
|
115
|
-
sub.cached = state;
|
|
116
|
-
sub.loaded = true;
|
|
117
|
-
// Notify all notifiers (including any added while the fetch was in progress)
|
|
118
|
-
for (const n of sub.notifiers) {
|
|
119
|
-
n(state);
|
|
120
|
-
}
|
|
121
|
-
await this.socket.subscribeState(item.stateId, handler);
|
|
122
|
-
return;
|
|
29
|
+
return this.addStateListener(item.stateId, item.mapping, callback);
|
|
123
30
|
}
|
|
124
31
|
}
|
|
125
32
|
catch (error) {
|
|
126
33
|
console.error('Error in StateOrObjectHandler:', item, error);
|
|
127
34
|
}
|
|
128
35
|
callback(undefined);
|
|
36
|
+
return emptySubscription;
|
|
129
37
|
}
|
|
130
|
-
async
|
|
131
|
-
|
|
132
|
-
|
|
38
|
+
async addObjectListener(objectId, property, callback) {
|
|
39
|
+
const notifyValue = (obj) => {
|
|
40
|
+
if (!obj) {
|
|
41
|
+
callback(undefined);
|
|
42
|
+
return;
|
|
43
|
+
}
|
|
44
|
+
const parts = property.split('.');
|
|
45
|
+
let current = obj;
|
|
46
|
+
for (const part of parts) {
|
|
47
|
+
if (current[part] === undefined) {
|
|
48
|
+
callback(undefined);
|
|
49
|
+
return;
|
|
50
|
+
}
|
|
51
|
+
current = current[part];
|
|
52
|
+
}
|
|
53
|
+
callback(current);
|
|
54
|
+
};
|
|
55
|
+
const existing = this.objectSubs.get(objectId);
|
|
56
|
+
if (existing) {
|
|
57
|
+
existing.notifiers.push(notifyValue);
|
|
58
|
+
if (existing.loaded) {
|
|
59
|
+
// Already have the value — notify immediately without re-fetching
|
|
60
|
+
notifyValue(existing.cached);
|
|
61
|
+
}
|
|
62
|
+
// If still loading, notifyValue will be called once the load completes
|
|
63
|
+
return { unsubscribe: () => existing.unsubscribe(notifyValue) };
|
|
133
64
|
}
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
65
|
+
const sub = {
|
|
66
|
+
notifiers: [notifyValue],
|
|
67
|
+
handler: null,
|
|
68
|
+
unsubscribe: null,
|
|
69
|
+
loaded: false,
|
|
70
|
+
cached: undefined,
|
|
71
|
+
};
|
|
72
|
+
this.objectSubs.set(objectId, sub);
|
|
73
|
+
sub.handler = (_id, obj) => {
|
|
74
|
+
sub.cached = obj;
|
|
75
|
+
for (const n of sub.notifiers) {
|
|
76
|
+
n(obj);
|
|
77
|
+
}
|
|
78
|
+
};
|
|
79
|
+
sub.unsubscribe = async (notifier) => {
|
|
80
|
+
const index = sub.notifiers.indexOf(notifier);
|
|
81
|
+
if (index !== -1) {
|
|
82
|
+
sub.notifiers.splice(index, 1);
|
|
83
|
+
}
|
|
84
|
+
if (sub.notifiers.length === 0) {
|
|
85
|
+
this.objectSubs.delete(objectId);
|
|
86
|
+
await this.socket.unsubscribeObject(objectId, sub.handler);
|
|
87
|
+
}
|
|
88
|
+
};
|
|
89
|
+
const obj = await this.socket.getObject(objectId);
|
|
90
|
+
sub.cached = obj;
|
|
91
|
+
sub.loaded = true;
|
|
92
|
+
// Notify all notifiers (including any added while getObject was in progress)
|
|
93
|
+
for (const n of sub.notifiers) {
|
|
94
|
+
n(obj);
|
|
95
|
+
}
|
|
96
|
+
await this.socket.subscribeObject(objectId, sub.handler);
|
|
97
|
+
return { unsubscribe: () => sub.unsubscribe(notifyValue) };
|
|
98
|
+
}
|
|
99
|
+
async addStateListener(stateId, mapping, callback) {
|
|
100
|
+
const notifyValue = (state) => {
|
|
101
|
+
let val = state?.val;
|
|
102
|
+
if (val === undefined || val === null) {
|
|
103
|
+
callback(undefined);
|
|
104
|
+
return;
|
|
105
|
+
}
|
|
106
|
+
if (mapping) {
|
|
107
|
+
if (typeof val === 'boolean') {
|
|
108
|
+
val = val.toString();
|
|
109
|
+
}
|
|
110
|
+
callback(mapping[val]);
|
|
111
|
+
}
|
|
112
|
+
else {
|
|
113
|
+
callback(val);
|
|
114
|
+
}
|
|
115
|
+
};
|
|
116
|
+
const existing = this.stateSubs.get(stateId);
|
|
117
|
+
if (existing) {
|
|
118
|
+
existing.notifiers.push(notifyValue);
|
|
119
|
+
if (existing.loaded) {
|
|
120
|
+
// Already have the value — notify immediately without re-fetching
|
|
121
|
+
notifyValue(existing.cached);
|
|
122
|
+
}
|
|
123
|
+
// If still loading, notifyValue will be called once the load completes
|
|
124
|
+
return { unsubscribe: () => existing.unsubscribe(notifyValue) };
|
|
125
|
+
}
|
|
126
|
+
const sub = {
|
|
127
|
+
notifiers: [notifyValue],
|
|
128
|
+
handler: null,
|
|
129
|
+
unsubscribe: null,
|
|
130
|
+
loaded: false,
|
|
131
|
+
cached: undefined,
|
|
132
|
+
};
|
|
133
|
+
this.stateSubs.set(stateId, sub);
|
|
134
|
+
sub.handler = (_id, state) => {
|
|
135
|
+
sub.cached = state;
|
|
136
|
+
for (const n of sub.notifiers) {
|
|
137
|
+
n(state);
|
|
138
|
+
}
|
|
139
|
+
};
|
|
140
|
+
sub.unsubscribe = notifier => {
|
|
141
|
+
const index = sub.notifiers.indexOf(notifier);
|
|
142
|
+
if (index !== -1) {
|
|
143
|
+
sub.notifiers.splice(index, 1);
|
|
144
|
+
}
|
|
145
|
+
if (sub.notifiers.length === 0) {
|
|
146
|
+
this.stateSubs.delete(stateId);
|
|
147
|
+
this.socket.unsubscribeState(stateId, sub.handler);
|
|
148
|
+
}
|
|
149
|
+
};
|
|
150
|
+
const state = await this.socket.getState(stateId);
|
|
151
|
+
sub.cached = state;
|
|
152
|
+
sub.loaded = true;
|
|
153
|
+
// Notify all notifiers (including any added while the fetch was in progress)
|
|
154
|
+
for (const n of sub.notifiers) {
|
|
155
|
+
n(state);
|
|
137
156
|
}
|
|
138
|
-
this.
|
|
157
|
+
await this.socket.subscribeState(stateId, sub.handler);
|
|
158
|
+
return { unsubscribe: () => sub.unsubscribe(notifyValue) };
|
|
139
159
|
}
|
|
140
160
|
}
|
|
141
161
|
//# sourceMappingURL=StateOrObjectHandler.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"StateOrObjectHandler.js","sourceRoot":"./src/","sources":["StateOrObjectHandler.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Connection, ObjectChangeHandler } from '@iobroker/adapter-react-v5';\nimport type { ValueOrStateOrObject } from '@iobroker/dm-utils';\n\ninterface ObjectSubscription {\n notifiers: ((obj?: ioBroker.Object | null) => void)[];\n handler: ObjectChangeHandler;\n loaded: boolean;\n cached: ioBroker.Object | null | undefined;\n}\n\ninterface StateSubscription {\n notifiers: ((state?: ioBroker.State | null) => void)[];\n handler: ioBroker.StateChangeHandler;\n loaded: boolean;\n cached: ioBroker.State | null | undefined;\n}\n\nexport class StateOrObjectHandler {\n private readonly objectSubs = new Map<string, ObjectSubscription>();\n private readonly stateSubs = new Map<string, StateSubscription>();\n\n constructor(private readonly socket: Connection) {}\n\n public async addListener<T extends ioBroker.StringOrTranslated | number | boolean>(\n item: ValueOrStateOrObject<T> | undefined,\n callback: (value: T | undefined) => void,\n ): Promise<void> {\n if (item === undefined) {\n callback(undefined);\n return;\n }\n\n if (typeof item !== 'object') {\n callback(item);\n return;\n }\n\n if ('en' in item) {\n callback(item as T);\n return;\n }\n\n try {\n if ('objectId' in item) {\n const notifyValue = (obj?: ioBroker.Object | null): void => {\n if (!obj) {\n callback(undefined);\n return;\n }\n\n const parts = item.property.split('.');\n let current: any = obj;\n for (const part of parts) {\n if (current[part] === undefined) {\n callback(undefined);\n return;\n }\n current = current[part];\n }\n\n callback(current as T);\n };\n\n const existing = this.objectSubs.get(item.objectId);\n if (existing) {\n existing.notifiers.push(notifyValue);\n if (existing.loaded) {\n // Already have the value — notify immediately without re-fetching\n notifyValue(existing.cached);\n }\n // If still loading, notifyValue will be called once the load completes\n return;\n }\n\n const sub: ObjectSubscription = {\n notifiers: [notifyValue],\n handler: null!,\n loaded: false,\n cached: undefined,\n };\n this.objectSubs.set(item.objectId, sub);\n\n const handler: ObjectChangeHandler = (_id, obj) => {\n sub.cached = obj;\n for (const n of sub.notifiers) {\n n(obj);\n }\n };\n sub.handler = handler;\n\n const obj = await this.socket.getObject(item.objectId);\n sub.cached = obj;\n sub.loaded = true;\n // Notify all notifiers (including any added while the fetch was in progress)\n for (const n of sub.notifiers) {\n n(obj);\n }\n await this.socket.subscribeObject(item.objectId, handler);\n return;\n }\n\n if ('stateId' in item) {\n const notifyValue = (state?: ioBroker.State | null): void => {\n let val = state?.val;\n if (val === undefined || val === null) {\n callback(undefined);\n return;\n }\n\n if (item.mapping) {\n if (typeof val === 'boolean') {\n val = val.toString();\n }\n\n callback(item.mapping[val]);\n } else {\n callback(val as T);\n }\n };\n\n const existing = this.stateSubs.get(item.stateId);\n if (existing) {\n existing.notifiers.push(notifyValue);\n if (existing.loaded) {\n // Already have the value — notify immediately without re-fetching\n notifyValue(existing.cached);\n }\n // If still loading, notifyValue will be called once the load completes\n return;\n }\n\n const sub: StateSubscription = {\n notifiers: [notifyValue],\n handler: null!,\n loaded: false,\n cached: undefined,\n };\n this.stateSubs.set(item.stateId, sub);\n\n const handler: ioBroker.StateChangeHandler = (_id, state) => {\n sub.cached = state;\n for (const n of sub.notifiers) {\n n(state);\n }\n };\n sub.handler = handler;\n\n const state = await this.socket.getState(item.stateId);\n sub.cached = state;\n sub.loaded = true;\n // Notify all notifiers (including any added while the fetch was in progress)\n for (const n of sub.notifiers) {\n n(state);\n }\n await this.socket.subscribeState(item.stateId, handler);\n return;\n }\n } catch (error) {\n console.error('Error in StateOrObjectHandler:', item, error);\n }\n\n callback(undefined);\n }\n\n public async unsubscribe(): Promise<void> {\n for (const [id, sub] of this.objectSubs) {\n await this.socket.unsubscribeObject(id, sub.handler);\n }\n this.objectSubs.clear();\n\n for (const [id, sub] of this.stateSubs) {\n this.socket.unsubscribeState(id, sub.handler);\n }\n this.stateSubs.clear();\n }\n}\n"]}
|
|
1
|
+
{"version":3,"file":"StateOrObjectHandler.js","sourceRoot":"./src/","sources":["StateOrObjectHandler.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Connection, ObjectChangeHandler } from '@iobroker/adapter-react-v5';\nimport type { ValueOrStateOrObject } from '@iobroker/dm-utils';\n\ntype ObjectSubscriptionNotifier = (obj?: ioBroker.Object | null) => void;\n\ninterface ObjectSubscription {\n notifiers: ObjectSubscriptionNotifier[];\n handler: ObjectChangeHandler;\n unsubscribe: (notifier: ObjectSubscriptionNotifier) => Promise<void>;\n loaded: boolean;\n cached: ioBroker.Object | null | undefined;\n}\n\ntype StateSubscriptionNotifier = (state?: ioBroker.State | null) => void;\n\ninterface StateSubscription {\n notifiers: StateSubscriptionNotifier[];\n handler: ioBroker.StateChangeHandler;\n unsubscribe: (notifier: StateSubscriptionNotifier) => void;\n loaded: boolean;\n cached: ioBroker.State | null | undefined;\n}\n\nexport interface StateOrObjectSubscription {\n unsubscribe: () => Promise<void> | void;\n}\n\nconst emptySubscription: StateOrObjectSubscription = {\n unsubscribe: () => {},\n};\n\nexport class StateOrObjectHandler {\n private readonly objectSubs = new Map<string, ObjectSubscription>();\n private readonly stateSubs = new Map<string, StateSubscription>();\n\n constructor(private readonly socket: Connection) {}\n\n public async addListener<T extends ioBroker.StringOrTranslated | number | boolean>(\n item: ValueOrStateOrObject<T> | undefined,\n callback: (value: T | undefined) => void,\n ): Promise<StateOrObjectSubscription> {\n if (item === undefined) {\n callback(undefined);\n return emptySubscription;\n }\n\n if (typeof item !== 'object') {\n callback(item);\n return emptySubscription;\n }\n\n if ('en' in item) {\n callback(item as T);\n return emptySubscription;\n }\n\n try {\n if ('objectId' in item) {\n return this.addObjectListener(item.objectId, item.property, callback);\n }\n\n if ('stateId' in item) {\n return this.addStateListener(item.stateId, item.mapping, callback);\n }\n } catch (error) {\n console.error('Error in StateOrObjectHandler:', item, error);\n }\n\n callback(undefined);\n return emptySubscription;\n }\n\n private async addObjectListener<T extends ioBroker.StringOrTranslated | number | boolean>(\n objectId: string,\n property: string,\n callback: (value: T | undefined) => void,\n ): Promise<StateOrObjectSubscription> {\n const notifyValue = (obj?: ioBroker.Object | null): void => {\n if (!obj) {\n callback(undefined);\n return;\n }\n\n const parts = property.split('.');\n let current: any = obj;\n for (const part of parts) {\n if (current[part] === undefined) {\n callback(undefined);\n return;\n }\n current = current[part];\n }\n\n callback(current as T);\n };\n\n const existing = this.objectSubs.get(objectId);\n if (existing) {\n existing.notifiers.push(notifyValue);\n if (existing.loaded) {\n // Already have the value — notify immediately without re-fetching\n notifyValue(existing.cached);\n }\n // If still loading, notifyValue will be called once the load completes\n return { unsubscribe: () => existing.unsubscribe(notifyValue) };\n }\n\n const sub: ObjectSubscription = {\n notifiers: [notifyValue],\n handler: null!,\n unsubscribe: null!,\n loaded: false,\n cached: undefined,\n };\n this.objectSubs.set(objectId, sub);\n\n sub.handler = (_id, obj) => {\n sub.cached = obj;\n for (const n of sub.notifiers) {\n n(obj);\n }\n };\n sub.unsubscribe = async notifier => {\n const index = sub.notifiers.indexOf(notifier);\n if (index !== -1) {\n sub.notifiers.splice(index, 1);\n }\n if (sub.notifiers.length === 0) {\n this.objectSubs.delete(objectId);\n await this.socket.unsubscribeObject(objectId, sub.handler);\n }\n };\n\n const obj = await this.socket.getObject(objectId);\n sub.cached = obj;\n sub.loaded = true;\n // Notify all notifiers (including any added while getObject was in progress)\n for (const n of sub.notifiers) {\n n(obj);\n }\n await this.socket.subscribeObject(objectId, sub.handler);\n return { unsubscribe: () => sub.unsubscribe(notifyValue) };\n }\n\n private async addStateListener<T extends ioBroker.StringOrTranslated | number | boolean>(\n stateId: string,\n mapping: Record<string | number, T> | undefined,\n callback: (value: T | undefined) => void,\n ): Promise<StateOrObjectSubscription> {\n const notifyValue = (state?: ioBroker.State | null): void => {\n let val = state?.val;\n if (val === undefined || val === null) {\n callback(undefined);\n return;\n }\n\n if (mapping) {\n if (typeof val === 'boolean') {\n val = val.toString();\n }\n\n callback(mapping[val]);\n } else {\n callback(val as T);\n }\n };\n\n const existing = this.stateSubs.get(stateId);\n if (existing) {\n existing.notifiers.push(notifyValue);\n if (existing.loaded) {\n // Already have the value — notify immediately without re-fetching\n notifyValue(existing.cached);\n }\n // If still loading, notifyValue will be called once the load completes\n return { unsubscribe: () => existing.unsubscribe(notifyValue) };\n }\n\n const sub: StateSubscription = {\n notifiers: [notifyValue],\n handler: null!,\n unsubscribe: null!,\n loaded: false,\n cached: undefined,\n };\n this.stateSubs.set(stateId, sub);\n\n sub.handler = (_id, state) => {\n sub.cached = state;\n for (const n of sub.notifiers) {\n n(state);\n }\n };\n sub.unsubscribe = notifier => {\n const index = sub.notifiers.indexOf(notifier);\n if (index !== -1) {\n sub.notifiers.splice(index, 1);\n }\n if (sub.notifiers.length === 0) {\n this.stateSubs.delete(stateId);\n this.socket.unsubscribeState(stateId, sub.handler);\n }\n };\n\n const state = await this.socket.getState(stateId);\n sub.cached = state;\n sub.loaded = true;\n // Notify all notifiers (including any added while the fetch was in progress)\n for (const n of sub.notifiers) {\n n(state);\n }\n await this.socket.subscribeState(stateId, sub.handler);\n return { unsubscribe: () => sub.unsubscribe(notifyValue) };\n }\n}\n"]}
|
package/build/hooks.js
CHANGED
|
@@ -2,8 +2,9 @@ import { useEffect, useState } from 'react';
|
|
|
2
2
|
export function useStateOrObject(item, stateOrObjectHandler) {
|
|
3
3
|
const [value, setValue] = useState();
|
|
4
4
|
useEffect(() => {
|
|
5
|
-
|
|
6
|
-
|
|
5
|
+
let subscription = undefined;
|
|
6
|
+
void stateOrObjectHandler.addListener(item, value => setValue(value)).then(sub => (subscription = sub));
|
|
7
|
+
return () => void subscription?.unsubscribe();
|
|
7
8
|
}, [stateOrObjectHandler, item]);
|
|
8
9
|
return value;
|
|
9
10
|
}
|
package/build/hooks.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"hooks.js","sourceRoot":"./src/","sources":["hooks.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAC;AAG5C,MAAM,UAAU,gBAAgB,CAC5B,IAAyC,EACzC,oBAA0C;IAE1C,MAAM,CAAC,KAAK,EAAE,QAAQ,CAAC,GAAG,QAAQ,EAAK,CAAC;IAExC,SAAS,CAAC,GAAG,EAAE;QACX,KAAK,oBAAoB,CAAC,WAAW,CAAC,IAAI,EAAE,KAAK,CAAC,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC;
|
|
1
|
+
{"version":3,"file":"hooks.js","sourceRoot":"./src/","sources":["hooks.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAC;AAG5C,MAAM,UAAU,gBAAgB,CAC5B,IAAyC,EACzC,oBAA0C;IAE1C,MAAM,CAAC,KAAK,EAAE,QAAQ,CAAC,GAAG,QAAQ,EAAK,CAAC;IAExC,SAAS,CAAC,GAAG,EAAE;QACX,IAAI,YAAY,GAA0C,SAAS,CAAC;QACpE,KAAK,oBAAoB,CAAC,WAAW,CAAC,IAAI,EAAE,KAAK,CAAC,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,YAAY,GAAG,GAAG,CAAC,CAAC,CAAC;QACxG,OAAO,GAAG,EAAE,CAAC,KAAK,YAAY,EAAE,WAAW,EAAE,CAAC;IAClD,CAAC,EAAE,CAAC,oBAAoB,EAAE,IAAI,CAAC,CAAC,CAAC;IAEjC,OAAO,KAAK,CAAC;AACjB,CAAC","sourcesContent":["import type { ValueOrStateOrObject } from '@iobroker/dm-utils';\nimport { useEffect, useState } from 'react';\nimport type { StateOrObjectHandler, StateOrObjectSubscription } from './StateOrObjectHandler';\n\nexport function useStateOrObject<T extends ioBroker.StringOrTranslated | number | boolean>(\n item: ValueOrStateOrObject<T> | undefined,\n stateOrObjectHandler: StateOrObjectHandler,\n): T | undefined {\n const [value, setValue] = useState<T>();\n\n useEffect(() => {\n let subscription: StateOrObjectSubscription | undefined = undefined;\n void stateOrObjectHandler.addListener(item, value => setValue(value)).then(sub => (subscription = sub));\n return () => void subscription?.unsubscribe();\n }, [stateOrObjectHandler, item]);\n\n return value;\n}\n"]}
|
|
@@ -12,11 +12,75 @@ export class DmProtocolV1 extends DmProtocolBase {
|
|
|
12
12
|
const devices = await this.send('dm:listDevices');
|
|
13
13
|
await callback(devices.map(d => ({ ...d, identifier: d.id })));
|
|
14
14
|
}
|
|
15
|
-
sendAction(command, messageToSend) {
|
|
16
|
-
|
|
15
|
+
async sendAction(command, messageToSend) {
|
|
16
|
+
const response = await this.send(command, messageToSend);
|
|
17
|
+
switch (response.type) {
|
|
18
|
+
case 'message':
|
|
19
|
+
return {
|
|
20
|
+
type: 'message',
|
|
21
|
+
message: response.message || '',
|
|
22
|
+
origin: response.origin, // origin was accidentally set to string in V1
|
|
23
|
+
};
|
|
24
|
+
case 'confirm':
|
|
25
|
+
return {
|
|
26
|
+
type: 'confirm',
|
|
27
|
+
confirm: response.confirm || '',
|
|
28
|
+
origin: response.origin, // origin was accidentally set to string in V1
|
|
29
|
+
};
|
|
30
|
+
case 'progress':
|
|
31
|
+
return {
|
|
32
|
+
type: 'progress',
|
|
33
|
+
progress: response.progress || { open: false, indeterminate: true },
|
|
34
|
+
origin: response.origin, // origin was accidentally set to string in V1
|
|
35
|
+
};
|
|
36
|
+
case 'form':
|
|
37
|
+
return {
|
|
38
|
+
type: 'form',
|
|
39
|
+
form: response.form || { title: '', schema: { type: 'panel', items: {} } },
|
|
40
|
+
origin: response.origin, // origin was accidentally set to string in V1
|
|
41
|
+
};
|
|
42
|
+
case 'result':
|
|
43
|
+
if (response.result.error) {
|
|
44
|
+
return {
|
|
45
|
+
type: 'result',
|
|
46
|
+
result: { error: response.result.error },
|
|
47
|
+
origin: response.origin, // origin was accidentally set to string in V1
|
|
48
|
+
};
|
|
49
|
+
}
|
|
50
|
+
switch (response.result.refresh) {
|
|
51
|
+
case true:
|
|
52
|
+
return {
|
|
53
|
+
type: 'result',
|
|
54
|
+
result: { refresh: true },
|
|
55
|
+
origin: response.origin, // origin was accidentally set to string in V1
|
|
56
|
+
};
|
|
57
|
+
case 'device':
|
|
58
|
+
return {
|
|
59
|
+
type: 'result',
|
|
60
|
+
result: { refresh: 'devices' },
|
|
61
|
+
origin: response.origin, // origin was accidentally set to string in V1
|
|
62
|
+
};
|
|
63
|
+
case 'instance':
|
|
64
|
+
return {
|
|
65
|
+
type: 'result',
|
|
66
|
+
result: { refresh: 'instance' },
|
|
67
|
+
origin: response.origin, // origin was accidentally set to string in V1
|
|
68
|
+
};
|
|
69
|
+
default:
|
|
70
|
+
return {
|
|
71
|
+
type: 'result',
|
|
72
|
+
result: { refresh: false },
|
|
73
|
+
origin: response.origin, // origin was accidentally set to string in V1
|
|
74
|
+
};
|
|
75
|
+
}
|
|
76
|
+
default:
|
|
77
|
+
throw new Error(`Unknown response type: ${response.type}`);
|
|
78
|
+
}
|
|
17
79
|
}
|
|
18
|
-
sendControl(command, messageToSend) {
|
|
19
|
-
|
|
80
|
+
async sendControl(command, messageToSend) {
|
|
81
|
+
const response = await this.send(command, messageToSend);
|
|
82
|
+
// contents matches, types unfortunately don't, so we need to cast here
|
|
83
|
+
return response;
|
|
20
84
|
}
|
|
21
85
|
}
|
|
22
86
|
//# sourceMappingURL=DmProtocolV1.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"DmProtocolV1.js","sourceRoot":"./src/","sources":["protocol/DmProtocolV1.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"DmProtocolV1.js","sourceRoot":"./src/","sources":["protocol/DmProtocolV1.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable class-methods-use-this */\nimport type { DeviceRefresh as DeviceRefreshV1 } from '@iobroker/dm-utils-v1';\nimport type * as V1 from '@iobroker/dm-utils-v1/build/types/api';\nimport type {\n CommunicationForm,\n ControlState,\n DeviceInfo,\n DmActionResponse,\n DmControlResponse,\n InstanceDetails,\n} from './api';\nimport { DmProtocolBase, type CommandName, type LoadDevicesCallback, type Message } from './DmProtocolBase';\n\ninterface DmResponseV1 {\n /* Type of message */\n type: 'message' | 'confirm' | 'progress' | 'result' | 'form';\n /* Origin */\n origin: string;\n}\n\ninterface DmControlResponseV1 extends DmResponseV1 {\n result: {\n error?: {\n code: number;\n message: string;\n };\n state?: ioBroker.State;\n deviceId: string;\n controlId: string;\n };\n}\n\ninterface DmActionResponseV1 extends DmResponseV1 {\n result: {\n refresh?: DeviceRefreshV1;\n error?: {\n code: number;\n message: string;\n };\n };\n message?: string;\n confirm?: string;\n form?: CommunicationForm;\n progress?:\n | {\n open: boolean;\n indeterminate: boolean;\n }\n | {\n open: boolean;\n progress: number;\n };\n}\n\nexport class DmProtocolV1 extends DmProtocolBase {\n public override convertInstanceDetails(details: any): InstanceDetails {\n if (details.apiVersion !== 'v1') {\n throw new Error(`Unsupported API version: ${details.apiVersion ?? 'unknown'}`);\n }\n\n const v1 = details as V1.InstanceDetails;\n return { ...v1, apiVersion: 'v3' };\n }\n\n public override async loadDevices(callback: LoadDevicesCallback): Promise<void> {\n // in V1, devices come in a single batch; thus we can simply call the callback with those\n const devices = await this.send<V1.DeviceInfo[]>('dm:listDevices');\n await callback(devices.map<DeviceInfo>(d => ({ ...d, identifier: d.id })));\n }\n\n public override async sendAction(command: CommandName, messageToSend: Message): Promise<DmActionResponse> {\n const response = await this.send<DmActionResponseV1>(command, messageToSend);\n switch (response.type) {\n case 'message':\n return {\n type: 'message',\n message: response.message || '',\n origin: response.origin as any, // origin was accidentally set to string in V1\n };\n case 'confirm':\n return {\n type: 'confirm',\n confirm: response.confirm || '',\n origin: response.origin as any, // origin was accidentally set to string in V1\n };\n case 'progress':\n return {\n type: 'progress',\n progress: response.progress || { open: false, indeterminate: true },\n origin: response.origin as any, // origin was accidentally set to string in V1\n };\n case 'form':\n return {\n type: 'form',\n form: response.form || { title: '', schema: { type: 'panel', items: {} } },\n origin: response.origin as any, // origin was accidentally set to string in V1\n };\n case 'result':\n if (response.result.error) {\n return {\n type: 'result',\n result: { error: response.result.error },\n origin: response.origin as any, // origin was accidentally set to string in V1\n };\n }\n switch (response.result.refresh) {\n case true:\n return {\n type: 'result',\n result: { refresh: true },\n origin: response.origin as any, // origin was accidentally set to string in V1\n };\n case 'device':\n return {\n type: 'result',\n result: { refresh: 'devices' },\n origin: response.origin as any, // origin was accidentally set to string in V1\n };\n case 'instance':\n return {\n type: 'result',\n result: { refresh: 'instance' },\n origin: response.origin as any, // origin was accidentally set to string in V1\n };\n default:\n return {\n type: 'result',\n result: { refresh: false },\n origin: response.origin as any, // origin was accidentally set to string in V1\n };\n }\n default:\n throw new Error(`Unknown response type: ${(response as any).type}`);\n }\n }\n\n public async sendControl(\n command: CommandName,\n messageToSend: { deviceId: string; controlId: string; state?: ControlState },\n ): Promise<DmControlResponse> {\n const response = await this.send<DmControlResponseV1>(command, messageToSend);\n // contents matches, types unfortunately don't, so we need to cast here\n return response as any;\n }\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@iobroker/dm-gui-components",
|
|
3
|
-
"version": "9.0.
|
|
3
|
+
"version": "9.0.3",
|
|
4
4
|
"description": "ReactJS components to develop admin interface for ioBroker device manager.",
|
|
5
5
|
"author": {
|
|
6
6
|
"name": "Jey Cee",
|
|
@@ -53,7 +53,7 @@
|
|
|
53
53
|
"homepage": "https://github.com/ioBroker/dm-gui-components#readme",
|
|
54
54
|
"dependencies": {
|
|
55
55
|
"@iobroker/adapter-react-v5": "^8.1.1",
|
|
56
|
-
"@iobroker/json-config": "^8.2.
|
|
56
|
+
"@iobroker/json-config": "^8.2.3"
|
|
57
57
|
},
|
|
58
58
|
"devDependencies": {
|
|
59
59
|
"@alcalzone/release-script": "^5.1.1",
|