1
+ from typing import Tuple , List
2
+
1
3
import base64
2
4
import re
3
5
17
19
len_ ,
18
20
this ,
19
21
)
22
+ try :
23
+ import heatshrink
24
+ except :
25
+ heatshrink = None
20
26
21
27
from .click_common import command , format_output
22
28
from .device import Device
@@ -87,33 +93,46 @@ def play_pronto(self, pronto: str, repeats: int = 1):
87
93
return self .play_raw (* self .pronto_to_raw (pronto , repeats ))
88
94
89
95
@classmethod
90
- def pronto_to_raw (cls , pronto : str , repeats : int = 1 ):
91
- """Play a Pronto Hex encoded IR command. Supports only raw Pronto format,
92
- starting with 0000.
96
+ def _parse_pronto (cls , pronto : str ) -> Tuple [List ["ProntoBurstPair" ],
97
+ List ["ProntoBurstPair" ],
98
+ int ]:
99
+ """Parses Pronto Hex encoded IR command and returns a tuple containing
100
+ a list of intro pairs, a list of repeat pairs and a signal carrier frequency
101
+ """
102
+ try :
103
+ pronto_data = Pronto .parse (bytearray .fromhex (pronto ))
104
+ except Exception as ex :
105
+ raise ChuangmiIrException ("Invalid Pronto command" ) from ex
106
+
107
+ return pronto_data .intro , pronto_data .repeat , int (round (pronto_data .frequency ))
108
+
109
+ @classmethod
110
+ def pronto_to_raw (cls , pronto : str , repeats : int = 1 ) -> Tuple [str , int ]:
111
+ """Takes a Pronto Hex encoded IR command and number of repeats
112
+ and returns a tuple containing a string encoded IR signal accepted by
113
+ controller and frequency.
114
+ Supports only raw Pronto format, starting with 0000.
93
115
94
116
:param str pronto: Pronto Hex string.
95
117
:param int repeats: Number of extra signal repeats.
96
118
"""
97
119
if repeats < 0 :
98
120
raise ChuangmiIrException ("Invalid repeats value" )
99
121
100
- try :
101
- pronto_data = Pronto .parse (bytearray .fromhex (pronto ))
102
- except Exception as ex :
103
- raise ChuangmiIrException ("Invalid Pronto command" ) from ex
122
+ intro_pairs , repeat_pairs , frequency = cls ._parse_pronto (pronto )
104
123
105
- if len (pronto_data . intro ) == 0 :
124
+ if len (intro_pairs ) == 0 :
106
125
repeats += 1
107
126
108
127
times = set ()
109
- for pair in pronto_data . intro + pronto_data . repeat * (1 if repeats else 0 ):
128
+ for pair in intro_pairs + repeat_pairs * (1 if repeats else 0 ):
110
129
times .add (pair .pulse )
111
130
times .add (pair .gap )
112
131
113
132
times = sorted (times )
114
133
times_map = {t : idx for idx , t in enumerate (times )}
115
134
edge_pairs = []
116
- for pair in pronto_data . intro + pronto_data . repeat * repeats :
135
+ for pair in intro_pairs + repeat_pairs * repeats :
117
136
edge_pairs .append (
118
137
{"pulse" : times_map [pair .pulse ], "gap" : times_map [pair .gap ]}
119
138
)
@@ -127,7 +146,7 @@ def pronto_to_raw(cls, pronto: str, repeats: int = 1):
127
146
)
128
147
).decode ()
129
148
130
- return signal_code , int ( round ( pronto_data . frequency ))
149
+ return signal_code , frequency
131
150
132
151
@command (
133
152
click .argument ("command" , type = str ),
@@ -185,6 +204,73 @@ def get_indicator_led(self):
185
204
return self .send ("get_indicatorLamp" )
186
205
187
206
207
+ class ChuangmiRemote (ChuangmiIr ):
208
+ """Class representing new type of Chuangmi IR Remote Controller
209
+ identified by model "chuangmi-remote-h102a03_". The new controller uses
210
+ different format for learned IR commands, which actually is the old
211
+ format but with additional layer of compression.
212
+ """
213
+
214
+ @classmethod
215
+ def pronto_to_raw (cls , pronto : str , repeats : int = 1 ) -> Tuple [str , int ]:
216
+ """Takes a Pronto Hex encoded IR command and number of repeats
217
+ and returns a tuple containing a string encoded IR signal accepted by
218
+ controller and frequency.
219
+ Supports only raw Pronto format, starting with 0000.
220
+
221
+ :raises ChuangmiIrException if heatshrink package is not installed
222
+
223
+ :param str pronto: Pronto Hex string.
224
+ :param int repeats: Number of extra signal repeats.
225
+ """
226
+ if heatshrink is None :
227
+ raise ChuangmiIrException ("Heatshrink library is missing" )
228
+ raw , frequency = super ().pronto_to_raw (pronto , repeats )
229
+ return base64 .b64encode (
230
+ heatshrink .encode ('learn{}' .format (raw ).encode ())
231
+ ).decode (), frequency
232
+
233
+
234
+ class ChuangmiRemoteV2 (ChuangmiIr ):
235
+ """Class representing new type of Chuangmi IR Remote Controller
236
+ identified by model "chuangmi-remote-v2". The new controller uses
237
+ different format for learned IR commands, which compresses an ASCII list
238
+ of comma separated edge timings.
239
+ """
240
+
241
+ @classmethod
242
+ def pronto_to_raw (cls , pronto : str , repeats : int = 1 ) -> Tuple [str , int ]:
243
+ """Takes a Pronto Hex encoded IR command and number of repeats
244
+ and returns a tuple containing a string encoded IR signal accepted by
245
+ controller and frequency.
246
+ Supports only raw Pronto format, starting with 0000.
247
+
248
+ :param str pronto: Pronto Hex string.
249
+ :param int repeats: Number of extra signal repeats."""
250
+ if heatshrink is None :
251
+ raise ChuangmiIrException ("Heatshrink library is missing" )
252
+
253
+ if repeats < 0 :
254
+ raise ChuangmiIrException ('Invalid repeats value' )
255
+
256
+ intro_pairs , repeat_pairs , frequency = cls ._parse_pronto (pronto )
257
+
258
+ if len (intro_pairs ) == 0 :
259
+ repeats += 1
260
+
261
+ timings = []
262
+ for pair in intro_pairs + repeat_pairs * repeats :
263
+ timings .append (pair .pulse )
264
+ timings .append (pair .gap )
265
+ timings [- 1 ] = 0
266
+
267
+ timings = '{}\0 ' .format (',' .join (map (str , timings ))).encode ()
268
+
269
+ return base64 .b64encode (
270
+ heatshrink .encode (timings )
271
+ ).decode (), frequency
272
+
273
+
188
274
class ProntoPulseAdapter (Adapter ):
189
275
def _decode (self , obj , context , path ):
190
276
return int (obj * context ._ .modulation_period )
0 commit comments