Rettelser og reset

This commit is contained in:
2026-04-19 11:28:28 +02:00
parent f0a4b4dfa7
commit 80407e98fb
8 changed files with 148 additions and 19 deletions

51
full_reset.sh Normal file
View File

@@ -0,0 +1,51 @@
#!/bin/bash
# ================================================================
# full_reset.sh — KOMPLET nulstilling af LineDance-systemet
#
# Kør dette script på APP-SERVEREN:
# bash full_reset.sh
#
# Herefter skal du selv:
# docker compose down && docker compose up -d --build
# ================================================================
set -e
RED='\033[0;31m'
GREEN='\033[0;32m'
YELLOW='\033[1;33m'
NC='\033[0m'
echo ""
echo -e "${RED}╔══════════════════════════════════════════════════════╗${NC}"
echo -e "${RED}║ KOMPLET NULSTILLING — LINEDANCE AFSPILLER ║${NC}"
echo -e "${RED}║ Sletter ALT: sange, danse, playlister, synk-data ║${NC}"
echo -e "${RED}╚══════════════════════════════════════════════════════╝${NC}"
echo ""
echo -e "${YELLOW}Dette kan IKKE fortrydes. Al data går tabt.${NC}"
echo ""
read -p "Skriv 'SLET ALT' for at fortsætte: " confirm
[ "$confirm" = "SLET ALT" ] || { echo "Afbrudt."; exit 1; }
COMPOSE_DIR="/opt/docker/linedanceafspiller/linedance-api"
# ── MySQL: drop og genskab tom database ───────────────────────
echo ""
echo -e "${YELLOW}▶ Dropper og genskaber MySQL-database...${NC}"
docker compose -f "$COMPOSE_DIR/docker-compose.yml" exec -T db \
mysql -u root -proot << 'MYSQL'
DROP DATABASE IF EXISTS linedance;
CREATE DATABASE linedance CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci;
MYSQL
echo -e "${GREEN} ✓ MySQL klar — tom database oprettet${NC}"
# ── Færdig ────────────────────────────────────────────────────
echo ""
echo -e "${GREEN}╔══════════════════════════════════════════════════════╗${NC}"
echo -e "${GREEN}║ ✓ Server-database nulstillet ║${NC}"
echo -e "${GREEN}╚══════════════════════════════════════════════════════╝${NC}"
echo ""
echo "Gør nu dette:"
echo " 1. Rebuild og genstart Docker:"
echo " cd $COMPOSE_DIR"
echo " docker compose down && docker compose up -d --build"
echo ""

View File

@@ -646,10 +646,11 @@ def get_playlist_with_songs(playlist_id: int) -> dict:
songs = conn.execute("""
SELECT ps.id as ps_id, ps.position, ps.status,
s.*, GROUP_CONCAT(sd.dance_name ORDER BY sd.dance_order) as dances
s.*, GROUP_CONCAT(d.name ORDER BY sd.dance_order) as dances
FROM playlist_songs ps
JOIN songs s ON s.id = ps.song_id
LEFT JOIN song_dances sd ON sd.song_id = s.id
LEFT JOIN dances d ON d.id = sd.dance_id
WHERE ps.playlist_id = ?
GROUP BY ps.id
ORDER BY ps.position

View File

@@ -54,10 +54,10 @@ class SyncManager:
def _run():
try:
payload = self._build_push_payload()
logger.info(f"Push: {len(payload['songs'])} sange, {len(payload['playlists'])} playlister")
logger.info(f"Push OK: {len(payload['songs'])} sange")
result = self._post("/sync/push", payload)
self._save_playlist_ids(result.get("playlist_id_map", {}))
logger.info(f"Push OK: {result}")
logger.info(f"Push OK: {result.get('songs_synced', '?')} sange synkroniseret")
if on_done:
on_done(result)
except Exception as e:
@@ -88,9 +88,7 @@ class SyncManager:
try:
result = self._get("/sync/pull")
pl_count = len(result.get("my_playlists", []))
logger.info(f"Pull: {len(result.get('dances', []))} danse, {pl_count} playlister")
for pl in result.get("my_playlists", []):
logger.info(f" Playliste fra server: '{pl['name']}' ({len(pl.get('songs',[]))} sange)")
logger.info(f"Pull OK: {pl_count} playlister")
self._apply_pull(result)
if on_done:
on_done(result)
@@ -104,16 +102,15 @@ class SyncManager:
"""Push og derefter pull i samme tråd."""
def _run():
try:
logger.info("push_and_pull: bygger payload...")
payload = self._build_push_payload()
logger.info(f"push_and_pull: sender {len(payload['songs'])} sange, {len(payload['playlists'])} playlister")
push_result = self._post("/sync/push", payload)
logger.info(f"push_and_pull: push OK — {push_result}")
pull_result = self._get("/sync/pull")
pl_count = len(pull_result.get("my_playlists", []))
logger.info(f"push_and_pull: pull OK — {pl_count} playlister")
for pl in pull_result.get("my_playlists", []):
logger.info(f" Playliste: '{pl['name']}' ({len(pl.get('songs',[]))} sange)")
logger.info(
f"Sync OK — {len(payload['songs'])} sange, "
f"{len(payload['playlists'])} playlister, "
f"{pl_count} server-playlister"
)
self._apply_pull(pull_result)
if on_done:
on_done({"push": push_result, "pull": pull_result})

View File

@@ -8,7 +8,7 @@ Start:
import sys
import os
APP_VERSION = "0.8.2"
APP_VERSION = "0.8.3"
sys.path.insert(0, os.path.dirname(__file__))

View File

@@ -312,9 +312,14 @@ class LibraryPanel(QWidget):
missing = song.get("file_missing", False)
dance_parts = []
choreos = song.get("dance_choreographers", [])
for i, d in enumerate(dances):
lvl = dance_levels[i] if i < len(dance_levels) else ""
dance_parts.append(f"{d} / {lvl}" if lvl else d)
choreo = choreos[i] if i < len(choreos) else ""
part = f"{d} / {lvl}" if lvl else d
if choreo:
part += f" · {choreo}"
dance_parts.append(part)
dance_str = " · " + " | ".join(dance_parts) if dance_parts else ""
prefix = "" if missing else ""

View File

@@ -144,7 +144,9 @@ class PlaylistManagerDialog(QDialog):
for row in data.get("songs", []):
with get_db() as conn:
dances = conn.execute(
"SELECT dance_name FROM song_dances WHERE song_id=? ORDER BY dance_order",
"""SELECT d.name FROM song_dances sd
JOIN dances d ON d.id = sd.dance_id
WHERE sd.song_id=? ORDER BY sd.dance_order""",
(row["id"],)
).fetchall()
songs.append({
@@ -157,7 +159,7 @@ class PlaylistManagerDialog(QDialog):
"local_path": row.get("local_path", ""),
"file_format": row.get("file_format", ""),
"file_missing": bool(row.get("file_missing", False)),
"dances": [d["dance_name"] for d in dances],
"dances": [d["name"] for d in dances],
})
self.playlist_loaded.emit(pl["name"], songs)
self._load_status.setText(f"✓ Indlæst: {pl['name']} ({len(songs)} sange)")
@@ -277,7 +279,9 @@ class PlaylistManagerDialog(QDialog):
# Hent danse
with get_db() as conn:
dances = conn.execute(
"SELECT dance_name FROM song_dances WHERE song_id=? ORDER BY dance_order",
"""SELECT d.name FROM song_dances sd
JOIN dances d ON d.id = sd.dance_id
WHERE sd.song_id=? ORDER BY sd.dance_order""",
(row["id"],)
).fetchall()
found.append({
@@ -290,7 +294,7 @@ class PlaylistManagerDialog(QDialog):
"local_path": row["local_path"],
"file_format": row["file_format"],
"file_missing": bool(row["file_missing"]),
"dances": [d["dance_name"] for d in dances],
"dances": [d["name"] for d in dances],
})
elif os.path.exists(p) and is_supported(p):
# Filen er ikke scannet endnu — høst tags og tilføj

View File

@@ -120,6 +120,27 @@ class TagEditorDialog(QDialog):
grp = QGroupBox(_("tags.dances"))
layout = QVBoxLayout(grp)
# Kolonneoverskrifter
hdr = QWidget()
hdr_layout = QHBoxLayout(hdr)
hdr_layout.setContentsMargins(0, 0, 0, 0)
hdr_layout.setSpacing(4)
lbl_dans = QLabel("Dans")
lbl_dans.setObjectName("result_count")
lbl_niveau = QLabel("Niveau")
lbl_niveau.setObjectName("result_count")
lbl_niveau.setFixedWidth(130)
lbl_choreo = QLabel("Koreograf")
lbl_choreo.setObjectName("result_count")
lbl_choreo.setFixedWidth(140)
lbl_btn = QLabel("") # plads til knapper
lbl_btn.setFixedWidth(72)
hdr_layout.addWidget(lbl_dans, stretch=2)
hdr_layout.addWidget(lbl_niveau)
hdr_layout.addWidget(lbl_choreo)
hdr_layout.addWidget(lbl_btn)
layout.addWidget(hdr)
# Eksisterende danse
scroll = QScrollArea()
scroll.setWidgetResizable(True)
@@ -175,6 +196,8 @@ class TagEditorDialog(QDialog):
edit = DanceLineEdit("Dans...", self)
edit.setText(name)
edit.setToolTip(name)
edit.textChanged.connect(lambda txt, e=edit: e.setToolTip(txt))
row_layout.addWidget(edit, stretch=2)
# Niveau-dropdown
@@ -195,8 +218,12 @@ class TagEditorDialog(QDialog):
choreo_edit.setText(choreographer)
choreo_edit.setPlaceholderText("Koreograf...")
choreo_edit.setFixedWidth(140)
choreo_edit.setToolTip(choreographer)
choreo_edit.textChanged.connect(
lambda txt, ce=choreo_edit: self._show_choreo_suggestions(txt, ce)
lambda txt, ce=choreo_edit: (
ce.setToolTip(txt),
self._show_choreo_suggestions(txt, ce)
)
)
row_layout.addWidget(choreo_edit)

44
reset_local.sh Normal file
View File

@@ -0,0 +1,44 @@
#!/bin/bash
# ================================================================
# reset_local.sh — Nulstil SQLite på desktop-maskinen
# Kør på den PC hvor LineDance-appen kører
# ================================================================
set -e
GREEN='\033[0;32m'
YELLOW='\033[1;33m'
NC='\033[0m'
echo ""
echo -e "${YELLOW}▶ Finder SQLite-database...${NC}"
SQLITE_DB=""
for candidate in \
"$HOME/.local/share/LineDanceAfspiller/library.db" \
"$HOME/.linedance/library.db" \
"$HOME/AppData/Local/LineDanceAfspiller/library.db"; do
if [ -f "$candidate" ]; then
SQLITE_DB="$candidate"
break
fi
done
if [ -z "$SQLITE_DB" ]; then
FOUND=$(find "$HOME" -name "library.db" 2>/dev/null | head -1)
[ -n "$FOUND" ] && SQLITE_DB="$FOUND"
fi
if [ -n "$SQLITE_DB" ]; then
BACKUP="${SQLITE_DB}.backup_$(date +%Y%m%d_%H%M%S)"
cp "$SQLITE_DB" "$BACKUP"
rm "$SQLITE_DB"
echo -e "${GREEN} ✓ Slettet: $SQLITE_DB${NC}"
echo " Backup gemt: $BACKUP"
else
echo -e "${GREEN} ✓ Ingen SQLite-fil fundet — appen starter allerede frisk${NC}"
fi
echo ""
echo -e "${GREEN}✓ Lokal database nulstillet${NC}"
echo "Start nu LineDance-appen — den genskaber databasen automatisk."
echo ""