recreated json file handling as connector model, fixing #10 and introducing #8 and #7

This commit is contained in:
2025-04-28 00:18:22 +02:00
parent dc27197129
commit 9ea44933a6
3 changed files with 113 additions and 98 deletions

View File

@@ -8,6 +8,7 @@ from tkinter import ttk
import json
from config import Config
from connector import JSONConnector
from windows import SettingsWindow, EditRecord, Window, show_error
@@ -35,13 +36,14 @@ class Application:
self.last_sort_field = "#3"
self.filter_active = tk.BooleanVar(value=False)
# model connector
self.model = JSONConnector()
# init paths to json and csv file
self.json_file_name = "brovski-adress-etiketten-verwaltung.json"
self.csv_file_name = "brovski-adress-etiketten.csv"
self.json_path = ""
self.csv_path = ""
self.load_config()
self.json_file = os.path.join(self.json_path, self.json_file_name)
self.csv_file = os.path.join(self.csv_path, self.csv_file_name)
# status bar content
@@ -71,7 +73,8 @@ class Application:
tk.Button(top_frame, text="Delete", command=self.delete_record, width=button_width).pack(side=tk.LEFT)
tk.Button(top_frame, text="Export CSV", command=self.export_csv, width=button_width).pack(side=tk.LEFT)
tk.Button(top_frame, text="Toggle Aktiv", command=self.toggle_active, width=button_width).pack(side=tk.LEFT)
tk.Checkbutton(top_frame, text="Filter aktive", variable=self.filter_active, command=self.populate_table).pack(side=tk.LEFT)
tk.Checkbutton(top_frame, text="Filter aktive", variable=self.filter_active, command=self.populate_table).pack(
side=tk.LEFT)
tk.Button(top_frame, text="Quit", command=self.on_close, width=button_width).pack(side=tk.RIGHT)
tk.Button(top_frame, text="Settings", command=self.show_settings, width=button_width).pack(side=tk.RIGHT)
@@ -96,17 +99,14 @@ class Application:
# bottom status bar
tk.Label(bottom_frame, textvariable=self.statusbar).pack(side=tk.LEFT)
self._load_json_file()
self.first_sort_after_start()
self.root.mainloop()
def load_config(self):
try:
self.json_path = self.config.get("json", "path")
self.csv_path = self.config.get("csv", "path")
if self.json_path == "" or self.csv_path == "":
if self.csv_path == "":
raise ValueError("Empty JSON or CSV path")
self.config_good = True
except NoSectionError:
@@ -132,55 +132,56 @@ class Application:
settings.wait_window()
def insert_record(self):
values = [
"x",
"Firma",
"Name",
"Strasse",
"Plz/Ort",
]
last_iid = self.table.get_children()[-1]
next_iid = int(last_iid) + 1
self.table.insert('', 'end', iid=next_iid, values=values)
self._save_json_file()
values = {
"aktiv": "x",
"firma": "Firma",
"name": "Name",
"strasse": "Strasse",
"plzort": "Plz/Ort"
}
self.model.create_new(values)
self.populate_table()
self.deselect_tree()
def delete_record(self):
if self.current_record is None:
return
if len(self.table.selection()) > 1:
show_error(message_title="Mehrere Adressen ausgewählt",
message="Es können nur einzelne Adressen gelöscht werden",
parent=self.root)
return
if messagebox.askyesno(
"Eintrag löschen?",
"Willst du diesen Eintrag wirklich löschen?\nDies kann nicht rückgängig gemacht werden"):
self.table.delete(self.current_record)
print(type(self.current_record))
self.model.delete_by_id(self.current_record)
self.deselect_tree()
self._save_json_file()
self.populate_table()
def update_record(self, data: list):
def update_record(self, record: dict):
if self.current_record is None:
return
values = {}
for idx, value in enumerate(data):
values[str(idx)] = value.get()
for key, value in values.items():
self.table.set(self.current_record, key, value)
self._save_json_file()
self.model.update_record(record)
self.populate_table()
def toggle_active(self):
items = self.table.selection()
if len(items) == 0:
selection = self.table.selection()
if len(selection) == 0:
return
for record_id in items:
values = self.table.item(record_id, "values")
active = values[0]
new_active = "x" if active == "" else ""
self.table.set(record_id, "0", new_active)
item_list = [int(x) for x in selection]
for address in self.address_list:
record_id = address.get("record_id")
if record_id in item_list:
address["aktiv"] = "x" if address["aktiv"] == "" else ""
self.model.update_record(address)
self.table.set(record_id, "0", address["aktiv"])
self._save_json_file()
self.update_status_bar()
def deselect_tree(self):
while len(self.table.selection()) > 0:
@@ -188,7 +189,11 @@ class Application:
self.current_record = None
def mouse_click(self, event):
self.current_record = self.table.focus()
id_string = self.table.focus()
if id_string is not None and id_string != "":
self.current_record = int(self.table.focus())
else:
self.current_record = None
region = self.table.identify("region", event.x, event.y)
match region:
case "heading":
@@ -202,75 +207,70 @@ class Application:
def click_on_header(self, event):
column = self.table.identify_column(event.x)
print(f"col: {column}")
if self.last_sort_field == column:
self.sort_order = False if self.sort_order else True
else:
self.sort_order = False
self.last_sort_field = column
self.address_list.sort(key=lambda x: (x[int(column[-1]) - 1]), reverse=self.sort_order)
# special case company sort
if column == "#2":
self.address_list.sort(key=lambda x: (x[1], x[2]), reverse=self.sort_order)
self.populate_table()
match column:
case "#1":
field = "aktiv"
case "#2":
field = "firma"
case "#3":
field = "name"
case "#4":
field = "strasse"
case "#5":
field = "plzort"
case _:
field = "name"
self.address_list = self.model.get_all_sorted_by(field, self.sort_order)
self.populate_table(reload=False)
def click_on_cell(self):
self.current_record = self.table.focus()
values = self.table.item(self.current_record, "values")
self.open_window_edit_records(values)
self.current_record = int(self.table.focus())
self.open_window_edit_records(self.current_record)
def _load_json_file(self):
try:
with open(self.json_file, "r", encoding="utf-8") as f:
self.address_list = json.load(f)
self.address_list.sort(key=lambda x: (x[0], x[1]))
except FileNotFoundError:
show_error(
message_title="Datei nicht gefunden",
message=f"{self.json_file_name} nicht gefunden, erstelle leere Datei unter {self.json_path}",
parent=self.root
)
self.address_list = [["", "firma", "name", "adresse", "plz/ort"]]
with open(self.json_file, "w", encoding="utf-8") as f:
json.dump(self.address_list, f)
self.populate_table()
def _save_json_file(self):
self.export_table_to_address_list()
try:
with open(self.json_file, "w", encoding="utf-8") as f:
json.dump(self.address_list, f, indent=4, sort_keys=True)
except FileNotFoundError:
show_error(
message_title="Unexpected Error: File not found?!",
message=f"{self.json_file_name} not found",
parent=self.root
)
def export_csv(self):
try:
with open(self.csv_file, "w", encoding="utf-8") as f:
writer = csv.writer(f, delimiter=",")
for address in self.address_list:
if address[0] != "x":
if address["aktiv"] != "x":
continue
if address[1] == "":
writer.writerow(address[2:])
else:
writer.writerow(address[1:])
line = []
if address["firma"] != "":
line.append(address["firma"])
line.append(address["name"])
line.append(address["strasse"])
line.append(address["plzort"])
writer.writerow(line)
except FileNotFoundError:
show_error(message_title="Unexpected error",
message=f"Could not write file {self.csv_file}",
parent=self.root
)
def populate_table(self):
def populate_table(self, reload=True):
if reload:
self.address_list = self.model.get_all()
if len(self.address_list) == 0:
return
self.delete_all_table_items()
for index, item in enumerate(self.address_list):
for address in self.address_list:
# skip inactive records if filter is true
if self.filter_active.get() and item[0] != "x":
if self.filter_active.get() and address["aktiv"] != "x":
continue
self.table.insert('', 'end', iid=index, values=item)
print(address)
self.table.insert('', 'end', iid=address["record_id"],
values=(address["aktiv"], address["firma"], address["name"], address["strasse"],
address["plzort"])
)
self.update_status_bar()
def export_table_to_address_list(self):
@@ -292,8 +292,8 @@ class Application:
window.grab_set()
return window
def open_window_edit_records(self, data):
window = EditRecord(self, self.root, data)
def open_window_edit_records(self, record_id: int):
window = EditRecord(self, self.root, record_id)
window.wm_transient(self.root)
window.wait_visibility()
window.grab_set()
@@ -306,12 +306,12 @@ class Application:
self.length_address_list = len(self.address_list)
count = 0
for address in self.address_list:
if address[0] == "x":
if address["aktiv"] == "x":
count += 1
self.length_address_list_active = count
def first_sort_after_start(self):
self.address_list.sort(key=lambda x: (x[1], x[2]))
self.address_list.sort(key=lambda x: (x["firma"], x["name"]))
self.populate_table()