Project

General

Profile

Bug #11136 » ft70.py

Dan Smith, 02/29/2024 09:24 PM

 
1
# Copyright 2010 Dan Smith <dsmith@danplanet.com>
2
# Copyright 2017 Nicolas Pike <nick@zbm2.com>
3
#
4
# This program is free software: you can redistribute it and/or modify
5
# it under the terms of the GNU General Public License as published by
6
# the Free Software Foundation, either version 3 of the License, or
7
# (at your option) any later version.
8
#
9
# This program is distributed in the hope that it will be useful,
10
# but WITHOUT ANY WARRANTY; without even the implied warranty of
11
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12
# GNU General Public License for more details.
13
#
14
# You should have received a copy of the GNU General Public License
15
# along with this program.  If not, see <http://www.gnu.org/licenses/>.
16

    
17
import logging
18

    
19
from chirp.drivers import yaesu_clone
20
from chirp import chirp_common, directory, bitwise
21
from chirp import errors
22
from chirp import memmap
23
from chirp.settings import RadioSettingGroup, RadioSetting, RadioSettings, \
24
    RadioSettingValueString, RadioSettingValueList, \
25
    InvalidValueError
26
from chirp import util
27

    
28
import string
29
LOG = logging.getLogger(__name__)
30

    
31
# Testing
32

    
33
# 37    PAG.ABK     Turn the pager answer back Function ON/OFF
34
# 38    PAG.CDR     Specify a personal code (receive)
35
# 39    PAG.CDT     Specify a personal code (transmit)
36
# 47    RX.MOD      Select the receive mode. Auto FM AM
37

    
38
MEM_SETTINGS_FORMAT = """
39

    
40
// FT-70DE New Model #5329
41
//
42
// Communications Mode ? AMS,FM DN,DW   TX vs RX?
43
// Mode not currently correctly stored in memories ? - ALL show as FM in memories
44
// SKIP test/where stored
45
// Check storage of steps
46
// Pager settings ?
47
// Triple check/ understand _memsize and _block_lengths
48
// Bank name label name size display 6 store 16? padded with 0xFF same for MYCALL and message
49
// CHIRP mode DIG not supported - is there a CHIRP Fusion mode? Auto?
50
// Check character set
51
// Supported Modes ?
52
// Supported Bands ?
53
// rf.has_dtcs_polarity = False - think radio supports DTCS polarity
54
// rf.memory_bounds = (1, 900) - should this be 0? as zero displays as blank
55
// RPT offsets (stored per band) not included.
56
// 59 Display radio firmware version info and Radio ID?
57
// Front Panel settings power etc?
58
// Banks and VFO?
59

    
60
// Features Required
61
// Default AMS and Memory name (in mem extras) to enabled.
62

    
63
// Bugs
64
// MYCALL and Opening Message errors if not 10 characters
65
// Values greater than one sometimes stored as whole bytes, these need to be refactored into bit fields
66
// to prevent accidental overwriting of adjacent values
67
// Bank Name length not checked on gui input - but first 6 characters are saved correctly.
68
// Extended characters entered as bank names on radio are corrupted in Chirp
69

    
70
// Missing
71
// 50 SCV.WTH Set the VFO scan frequency range. BAND / ALL - NOT FOUND
72
// 49 SCM.WTH Set the memory scan frequency range. ALL / BAND - NOT FOUND
73

    
74
// Radio Questions
75
// Temp unit C/F not saved by radio, always goes back to C ?
76
// 44 RF SQL Adjusts the RF Squelch threshold level. OFF / S1 - S9? Default is OFF - Based on RF strength - for AM? How
77
// is this different from F, Monitor, Dial Squelch?
78
// Password setting on radio allows letters (DTMF), but letters cannot be entered at the radio's password prompt?
79
// 49 SCM.WTH Set the memory scan frequency range. ALL / BAND Defaults to ALL Not Band as stated in the manual.
80

    
81
   #seekto 0x049a;
82
    struct {
83
    u8 unknown0:4,
84
    squelch:4;              // Squelch F, Monitor, Dial Adjust the Squelch level
85
    } squelch_settings;
86

    
87
    #seekto 0x04ba;
88
    struct {
89
    u8 unknown:3,
90
    scan_resume:5;          // 52 SCN.RSM   Configure the scan stop mode settings. 2.0 S - 5.0 S - 10.0 S / BUSY / HOLD
91
    u8 unknown1:3,
92
    dw_resume_interval:5;   // 22 DW RSM    Configure the scan stop mode settings for Dual Receive. 2.0S-10.0S/BUSY/HOLD
93
    u8 unknown2;
94
    u8 unknown3:3,
95
    apo:5;                  // 02 APO       Set the length of time until the transceiver turns off automatically.
96
    u8 unknown4:6,
97
    gm_ring:2;              // 24 GM RNG    Select the beep option while receiving digital GM info. OFF/IN RNG/ALWAYS
98
    u8 temp_cf;             // Placeholder as not found
99
    u8 unknown5;
100
    } first_settings;
101

    
102
    #seekto 0x04ed;
103
    struct {
104
    u8 unknown1:1,
105
    unknown2:1,
106
    unknown3:1,
107
    unknown4:1,
108
    unknown5:1,
109
    unknown6:1,
110
    unknown7:1,
111
    unknown8:1;
112
    } test_bit_field;
113

    
114
    #seekto 0x04c0;
115
    struct {
116
    u8 unknown1:5,
117
    beep_level:3;           // 05 BEP.LVL   Beep volume setting LEVEL1 - LEVEL4 - LEVEL7
118
    u8 unknown2:6,
119
    beep_select:2;          // 04 BEEP      Sets the beep sound function OFF / KEY+SC / KEY
120
    } beep_settings;
121

    
122
    #seekto 0x04ce;
123
    struct {
124
    u8 lcd_dimmer;                      // 14 DIMMER    LCD Dimmer
125
    u8 dtmf_delay;                      // 18 DT DLY    DTMF delay
126
    u8 unknown0[3];
127
    u8 unknown1:4,
128
    unknown1_2:4;
129
    u8 lamp;                            // 28 LAMP      Set the duration time of the backlight and keys to be lit
130
    u8 lock;                            // 30 LOCK      Configure the lock mode setting. KEY/DIAL/K+D/PTT/K+P/D+P/ALL
131
    u8 unknown2_1;
132
    u8 mic_gain;                        // 31 MCGAIN    Adjust the microphone gain level
133
    u8 unknown2_3;
134
    u8 dw_interval;                     // 21 DW INT Set the priority memory ch mon int during Dual RX 0.1S-5.0S-10.0S
135
    u8 ptt_delay;                       // 42 PTT.DLY   Set the PTT delay time. OFF / 20 ms / 50 ms / 100 ms / 200 ms
136
    u8 rx_save;                         // 48 RX.SAVE   Set the battery save time. OFF / 0.2 s - 60.0 s
137
    u8 scan_restart;                    // 53 SCN.STR   Set the scanning restart time.  0.1 s - 2.0 s - 10.0 s
138
    u8 unknown2_5;
139
    u8 unknown2_6;
140
    u8 unknown4[5];
141
    u8 tot;                             // 56 TOT       Set the transmission timeout timer
142
    u8 unknown5[3];          // 26
143
    u8 vfo_mode:1,                      // 60 VFO.MOD   Set freq setting range in the VFO mode by DIAL knob. ALL / BAND
144
    unknown7:1,
145
    scan_lamp:1,                        // 51 SCN.LMP   Set the scan lamp ON or OFF when scanning stops On/Off
146
    unknown8:1,
147
    ars:1,                              // 45 RPT.ARS   Turn the ARS function on/off.
148
    dtmf_speed:1,                       // 20 DT SPD    Set DTMF speed
149
    unknown8_1:1,
150
    dtmf_mode:1;                        // DTMF Mode set from front panel
151
    u8 busy_led:1,                      // Not Supported ?
152
    unknown8_2:1,
153
    unknown8_3:1,
154
    bclo:1,                             // 03 BCLO      Turns the busy channel lockout function on/off.
155
    beep_edge:1,                        // 06 BEP.Edg   Sets the beep sound ON or OFF when a band edge is encountered.
156
    unknown8_6:1,
157
    unknown8_7:1,
158
    unknown8_8:1;            // 28
159
    u8 unknown9_1:1,
160
    unknown9_2:1,
161
    unknown9_3:1,
162
    unknown9_4:1,
163
    unknown9_5:1,
164
    password:1,                         // Placeholder location
165
    home_rev:1,                         // 26 HOME/REV   Select the function of the [HOME/REV] key.
166
    moni:1;                             // 32 Mon/T-Call Select the function of the [MONI/T-CALL] switch.
167
    u8 gm_interval:4,       // 30       // 25 GM INT Set tx interval of digital GM information. OFF / NORMAL / LONG
168
    unknown10:4;
169
    u8 unknown11;
170
    u8 unknown12:1,
171
    unknown12_2:1,
172
    unknown12_3:1,
173
    unknown12_4:1,
174
    home_vfo:1,                         // 27 HOME->VFO  Turn transfer VFO to the Home channel ON or OFF.
175
    unknown12_6:1,
176
    unknown12_7:1,
177
    dw_rt:1;                // 32       // 23 DW RVT Turn "Priority Channel Revert" feature ON or OFF during Dual Rx.
178
    u8 unknown33;
179
    u8 unknown34;
180
    u8 unknown35;
181
    u8 unknown36;
182
    u8 unknown37;
183
    u8 unknown38;
184
    u8 unknown39;
185
    u8 unknown40;
186
    u8 unknown41;
187
    u8 unknown42;
188
    u8 unknown43;
189
    u8 unknown44;
190
    u8 unknown45;
191
    u8 prog_key1;           // P1 Set Mode Items to the Programmable Key
192
    u8 prog_key2;           // P2 Set Mode Items to the Programmable Key
193
    u8 unknown48;
194
    u8 unknown49;
195
    u8 unknown50;
196
    } scan_settings;
197

    
198
    #seekto 0x064b;
199
    struct {
200
    u8 unknown1:1,
201
    unknown2:1,
202
    unknown3:1,
203
    unknown4:1,
204
    vfo_scan_width:1,       // Placeholder as not found - 50 SCV.WTH Set the VFO scan frequency range. BAND / ALL
205
    memory_scan_width:1,    // Placeholder as not found - 49 SCM.WTH Set the memory scan frequency range. ALL / BAND
206
    unknown7:1,
207
    unknown8:1;
208
    } scan_settings_1;
209

    
210
    #seekto 0x06B6;
211
    struct {
212
    u8 unknown1:3,
213
    volume:5;               // # VOL and Dial  Adjust the volume level
214
    } scan_settings_2;
215

    
216
    #seekto 0x0690;         // Memory or VFO Settings Map?
217
    struct {
218
    u8 unknown[48];         // Array cannot be 64 elements!
219
    u8 unknown1[16];        // Exception: Not implemented for chirp.bitwise.structDataElement
220
    } vfo_info_1;
221

    
222
    #seekto 0x0710;         // Backup Memory or VFO Settings Map?
223
    struct {
224
    u8 unknown[48];
225
    u8 unknown1[16];
226
    } vfo_backup_info_1;
227

    
228
    #seekto 0x047e;
229
    struct {
230
    u8 unknown1;
231
    u8 flag;
232
    u16 unknown2;
233
    struct {
234
    char padded_string[6];              // 36 OPN.MSG   Select MSG then key vm to edit it
235
    } message;
236
    } opening_message;                  // 36 OPN.MSG   Select the Opening Message when transceiver is ON. OFF/MSG/DC
237

    
238
    #seekto 0x094a;                     // DTMF Memories
239
    struct {
240
    u8 memory[16];
241
    } dtmf[10];
242

    
243
    #seekto 0x154a;
244
    struct {
245
    u16 channel[100];
246
    } bank_members[24];
247

    
248
    #seekto 0x54a;
249
    struct {
250
    u16 in_use;
251
    } bank_used[24];
252

    
253
    #seekto 0x0EFE;
254
    struct {
255
    u8 unknown[2];
256
    u8 name[6];
257
    u8 unknown1[10];
258
    } bank_info[24];
259

    
260
    #seekto 0xCF30;
261
    struct {
262
    u8 unknown0;
263
    u8 unknown1;
264
    u8 unknown2;
265
    u8 unknown3;
266
    u8 unknown4;
267
    u8 unknown5;
268
    u8 unknown6;
269
    u8 digital_popup;                   // 15 DIG.POP   Call sign display pop up time
270
    } digital_settings_more;
271

    
272
    #seekto 0xCF7C;
273
    struct {
274
    u8 unknown0:6,
275
    ams_tx_mode:2;                      // AMS TX Mode  Short Press AMS button AMS TX Mode
276
    u8 unknown1;
277
    u8 unknown2:7,
278
    standby_beep:1;                     // 07 BEP.STB   Standby Beep in the digital C4FM mode. On/Off
279
    u8 unknown3;
280
    u8 unknown4:6,
281
    gm_ring:2;                          // 24 GM RNG Select beep option while rx digital GM info. OFF/IN RNG/ALWAYS
282
    u8 unknown5;
283
    u8 rx_dg_id;                        // RX DG-ID     Long Press Mode Key, Mode Key to select, Dial
284
    u8 tx_dg_id;                        // TX DG-ID     Long Press Mode Key, Dial
285
    u8 unknown6:7,
286
    vw_mode:1;                          // 16 DIG VW    Turn the VW mode selection ON or OFF
287
    u8 unknown7;
288
    } digital_settings;
289

    
290
    // ^^^ All above referenced U8's have been refactored to minimum number of bits.
291

    
292
    """
293

    
294
MEM_FORMAT = """
295
    #seekto 0x2D4A;
296
    struct {                            // 32 Bytes per memory entry
297
    u8 display_tag:1,                   // 0 Display Freq, 1 Display Name
298
    unknown0:1,                         // Mode if AMS not selected????????
299
    deviation:1,                        // 0 Full deviation (Wide), 1 Half deviation (Narrow)
300
    clock_shift:1,                      // 0 None, 1 CPU clock shifted
301
    unknown1:4;                                                                         // 1
302
    u8 mode:2,                          // FM,AM,WFM only? - check
303
    duplex:2,                           // Works
304
    tune_step:4;                        // Works - check all steps? 7 = Auto            // 1
305
    bbcd freq[3];                       // Works                                        // 3
306
    u8 power:2,                         // Works
307
    unknown2:1,                         // 0 FM, 1 Digital - If AMS off
308
    ams:1,                              // 0 AMS off, 1 AMS on ?
309
    tone_mode:4;                        // Works                                        // 1
310
    u8 charsetbits[2];                                                                  // 2
311
    char label[6];                      // Works - Can only input 6 on screen           // 6
312
    char unknown7[10];                  // Rest of label ???                            // 10
313
    bbcd offset[3];                     // Works                                        // 3
314
    u8 unknown5:2,
315
    tone:6;                             // Works                                       // 1
316
    u8 unknown6:1,
317
    dcs:7;                              // Works                                        // 1
318
    u8 unknown9;
319
    u8 ams_on_dn_vw_fm:2,               // AMS DN, AMS VW, AMS FM
320
    unknown8_3:1,
321
    unknown8_4:1,
322
    smeter:4;
323
    u8 unknown10:2,
324
       att:1,
325
       auto_step:1,
326
       auto_mode:1,
327
       unknown11:2,
328
       bell:1;
329
    } memory[%d];                        // DN, VW, FM, AM
330
                                        // AMS DN, AMS VW, AMS FM
331

    
332
    #seekto 0x280A;
333
    struct {
334
    u8 nosubvfo:1,
335
    unknown:3,
336
    pskip:1,                            // PSkip (Select?)
337
    skip:1,                             // Skip memory during scan
338
    used:1,                             // Memory used
339
    valid:1;                            // Always 1?
340
    } flag[%d];
341
    """
342

    
343
MEM_CALLSIGN_FORMAT = """
344
#seekto 0x0ced0;
345
    struct {
346
    char callsign[10];              // 63 MYCALL    Set the call sign. (up to 10 characters)
347
    u16 charset;                    // character set ID
348
    } my_call;
349
    """
350

    
351
MEM_CHECKSUM_FORMAT = """
352
    #seekto 0xFECA;
353
    u8 checksum;
354
    """
355

    
356
TMODES = ["", "Tone", "TSQL", "DTCS"]
357
DUPLEX = ["", "-", "+", "split"]
358

    
359
MODES = ["FM", "AM", "NFM"]
360

    
361
STEPS = [0, 5, 6.25, 10, 12.5, 15, 20, 25, 50, 100]  # 0 = auto
362
RFSQUELCH = ["OFF", "S1", "S2", "S3", "S4", "S5", "S6", "S7", "S8"]
363

    
364
SKIPS = ["", "S", "P"]
365
FT70_DTMF_CHARS = list("0123456789ABCDEF-")
366

    
367
CHARSET = ["%i" % int(x) for x in range(0, 10)] + \
368
          [chr(x) for x in range(ord("A"), ord("Z") + 1)] + \
369
          [" ", ] + \
370
          [chr(x) for x in range(ord("a"), ord("z") + 1)] + \
371
          list(".,:;*#_-/&()@!?^ ") + list("\x00" * 100)
372

    
373
POWER_LEVELS = [chirp_common.PowerLevel("Hi", watts=5.00),
374
                chirp_common.PowerLevel("Mid", watts=2.00),
375
                chirp_common.PowerLevel("Low", watts=.50)]
376

    
377

    
378
class FT70Bank(chirp_common.NamedBank):
379
    """A FT70 bank"""
380

    
381
    def get_name(self):
382
        _bank = self._model._radio._memobj.bank_info[self.index]
383
        name = ""
384
        for i in _bank.name:
385
            if i == 0xff:
386
                break
387
            name += chr(i & 0xFF)
388
        return name.rstrip()
389

    
390
    def set_name(self, name):
391
        _bank = self._model._radio._memobj.bank_info[self.index]
392
        _bank.name = [ord(x) for x in name.ljust(6, chr(0xFF))[:6]]
393

    
394

    
395
class FT70BankModel(chirp_common.BankModel):
396
    """A FT70 bank model"""
397

    
398
    def __init__(self, radio, name='Banks'):
399
        super(FT70BankModel, self).__init__(radio, name)
400

    
401
        _banks = self._radio._memobj.bank_info
402
        self._bank_mappings = []
403
        for index, _bank in enumerate(_banks):
404
            bank = FT70Bank(self, "%i" % index, "BANK-%i" % index)
405
            bank.index = index
406
            self._bank_mappings.append(bank)
407

    
408
    def get_num_mappings(self):
409
        return len(self._bank_mappings)
410

    
411
    def get_mappings(self):
412
        return self._bank_mappings
413

    
414
    def _channel_numbers_in_bank(self, bank):
415
        _bank_used = self._radio._memobj.bank_used[bank.index]
416
        if _bank_used.in_use == 0xFFFF:
417
            return set()
418

    
419
        _members = self._radio._memobj.bank_members[bank.index]
420
        return set([int(ch) + 1 for ch in _members.channel if ch != 0xFFFF])
421

    
422
    def _update_bank_with_channel_numbers(self, bank, channels_in_bank):
423
        _members = self._radio._memobj.bank_members[bank.index]
424
        if len(channels_in_bank) > len(_members.channel):
425
            raise Exception("Too many entries in bank %d" % bank.index)
426

    
427
        empty = 0
428
        for index, channel_number in enumerate(sorted(channels_in_bank)):
429
            _members.channel[index] = channel_number - 1
430
            empty = index + 1
431
        for index in range(empty, len(_members.channel)):
432
            _members.channel[index] = 0xFFFF
433

    
434
    def add_memory_to_mapping(self, memory, bank):
435
        channels_in_bank = self._channel_numbers_in_bank(bank)
436
        channels_in_bank.add(memory.number)
437
        self._update_bank_with_channel_numbers(bank, channels_in_bank)
438

    
439
        _bank_used = self._radio._memobj.bank_used[bank.index]
440
        _bank_used.in_use = 0x06
441

    
442
    def remove_memory_from_mapping(self, memory, bank):
443
        channels_in_bank = self._channel_numbers_in_bank(bank)
444
        try:
445
            channels_in_bank.remove(memory.number)
446
        except KeyError:
447
            raise Exception("Memory %i is not in bank %s. Cannot remove" %
448
                            (memory.number, bank))
449
        self._update_bank_with_channel_numbers(bank, channels_in_bank)
450

    
451
        if not channels_in_bank:
452
            _bank_used = self._radio._memobj.bank_used[bank.index]
453
            _bank_used.in_use = 0xFFFF
454

    
455
    def get_mapping_memories(self, bank):
456
        memories = []
457
        for channel in self._channel_numbers_in_bank(bank):
458
            memories.append(self._radio.get_memory(channel))
459

    
460
        return memories
461

    
462
    def get_memory_mappings(self, memory):
463
        banks = []
464
        for bank in self.get_mappings():
465
            if memory.number in self._channel_numbers_in_bank(bank):
466
                banks.append(bank)
467

    
468
        return banks
469

    
470

    
471
@directory.register
472
class FT70Radio(yaesu_clone.YaesuCloneModeRadio):
473
    """Yaesu FT-70DE"""
474
    BAUD_RATE = 38400
475
    VENDOR = "Yaesu"
476
    MODEL = "FT-70D"
477
    FORMATS = [directory.register_format('FT-70D ADMS-10', '*.ft70d')]
478

    
479
    _model = b"AH51G"
480
    _adms_ext = '.ft70d'
481

    
482
    _memsize = 65227  # 65227 read from dump
483
    _block_lengths = [10, 65217]
484
    _block_size = 32
485
    _mem_params = (900,  # size of memories array
486
                   900,  # size of flags array
487
                   )
488

    
489
    _has_vibrate = False
490
    _has_af_dual = True
491

    
492
    _BEEP_SELECT = ("Off", "Key+Scan", "Key")
493
    _OPENING_MESSAGE = ("Off", "DC", "Message")
494
    _MIC_GAIN = ("Level 1", "Level 2", "Level 3", "Level 4", "Level 5", "Level 6", "Level 7", "Level 8", "Level 9")
495
    _AMS_TX_MODE = ("TX Auto", "TX DIGITAL", "TX FM")
496
    _VW_MODE = ("On", "Off")
497
    _DIG_POP_UP = ("Off", "2sec", "4sec", "6sec", "8sec", "10sec", "20sec", "30sec", "60sec", "Continuous")
498
    _STANDBY_BEEP = ("On", "Off")
499
    _SCAN_RESUME = ["%.1fs" % (0.5 * x) for x in range(4, 21)] + \
500
                   ["Busy", "Hold"]
501
    _SCAN_RESTART = ["%.1fs" % (0.1 * x) for x in range(1, 10)] + \
502
                    ["%.1fs" % (0.5 * x) for x in range(2, 21)]
503
    _LAMP_KEY = ["Key %d sec" % x
504
                 for x in range(2, 11)] + ["Continuous", "OFF"]
505
    _LCD_DIMMER = ["Level %d" % x for x in range(1, 7)]
506
    _TOT_TIME = ["Off"] + ["%.1f min" % (0.5 * x) for x in range(1, 21)]
507
    _OFF_ON = ("Off", "On")
508
    _ON_OFF = ("On", "Off")
509
    _DTMF_MODE = ("Manual", "Auto")
510
    _DTMF_SPEED = ("50ms", "100ms")
511
    _DTMF_DELAY = ("50ms", "250ms", "450ms", "750ms", "1000ms")
512
    _TEMP_CF = ("Centigrade", "Fahrenheit")
513
    _APO_SELECT = ("Off", "0.5H", "1.0H", "1.5H", "2.0H", "2.5H", "3.0H", "3.5H", "4.0H", "4.5H", "5.0H",
514
                   "5.5H", "6.0H", "6.5H", "7.0H", "7.5H", "8.0H", "8.5H", "9.0H", "9.5H", "10.0H", "10.5H",
515
                   "11.0H", "11.5H", "12.0H")
516
    _MONI_TCALL = ("Monitor", "Tone-CALL")
517
    _HOME_REV = ("Home", "Reverse")
518
    _LOCK = ("KEY", "DIAL", "Key+Dial", "PTT", "Key+PTT", "Dial+PTT", "ALL")
519
    _PTT_DELAY = ("Off", "20 ms", "50 ms", "100 ms", "200 ms")
520
    _BEEP_LEVEL = ("Level 1", "Level 2", "Level 3", "Level 4", "Level 5", "Level 6", "Level 7")
521
    _SET_MODE = ("Level 1", "Level 2", "Level 3", "Level 4", "Level 5", "Level 6", "Level 7")
522
    _RX_SAVE = ("OFF", "0.2s", ".3s", ".4s", ".5s", ".6s", ".7s", ".8s", ".9s", "1.0s", "1.5s",
523
                "2.0s", "2.5s", "3.0s", "3.5s", "4.0s", "4.5s", "5.0s", "5.5s", "6.0s", "6.5s", "7.0s",
524
                "7.5s", "8.0s", "8.5s", "9.0s", "10.0s", "15s", "20s", "25s", "30s", "35s", "40s", "45s", "50s", "55s",
525
                "60s")
526
    _VFO_MODE = ("ALL", "BAND")
527
    _VFO_SCAN_MODE = ("BAND", "ALL")
528
    _MEMORY_SCAN_MODE = ("BAND", "ALL")
529

    
530
    _VOLUME = ["Level %d" % x for x in range(0, 32)]
531
    _SQUELCH = ["Level %d" % x for x in range(0, 16)]
532

    
533
    _DG_ID = ["%d" % x for x in range(0, 100)]
534
    _GM_RING = ("OFF", "IN RING", "AlWAYS")
535
    _GM_INTERVAL = ("LONG", "NORMAL", "OFF")
536

    
537
    _MYCALL_CHR_SET = list(string.ascii_uppercase) + list(string.digits) + ['-', '/']
538

    
539
    @classmethod
540
    def match_model(cls, filedata, filename):
541
        if filename.endswith(cls._adms_ext):
542
            return True
543
        else:
544
            return super().match_model(filedata, filename)
545

    
546
    @classmethod
547
    def get_prompts(cls):
548
        rp = chirp_common.RadioPrompts()
549

    
550
        rp.pre_download = _(
551
            "1. Turn radio on.\n"
552
            "2. Connect cable to DATA terminal.\n"
553
            "3. Unclip battery.\n"
554
            "4. Press and hold in the [AMS] key and power key while clipping"
555
            " \n in back battery the"
556
            "(\"ADMS\" will appear on the display).\n"
557
            "5. <b>After clicking OK</b>, press the [BAND] key.\n")
558
        rp.pre_upload = _(
559
            "1. Turn radio on.\n"
560
            "2. Connect cable to DATA terminal.\n"
561
            "3. Unclip battery.\n"
562
            "4. Press and hold in the [AMS] key and power key while clipping"
563
            " \n in back battery the "
564
            "(\"ADMS\" will appear on the display).\n"
565
            "5. Press the [MODE] key (\"-WAIT-\" will appear on the LCD).\n"
566
            "<b>Then click OK</b>")
567
        return rp
568

    
569
    def process_mmap(self):
570

    
571
        mem_format = MEM_SETTINGS_FORMAT + MEM_FORMAT + MEM_CALLSIGN_FORMAT + MEM_CHECKSUM_FORMAT
572

    
573
        self._memobj = bitwise.parse(mem_format % self._mem_params, self._mmap)
574

    
575
    def get_features(self):
576
        rf = chirp_common.RadioFeatures()
577
        rf.has_dtcs_polarity = False
578
        rf.valid_modes = list(MODES)
579
        rf.valid_tmodes = list(TMODES)
580
        rf.valid_duplexes = list(DUPLEX)
581
        rf.valid_tuning_steps = list(STEPS)
582
        rf.valid_bands = [(500000, 999900000)]
583
        rf.valid_skips = SKIPS
584
        rf.valid_power_levels = POWER_LEVELS
585
        rf.valid_characters = "".join(CHARSET)
586
        rf.valid_name_length = 6
587
        rf.memory_bounds = (1, 900)
588
        rf.can_odd_split = True
589
        rf.has_ctone = False
590
        rf.has_bank_names = True
591
        rf.has_settings = True
592
        return rf
593

    
594
    def get_raw_memory(self, number):
595
        return "\n".join([repr(self._memobj.memory[number - 1]),
596
                          repr(self._memobj.flag[number - 1])])
597

    
598
    def _checksums(self):
599
        return [yaesu_clone.YaesuChecksum(0x0000, 0xFEC9)]  # The whole file -2 bytes
600

    
601
    @staticmethod
602
    def _add_ff_pad(val, length):
603
        return val.ljust(length, "\xFF")[:length]
604

    
605
    @classmethod
606
    def _strip_ff_pads(cls, messages):
607
        result = []
608
        for msg_text in messages:
609
            result.append(str(msg_text).rstrip("\xFF"))
610
        return result
611

    
612
    def get_memory(self, number):
613
        flag = self._memobj.flag[number - 1]
614
        _mem = self._memobj.memory[number - 1]
615

    
616
        mem = chirp_common.Memory()
617
        mem.number = number
618
        if not flag.used:
619
            mem.empty = True
620
        if not flag.valid:
621
            mem.empty = True
622
            return mem
623
        mem.freq = chirp_common.fix_rounded_step(int(_mem.freq) * 1000)
624
        mem.offset = int(_mem.offset) * 1000
625
        mem.rtone = mem.ctone = chirp_common.TONES[_mem.tone]
626
        self._get_tmode(mem, _mem)
627
        mem.duplex = DUPLEX[_mem.duplex]
628
        if mem.duplex == "split":
629
            mem.offset = chirp_common.fix_rounded_step(mem.offset)
630
        mem.mode = self._decode_mode(_mem)
631
        mem.dtcs = chirp_common.DTCS_CODES[_mem.dcs]
632
        mem.tuning_step = STEPS[_mem.tune_step]
633
        mem.power = self._decode_power_level(_mem)
634
        mem.skip = flag.pskip and "P" or flag.skip and "S" or ""
635
        mem.name = self._decode_label(_mem)
636

    
637
        return mem
638

    
639
    def _decode_label(self, mem):
640
        return str(mem.label).rstrip("\xFF")
641

    
642
    def _encode_label(self, mem):
643
        return self._add_ff_pad(mem.name.rstrip(), 6)
644

    
645
    def _encode_charsetbits(self, mem):
646
        # We only speak English here in chirpville
647
        return [0x00, 0x00]
648

    
649
    def _decode_power_level(self, mem):  # 3 High 2 Mid 1 Low
650
        return POWER_LEVELS[3 - mem.power]
651

    
652
    def _encode_power_level(self, mem):
653
        return 3 - POWER_LEVELS.index(mem.power)
654

    
655
    def _decode_mode(self, mem):
656
        mode = MODES[mem.mode]
657
        if mode == 'FM' and int(mem.deviation):
658
            return 'NFM'
659
        else:
660
            return mode
661

    
662
    def _encode_mode(self, mem):
663
        mode = mem.mode
664
        if mode == 'NFM':
665
            # Narrow is handled by a separate flag
666
            mode = 'FM'
667
        return MODES.index(mode)
668

    
669
    def _get_tmode(self, mem, _mem):
670
        mem.tmode = TMODES[_mem.tone_mode]
671

    
672
    def _set_tmode(self, _mem, mem):
673
        _mem.tone_mode = TMODES.index(mem.tmode)
674

    
675
    def _set_mode(self, _mem, mem):
676
        _mem.deviation = mem.mode == 'NFM'
677
        _mem.mode = self._encode_mode(mem)
678

    
679
    def _debank(self, mem):
680
        bm = self.get_bank_model()
681
        for bank in bm.get_memory_mappings(mem):
682
            bm.remove_memory_from_mapping(mem, bank)
683

    
684
    def set_memory(self, mem):
685
        _mem = self._memobj.memory[mem.number - 1]
686
        flag = self._memobj.flag[mem.number - 1]
687

    
688
        self._debank(mem)
689

    
690
        if not mem.empty and not flag.valid:
691
            self._wipe_memory(_mem)
692

    
693
        if mem.empty and flag.valid and not flag.used:
694
            flag.valid = False
695
            return
696
        flag.used = not mem.empty
697
        flag.valid = flag.used
698

    
699
        if mem.empty:
700
            return
701

    
702
        if mem.freq < 30000000 or \
703
                (mem.freq > 88000000 and mem.freq < 108000000) or \
704
                mem.freq > 580000000:
705
            flag.nosubvfo = True  # Masked from VFO B
706
        else:
707
            flag.nosubvfo = False  # Available in both VFOs
708

    
709
        _mem.freq = int(mem.freq / 1000)
710
        _mem.offset = int(mem.offset / 1000)
711
        _mem.tone = chirp_common.TONES.index(mem.rtone)
712
        self._set_tmode(_mem, mem)
713
        _mem.duplex = DUPLEX.index(mem.duplex)
714
        self._set_mode(_mem, mem)
715
        _mem.dcs = chirp_common.DTCS_CODES.index(mem.dtcs)
716
        _mem.tune_step = STEPS.index(mem.tuning_step)
717

    
718
        if mem.power:
719
            _mem.power = self._encode_power_level(mem)
720
        else:
721
            _mem.power = 3  # Set 3 - High power as the default
722

    
723
        _mem.label = self._encode_label(mem)
724
        charsetbits = self._encode_charsetbits(mem)
725
        _mem.charsetbits[0], _mem.charsetbits[1] = charsetbits
726

    
727
        flag.skip = mem.skip == "S"
728
        flag.pskip = mem.skip == "P"
729

    
730
        _mem.display_tag = 1    # Always Display Memory Name (For the moment..)
731

    
732
    @classmethod
733
    def _wipe_memory(cls, mem):
734
        mem.set_raw("\x00" * (mem.size() // 8))
735
        mem.unknown1 = 0x05
736

    
737
    def get_bank_model(self):
738
        return FT70BankModel(self)
739

    
740
    def _get_dtmf_settings(self):
741
        menu = RadioSettingGroup("dtmf_settings", "DTMF")
742
        dtmf = self._memobj.scan_settings
743

    
744
        val = RadioSettingValueList(
745
            self._DTMF_MODE,
746
            self._DTMF_MODE[dtmf.dtmf_mode])
747
        rs = RadioSetting("scan_settings.dtmf_mode", "DTMF Mode", val)
748
        menu.append(rs)
749

    
750
        val = RadioSettingValueList(
751
            self._DTMF_DELAY,
752
            self._DTMF_DELAY[dtmf.dtmf_delay])
753
        rs = RadioSetting(
754
            "scan_settings.dtmf_delay", "DTMF Delay", val)
755
        menu.append(rs)
756

    
757
        val = RadioSettingValueList(
758
            self._DTMF_SPEED,
759
            self._DTMF_SPEED[dtmf.dtmf_speed])
760
        rs = RadioSetting(
761
            "scan_settings.dtmf_speed", "DTMF Speed", val)
762
        menu.append(rs)
763

    
764
        for i in range(10):
765

    
766
            name = "dtmf_%02d" % (i + 1)
767
            if i == 9:
768
                name = "dtmf_%02d" % 0
769

    
770
            dtmfsetting = self._memobj.dtmf[i]
771
            dtmfstr = ""
772
            for c in dtmfsetting.memory:
773
                if c == 0xFF:
774
                    break
775
                if c < len(FT70_DTMF_CHARS):
776
                    dtmfstr += FT70_DTMF_CHARS[c]
777
            dtmfentry = RadioSettingValueString(0, 16, dtmfstr)
778
            dtmfentry.set_charset(
779
                FT70_DTMF_CHARS + list("abcdef "))  # Allow input in lowercase, space ? validation fails otherwise
780
            rs = RadioSetting(name, name.upper(), dtmfentry)
781
            rs.set_apply_callback(self.apply_dtmf, i)
782
            menu.append(rs)
783

    
784
        return menu
785

    
786
    def _get_display_settings(self):
787
        menu = RadioSettingGroup("display_settings", "Display")
788
        scan_settings = self._memobj.scan_settings
789

    
790
        val = RadioSettingValueList(
791
            self._LAMP_KEY,
792
            self._LAMP_KEY[scan_settings.lamp])
793
        rs = RadioSetting("scan_settings.lamp", "Lamp", val)
794
        menu.append(rs)
795

    
796
        val = RadioSettingValueList(
797
            self._LCD_DIMMER,
798
            self._LCD_DIMMER[scan_settings.lcd_dimmer])
799
        rs = RadioSetting("scan_settings.lcd_dimmer", "LCD Dimmer", val)
800
        menu.append(rs)
801

    
802
        opening_message = self._memobj.opening_message
803
        val = RadioSettingValueList(
804
            self._OPENING_MESSAGE,
805
            self._OPENING_MESSAGE[opening_message.flag])
806
        rs = RadioSetting("opening_message.flag", "Opening Msg Mode", val)
807
        menu.append(rs)
808

    
809
        return menu
810

    
811
    def _get_config_settings(self):
812
        menu = RadioSettingGroup("config_settings", "Config")
813
        scan_settings = self._memobj.scan_settings
814

    
815
        # 02 APO    Set the length of time until the transceiver turns off automatically.
816

    
817
        first_settings = self._memobj.first_settings
818
        val = RadioSettingValueList(
819
            self._APO_SELECT,
820
            self._APO_SELECT[first_settings.apo])
821
        rs = RadioSetting("first_settings.apo", "APO", val)
822
        menu.append(rs)
823

    
824
        # 03 BCLO   Turns the busy channel lockout function on/off.
825

    
826
        val = RadioSettingValueList(
827
            self._OFF_ON,
828
            self._OFF_ON[scan_settings.bclo])
829
        rs = RadioSetting("scan_settings.bclo", "Busy Channel Lockout", val)
830
        menu.append(rs)
831

    
832
        # 04 BEEP   Sets the beep sound function.
833

    
834
        beep_settings = self._memobj.beep_settings
835
        val = RadioSettingValueList(
836
            self._BEEP_SELECT,
837
            self._BEEP_SELECT[beep_settings.beep_select])
838
        rs = RadioSetting("beep_settings.beep_select", "Beep", val)
839
        menu.append(rs)
840

    
841
        # 05 BEP.LVL    Beep volume setting LEVEL1 - LEVEL4 - LEVEL7
842

    
843
        val = RadioSettingValueList(
844
            self._BEEP_LEVEL,
845
            self._BEEP_LEVEL[beep_settings.beep_level])
846
        rs = RadioSetting("beep_settings", "Beep Level", val)
847
        menu.append(rs)
848

    
849
        # 06 BEP.Edg    Sets the beep sound ON or OFF when a band edge is encountered.
850

    
851
        val = RadioSettingValueList(
852
            self._OFF_ON,
853
            self._OFF_ON[scan_settings.beep_edge])
854
        rs = RadioSetting("scan_settings.beep_edge", "Beep Band Edge", val)
855
        menu.append(rs)
856

    
857
        # 10 Bsy.LED    Turn the MODE/STATUS Indicator ON or OFF while receiving signals.
858

    
859
        val = RadioSettingValueList(
860
            self._ON_OFF,
861
            self._ON_OFF[scan_settings.busy_led])
862
        rs = RadioSetting("scan_settings.busy_led", "Busy LED", val)
863
        menu.append(rs)
864

    
865
        # 26 HOME/REV   Select the function of the [HOME/REV] key.
866

    
867
        val = RadioSettingValueList(
868
            self._HOME_REV,
869
            self._HOME_REV[scan_settings.home_rev])
870
        rs = RadioSetting("scan_settings.home_rev", "HOME/REV", val)
871
        menu.append(rs)
872

    
873
        # 27 HOME->VFO  Turn transfer VFO to the Home channel ON or OFF.
874

    
875
        val = RadioSettingValueList(
876
            self._OFF_ON,
877
            self._OFF_ON[scan_settings.home_vfo])
878
        rs = RadioSetting("scan_settings.home_vfo", "Home->VFO", val)
879
        menu.append(rs)
880

    
881
        # 30 LOCK       Configure the lock mode setting. KEY / DIAL / K+D / PTT / K+P / D+P / ALL
882

    
883
        val = RadioSettingValueList(
884
            self._LOCK,
885
            self._LOCK[scan_settings.lock])
886
        rs = RadioSetting("scan_settings.lock", "Lock Mode", val)
887
        menu.append(rs)
888

    
889
        # 32 Mon/T-Call Select the function of the [MONI/T-CALL] switch.
890

    
891
        val = RadioSettingValueList(
892
            self._MONI_TCALL,
893
            self._MONI_TCALL[scan_settings.moni])
894
        rs = RadioSetting("scan_settings.moni", "MONI/T-CALL", val)
895
        menu.append(rs)
896

    
897
        # 42 PTT.DLY    Set the PTT delay time. OFF / 20 ms / 50 ms / 100 ms / 200 ms
898

    
899
        val = RadioSettingValueList(
900
            self._PTT_DELAY,
901
            self._PTT_DELAY[scan_settings.ptt_delay])
902
        rs = RadioSetting("scan_settings.ptt_delay", "PTT Delay", val)
903
        menu.append(rs)
904

    
905
        # 45 RPT.ARS    Turn the ARS function on/off.
906

    
907
        val = RadioSettingValueList(
908
            self._OFF_ON,
909
            self._OFF_ON[scan_settings.ars])
910
        rs = RadioSetting("scan_settings.ars", "ARS", val)
911
        menu.append(rs)
912

    
913
        # 48 RX.SAVE    Set the battery save time. OFF / 0.2 s - 60.0 s
914

    
915
        val = RadioSettingValueList(
916
            self._RX_SAVE,
917
            self._RX_SAVE[scan_settings.rx_save])
918
        rs = RadioSetting("scan_settings.rx_save", "RX SAVE", val)
919
        menu.append(rs)
920

    
921
        # 60 VFO.MOD    Set the frequency setting range in the VFO mode by DIAL knob. ALL / BAND
922

    
923
        val = RadioSettingValueList(
924
            self._VFO_MODE,
925
            self._VFO_MODE[scan_settings.vfo_mode])
926
        rs = RadioSetting("scan_settings.vfo_mode", "VFO MODE", val)
927
        menu.append(rs)
928

    
929
        # 56 TOT        Set the timeout timer.
930

    
931
        val = RadioSettingValueList(
932
            self._TOT_TIME,
933
            self._TOT_TIME[scan_settings.tot])
934
        rs = RadioSetting("scan_settings.tot", "Transmit Timeout (TOT)", val)
935
        menu.append(rs)
936

    
937
        # 31 MCGAIN     Adjust the microphone gain level
938

    
939
        val = RadioSettingValueList(
940
            self._MIC_GAIN,
941
            self._MIC_GAIN[scan_settings.mic_gain])
942
        rs = RadioSetting("scan_settings.mic_gain", "Mic Gain", val)
943
        menu.append(rs)
944

    
945
        # VOLUME       Adjust the volume level
946

    
947
        scan_settings_2 = self._memobj.scan_settings_2
948
        val = RadioSettingValueList(
949
            self._VOLUME,
950
            self._VOLUME[scan_settings_2.volume])
951
        rs = RadioSetting("scan_settings_2.volume", "Volume", val)
952
        menu.append(rs)
953

    
954
        # Squelch       F key, Hold Monitor, Dial to adjust squelch level
955

    
956
        squelch_settings = self._memobj.squelch_settings
957
        val = RadioSettingValueList(
958
            self._SQUELCH,
959
            self._SQUELCH[squelch_settings.squelch])
960
        rs = RadioSetting("squelch_settings.squelch", "Squelch", val)
961
        menu.append(rs)
962

    
963
        return menu
964

    
965
    def _get_digital_settings(self):
966
        menu = RadioSettingGroup("digital_settings", "Digital")
967

    
968
        # MYCALL
969
        mycall = self._memobj.my_call
970
        mycallstr = str(mycall.callsign).rstrip("\xFF")
971

    
972
        mycallentry = RadioSettingValueString(0, 10, mycallstr, False, charset=self._MYCALL_CHR_SET)
973
        rs = RadioSetting('mycall.callsign', 'MYCALL', mycallentry)
974
        rs.set_apply_callback(self.apply_mycall, mycall)
975
        menu.append(rs)
976

    
977
        # Short Press AMS button AMS TX Mode
978

    
979
        digital_settings = self._memobj.digital_settings
980
        val = RadioSettingValueList(
981
            self._AMS_TX_MODE,
982
            self._AMS_TX_MODE[digital_settings.ams_tx_mode])
983
        rs = RadioSetting("digital_settings.ams_tx_mode", "AMS TX Mode", val)
984
        menu.append(rs)
985

    
986
        # 16 DIG VW  Turn the VW mode selection ON or OFF.
987

    
988
        val = RadioSettingValueList(
989
            self._VW_MODE,
990
            self._VW_MODE[digital_settings.vw_mode])
991
        rs = RadioSetting("digital_settings.vw_mode", "VW Mode", val)
992
        menu.append(rs)
993

    
994
        # TX DG-ID Long Press Mode Key, Dial
995

    
996
        val = RadioSettingValueList(
997
            self._DG_ID,
998
            self._DG_ID[digital_settings.tx_dg_id])
999
        rs = RadioSetting("digital_settings.tx_dg_id", "TX DG-ID", val)
1000
        menu.append(rs)
1001

    
1002
        # RX DG-ID Long Press Mode Key, Mode Key to select, Dial
1003

    
1004
        val = RadioSettingValueList(
1005
            self._DG_ID,
1006
            self._DG_ID[digital_settings.rx_dg_id])
1007
        rs = RadioSetting("digital_settings.rx_dg_id", "RX DG-ID", val)
1008
        menu.append(rs)
1009

    
1010
        # 15 DIG.POP    Call sign display pop up time
1011

    
1012
        # 00 OFF     00
1013
        # 0A 2s      10
1014
        # 0B 4s      11
1015
        # 0C 6s      12
1016
        # 0D 8s      13
1017
        # 0E 10s     14
1018
        # 0F 20s     15
1019
        # 10 30s     16
1020
        # 11 60s     17
1021
        # 12 CONT    18
1022

    
1023
        digital_settings_more = self._memobj.digital_settings_more
1024

    
1025
        val = RadioSettingValueList(
1026
            self._DIG_POP_UP,
1027
            self._DIG_POP_UP[
1028
                0 if digital_settings_more.digital_popup == 0 else digital_settings_more.digital_popup - 9])
1029

    
1030
        rs = RadioSetting("digital_settings_more.digital_popup", "Digital Popup", val)
1031
        rs.set_apply_callback(self.apply_digital_popup, digital_settings_more)
1032
        menu.append(rs)
1033

    
1034
        # 07  BEP.STB    Standby Beep in the digital C4FM mode. On/Off
1035

    
1036
        val = RadioSettingValueList(
1037
            self._STANDBY_BEEP,
1038
            self._STANDBY_BEEP[digital_settings.standby_beep])
1039
        rs = RadioSetting("digital_settings.standby_beep", "Standby Beep", val)
1040
        menu.append(rs)
1041

    
1042
        return menu
1043

    
1044
    def _get_gm_settings(self):
1045
        menu = RadioSettingGroup("first_settings", "Group Monitor")
1046

    
1047
        # 24 GM RNG Select the beep option while receiving digital GM information. OFF / IN RNG /ALWAYS
1048

    
1049
        first_settings = self._memobj.first_settings
1050
        val = RadioSettingValueList(
1051
            self._GM_RING,
1052
            self._GM_RING[first_settings.gm_ring])
1053
        rs = RadioSetting("first_settings.gm_ring", "GM Ring", val)
1054
        menu.append(rs)
1055

    
1056
        # 25 GM INT Set the transmission interval of digital GM information. OFF / NORMAL / LONG
1057

    
1058
        scan_settings = self._memobj.scan_settings
1059
        val = RadioSettingValueList(
1060
            self._GM_INTERVAL,
1061
            self._GM_INTERVAL[scan_settings.gm_interval])
1062
        rs = RadioSetting("scan_settings.gm_interval", "GM Interval", val)
1063
        menu.append(rs)
1064

    
1065
        return menu
1066

    
1067
    def _get_scan_settings(self):
1068
        menu = RadioSettingGroup("scan_settings", "Scan")
1069
        scan_settings = self._memobj.scan_settings
1070

    
1071
        # 23 DW RVT     Turn the "Priority Channel Revert" feature ON or OFF during Dual Receive.
1072

    
1073
        val = RadioSettingValueList(
1074
            self._OFF_ON,
1075
            self._OFF_ON[scan_settings.dw_rt])
1076
        rs = RadioSetting("scan_settings.dw_rt", "Dual Watch Priority Channel Revert", val)
1077
        menu.append(rs)
1078

    
1079
        # 21 DW INT Set the priority memory channel monitoring interval during Dual Receive. 0.1S - 5.0S - 10.0S
1080

    
1081
        val = RadioSettingValueList(
1082
            self._SCAN_RESTART,
1083
            self._SCAN_RESTART[scan_settings.dw_interval])
1084
        rs = RadioSetting("scan_settings.dw_interval", "Dual Watch Interval", val)
1085
        menu.append(rs)
1086

    
1087
        # 22 DW RSM Configure the scan stop mode settings for Dual Receive. 2.0S - 10.0 S / BUSY / HOLD
1088

    
1089
        first_settings = self._memobj.first_settings
1090
        val = RadioSettingValueList(
1091
            self._SCAN_RESUME,
1092
            self._SCAN_RESUME[first_settings.dw_resume_interval])
1093
        rs = RadioSetting("first_settings.dw_resume_interval", "Dual Watch Resume Interval", val)
1094
        menu.append(rs)
1095

    
1096
        # 51 SCN.LMP   Set the scan lamp ON or OFF when scanning stops. OFF / ON
1097

    
1098
        val = RadioSettingValueList(
1099
            self._OFF_ON,
1100
            self._OFF_ON[scan_settings.scan_lamp])
1101
        rs = RadioSetting("scan_settings.scan_lamp", "Scan Lamp", val)
1102
        menu.append(rs)
1103

    
1104
        # 53 SCN.STR   Set the scanning restart time.  0.1 S - 2.0 S - 10.0 S
1105

    
1106
        val = RadioSettingValueList(
1107
            self._SCAN_RESTART,
1108
            self._SCAN_RESTART[scan_settings.scan_restart])
1109
        rs = RadioSetting("scan_settings.scan_restart", "Scan Restart", val)
1110
        menu.append(rs)
1111

    
1112
        # Scan Width Section
1113

    
1114
        # 50 SCV.WTH Set the VFO scan frequency range. BAND / ALL  - NOT FOUND!
1115

    
1116
        # Scan Resume Section
1117

    
1118
        # 52 SCN.RSM    Configure the scan stop mode settings. 2.0 S - 5.0 S - 10.0 S / BUSY / HOLD
1119

    
1120
        first_settings = self._memobj.first_settings
1121
        val = RadioSettingValueList(
1122
            self._SCAN_RESUME,
1123
            self._SCAN_RESUME[first_settings.scan_resume])
1124
        rs = RadioSetting("first_settings.scan_resume", "Scan Resume", val)
1125
        menu.append(rs)
1126

    
1127
        return menu
1128

    
1129
    def _get_settings(self):
1130
        top = RadioSettings(
1131
            self._get_config_settings(),
1132
            self._get_digital_settings(),
1133
            self._get_display_settings(),
1134
            self._get_dtmf_settings(),
1135
            self._get_gm_settings(),
1136
            self._get_scan_settings()
1137
        )
1138
        return top
1139

    
1140
    def get_settings(self):
1141
        try:
1142
            return self._get_settings()
1143
        except:
1144
            import traceback
1145
            LOG.error("Failed to parse settings: %s", traceback.format_exc())
1146
            return None
1147

    
1148
    @classmethod
1149
    def apply_ff_padded_string(cls, setting, obj):
1150
        setattr(obj, "padded_string", cls._add_ff_pad(setting.value.get_value().rstrip(), 6))
1151

    
1152
    def set_settings(self, settings):
1153
        _mem = self._memobj
1154
        for element in settings:
1155
            if not isinstance(element, RadioSetting):
1156
                self.set_settings(element)
1157
                continue
1158
            if not element.changed():
1159
                continue
1160
            try:
1161
                if element.has_apply_callback():
1162
                    LOG.debug("Using apply callback")
1163
                    try:
1164
                        element.run_apply_callback()
1165
                    except NotImplementedError as e:
1166
                        LOG.error(e)
1167
                    continue
1168

    
1169
                # Find the object containing setting.
1170
                obj = _mem
1171
                bits = element.get_name().split(".")
1172
                setting = bits[-1]
1173
                for name in bits[:-1]:
1174
                    if name.endswith("]"):
1175
                        name, index = name.split("[")
1176
                        index = int(index[:-1])
1177
                        obj = getattr(obj, name)[index]
1178
                    else:
1179
                        obj = getattr(obj, name)
1180

    
1181
                try:
1182
                    old_val = getattr(obj, setting)
1183
                    LOG.debug("Setting %s(%r) <= %s" % (
1184
                        element.get_name(), old_val, element.value))
1185
                    setattr(obj, setting, element.value)
1186
                except AttributeError as e:
1187
                    LOG.error("Setting %s is not in the memory map: %s" %
1188
                              (element.get_name(), e))
1189
            except Exception:
1190
                LOG.debug(element.get_name())
1191
                raise
1192

    
1193
    def apply_volume(cls, setting, vfo):
1194
        val = setting.value.get_value()
1195
        cls._memobj.vfo_info[(vfo * 2)].volume = val
1196
        cls._memobj.vfo_info[(vfo * 2) + 1].volume = val
1197

    
1198
    def apply_dtmf(cls, setting, i):
1199
        rawval = setting.value.get_value().upper().rstrip()
1200
        val = [FT70_DTMF_CHARS.index(x) for x in rawval]
1201
        for x in range(len(val), 16):
1202
            val.append(0xFF)
1203
        cls._memobj.dtmf[i].memory = val
1204

    
1205
    def apply_digital_popup(cls, setting, obj):
1206
        rawval = setting.value.get_value()
1207
        val = 0 if cls._DIG_POP_UP.index(rawval) == 0 else cls._DIG_POP_UP.index(rawval) + 9
1208
        obj.digital_popup = val
1209

    
1210
    def apply_mycall(cls, setting, obj):
1211
        cs = setting.value.get_value()
1212
        if cs[0] in ('-', '/'):
1213
            raise InvalidValueError("First character of call sign can't be - or /:  {0:s}".format(cs))
1214
        else:
1215
            obj.callsign = cls._add_ff_pad(cs.rstrip(), 10)
1216

    
1217
    def load_mmap(self, filename):
1218
        if filename.lower().endswith(self._adms_ext):
1219
            with open(filename, 'rb') as f:
1220
                self._adms_header = f.read(0xED)
1221
                if b'=ADMS10, Version=1.0.1.0' not in self._adms_header:
1222
                    raise errors.ImageDetectFailed(
1223
                        'Unsupported version found in ADMS file')
1224
                LOG.debug('ADMS Header:\n%s',
1225
                          util.hexprint(self._adms_header))
1226
                self._mmap = memmap.MemoryMapBytes(self._model + f.read())
1227
                LOG.info('Loaded ADMS file')
1228
            self.process_mmap()
1229
        else:
1230
            chirp_common.CloneModeRadio.load_mmap(self, filename)
1231

    
1232
    def save_mmap(self, filename):
1233
        if filename.lower().endswith(self._adms_ext):
1234
            if not hasattr(self, '_adms_header'):
1235
                raise Exception('Unable to save .img to %s' % self._adms_ext)
1236
            with open(filename, 'wb') as f:
1237
                f.write(self._adms_header)
1238
                f.write(self._mmap.get_packed()[5:])
1239
                LOG.info('Wrote file')
1240
        else:
1241
            chirp_common.CloneModeRadio.save_mmap(self, filename)
(1-1/4)