@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.
@@ -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
- const notifyValue = (obj) => {
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
- const notifyValue = (state) => {
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 unsubscribe() {
131
- for (const [id, sub] of this.objectSubs) {
132
- await this.socket.unsubscribeObject(id, sub.handler);
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
- this.objectSubs.clear();
135
- for (const [id, sub] of this.stateSubs) {
136
- this.socket.unsubscribeState(id, sub.handler);
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.stateSubs.clear();
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
- void stateOrObjectHandler.addListener(item, value => setValue(value));
6
- return () => void stateOrObjectHandler.unsubscribe();
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
  }
@@ -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;QACtE,OAAO,GAAG,EAAE,CAAC,KAAK,oBAAoB,CAAC,WAAW,EAAE,CAAC;IACzD,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 } 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 void stateOrObjectHandler.addListener(item, value => setValue(value));\n return () => void stateOrObjectHandler.unsubscribe();\n }, [stateOrObjectHandler, item]);\n\n return value;\n}\n"]}
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
- return this.send(command, messageToSend);
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
- return this.send(command, messageToSend);
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":"AAGA,OAAO,EAAE,cAAc,EAA4D,MAAM,kBAAkB,CAAC;AAE5G,MAAM,OAAO,YAAa,SAAQ,cAAc;IAC5B,sBAAsB,CAAC,OAAY;QAC/C,IAAI,OAAO,CAAC,UAAU,KAAK,IAAI,EAAE,CAAC;YAC9B,MAAM,IAAI,KAAK,CAAC,4BAA4B,OAAO,CAAC,UAAU,IAAI,SAAS,EAAE,CAAC,CAAC;QACnF,CAAC;QAED,MAAM,EAAE,GAAG,OAA6B,CAAC;QACzC,OAAO,EAAE,GAAG,EAAE,EAAE,UAAU,EAAE,IAAI,EAAE,CAAC;IACvC,CAAC;IAEe,KAAK,CAAC,WAAW,CAAC,QAA6B;QAC3D,yFAAyF;QACzF,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,IAAI,CAAkB,gBAAgB,CAAC,CAAC;QACnE,MAAM,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAa,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,UAAU,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC;IAC/E,CAAC;IAEe,UAAU,CAAC,OAAoB,EAAE,aAAsB;QACnE,OAAO,IAAI,CAAC,IAAI,CAAmB,OAAO,EAAE,aAAa,CAAC,CAAC;IAC/D,CAAC;IAEM,WAAW,CACd,OAAoB,EACpB,aAA4E;QAE5E,OAAO,IAAI,CAAC,IAAI,CAAoB,OAAO,EAAE,aAAa,CAAC,CAAC;IAChE,CAAC;CACJ","sourcesContent":["/* eslint-disable class-methods-use-this */\nimport type * as V1 from '@iobroker/dm-utils-v1/build/types/api';\nimport type { ControlState, DeviceInfo, DmActionResponse, DmControlResponse, InstanceDetails } from './api';\nimport { DmProtocolBase, type CommandName, type LoadDevicesCallback, type Message } from './DmProtocolBase';\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 sendAction(command: CommandName, messageToSend: Message): Promise<DmActionResponse> {\n return this.send<DmActionResponse>(command, messageToSend);\n }\n\n public sendControl(\n command: CommandName,\n messageToSend: { deviceId: string; controlId: string; state?: ControlState },\n ): Promise<DmControlResponse> {\n return this.send<DmControlResponse>(command, messageToSend);\n }\n}\n"]}
1
+ {"version":3,"file":"DmProtocolV1.js","sourceRoot":"./src/","sources":["protocol/DmProtocolV1.ts"],"names":[],"mappings":"AAWA,OAAO,EAAE,cAAc,EAA4D,MAAM,kBAAkB,CAAC;AA2C5G,MAAM,OAAO,YAAa,SAAQ,cAAc;IAC5B,sBAAsB,CAAC,OAAY;QAC/C,IAAI,OAAO,CAAC,UAAU,KAAK,IAAI,EAAE,CAAC;YAC9B,MAAM,IAAI,KAAK,CAAC,4BAA4B,OAAO,CAAC,UAAU,IAAI,SAAS,EAAE,CAAC,CAAC;QACnF,CAAC;QAED,MAAM,EAAE,GAAG,OAA6B,CAAC;QACzC,OAAO,EAAE,GAAG,EAAE,EAAE,UAAU,EAAE,IAAI,EAAE,CAAC;IACvC,CAAC;IAEe,KAAK,CAAC,WAAW,CAAC,QAA6B;QAC3D,yFAAyF;QACzF,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,IAAI,CAAkB,gBAAgB,CAAC,CAAC;QACnE,MAAM,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAa,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,UAAU,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC;IAC/E,CAAC;IAEe,KAAK,CAAC,UAAU,CAAC,OAAoB,EAAE,aAAsB;QACzE,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,IAAI,CAAqB,OAAO,EAAE,aAAa,CAAC,CAAC;QAC7E,QAAQ,QAAQ,CAAC,IAAI,EAAE,CAAC;YACpB,KAAK,SAAS;gBACV,OAAO;oBACH,IAAI,EAAE,SAAS;oBACf,OAAO,EAAE,QAAQ,CAAC,OAAO,IAAI,EAAE;oBAC/B,MAAM,EAAE,QAAQ,CAAC,MAAa,EAAE,8CAA8C;iBACjF,CAAC;YACN,KAAK,SAAS;gBACV,OAAO;oBACH,IAAI,EAAE,SAAS;oBACf,OAAO,EAAE,QAAQ,CAAC,OAAO,IAAI,EAAE;oBAC/B,MAAM,EAAE,QAAQ,CAAC,MAAa,EAAE,8CAA8C;iBACjF,CAAC;YACN,KAAK,UAAU;gBACX,OAAO;oBACH,IAAI,EAAE,UAAU;oBAChB,QAAQ,EAAE,QAAQ,CAAC,QAAQ,IAAI,EAAE,IAAI,EAAE,KAAK,EAAE,aAAa,EAAE,IAAI,EAAE;oBACnE,MAAM,EAAE,QAAQ,CAAC,MAAa,EAAE,8CAA8C;iBACjF,CAAC;YACN,KAAK,MAAM;gBACP,OAAO;oBACH,IAAI,EAAE,MAAM;oBACZ,IAAI,EAAE,QAAQ,CAAC,IAAI,IAAI,EAAE,KAAK,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,KAAK,EAAE,EAAE,EAAE,EAAE;oBAC1E,MAAM,EAAE,QAAQ,CAAC,MAAa,EAAE,8CAA8C;iBACjF,CAAC;YACN,KAAK,QAAQ;gBACT,IAAI,QAAQ,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;oBACxB,OAAO;wBACH,IAAI,EAAE,QAAQ;wBACd,MAAM,EAAE,EAAE,KAAK,EAAE,QAAQ,CAAC,MAAM,CAAC,KAAK,EAAE;wBACxC,MAAM,EAAE,QAAQ,CAAC,MAAa,EAAE,8CAA8C;qBACjF,CAAC;gBACN,CAAC;gBACD,QAAQ,QAAQ,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;oBAC9B,KAAK,IAAI;wBACL,OAAO;4BACH,IAAI,EAAE,QAAQ;4BACd,MAAM,EAAE,EAAE,OAAO,EAAE,IAAI,EAAE;4BACzB,MAAM,EAAE,QAAQ,CAAC,MAAa,EAAE,8CAA8C;yBACjF,CAAC;oBACN,KAAK,QAAQ;wBACT,OAAO;4BACH,IAAI,EAAE,QAAQ;4BACd,MAAM,EAAE,EAAE,OAAO,EAAE,SAAS,EAAE;4BAC9B,MAAM,EAAE,QAAQ,CAAC,MAAa,EAAE,8CAA8C;yBACjF,CAAC;oBACN,KAAK,UAAU;wBACX,OAAO;4BACH,IAAI,EAAE,QAAQ;4BACd,MAAM,EAAE,EAAE,OAAO,EAAE,UAAU,EAAE;4BAC/B,MAAM,EAAE,QAAQ,CAAC,MAAa,EAAE,8CAA8C;yBACjF,CAAC;oBACN;wBACI,OAAO;4BACH,IAAI,EAAE,QAAQ;4BACd,MAAM,EAAE,EAAE,OAAO,EAAE,KAAK,EAAE;4BAC1B,MAAM,EAAE,QAAQ,CAAC,MAAa,EAAE,8CAA8C;yBACjF,CAAC;gBACV,CAAC;YACL;gBACI,MAAM,IAAI,KAAK,CAAC,0BAA2B,QAAgB,CAAC,IAAI,EAAE,CAAC,CAAC;QAC5E,CAAC;IACL,CAAC;IAEM,KAAK,CAAC,WAAW,CACpB,OAAoB,EACpB,aAA4E;QAE5E,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,IAAI,CAAsB,OAAO,EAAE,aAAa,CAAC,CAAC;QAC9E,uEAAuE;QACvE,OAAO,QAAe,CAAC;IAC3B,CAAC;CACJ","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.2",
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.1"
56
+ "@iobroker/json-config": "^8.2.3"
57
57
  },
58
58
  "devDependencies": {
59
59
  "@alcalzone/release-script": "^5.1.1",