kopia lustrzana https://github.com/NanoVNA-Saver/nanovna-saver
				
				
				
			
		
			
				
	
	
		
			220 wiersze
		
	
	
		
			7.5 KiB
		
	
	
	
		
			Python
		
	
	
			
		
		
	
	
			220 wiersze
		
	
	
		
			7.5 KiB
		
	
	
	
		
			Python
		
	
	
| #  NanoVNASaver
 | |
| #
 | |
| #  A python program to view and export Touchstone data from a NanoVNA
 | |
| #  Copyright (C) 2019, 2020  Rune B. Broberg
 | |
| #  Copyright (C) 2020 NanoVNA-Saver Authors
 | |
| #
 | |
| #  This program is free software: you can redistribute it and/or modify
 | |
| #  it under the terms of the GNU General Public License as published by
 | |
| #  the Free Software Foundation, either version 3 of the License, or
 | |
| #  (at your option) any later version.
 | |
| #
 | |
| #  This program is distributed in the hope that it will be useful,
 | |
| #  but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| #  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
| #  GNU General Public License for more details.
 | |
| #
 | |
| #  You should have received a copy of the GNU General Public License
 | |
| #  along with this program.  If not, see <https://www.gnu.org/licenses/>.
 | |
| import logging
 | |
| import re
 | |
| import typing
 | |
| from typing import List, Tuple
 | |
| 
 | |
| from PyQt5 import QtCore, QtGui
 | |
| from PyQt5.QtCore import QModelIndex
 | |
| 
 | |
| logger = logging.getLogger(__name__)
 | |
| 
 | |
| 
 | |
| class BandsModel(QtCore.QAbstractTableModel):
 | |
|     bands: List[Tuple[str, int, int]] = []
 | |
|     enabled = False
 | |
|     color = QtGui.QColor(128, 128, 128, 48)
 | |
| 
 | |
|     # These bands correspond broadly to the Danish Amateur Radio allocation
 | |
|     default_bands = ["2200 m;135700;137800",
 | |
|                      "630 m;472000;479000",
 | |
|                      "160 m;1800000;2000000",
 | |
|                      "80 m;3500000;3800000",
 | |
|                      "60 m;5250000;5450000",
 | |
|                      "40 m;7000000;7200000",
 | |
|                      "30 m;10100000;10150000",
 | |
|                      "20 m;14000000;14350000",
 | |
|                      "17 m;18068000;18168000",
 | |
|                      "15 m;21000000;21450000",
 | |
|                      "12 m;24890000;24990000",
 | |
|                      "10 m;28000000;29700000",
 | |
|                      "6 m;50000000;52000000",
 | |
|                      "4 m;69887500;70512500",
 | |
|                      "2 m;144000000;146000000",
 | |
|                      "70 cm;432000000;438000000",
 | |
|                      "23 cm;1240000000;1300000000",
 | |
|                      "13 cm;2320000000;2450000000"]
 | |
| 
 | |
|     def __init__(self):
 | |
|         super().__init__()
 | |
|         self.settings = QtCore.QSettings(QtCore.QSettings.IniFormat,
 | |
|                                          QtCore.QSettings.UserScope,
 | |
|                                          "NanoVNASaver", "Bands")
 | |
|         self.settings.setIniCodec("UTF-8")
 | |
|         self.enabled = self.settings.value("ShowBands", False, bool)
 | |
| 
 | |
|         stored_bands: List[str] = self.settings.value("bands", self.default_bands)
 | |
|         if stored_bands:
 | |
|             for b in stored_bands:
 | |
|                 (name, start, end) = b.split(";")
 | |
|                 self.bands.append((name, int(start), int(end)))
 | |
| 
 | |
|     def saveSettings(self):
 | |
|         stored_bands = []
 | |
|         for b in self.bands:
 | |
|             stored_bands.append(b[0] + ";" + str(b[1]) + ";" + str(b[2]))
 | |
|         self.settings.setValue("bands", stored_bands)
 | |
|         self.settings.sync()
 | |
| 
 | |
|     def resetBands(self):
 | |
|         self.bands = []
 | |
|         for b in self.default_bands:
 | |
|             (name, start, end) = b.split(";")
 | |
|             self.bands.append((name, int(start), int(end)))
 | |
|         self.layoutChanged.emit()
 | |
|         self.saveSettings()
 | |
| 
 | |
|     def columnCount(self, parent: QModelIndex = ...) -> int:
 | |
|         return 3
 | |
| 
 | |
|     def rowCount(self, parent: QModelIndex = ...) -> int:
 | |
|         return len(self.bands)
 | |
| 
 | |
|     def data(self, index: QModelIndex, role: int = ...) -> QtCore.QVariant:
 | |
|         if (role == QtCore.Qt.DisplayRole or
 | |
|                 role == QtCore.Qt.ItemDataRole or role == QtCore.Qt.EditRole):
 | |
|             return QtCore.QVariant(self.bands[index.row()][index.column()])
 | |
|         if role == QtCore.Qt.TextAlignmentRole:
 | |
|             if index.column() == 0:
 | |
|                 return QtCore.QVariant(QtCore.Qt.AlignCenter)
 | |
|             return QtCore.QVariant(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
 | |
|         return QtCore.QVariant()
 | |
| 
 | |
|     def setData(self, index: QModelIndex, value: typing.Any, role: int = ...) -> bool:
 | |
|         if role == QtCore.Qt.EditRole and index.isValid():
 | |
|             t = self.bands[index.row()]
 | |
|             name = t[0]
 | |
|             start = t[1]
 | |
|             end = t[2]
 | |
|             if index.column() == 0:
 | |
|                 name = value
 | |
|             elif index.column() == 1:
 | |
|                 start = value
 | |
|             elif index.column() == 2:
 | |
|                 end = value
 | |
|             self.bands[index.row()] = (name, start, end)
 | |
|             self.dataChanged.emit(index, index)
 | |
|             self.saveSettings()
 | |
|             return True
 | |
|         return False
 | |
| 
 | |
|     def index(self, row: int, column: int, parent: QModelIndex = ...) -> QModelIndex:
 | |
|         return self.createIndex(row, column)
 | |
| 
 | |
|     def addRow(self):
 | |
|         self.bands.append(("New", 0, 0))
 | |
|         self.dataChanged.emit(self.index(len(self.bands), 0), self.index(len(self.bands), 2))
 | |
|         self.layoutChanged.emit()
 | |
| 
 | |
|     def removeRow(self, row: int, parent: QModelIndex = ...) -> bool:
 | |
|         self.bands.remove(self.bands[row])
 | |
|         self.layoutChanged.emit()
 | |
|         self.saveSettings()
 | |
|         return True
 | |
| 
 | |
|     def headerData(self, section: int,
 | |
|                    orientation: QtCore.Qt.Orientation, role: int = ...):
 | |
|         if (role == QtCore.Qt.DisplayRole and
 | |
|                 orientation == QtCore.Qt.Horizontal):
 | |
|             if section == 0:
 | |
|                 return "Band"
 | |
|             if section == 1:
 | |
|                 return "Start (Hz)"
 | |
|             if section == 2:
 | |
|                 return "End (Hz)"
 | |
|             return "Invalid"
 | |
|         super().headerData(section, orientation, role)
 | |
| 
 | |
|     def flags(self, index: QModelIndex) -> QtCore.Qt.ItemFlags:
 | |
|         if index.isValid():
 | |
|             return QtCore.Qt.ItemFlags(
 | |
|                 QtCore.Qt.ItemIsEditable |
 | |
|                 QtCore.Qt.ItemIsEnabled |
 | |
|                 QtCore.Qt.ItemIsSelectable)
 | |
|         super().flags(index)
 | |
| 
 | |
|     def setColor(self, color):
 | |
|         self.color = color
 | |
| 
 | |
| 
 | |
| class Version:
 | |
|     RXP = re.compile(r"""^
 | |
|         \D*
 | |
|         (?P<major>\d+)\.
 | |
|         (?P<minor>\d+)\.
 | |
|         (?P<revision>\d+)
 | |
|         (?P<note>.*)
 | |
|         $""", re.VERBOSE)
 | |
| 
 | |
|     def __init__(self, vstring: str = "0.0.0"):
 | |
|         self.data = {
 | |
|             "major": 0,
 | |
|             "minor": 0,
 | |
|             "revision": 0,
 | |
|             "note": "",
 | |
|         }
 | |
|         try:
 | |
|             self.data = Version.RXP.search(vstring).groupdict()
 | |
|             for name in ("major", "minor", "revision"):
 | |
|                 self.data[name] = int(self.data[name])
 | |
|         except AttributeError:
 | |
|             logger.error("Unable to parse version: %s", vstring)
 | |
| 
 | |
|     def __gt__(self, other: "Version") -> bool:
 | |
|         l, r = self.data, other.data
 | |
|         for name in ("major", "minor", "revision"):
 | |
|             if l[name] > r[name]:
 | |
|                 return True
 | |
|             if l[name] < r[name]:
 | |
|                 return False
 | |
|         return False
 | |
| 
 | |
|     def __lt__(self, other: "Version") -> bool:
 | |
|         return other > self
 | |
| 
 | |
|     def __ge__(self, other: "Version") -> bool:
 | |
|         return self > other or self == other
 | |
| 
 | |
|     def __le__(self, other: "Version") -> bool:
 | |
|         return self < other or self == other
 | |
| 
 | |
|     def __eq__(self, other: "Version") -> bool:
 | |
|         return self.data == other.data
 | |
| 
 | |
|     def __str__(self) -> str:
 | |
|         return (f'{self.data["major"]}.{self.data["minor"]}'
 | |
|                 f'.{self.data["revision"]}{self.data["note"]}')
 | |
| 
 | |
|     @property
 | |
|     def major(self) -> int:
 | |
|         return self.data["major"]
 | |
| 
 | |
|     @property
 | |
|     def minor(self) -> int:
 | |
|         return self.data["minor"]
 | |
| 
 | |
|     @property
 | |
|     def revision(self) -> int:
 | |
|         return self.data["revision"]
 | |
| 
 | |
|     @property
 | |
|     def note(self) -> str:
 | |
|         return self.data["note"]
 |