1
0

pattern_model.py 3.3 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192
  1. from PySide6.QtCore import QAbstractListModel, Qt, Slot, Signal
  2. from PySide6.QtQml import QmlElement
  3. from pathlib import Path
  4. QML_IMPORT_NAME = "DuneWeaver"
  5. QML_IMPORT_MAJOR_VERSION = 1
  6. @QmlElement
  7. class PatternModel(QAbstractListModel):
  8. """Model for pattern list with direct file system access"""
  9. NameRole = Qt.UserRole + 1
  10. PathRole = Qt.UserRole + 2
  11. PreviewRole = Qt.UserRole + 3
  12. def __init__(self):
  13. super().__init__()
  14. self._patterns = []
  15. self._filtered_patterns = []
  16. # Look for patterns in the parent directory (main dune-weaver folder)
  17. self.patterns_dir = Path("../patterns")
  18. self.cache_dir = Path("../patterns/cached_images")
  19. self.refresh()
  20. def roleNames(self):
  21. return {
  22. self.NameRole: b"name",
  23. self.PathRole: b"path",
  24. self.PreviewRole: b"preview"
  25. }
  26. def rowCount(self, parent=None):
  27. return len(self._filtered_patterns)
  28. def data(self, index, role):
  29. if not index.isValid() or index.row() >= len(self._filtered_patterns):
  30. return None
  31. pattern = self._filtered_patterns[index.row()]
  32. if role == self.NameRole:
  33. return pattern["name"]
  34. elif role == self.PathRole:
  35. return pattern["path"]
  36. elif role == self.PreviewRole:
  37. # For patterns in subdirectories, check both flattened and hierarchical cache structures
  38. pattern_name = pattern["name"]
  39. # Use PNG format only for kiosk compatibility
  40. # First try hierarchical structure (preserving subdirectories)
  41. preview_path_hierarchical = self.cache_dir / f"{pattern_name}.png"
  42. if preview_path_hierarchical.exists():
  43. return str(preview_path_hierarchical.absolute())
  44. # Then try flattened structure (replace / with _)
  45. preview_name_flat = pattern_name.replace("/", "_").replace("\\", "_")
  46. preview_path_flat = self.cache_dir / f"{preview_name_flat}.png"
  47. if preview_path_flat.exists():
  48. return str(preview_path_flat.absolute())
  49. return ""
  50. return None
  51. @Slot()
  52. def refresh(self):
  53. print(f"Loading patterns from: {self.patterns_dir.absolute()}")
  54. self.beginResetModel()
  55. patterns = []
  56. for file_path in self.patterns_dir.rglob("*.thr"):
  57. relative = file_path.relative_to(self.patterns_dir)
  58. patterns.append({
  59. "name": str(relative),
  60. "path": str(file_path)
  61. })
  62. self._patterns = sorted(patterns, key=lambda x: x["name"])
  63. self._filtered_patterns = self._patterns.copy()
  64. print(f"Loaded {len(self._patterns)} patterns")
  65. self.endResetModel()
  66. @Slot(str)
  67. def filter(self, search_text):
  68. self.beginResetModel()
  69. if not search_text:
  70. self._filtered_patterns = self._patterns.copy()
  71. else:
  72. search_lower = search_text.lower()
  73. self._filtered_patterns = [
  74. p for p in self._patterns
  75. if search_lower in p["name"].lower()
  76. ]
  77. self.endResetModel()