not1mm 24.9.5__py3-none-any.whl → 24.9.8__py3-none-any.whl

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.
not1mm/__main__.py CHANGED
@@ -1936,7 +1936,7 @@ class MainWindow(QtWidgets.QMainWindow):
1936
1936
  " "
1937
1937
  )[:19]
1938
1938
  self.contact["Call"] = self.callsign.text()
1939
- if self.contact.get("Mode") not in ("FT8", "FT4"):
1939
+ if self.contact.get("Mode") not in ("FT8", "FT4", "RTTY"):
1940
1940
  self.contact["Freq"] = round(
1941
1941
  float(self.radio_state.get("vfoa", 0.0)) / 1000, 2
1942
1942
  )
@@ -3214,14 +3214,21 @@ class MainWindow(QtWidgets.QMainWindow):
3214
3214
  cmd["bw"] = bw
3215
3215
  self.multicast_interface.send_as_json(cmd)
3216
3216
  if self.n1mm:
3217
+ self.n1mm.radio_info["Freq"] = vfo[:-1]
3218
+ self.n1mm.radio_info["TXFreq"] = vfo[:-1]
3219
+ self.n1mm.radio_info["Mode"] = mode
3220
+ self.n1mm.radio_info["OpCall"] = self.current_op
3221
+ self.n1mm.radio_info["IsRunning"] = str(
3222
+ self.pref.get("run_state", False)
3223
+ )
3217
3224
  if self.n1mm.send_radio_packets:
3218
- self.n1mm.radio_info["Freq"] = vfo[:-1]
3219
- self.n1mm.radio_info["TXFreq"] = vfo[:-1]
3220
- self.n1mm.radio_info["Mode"] = mode
3221
- self.n1mm.radio_info["OpCall"] = self.current_op
3222
- self.n1mm.radio_info["IsRunning"] = str(
3223
- self.pref.get("run_state", False)
3224
- )
3225
+ # self.n1mm.radio_info["Freq"] = vfo[:-1]
3226
+ # self.n1mm.radio_info["TXFreq"] = vfo[:-1]
3227
+ # self.n1mm.radio_info["Mode"] = mode
3228
+ # self.n1mm.radio_info["OpCall"] = self.current_op
3229
+ # self.n1mm.radio_info["IsRunning"] = str(
3230
+ # self.pref.get("run_state", False)
3231
+ # )
3225
3232
  self.n1mm.send_radio()
3226
3233
  except TypeError as err:
3227
3234
  logger.debug(f"{err=} {vfo=} {the_dict=}")
@@ -347,6 +347,11 @@
347
347
  <string>STEW PERRY TOPBAND</string>
348
348
  </property>
349
349
  </item>
350
+ <item>
351
+ <property name="text">
352
+ <string>WEEKLY RTTY</string>
353
+ </property>
354
+ </item>
350
355
  <item>
351
356
  <property name="text">
352
357
  <string>WINTER FIELD DAY</string>
@@ -841,7 +846,7 @@
841
846
  <enum>Qt::Horizontal</enum>
842
847
  </property>
843
848
  <property name="standardButtons">
844
- <set>QDialogButtonBox::Cancel|QDialogButtonBox::Ok</set>
849
+ <set>QDialogButtonBox::StandardButton::Cancel|QDialogButtonBox::StandardButton::Ok</set>
845
850
  </property>
846
851
  <property name="centerButtons">
847
852
  <bool>true</bool>
not1mm/lib/database.py CHANGED
@@ -663,8 +663,8 @@ class DataBase:
663
663
  logger.debug("%s", exception)
664
664
  return {}
665
665
 
666
- def fetch_call_exists(self, call) -> dict:
667
- """returns a dict key of nr_count"""
666
+ def fetch_call_exists(self, call: str) -> dict:
667
+ """returns a dict key of call_count"""
668
668
  try:
669
669
  with sqlite3.connect(self.database) as conn:
670
670
  conn.row_factory = self.row_factory
@@ -172,7 +172,7 @@ def gen_adif(self, cabrillo_name: str, contest_id=""):
172
172
  file=file_descriptor,
173
173
  )
174
174
  # ------------CQ 160---------------
175
- elif cabrillo_name in ("CQ-160-CW", "CQ-160-SSB"):
175
+ elif cabrillo_name in ("CQ-160-CW", "CQ-160-SSB", "WEEKLY-RTTY"):
176
176
  rcv = f"{contact.get('Exchange1', '')}"
177
177
  if len(rcv) > 1:
178
178
  print(
not1mm/lib/version.py CHANGED
@@ -1,3 +1,3 @@
1
1
  """It's the version"""
2
2
 
3
- __version__ = "24.9.5"
3
+ __version__ = "24.9.8"
@@ -0,0 +1,442 @@
1
+ """Weekly RTTY Test"""
2
+
3
+ # Geographic Focus: North America
4
+ # Participation: Worldwide
5
+ # Mode: RTTY
6
+ # Bands: 80, 40, 20, 15, 10m
7
+ # Classes: Single-Op (QRP/Low)
8
+ # Max power: LP: 100 watts
9
+ # QRP: 5 watts
10
+ # Exchange: Name + (state/province/country)
11
+ # Work stations: Once per band
12
+ # QSO Points: 1 point per QSO
13
+ # Multipliers: Each callsign once
14
+ # Score Calculation: Total score = total QSO points x total mults
15
+ # Submit logs by: September 8, 2024
16
+ # E-mail logs to: (none)
17
+ # Post log summary at: http://www.3830scores.com/
18
+ # Mail logs to: (none)
19
+ # Find rules at: https://radiosport.world/wrt.html
20
+
21
+ import datetime
22
+ import logging
23
+ import platform
24
+
25
+ from pathlib import Path
26
+
27
+ from PyQt6 import QtWidgets
28
+
29
+ from not1mm.lib.ham_utility import get_logged_band
30
+ from not1mm.lib.plugin_common import gen_adif, get_points
31
+ from not1mm.lib.version import __version__
32
+
33
+ logger = logging.getLogger(__name__)
34
+
35
+ ALTEREGO = None
36
+
37
+ EXCHANGE_HINT = "Name + SPC"
38
+
39
+ name = "WEEKLY RTTY TEST"
40
+ cabrillo_name = "WEEKLY-RTTY"
41
+ mode = "RTTY" # CW SSB BOTH RTTY
42
+ # columns = [0, 1, 2, 3, 4, 10, 11, 14, 15]
43
+ columns = [
44
+ "YYYY-MM-DD HH:MM:SS",
45
+ "Call",
46
+ "Freq",
47
+ "Exchange1",
48
+ "M1",
49
+ "PTS",
50
+ ]
51
+
52
+ advance_on_space = [True, True, True, True, True]
53
+
54
+ # 1 once per contest, 2 work each band, 3 each band/mode, 4 no dupe checking
55
+ dupe_type = 2
56
+
57
+
58
+ def init_contest(self):
59
+ """setup plugin"""
60
+ set_tab_next(self)
61
+ set_tab_prev(self)
62
+ interface(self)
63
+ self.next_field = self.other_1
64
+
65
+
66
+ def interface(self):
67
+ """Setup user interface"""
68
+ self.field1.hide()
69
+ self.field2.hide()
70
+ self.field3.show()
71
+ self.field4.show()
72
+ namefield = self.field3.findChild(QtWidgets.QLabel)
73
+ namefield.setText("Name")
74
+ self.field3.setAccessibleName("Name")
75
+ spc = self.field4.findChild(QtWidgets.QLabel)
76
+ spc.setText("SPC")
77
+ self.field4.setAccessibleName("SPC")
78
+
79
+
80
+ def reset_label(self):
81
+ """reset label after field cleared"""
82
+
83
+
84
+ def set_tab_next(self):
85
+ """Set TAB Advances"""
86
+ self.tab_next = {
87
+ self.callsign: self.field3.findChild(QtWidgets.QLineEdit),
88
+ self.field3.findChild(QtWidgets.QLineEdit): self.field4.findChild(
89
+ QtWidgets.QLineEdit
90
+ ),
91
+ self.field4.findChild(QtWidgets.QLineEdit): self.callsign,
92
+ }
93
+
94
+
95
+ def set_tab_prev(self):
96
+ """Set TAB Advances"""
97
+ self.tab_prev = {
98
+ self.callsign: self.field4.findChild(QtWidgets.QLineEdit),
99
+ self.field3.findChild(QtWidgets.QLineEdit): self.callsign,
100
+ self.field4.findChild(QtWidgets.QLineEdit): self.field3.findChild(
101
+ QtWidgets.QLineEdit
102
+ ),
103
+ }
104
+
105
+
106
+ def set_contact_vars(self):
107
+ """Contest Specific"""
108
+ self.contact["SNT"] = self.sent.text()
109
+ self.contact["RCV"] = self.receive.text()
110
+ self.contact["Name"] = self.other_1.text().upper()
111
+ self.contact["Sect"] = self.other_2.text().upper()
112
+ self.contact["SentNr"] = self.contest_settings.get("SentExchange", 0)
113
+ self.contact["Exchange1"] = (
114
+ f"{self.other_1.text().upper()} {self.other_2.text().upper()}"
115
+ )
116
+
117
+ result = self.database.fetch_call_exists(self.callsign.text())
118
+ # result = ALTEREGO.database.fetch_call_exists(self.callsign.text())
119
+ if result.get("call_count", ""):
120
+ self.contact["IsMultiplier1"] = 0
121
+ else:
122
+ self.contact["IsMultiplier1"] = 1
123
+
124
+
125
+ def predupe(self):
126
+ """called after callsign entered"""
127
+
128
+
129
+ def prefill(self):
130
+ """Fill sentnr"""
131
+
132
+
133
+ def points(self):
134
+ """Calc point"""
135
+ return 1
136
+
137
+
138
+ def show_mults(self):
139
+ """Return display string for mults"""
140
+ result = self.database.fetch_section_band_count_nodx()
141
+ if result:
142
+ return int(result.get("sb_count", 0))
143
+ return 0
144
+
145
+
146
+ def show_qso(self):
147
+ """Return qso count"""
148
+ result = self.database.fetch_qso_count()
149
+ if result:
150
+ return int(result.get("qsos", 0))
151
+ return 0
152
+
153
+
154
+ def calc_score(self):
155
+ """Return calculated score"""
156
+ result = self.database.fetch_points()
157
+ if result is not None:
158
+ score = result.get("Points", "0")
159
+ if score is None:
160
+ score = "0"
161
+ contest_points = int(score)
162
+ mults = show_mults(self)
163
+ return contest_points * mults
164
+ return 0
165
+
166
+
167
+ def adif(self):
168
+ """Call the generate ADIF function"""
169
+ gen_adif(self, cabrillo_name, cabrillo_name)
170
+
171
+
172
+ def cabrillo(self):
173
+ """Generates Cabrillo file. Maybe."""
174
+ # https://www.cqwpx.com/cabrillo.htm
175
+ logger.debug("******Cabrillo*****")
176
+ logger.debug("Station: %s", f"{self.station}")
177
+ logger.debug("Contest: %s", f"{self.contest_settings}")
178
+ now = datetime.datetime.now()
179
+ date_time = now.strftime("%Y-%m-%d_%H-%M-%S")
180
+ filename = (
181
+ str(Path.home())
182
+ + "/"
183
+ + f"{self.station.get('Call','').upper()}_{cabrillo_name}_{date_time}.log"
184
+ )
185
+ logger.debug("%s", filename)
186
+ log = self.database.fetch_all_contacts_asc()
187
+ try:
188
+ with open(filename, "w", encoding="ascii") as file_descriptor:
189
+ print("START-OF-LOG: 3.0", end="\r\n", file=file_descriptor)
190
+ print(
191
+ f"CREATED-BY: Not1MM v{__version__}",
192
+ end="\r\n",
193
+ file=file_descriptor,
194
+ )
195
+ print(
196
+ f"CONTEST: {cabrillo_name}",
197
+ end="\r\n",
198
+ file=file_descriptor,
199
+ )
200
+ if self.station.get("Club", ""):
201
+ print(
202
+ f"CLUB: {self.station.get('Club', '').upper()}",
203
+ end="\r\n",
204
+ file=file_descriptor,
205
+ )
206
+ print(
207
+ f"CALLSIGN: {self.station.get('Call','')}",
208
+ end="\r\n",
209
+ file=file_descriptor,
210
+ )
211
+ print(
212
+ f"LOCATION: {self.station.get('ARRLSection', '')}",
213
+ end="\r\n",
214
+ file=file_descriptor,
215
+ )
216
+ # print(
217
+ # f"ARRL-SECTION: {self.pref.get('section', '')}",
218
+ # end="\r\n",
219
+ # file=file_descriptor,
220
+ # )
221
+ print(
222
+ f"CATEGORY-OPERATOR: {self.contest_settings.get('OperatorCategory','')}",
223
+ end="\r\n",
224
+ file=file_descriptor,
225
+ )
226
+ print(
227
+ f"CATEGORY-ASSISTED: {self.contest_settings.get('AssistedCategory','')}",
228
+ end="\r\n",
229
+ file=file_descriptor,
230
+ )
231
+ print(
232
+ f"CATEGORY-BAND: {self.contest_settings.get('BandCategory','')}",
233
+ end="\r\n",
234
+ file=file_descriptor,
235
+ )
236
+ print(
237
+ f"CATEGORY-MODE: {self.contest_settings.get('ModeCategory','')}",
238
+ end="\r\n",
239
+ file=file_descriptor,
240
+ )
241
+ print(
242
+ f"CATEGORY-TRANSMITTER: {self.contest_settings.get('TransmitterCategory','')}",
243
+ end="\r\n",
244
+ file=file_descriptor,
245
+ )
246
+ if self.contest_settings.get("OverlayCategory", "") != "N/A":
247
+ print(
248
+ f"CATEGORY-OVERLAY: {self.contest_settings.get('OverlayCategory','')}",
249
+ end="\r\n",
250
+ file=file_descriptor,
251
+ )
252
+ print(
253
+ f"GRID-LOCATOR: {self.station.get('GridSquare','')}",
254
+ end="\r\n",
255
+ file=file_descriptor,
256
+ )
257
+ # print(
258
+ # f"CATEGORY: {None}",
259
+ # end="\r\n",
260
+ # file=file_descriptor,
261
+ # )
262
+ print(
263
+ f"CATEGORY-POWER: {self.contest_settings.get('PowerCategory','')}",
264
+ end="\r\n",
265
+ file=file_descriptor,
266
+ )
267
+
268
+ print(
269
+ f"CLAIMED-SCORE: {calc_score(self)}",
270
+ end="\r\n",
271
+ file=file_descriptor,
272
+ )
273
+ ops = f"@{self.station.get('Call','')}"
274
+ list_of_ops = self.database.get_ops()
275
+ for op in list_of_ops:
276
+ ops += f", {op.get('Operator', '')}"
277
+ print(
278
+ f"OPERATORS: {ops}",
279
+ end="\r\n",
280
+ file=file_descriptor,
281
+ )
282
+ print(
283
+ f"NAME: {self.station.get('Name', '')}",
284
+ end="\r\n",
285
+ file=file_descriptor,
286
+ )
287
+ print(
288
+ f"ADDRESS: {self.station.get('Street1', '')}",
289
+ end="\r\n",
290
+ file=file_descriptor,
291
+ )
292
+ print(
293
+ f"ADDRESS-CITY: {self.station.get('City', '')}",
294
+ end="\r\n",
295
+ file=file_descriptor,
296
+ )
297
+ print(
298
+ f"ADDRESS-STATE-PROVINCE: {self.station.get('State', '')}",
299
+ end="\r\n",
300
+ file=file_descriptor,
301
+ )
302
+ print(
303
+ f"ADDRESS-POSTALCODE: {self.station.get('Zip', '')}",
304
+ end="\r\n",
305
+ file=file_descriptor,
306
+ )
307
+ print(
308
+ f"ADDRESS-COUNTRY: {self.station.get('Country', '')}",
309
+ end="\r\n",
310
+ file=file_descriptor,
311
+ )
312
+ print(
313
+ f"EMAIL: {self.station.get('Email', '')}",
314
+ end="\r\n",
315
+ file=file_descriptor,
316
+ )
317
+ for contact in log:
318
+ the_date_and_time = contact.get("TS", "")
319
+ themode = contact.get("Mode", "")
320
+ if themode == "RTTY":
321
+ themode = "RY"
322
+ frequency = str(int(contact.get("Freq", "0"))).rjust(5)
323
+
324
+ loggeddate = the_date_and_time[:10]
325
+ loggedtime = the_date_and_time[11:13] + the_date_and_time[14:16]
326
+ exch1 = ""
327
+ exch2 = ""
328
+ if " " in str(contact.get("Exchange1", "")):
329
+ exch1, exch2 = str(contact.get("Exchange1", "")).strip().split(" ")
330
+ print(
331
+ f"QSO: {frequency} {themode} {loggeddate} {loggedtime} "
332
+ f"{contact.get('StationPrefix', '').ljust(13)} "
333
+ f"{str(contact.get('SentNr', '')).upper()} "
334
+ f"{contact.get('Call', '').ljust(13)} "
335
+ f"{exch1.ljust(10)} "
336
+ f"{exch2.ljust(5)} ",
337
+ end="\r\n",
338
+ file=file_descriptor,
339
+ )
340
+ print("END-OF-LOG:", end="\r\n", file=file_descriptor)
341
+ self.show_message_box(f"Cabrillo saved to: {filename}")
342
+ except IOError as exception:
343
+ logger.critical("cabrillo: IO error: %s, writing to %s", exception, filename)
344
+ self.show_message_box(f"Error saving Cabrillo: {exception} {filename}")
345
+ return
346
+
347
+
348
+ def recalculate_mults(self):
349
+ """Recalculates multipliers after change in logged qso."""
350
+
351
+ all_contacts = self.database.fetch_all_contacts_asc()
352
+ for contact in all_contacts:
353
+ time_stamp = contact.get("TS", "")
354
+ call = contact.get("Call", "")
355
+ query = (
356
+ f"select count(*) as call_count from dxlog where TS < '{time_stamp}' "
357
+ f"and Call = '{call}' "
358
+ f"and ContestNR = {self.pref.get('contest', '1')};"
359
+ )
360
+ result = self.database.exec_sql(query)
361
+ count = result.get("call_count", 1)
362
+ if count == 0:
363
+ contact["IsMultiplier1"] = 1
364
+ else:
365
+ contact["IsMultiplier1"] = 0
366
+ self.database.change_contact(contact)
367
+ cmd = {}
368
+ cmd["cmd"] = "UPDATELOG"
369
+ cmd["station"] = platform.node()
370
+ self.multicast_interface.send_as_json(cmd)
371
+
372
+
373
+ def set_self(the_outie):
374
+ """..."""
375
+ globals()["ALTEREGO"] = the_outie
376
+
377
+
378
+ def ft8_handler(the_packet: dict):
379
+ """Process FT8 QSO packets
380
+
381
+ FlDigi
382
+ {
383
+ 'FREQ': '7.029500',
384
+ 'CALL': 'DL2DSL',
385
+ 'MODE': 'RTTY',
386
+ 'NAME': 'BOB',result = ALTEREGO.database.fetch_call_exists(the_packet.get("CALL", ""))
387
+ if result.get("call_count", ""):
388
+ ALTEREGO.contact["IsMultiplier1"] = 0
389
+ else:
390
+ ALTEREGO.contact["IsMultiplier1"] = 1
391
+ 'QSO_DATE': '20240904',
392
+ 'QSO_DATE_OFF': '20240904',
393
+ 'TIME_OFF': '212825',
394
+ 'TIME_ON': '212800',
395
+ 'RST_RCVD': '599',
396
+ 'RST_SENT': '599',
397
+ 'BAND': '40M',
398
+ 'COUNTRY': 'FED. REP. OF GERMANY',
399
+ 'CQZ': '14',
400
+ 'STX': '000',
401
+ 'STX_STRING': '1D ORG',
402
+ 'CLASS': '1D',
403
+ 'ARRL_SECT': 'DX',
404
+ 'TX_PWR': '0',
405
+ 'OPERATOR': 'K6GTE',
406
+ 'STATION_CALLSIGN': 'K6GTE',
407
+ 'MY_GRIDSQUARE': 'DM13AT',
408
+ 'MY_CITY': 'ANAHEIM, CA',
409
+ 'MY_STATE': 'CA'
410
+ }
411
+
412
+ """
413
+ logger.debug(f"{the_packet=}")
414
+ if ALTEREGO is not None:
415
+ ALTEREGO.callsign.setText(the_packet.get("CALL"))
416
+ ALTEREGO.contact["Call"] = the_packet.get("CALL", "")
417
+ ALTEREGO.contact["SNT"] = ALTEREGO.sent.text()
418
+ ALTEREGO.contact["RCV"] = ALTEREGO.receive.text()
419
+ ALTEREGO.contact["Exchange1"] = the_packet.get("SRX_STRING", "")
420
+ exch1 = ""
421
+ exch2 = ""
422
+ if " " in str(the_packet.get("SRX_STRING", "")):
423
+ exch1, exch2 = str(the_packet.get("SRX_STRING", "")).strip().split(" ")
424
+ ALTEREGO.other_1.setText(exch1)
425
+ ALTEREGO.other_2.setText(exch2)
426
+ # ALTEREGO.contact["Sect"] = the_packet.get("ARRL_SECT", "ERR")
427
+ ALTEREGO.contact["Mode"] = the_packet.get("MODE", "ERR")
428
+ ALTEREGO.contact["Freq"] = round(float(the_packet.get("FREQ", "0.0")) * 1000, 2)
429
+ ALTEREGO.contact["QSXFreq"] = round(
430
+ float(the_packet.get("FREQ", "0.0")) * 1000, 2
431
+ )
432
+ ALTEREGO.contact["Band"] = get_logged_band(
433
+ str(int(float(the_packet.get("FREQ", "0.0")) * 1000000))
434
+ )
435
+
436
+ result = ALTEREGO.database.fetch_call_exists(the_packet.get("CALL", ""))
437
+ if result.get("call_count", ""):
438
+ ALTEREGO.contact["IsMultiplier1"] = 0
439
+ else:
440
+ ALTEREGO.contact["IsMultiplier1"] = 1
441
+ logger.debug(f"{ALTEREGO.contact=}")
442
+ ALTEREGO.save_contact()
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: not1mm
3
- Version: 24.9.5
3
+ Version: 24.9.8
4
4
  Summary: NOT1MM Logger
5
5
  Author-email: Michael Bridak <michael.bridak@gmail.com>
6
6
  Project-URL: Homepage, https://github.com/mbridak/not1mm
@@ -100,7 +100,7 @@ Requires-Dist: Levenshtein
100
100
  - [Cluster](#cluster)
101
101
  - [N1MM Packets](#n1mm-packets)
102
102
  - [Bands](#bands)
103
- - [Logging WSJT-X FT8/FT4 contacts](#logging-wsjt-x-ft8ft4-contacts)
103
+ - [Logging WSJT-X FT8/FT4 and FLDIGI RTTY contacts](#logging-wsjt-x-ft8ft4-and-fldigi-rtty-contacts)
104
104
  - [Sending CW](#sending-cw)
105
105
  - [Editing macro keys](#editing-macro-keys)
106
106
  - [Macro substitutions](#macro-substitutions)
@@ -169,9 +169,9 @@ when the program craps the bed. I'm only one guy, so if you see a bug let me kno
169
169
 
170
170
  ### Data and RTTY
171
171
 
172
- I don't do much of any digital or RTTY operating. This is why you don't see RTTY in
173
- the list of working contests. The Lord helps those who burn people at the... I
174
- mean, who help themselves. Feel free to fill in that hole with a pull request.
172
+ I've recently added portions of code to watch for WSTJ-X and fldigi QSOs. I've added
173
+ the Weekly RTTY Test, So RTTY could be tested. Also added FT8/4 and RTTY to ARRL Field
174
+ Day. Praying that it works. I'll add more details later.
175
175
 
176
176
  ### Other not so supported contests
177
177
 
@@ -222,10 +222,13 @@ generated, 'cause I'm lazy, list of those who've submitted PR's.
222
222
  - Phone Weekly Test
223
223
  - RAC Canada Day
224
224
  - Stew Perry Topband
225
+ - Weekly RTTY
225
226
  - Winter Field Day
226
227
 
227
228
  ## Recent Changes
228
229
 
230
+ - [24-9-8] Correct n1mm contact packet info.
231
+ - [24-9-6] Added the Weekly RTTY
229
232
  - [24-9-5] Added FlDigi support for Field Day.
230
233
  - [24-9-3] Added WSJT-X FT8 mode contacts to ARRL Field Day.
231
234
 
@@ -604,12 +607,20 @@ appear. Those without will not.
604
607
 
605
608
  ![Bands Configuration Screen](https://github.com/mbridak/not1mm/raw/master/pic/configure_bands.png)
606
609
 
607
- ## Logging WSJT-X FT8/FT4 contacts
610
+ ## Logging WSJT-X FT8/FT4 and FLDIGI RTTY contacts
608
611
 
609
- **Currently only working for ARRL Field Day.**
612
+ **FT8/FT4 Currently only working for ARRL Field Day.**
610
613
 
611
614
  not1mm listens for WSJT-X UDP traffic on the default localhost:2237. No setup is
612
- needed to be done on not1mm's side.
615
+ needed to be done on not1mm's side. That's good because I'm lazy.
616
+
617
+ not1mm polls for fldigi QSOs via it's XMLRPC interface. It does this in a rather stupid
618
+ way. It just keeps asking what was the last QSO and compares it to the previous response.
619
+ If it's different, it's new. I've added the Weekly RTTY Test so this can be tested.
620
+
621
+ Unlike WSJT, fldigi needs to be setup for this to work. The XMLRPC interface needs to be
622
+ active. And in fldigi's config dialog go to CONTESTS -> General -> CONTEST and select
623
+ Generic Contest. Make sure the Text Capture Order field says CALL EXCHANGE.
613
624
 
614
625
  ## Sending CW
615
626
 
@@ -1,5 +1,5 @@
1
1
  not1mm/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
2
- not1mm/__main__.py,sha256=C9DOp1FLdRj1Oako2gW9Y6rlfHniSp379RIhIKiYgEw,124001
2
+ not1mm/__main__.py,sha256=hjdL9Z4TiD5uKJj4zr8d4sWpFqn3aVqn0biofGjVS7g,124411
3
3
  not1mm/bandmap.py,sha256=1b5tXCfGTnpqqn6hPNt7zRA8SmuwSXzSwNHZXhCRt70,31434
4
4
  not1mm/checkwindow.py,sha256=aI-nr8OF90IWV7R_XRdmitvBJ9M85evCs72HoU3Jnvc,10374
5
5
  not1mm/fsutils.py,sha256=ukHKxKTeNKxKwqRaJjtzRShL4X5Xl0jRBbADyy3Ifp8,1701
@@ -30,7 +30,7 @@ not1mm/data/k6gte.not1mm-64.png,sha256=6ku45Gq1g5ezh04F07osoKRtanb3e4kbx5XdIEh3N
30
30
  not1mm/data/logwindow.ui,sha256=vfkNdzJgFs3tTOBKLDavF2zVMvNHWOZ82fAErRi6pQY,1436
31
31
  not1mm/data/logwindowx.ui,sha256=9FzDJtLRpagvAWcDjFdB9NnvNZ4bVxdTNHy1Jit2ido,1610
32
32
  not1mm/data/main.ui,sha256=Y8F0gOqMTcsxBmc3QvWkJMxhYFpxUJQl-btGiM1-31w,56506
33
- not1mm/data/new_contest.ui,sha256=6j4OYcTjSx-5ywvBf4GQfuAEaavCJVPVwzj_es5ajsc,22138
33
+ not1mm/data/new_contest.ui,sha256=x4KkVqbaqrv-808yC7BR5ZBpMal81GF6uzKCrod8ick,22278
34
34
  not1mm/data/not1mm.html,sha256=c9-mfjMwDt4f5pySUruz2gREW33CQ2_rCddM2z5CZQo,23273
35
35
  not1mm/data/opon.ui,sha256=mC4OhoVIfR1H9IqHAKXliPMm8VOVmxSEadpsFQ7XnS4,2247
36
36
  not1mm/data/pickcontest.ui,sha256=_9JFiJw4l-bRRgNDtVg1DpxreylYd_UqEq0wfcr9gJc,1600
@@ -94,7 +94,7 @@ not1mm/lib/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
94
94
  not1mm/lib/about.py,sha256=sWycfGcruN3SaEe4JmaJ61K6D8Itq0WxpUYT-lEcmYM,416
95
95
  not1mm/lib/cat_interface.py,sha256=aazvNTSeZAROq3KL8gPx-I95iVez2IiIOSk22qeqVCQ,19502
96
96
  not1mm/lib/cwinterface.py,sha256=Q8p3pScOHczZ8ptICfH1Yu6rCEwQJLgrNwYMN76B2i8,3389
97
- not1mm/lib/database.py,sha256=0dx6SX40OiPFent2ePAGsFj_XmOVMNBHaqd0QtAiewY,43129
97
+ not1mm/lib/database.py,sha256=TZjMixyPACZginoJkemXOdA4kAYMdSirdodkmN0Fv10,43136
98
98
  not1mm/lib/edit_contact.py,sha256=Ki9bGPpqyQQBB1cU8VIBDCal3lbXeQ6qxhzklmhE2_w,353
99
99
  not1mm/lib/edit_macro.py,sha256=raKWBwsHInj5EUKmvyLQ6gqc3ZFDlstsD3xqoM4PC8E,517
100
100
  not1mm/lib/edit_opon.py,sha256=j3qJ1aBsQoIOnQ9yiBl3lyeISvKTP0I_rtBYBPAfgeI,359
@@ -107,11 +107,11 @@ not1mm/lib/multicast.py,sha256=bnFUNHyy82GmIb3_88EPBVVssj7-HzkJPaH671cK8Qw,3249
107
107
  not1mm/lib/n1mm.py,sha256=H54mpgJF0GAmKavM-nb5OAq2SJFWYkux4eMWWiSRxJc,6288
108
108
  not1mm/lib/new_contest.py,sha256=IznTDMq7yXHB6zBoGUEC_WDYPCPpsSZW4wwMJi16zK0,816
109
109
  not1mm/lib/playsound.py,sha256=kxkcitBFbZCXJ2wxQ1lxg4rBwfxiSpuNpJSXHOPCoXA,9241
110
- not1mm/lib/plugin_common.py,sha256=wuG7B0OJx9zYc5Gew3fdt_lNyan8Ul9KNlPQ7PDKGsU,9114
110
+ not1mm/lib/plugin_common.py,sha256=yefvcX61fXSjs__OPssJqVlZyg1AlcV1VDkl2MQP6kk,9129
111
111
  not1mm/lib/select_contest.py,sha256=WsptLuwkouIHeocJL3oZ6-eUfEnhpwdc-x7eMZ_TIVM,359
112
112
  not1mm/lib/settings.py,sha256=MWiKXbasaFbzeHTjfzTaTqbCBrIijudP_-0a5jNmUAA,9265
113
113
  not1mm/lib/super_check_partial.py,sha256=p5l3u2ZOCBtlWgbvskC50FpuoaIpR07tfC6zTdRWbh4,2334
114
- not1mm/lib/version.py,sha256=F98YdDdv2B3tvIDuoqpiagUBxJ2s48EdRr4FvAbwOnc,47
114
+ not1mm/lib/version.py,sha256=5j2EOLjnhLXJzoesZbLU9w4RWqMYEr-RYE7LKFH1EDo,47
115
115
  not1mm/lib/versiontest.py,sha256=8vDNptuBBunn-1IGkjNaquehqBYUJyjrPSF8Igmd4_Y,1286
116
116
  not1mm/plugins/10_10_fall_cw.py,sha256=IttjX1yy4nDdACGsiYlPteFG8eVseX_WtoFio6bqHE8,10953
117
117
  not1mm/plugins/10_10_spring_cw.py,sha256=ThCptdM3dX4ywhoy2JRcOEyHSqcJolFaT7O_PYzM1Mg,10958
@@ -148,10 +148,11 @@ not1mm/plugins/naqp_ssb.py,sha256=VLWVrSzI0UP1AhSXYn61eZ7or1rz6a_pS_xCKfgS4Jw,11
148
148
  not1mm/plugins/phone_weekly_test.py,sha256=fLpMe03WB9_KgRl6vMgQQt_aktFdqfNt2Sw81CTRAUs,12325
149
149
  not1mm/plugins/ref_cw.py,sha256=aWjHHkqIKutjRUtzh09y5haFfnZK9poRQDWRQMDRxxU,16326
150
150
  not1mm/plugins/stew_perry_topband.py,sha256=CKBQbYl4ETxhXJd2dma4fg_C5pag_s7Nf61SCztZtqE,10668
151
+ not1mm/plugins/weekly_rtty.py,sha256=DQcy3SY0Zn56EdlYGf3NxrRhTnkNa5IqRQPRQdKDSPs,14255
151
152
  not1mm/plugins/winter_field_day.py,sha256=4rcfRtobwjHO6BNL3WOTHzBmyyeuX79BNGBG8PfjrI8,10238
152
- not1mm-24.9.5.dist-info/LICENSE,sha256=OXLcl0T2SZ8Pmy2_dmlvKuetivmyPd5m1q-Gyd-zaYY,35149
153
- not1mm-24.9.5.dist-info/METADATA,sha256=ARDMBkO0MbKxknvDqFFkeCqup5PWor3Ag0hZtqDWl_8,29781
154
- not1mm-24.9.5.dist-info/WHEEL,sha256=cVxcB9AmuTcXqmwrtPhNK88dr7IR_b6qagTj0UvIEbY,91
155
- not1mm-24.9.5.dist-info/entry_points.txt,sha256=pMcZk_0dxFgLkcUkF0Q874ojpwOmF3OL6EKw9LgvocM,47
156
- not1mm-24.9.5.dist-info/top_level.txt,sha256=0YmTxEcDzQlzXub-lXASvoLpg_mt1c2thb5cVkDf5J4,7
157
- not1mm-24.9.5.dist-info/RECORD,,
153
+ not1mm-24.9.8.dist-info/LICENSE,sha256=OXLcl0T2SZ8Pmy2_dmlvKuetivmyPd5m1q-Gyd-zaYY,35149
154
+ not1mm-24.9.8.dist-info/METADATA,sha256=W-lMQ73MBag75TDiTgO4Xl-3OaHh-wfjh6XU8DVSci8,30461
155
+ not1mm-24.9.8.dist-info/WHEEL,sha256=cVxcB9AmuTcXqmwrtPhNK88dr7IR_b6qagTj0UvIEbY,91
156
+ not1mm-24.9.8.dist-info/entry_points.txt,sha256=pMcZk_0dxFgLkcUkF0Q874ojpwOmF3OL6EKw9LgvocM,47
157
+ not1mm-24.9.8.dist-info/top_level.txt,sha256=0YmTxEcDzQlzXub-lXASvoLpg_mt1c2thb5cVkDf5J4,7
158
+ not1mm-24.9.8.dist-info/RECORD,,