mirror of https://github.com/gmbrax/Pilgrim.git
				
				
				
			Merge pull request #11 from gmbrax/feat/photo-sidebar-tui
Feat/photo sidebar tui
This commit is contained in:
		
						commit
						9ab190741a
					
				|  | @ -1,6 +1,6 @@ | ||||||
| from typing import Any | from typing import Any | ||||||
| 
 | 
 | ||||||
| from sqlalchemy import Column, Integer, String, ForeignKey | from sqlalchemy import Column, Integer, String, ForeignKey, DateTime | ||||||
| from sqlalchemy.orm import relationship | from sqlalchemy.orm import relationship | ||||||
| 
 | 
 | ||||||
| from pilgrim.models.photo_in_entry import photo_entry_association | from pilgrim.models.photo_in_entry import photo_entry_association | ||||||
|  | @ -12,7 +12,7 @@ class Entry(Base): | ||||||
|     id = Column(Integer, primary_key=True) |     id = Column(Integer, primary_key=True) | ||||||
|     title = Column(String) |     title = Column(String) | ||||||
|     text = Column(String) |     text = Column(String) | ||||||
|     date = Column(String) |     date = Column(DateTime) | ||||||
|     photos = relationship( |     photos = relationship( | ||||||
|         "Photo", |         "Photo", | ||||||
|         secondary=photo_entry_association, |         secondary=photo_entry_association, | ||||||
|  |  | ||||||
|  | @ -1,4 +1,6 @@ | ||||||
|  | from datetime import datetime | ||||||
| from typing import List | from typing import List | ||||||
|  | 
 | ||||||
| from ..models.entry import Entry | from ..models.entry import Entry | ||||||
| from ..models.travel_diary import TravelDiary | from ..models.travel_diary import TravelDiary | ||||||
| 
 | 
 | ||||||
|  | @ -7,7 +9,7 @@ class EntryService: | ||||||
|     def __init__(self,session): |     def __init__(self,session): | ||||||
|         self.session = session |         self.session = session | ||||||
| 
 | 
 | ||||||
|     def create(self, travel_diary_id:int, title: str, text: str, date: str, ): |     def create(self, travel_diary_id:int, title: str, text: str, date: datetime, ): | ||||||
|         travel_diary = self.session.query(TravelDiary).filter(TravelDiary.id == travel_diary_id).first() |         travel_diary = self.session.query(TravelDiary).filter(TravelDiary.id == travel_diary_id).first() | ||||||
|         if not travel_diary: |         if not travel_diary: | ||||||
|             return None |             return None | ||||||
|  |  | ||||||
|  | @ -1,4 +1,5 @@ | ||||||
| from pilgrim.service.entry_service import EntryService | from pilgrim.service.entry_service import EntryService | ||||||
|  | from pilgrim.service.photo_service import PhotoService | ||||||
| from pilgrim.service.travel_diary_service import TravelDiaryService | from pilgrim.service.travel_diary_service import TravelDiaryService | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -17,3 +18,7 @@ class ServiceManager: | ||||||
|         if self.session is not None: |         if self.session is not None: | ||||||
|             return TravelDiaryService(self.session) |             return TravelDiaryService(self.session) | ||||||
|         return None |         return None | ||||||
|  |     def get_photo_service(self): | ||||||
|  |         if self.session is not None: | ||||||
|  |             return PhotoService(self.session) | ||||||
|  |         return None | ||||||
|  | @ -1,15 +1,19 @@ | ||||||
| from typing import Optional, List | from typing import Optional, List | ||||||
| import asyncio | import asyncio | ||||||
| from datetime import datetime | from datetime import datetime | ||||||
|  | from pathlib import Path | ||||||
| 
 | 
 | ||||||
| from textual.app import ComposeResult | from textual.app import ComposeResult | ||||||
| from textual.screen import Screen | from textual.screen import Screen | ||||||
| from textual.widgets import Header, Footer, Static, TextArea | from textual.widgets import Header, Footer, Static, TextArea, OptionList, Input, Button | ||||||
| from textual.binding import Binding | from textual.binding import Binding | ||||||
| from textual.containers import Container, Horizontal | from textual.containers import Container, Horizontal, Vertical, ScrollableContainer | ||||||
| 
 | 
 | ||||||
| from pilgrim.models.entry import Entry | from pilgrim.models.entry import Entry | ||||||
| from pilgrim.models.travel_diary import TravelDiary | from pilgrim.models.travel_diary import TravelDiary | ||||||
|  | from pilgrim.models.photo import Photo | ||||||
|  | from pilgrim.ui.screens.modals.add_photo_modal import AddPhotoModal | ||||||
|  | from pilgrim.ui.screens.modals.file_picker_modal import FilePickerModal | ||||||
| from pilgrim.ui.screens.rename_entry_modal import RenameEntryModal | from pilgrim.ui.screens.rename_entry_modal import RenameEntryModal | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -21,13 +25,16 @@ class EditEntryScreen(Screen): | ||||||
|         Binding("ctrl+n", "next_entry", "Next/New Entry"), |         Binding("ctrl+n", "next_entry", "Next/New Entry"), | ||||||
|         Binding("ctrl+b", "prev_entry", "Previous Entry"), |         Binding("ctrl+b", "prev_entry", "Previous Entry"), | ||||||
|         Binding("ctrl+r", "rename_entry", "Rename Entry"), |         Binding("ctrl+r", "rename_entry", "Rename Entry"), | ||||||
|         Binding("escape", "back_to_list", "Back to List") |         Binding("escape", "back_to_list", "Back to List"), | ||||||
|  |         Binding("f8", "toggle_sidebar", "Toggle Sidebar"), | ||||||
|  |         Binding("f9", "toggle_focus", "Focus Sidebar/Editor"), | ||||||
|     ] |     ] | ||||||
| 
 | 
 | ||||||
|     def __init__(self, diary_id: int = 1): |     def __init__(self, diary_id: int = 1): | ||||||
|  |         print("DEBUG: EditEntryScreen INIT") | ||||||
|         super().__init__() |         super().__init__() | ||||||
|         self.diary_id = diary_id |         self.diary_id = diary_id | ||||||
|         self.diary_name = f"Diary {diary_id}"  # Use a better default name |         self.diary_name = f"Diary {diary_id}" | ||||||
|         self.current_entry_index = 0 |         self.current_entry_index = 0 | ||||||
|         self.entries: List[Entry] = [] |         self.entries: List[Entry] = [] | ||||||
|         self.is_new_entry = False |         self.is_new_entry = False | ||||||
|  | @ -38,6 +45,9 @@ class EditEntryScreen(Screen): | ||||||
|         self._updating_display = False |         self._updating_display = False | ||||||
|         self._original_content = "" |         self._original_content = "" | ||||||
|         self.is_refreshing = False |         self.is_refreshing = False | ||||||
|  |         self.sidebar_visible = False | ||||||
|  |         self.sidebar_focused = False | ||||||
|  |         self._sidebar_opened_once = False | ||||||
| 
 | 
 | ||||||
|         # Main header |         # Main header | ||||||
|         self.header = Header(name="Pilgrim v6", classes="EditEntryScreen-header") |         self.header = Header(name="Pilgrim v6", classes="EditEntryScreen-header") | ||||||
|  | @ -60,6 +70,27 @@ class EditEntryScreen(Screen): | ||||||
|         # Text area |         # Text area | ||||||
|         self.text_entry = TextArea(id="text_entry", classes="EditEntryScreen-text-entry") |         self.text_entry = TextArea(id="text_entry", classes="EditEntryScreen-text-entry") | ||||||
| 
 | 
 | ||||||
|  |         # Sidebar widgets | ||||||
|  |         self.sidebar_title = Static("📸 Photos", classes="EditEntryScreen-sidebar-title") | ||||||
|  |         self.photo_list = OptionList(id="photo_list", classes="EditEntryScreen-sidebar-photo-list") | ||||||
|  |         self.photo_info = Static("", classes="EditEntryScreen-sidebar-photo-info") | ||||||
|  |         self.help_text = Static("", classes="EditEntryScreen-sidebar-help") | ||||||
|  | 
 | ||||||
|  |         # Sidebar container: photo list and info in a flexible container, help_text fixed at bottom | ||||||
|  |         self.sidebar_content = Vertical( | ||||||
|  |             self.photo_list, | ||||||
|  |             self.photo_info, | ||||||
|  |             id="sidebar_content", | ||||||
|  |             classes="EditEntryScreen-sidebar-content" | ||||||
|  |         ) | ||||||
|  |         self.sidebar = Vertical( | ||||||
|  |             self.sidebar_title, | ||||||
|  |             self.sidebar_content, | ||||||
|  |             self.help_text,  # Always at the bottom, never scrolls | ||||||
|  |             id="sidebar", | ||||||
|  |             classes="EditEntryScreen-sidebar" | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|         # Main container |         # Main container | ||||||
|         self.main = Container( |         self.main = Container( | ||||||
|             self.sub_header, |             self.sub_header, | ||||||
|  | @ -71,16 +102,32 @@ class EditEntryScreen(Screen): | ||||||
|         # Footer |         # Footer | ||||||
|         self.footer = Footer(classes="EditEntryScreen-footer") |         self.footer = Footer(classes="EditEntryScreen-footer") | ||||||
| 
 | 
 | ||||||
|  |     def _update_footer_context(self): | ||||||
|  |         """Forces footer refresh to show updated bindings""" | ||||||
|  |         self.refresh() | ||||||
|  | 
 | ||||||
|     def compose(self) -> ComposeResult: |     def compose(self) -> ComposeResult: | ||||||
|  |         print("DEBUG: EditEntryScreen COMPOSE", getattr(self, 'sidebar_visible', None)) | ||||||
|         yield self.header |         yield self.header | ||||||
|         yield self.main |         yield Horizontal( | ||||||
|  |             self.main, | ||||||
|  |             self.sidebar, | ||||||
|  |             id="content_container", | ||||||
|  |             classes="EditEntryScreen-content-container" | ||||||
|  |         ) | ||||||
|         yield self.footer |         yield self.footer | ||||||
| 
 | 
 | ||||||
|     def on_mount(self) -> None: |     def on_mount(self) -> None: | ||||||
|         """Called when the screen is mounted""" |         """Called when the screen is mounted""" | ||||||
|  |         self.sidebar.display = False | ||||||
|  |         self.sidebar_visible = False | ||||||
|  |          | ||||||
|         # First update diary info, then refresh entries |         # First update diary info, then refresh entries | ||||||
|         self.update_diary_info() |         self.update_diary_info() | ||||||
|         self.refresh_entries() |         self.refresh_entries() | ||||||
|  |          | ||||||
|  |         # Initialize footer with editor context | ||||||
|  |         self._update_footer_context() | ||||||
| 
 | 
 | ||||||
|     def update_diary_info(self): |     def update_diary_info(self): | ||||||
|         """Updates diary information""" |         """Updates diary information""" | ||||||
|  | @ -93,17 +140,14 @@ class EditEntryScreen(Screen): | ||||||
|                 self.diary_name = diary.name |                 self.diary_name = diary.name | ||||||
|                 self.diary_info.update(f"Diary: {self.diary_name}") |                 self.diary_info.update(f"Diary: {self.diary_name}") | ||||||
|             else: |             else: | ||||||
|                 # If diary not found, try to get a default name |  | ||||||
|                 self.diary_name = f"Diary {self.diary_id}" |                 self.diary_name = f"Diary {self.diary_id}" | ||||||
|                 self.diary_info.update(f"Diary: {self.diary_name}") |                 self.diary_info.update(f"Diary: {self.diary_name}") | ||||||
|                 self.notify(f"Diary {self.diary_id} not found, using default name") |                 self.notify(f"Diary {self.diary_id} not found, using default name") | ||||||
|         except Exception as e: |         except Exception as e: | ||||||
|             # If there's an error, use a default name but don't break the app |  | ||||||
|             self.diary_name = f"Diary {self.diary_id}" |             self.diary_name = f"Diary {self.diary_id}" | ||||||
|             self.diary_info.update(f"Diary: {self.diary_name}") |             self.diary_info.update(f"Diary: {self.diary_name}") | ||||||
|             self.notify(f"Error loading diary info: {str(e)}") |             self.notify(f"Error loading diary info: {str(e)}") | ||||||
|          |          | ||||||
|         # Always ensure the diary info is updated |  | ||||||
|         self._ensure_diary_info_updated() |         self._ensure_diary_info_updated() | ||||||
| 
 | 
 | ||||||
|     def _ensure_diary_info_updated(self): |     def _ensure_diary_info_updated(self): | ||||||
|  | @ -111,7 +155,6 @@ class EditEntryScreen(Screen): | ||||||
|         try: |         try: | ||||||
|             self.diary_info.update(f"Diary: {self.diary_name}") |             self.diary_info.update(f"Diary: {self.diary_name}") | ||||||
|         except Exception as e: |         except Exception as e: | ||||||
|             # If even this fails, at least try to show something |  | ||||||
|             self.diary_info.update(f"Diary: {self.diary_id}") |             self.diary_info.update(f"Diary: {self.diary_id}") | ||||||
| 
 | 
 | ||||||
|     def refresh_entries(self): |     def refresh_entries(self): | ||||||
|  | @ -120,14 +163,10 @@ class EditEntryScreen(Screen): | ||||||
|             service_manager = self.app.service_manager |             service_manager = self.app.service_manager | ||||||
|             entry_service = service_manager.get_entry_service() |             entry_service = service_manager.get_entry_service() | ||||||
| 
 | 
 | ||||||
|             # Get all entries for this diary |  | ||||||
|             all_entries = entry_service.read_all() |             all_entries = entry_service.read_all() | ||||||
|             self.entries = [entry for entry in all_entries if entry.fk_travel_diary_id == self.diary_id] |             self.entries = [entry for entry in all_entries if entry.fk_travel_diary_id == self.diary_id] | ||||||
|              |  | ||||||
|             # Sort by ID |  | ||||||
|             self.entries.sort(key=lambda x: x.id) |             self.entries.sort(key=lambda x: x.id) | ||||||
| 
 | 
 | ||||||
|             # Update next entry ID |  | ||||||
|             if self.entries: |             if self.entries: | ||||||
|                 self.next_entry_id = max(entry.id for entry in self.entries) + 1 |                 self.next_entry_id = max(entry.id for entry in self.entries) + 1 | ||||||
|             else: |             else: | ||||||
|  | @ -139,41 +178,8 @@ class EditEntryScreen(Screen): | ||||||
|         except Exception as e: |         except Exception as e: | ||||||
|             self.notify(f"Error loading entries: {str(e)}") |             self.notify(f"Error loading entries: {str(e)}") | ||||||
|          |          | ||||||
|         # Ensure diary info is updated even if entries fail to load |  | ||||||
|         self._ensure_diary_info_updated() |         self._ensure_diary_info_updated() | ||||||
| 
 | 
 | ||||||
|     async def async_refresh_entries(self): |  | ||||||
|         """Asynchronous version of refresh""" |  | ||||||
|         if self.is_refreshing: |  | ||||||
|             return |  | ||||||
| 
 |  | ||||||
|         self.is_refreshing = True |  | ||||||
| 
 |  | ||||||
|         try: |  | ||||||
|             service_manager = self.app.service_manager |  | ||||||
|             entry_service = service_manager.get_entry_service() |  | ||||||
| 
 |  | ||||||
|             # For now, use synchronous method since mock doesn't have async |  | ||||||
|             all_entries = entry_service.read_all() |  | ||||||
|             self.entries = [entry for entry in all_entries if entry.fk_travel_diary_id == self.diary_id] |  | ||||||
|              |  | ||||||
|             # Sort by ID |  | ||||||
|             self.entries.sort(key=lambda x: x.id) |  | ||||||
| 
 |  | ||||||
|             # Update next entry ID |  | ||||||
|             if self.entries: |  | ||||||
|                 self.next_entry_id = max(entry.id for entry in self.entries) + 1 |  | ||||||
|             else: |  | ||||||
|                 self.next_entry_id = 1 |  | ||||||
| 
 |  | ||||||
|             self._update_entry_display() |  | ||||||
|             self._update_sub_header() |  | ||||||
| 
 |  | ||||||
|         except Exception as e: |  | ||||||
|             self.notify(f"Error loading entries: {str(e)}") |  | ||||||
|         finally: |  | ||||||
|             self.is_refreshing = False |  | ||||||
| 
 |  | ||||||
|     def _update_status_indicator(self, text: str, css_class: str): |     def _update_status_indicator(self, text: str, css_class: str): | ||||||
|         """Helper to update status indicator text and class.""" |         """Helper to update status indicator text and class.""" | ||||||
|         self.status_indicator.update(text) |         self.status_indicator.update(text) | ||||||
|  | @ -211,6 +217,8 @@ class EditEntryScreen(Screen): | ||||||
|         """Finishes the display update by reactivating change detection""" |         """Finishes the display update by reactivating change detection""" | ||||||
|         self._updating_display = False |         self._updating_display = False | ||||||
|         self._update_sub_header() |         self._update_sub_header() | ||||||
|  |         if self.sidebar_visible: | ||||||
|  |             self._update_sidebar_content() | ||||||
| 
 | 
 | ||||||
|     def _update_entry_display(self): |     def _update_entry_display(self): | ||||||
|         """Updates the display of the current entry""" |         """Updates the display of the current entry""" | ||||||
|  | @ -237,6 +245,305 @@ class EditEntryScreen(Screen): | ||||||
| 
 | 
 | ||||||
|         self.call_after_refresh(self._finish_display_update) |         self.call_after_refresh(self._finish_display_update) | ||||||
| 
 | 
 | ||||||
|  |     def _update_sidebar_content(self): | ||||||
|  |         """Updates the sidebar content with photos for the current diary""" | ||||||
|  |         photos = self._load_photos_for_diary(self.diary_id) | ||||||
|  | 
 | ||||||
|  |         # Clear existing options safely | ||||||
|  |         self.photo_list.clear_options() | ||||||
|  | 
 | ||||||
|  |         # Add 'Ingest Photo' option at the top | ||||||
|  |         self.photo_list.add_option("➕ Ingest Photo") | ||||||
|  | 
 | ||||||
|  |         if not photos: | ||||||
|  |             self.photo_info.update("No photos found for this diary") | ||||||
|  |             self.help_text.update("📸 No photos available\n\nUse Photo Manager to add photos") | ||||||
|  |             return | ||||||
|  | 
 | ||||||
|  |         # Add photos to the list | ||||||
|  |         for photo in photos: | ||||||
|  |             self.photo_list.add_option(f"📷 {photo.name}") | ||||||
|  | 
 | ||||||
|  |         self.photo_info.update(f"📸 {len(photos)} photos in diary") | ||||||
|  |          | ||||||
|  |         # English, visually distinct help text | ||||||
|  |         help_text = ( | ||||||
|  |             "[b]⌨️  Sidebar Shortcuts[/b]\n" | ||||||
|  |             "[b][green]i[/green][/b]: Insert photo into entry\n" | ||||||
|  |             "[b][green]n[/green][/b]: Add new photo\n" | ||||||
|  |             "[b][green]d[/green][/b]: Delete selected photo\n" | ||||||
|  |             "[b][green]e[/green][/b]: Edit selected photo\n" | ||||||
|  |             "[b][yellow]Tab[/yellow][/b]: Back to editor\n" | ||||||
|  |             "[b][yellow]F8[/yellow][/b]: Show/hide sidebar\n" | ||||||
|  |             "[b][yellow]F9[/yellow][/b]: Focus Sidebar/Editor" | ||||||
|  |         ) | ||||||
|  |         self.help_text.update(help_text) | ||||||
|  | 
 | ||||||
|  |     def _load_photos_for_diary(self, diary_id: int) -> List[Photo]: | ||||||
|  |         """Loads all photos for the specific diary""" | ||||||
|  |         try: | ||||||
|  |             service_manager = self.app.service_manager | ||||||
|  |             photo_service = service_manager.get_photo_service() | ||||||
|  |              | ||||||
|  |             all_photos = photo_service.read_all() | ||||||
|  |             photos = [photo for photo in all_photos if photo.fk_travel_diary_id == diary_id] | ||||||
|  |             photos.sort(key=lambda x: x.id) | ||||||
|  |             return photos | ||||||
|  |         except Exception as e: | ||||||
|  |             self.notify(f"Error loading photos: {str(e)}") | ||||||
|  |             return [] | ||||||
|  | 
 | ||||||
|  |     def action_toggle_sidebar(self): | ||||||
|  |         """Toggles the sidebar visibility""" | ||||||
|  |         print("DEBUG: TOGGLE SIDEBAR", self.sidebar_visible) | ||||||
|  |         self.sidebar_visible = not self.sidebar_visible | ||||||
|  |          | ||||||
|  |         if self.sidebar_visible: | ||||||
|  |             self.sidebar.display = True | ||||||
|  |             self._update_sidebar_content() | ||||||
|  |             # Notification when opening the sidebar for the first time | ||||||
|  |             if not self._sidebar_opened_once: | ||||||
|  |                 self.notify( | ||||||
|  |                     "Sidebar opened! Context-specific shortcuts are always visible in the sidebar help panel.", | ||||||
|  |                     severity="info" | ||||||
|  |                 ) | ||||||
|  |                 self._sidebar_opened_once = True | ||||||
|  |         else: | ||||||
|  |             self.sidebar.display = False | ||||||
|  |             self.sidebar_focused = False  # Reset focus when hiding | ||||||
|  |             self.text_entry.focus()  # Return focus to editor | ||||||
|  |          | ||||||
|  |         # Update footer after context change | ||||||
|  |         self._update_footer_context() | ||||||
|  |         self.refresh(layout=True) | ||||||
|  | 
 | ||||||
|  |     def action_toggle_focus(self): | ||||||
|  |         """Toggles focus between editor and sidebar""" | ||||||
|  |         print("DEBUG: TOGGLE FOCUS", self.sidebar_visible, self.sidebar_focused) | ||||||
|  |         if not self.sidebar_visible: | ||||||
|  |             # If sidebar is not visible, show it and focus it | ||||||
|  |             self.action_toggle_sidebar() | ||||||
|  |             return | ||||||
|  |          | ||||||
|  |         self.sidebar_focused = not self.sidebar_focused | ||||||
|  |         if self.sidebar_focused: | ||||||
|  |             self.photo_list.focus() | ||||||
|  |         else: | ||||||
|  |             self.text_entry.focus() | ||||||
|  |          | ||||||
|  |         # Update footer after focus change | ||||||
|  |         self._update_footer_context() | ||||||
|  | 
 | ||||||
|  |     def action_insert_photo(self): | ||||||
|  |         """Insert selected photo into text""" | ||||||
|  |         if not self.sidebar_focused or not self.sidebar_visible: | ||||||
|  |             self.notify("Use F9 to focus the sidebar before using this shortcut.", severity="warning") | ||||||
|  |             return | ||||||
|  |              | ||||||
|  |         # Get selected photo | ||||||
|  |         if self.photo_list.highlighted is None: | ||||||
|  |             self.notify("No photo selected", severity="warning") | ||||||
|  |             return | ||||||
|  |              | ||||||
|  |         photos = self._load_photos_for_diary(self.diary_id) | ||||||
|  |         if self.photo_list.highlighted >= len(photos): | ||||||
|  |             return | ||||||
|  |              | ||||||
|  |         selected_photo = photos[self.photo_list.highlighted] | ||||||
|  |          | ||||||
|  |         # Insert photo reference into text | ||||||
|  |         photo_ref = f"\n[📷 {selected_photo.name}]({selected_photo.filepath})\n" | ||||||
|  |         if selected_photo.caption: | ||||||
|  |             photo_ref += f"*{selected_photo.caption}*\n" | ||||||
|  |          | ||||||
|  |         # Insert at cursor position or at end | ||||||
|  |         current_text = self.text_entry.text | ||||||
|  |         cursor_position = len(current_text)  # Insert at end for now | ||||||
|  |         new_text = current_text + photo_ref | ||||||
|  |         self.text_entry.text = new_text | ||||||
|  |          | ||||||
|  |         self.notify(f"Inserted photo: {selected_photo.name}") | ||||||
|  | 
 | ||||||
|  |     def action_ingest_new_photo(self): | ||||||
|  |         """Ingest a new photo using modal""" | ||||||
|  |         if not self.sidebar_focused or not self.sidebar_visible: | ||||||
|  |             self.notify("Use F9 to focus the sidebar before using this shortcut.", severity="warning") | ||||||
|  |             return | ||||||
|  |          | ||||||
|  |         # Open add photo modal | ||||||
|  |         self.app.push_screen( | ||||||
|  |             AddPhotoModal(diary_id=self.diary_id), | ||||||
|  |             self.handle_add_photo_result | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|  |     def handle_add_photo_result(self, result: dict | None) -> None: | ||||||
|  |         """Callback that processes the add photo modal result.""" | ||||||
|  |         if result is None: | ||||||
|  |             self.notify("Add photo cancelled") | ||||||
|  |             return | ||||||
|  | 
 | ||||||
|  |         # Schedule async creation | ||||||
|  |         self.call_later(self._async_create_photo, result) | ||||||
|  | 
 | ||||||
|  |     async def _async_create_photo(self, photo_data: dict): | ||||||
|  |         """Creates a new photo asynchronously""" | ||||||
|  |         try: | ||||||
|  |             service_manager = self.app.service_manager | ||||||
|  |             photo_service = service_manager.get_photo_service() | ||||||
|  | 
 | ||||||
|  |             current_date = datetime.now().strftime("%Y-%m-%d %H:%M:%S") | ||||||
|  |              | ||||||
|  |             new_photo = photo_service.create( | ||||||
|  |                 filepath=Path(photo_data["filepath"]), | ||||||
|  |                 name=photo_data["name"], | ||||||
|  |                 travel_diary_id=self.diary_id, | ||||||
|  |                 addition_date=current_date, | ||||||
|  |                 caption=photo_data["caption"] | ||||||
|  |             ) | ||||||
|  | 
 | ||||||
|  |             if new_photo: | ||||||
|  |                 self.notify(f"Photo '{new_photo.name}' added successfully!") | ||||||
|  |                 # Refresh sidebar content | ||||||
|  |                 if self.sidebar_visible: | ||||||
|  |                     self._update_sidebar_content() | ||||||
|  |             else: | ||||||
|  |                 self.notify("Error creating photo") | ||||||
|  | 
 | ||||||
|  |         except Exception as e: | ||||||
|  |             self.notify(f"Error creating photo: {str(e)}") | ||||||
|  | 
 | ||||||
|  |     def action_delete_photo(self): | ||||||
|  |         """Delete selected photo""" | ||||||
|  |         if not self.sidebar_focused or not self.sidebar_visible: | ||||||
|  |             self.notify("Use F9 to focus the sidebar before using this shortcut.", severity="warning") | ||||||
|  |             return | ||||||
|  |              | ||||||
|  |         if self.photo_list.highlighted is None: | ||||||
|  |             self.notify("No photo selected", severity="warning") | ||||||
|  |             return | ||||||
|  |              | ||||||
|  |         photos = self._load_photos_for_diary(self.diary_id) | ||||||
|  |         if self.photo_list.highlighted >= len(photos): | ||||||
|  |             return | ||||||
|  |              | ||||||
|  |         selected_photo = photos[self.photo_list.highlighted] | ||||||
|  |          | ||||||
|  |         # Confirm deletion | ||||||
|  |         self.notify(f"Deleting photo: {selected_photo.name}") | ||||||
|  |          | ||||||
|  |         # Schedule async deletion | ||||||
|  |         self.call_later(self._async_delete_photo, selected_photo) | ||||||
|  | 
 | ||||||
|  |     async def _async_delete_photo(self, photo: Photo): | ||||||
|  |         """Deletes a photo asynchronously""" | ||||||
|  |         try: | ||||||
|  |             service_manager = self.app.service_manager | ||||||
|  |             photo_service = service_manager.get_photo_service() | ||||||
|  | 
 | ||||||
|  |             result = photo_service.delete(photo) | ||||||
|  | 
 | ||||||
|  |             if result: | ||||||
|  |                 self.notify(f"Photo '{photo.name}' deleted successfully!") | ||||||
|  |                 # Refresh sidebar content | ||||||
|  |                 if self.sidebar_visible: | ||||||
|  |                     self._update_sidebar_content() | ||||||
|  |             else: | ||||||
|  |                 self.notify("Error deleting photo") | ||||||
|  | 
 | ||||||
|  |         except Exception as e: | ||||||
|  |             self.notify(f"Error deleting photo: {str(e)}") | ||||||
|  | 
 | ||||||
|  |     def action_edit_photo(self): | ||||||
|  |         """Edit selected photo using modal""" | ||||||
|  |         if not self.sidebar_focused or not self.sidebar_visible: | ||||||
|  |             self.notify("Use F9 to focus the sidebar before using this shortcut.", severity="warning") | ||||||
|  |             return | ||||||
|  |              | ||||||
|  |         if self.photo_list.highlighted is None: | ||||||
|  |             self.notify("No photo selected", severity="warning") | ||||||
|  |             return | ||||||
|  |              | ||||||
|  |         photos = self._load_photos_for_diary(self.diary_id) | ||||||
|  |         if self.photo_list.highlighted >= len(photos): | ||||||
|  |             return | ||||||
|  |              | ||||||
|  |         selected_photo = photos[self.photo_list.highlighted] | ||||||
|  |          | ||||||
|  |         # Open edit photo modal | ||||||
|  |         self.app.push_screen( | ||||||
|  |             EditPhotoModal(photo=selected_photo), | ||||||
|  |             self.handle_edit_photo_result | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|  |     def handle_edit_photo_result(self, result: dict | None) -> None: | ||||||
|  |         """Callback that processes the edit photo modal result.""" | ||||||
|  |         if result is None: | ||||||
|  |             self.notify("Edit photo cancelled") | ||||||
|  |             return | ||||||
|  | 
 | ||||||
|  |         # Get the selected photo | ||||||
|  |         photos = self._load_photos_for_diary(self.diary_id) | ||||||
|  |         if self.photo_list.highlighted is None or self.photo_list.highlighted >= len(photos): | ||||||
|  |             self.notify("Photo no longer available", severity="error") | ||||||
|  |             return | ||||||
|  |              | ||||||
|  |         selected_photo = photos[self.photo_list.highlighted] | ||||||
|  |          | ||||||
|  |         # Schedule async update | ||||||
|  |         self.call_later(self._async_update_photo, selected_photo, result) | ||||||
|  | 
 | ||||||
|  |     async def _async_update_photo(self, original_photo: Photo, photo_data: dict): | ||||||
|  |         """Updates a photo asynchronously""" | ||||||
|  |         try: | ||||||
|  |             service_manager = self.app.service_manager | ||||||
|  |             photo_service = service_manager.get_photo_service() | ||||||
|  | 
 | ||||||
|  |             # Create updated photo object | ||||||
|  |             updated_photo = Photo( | ||||||
|  |                 filepath=photo_data["filepath"], | ||||||
|  |                 name=photo_data["name"], | ||||||
|  |                 addition_date=original_photo.addition_date, | ||||||
|  |                 caption=photo_data["caption"], | ||||||
|  |                 entries=original_photo.entries, | ||||||
|  |                 id=original_photo.id | ||||||
|  |             ) | ||||||
|  | 
 | ||||||
|  |             result = photo_service.update(original_photo, updated_photo) | ||||||
|  | 
 | ||||||
|  |             if result: | ||||||
|  |                 self.notify(f"Photo '{updated_photo.name}' updated successfully!") | ||||||
|  |                 # Refresh sidebar content | ||||||
|  |                 if self.sidebar_visible: | ||||||
|  |                     self._update_sidebar_content() | ||||||
|  |             else: | ||||||
|  |                 self.notify("Error updating photo") | ||||||
|  | 
 | ||||||
|  |         except Exception as e: | ||||||
|  |             self.notify(f"Error updating photo: {str(e)}") | ||||||
|  | 
 | ||||||
|  |     def on_option_list_option_selected(self, event: OptionList.OptionSelected) -> None: | ||||||
|  |         """Handles photo selection in the sidebar""" | ||||||
|  |         if not self.sidebar_visible: | ||||||
|  |             return | ||||||
|  |         # If 'Ingest Photo' is selected (always index 0) | ||||||
|  |         if event.option_index == 0: | ||||||
|  |             self.action_ingest_new_photo() | ||||||
|  |             return | ||||||
|  |         photos = self._load_photos_for_diary(self.diary_id) | ||||||
|  |         # Adjust index because of 'Ingest Photo' at the top | ||||||
|  |         photo_index = event.option_index - 1 | ||||||
|  |         if not photos or photo_index >= len(photos): | ||||||
|  |             return | ||||||
|  |         selected_photo = photos[photo_index] | ||||||
|  |         self.notify(f"Selected photo: {selected_photo.name}") | ||||||
|  |         # Update photo info with details | ||||||
|  |         photo_details = f"📷 {selected_photo.name}\n" | ||||||
|  |         photo_details += f"📅 {selected_photo.addition_date}\n" | ||||||
|  |         if selected_photo.caption: | ||||||
|  |             photo_details += f"💬 {selected_photo.caption}\n" | ||||||
|  |         photo_details += f"📁 {selected_photo.filepath}" | ||||||
|  |         self.photo_info.update(photo_details) | ||||||
|  | 
 | ||||||
|     def on_text_area_changed(self, event) -> None: |     def on_text_area_changed(self, event) -> None: | ||||||
|         """Detects text changes to mark as unsaved""" |         """Detects text changes to mark as unsaved""" | ||||||
|         if (hasattr(self, 'text_entry') and not self.text_entry.read_only and |         if (hasattr(self, 'text_entry') and not self.text_entry.read_only and | ||||||
|  | @ -251,6 +558,17 @@ class EditEntryScreen(Screen): | ||||||
|                     self.has_unsaved_changes = False |                     self.has_unsaved_changes = False | ||||||
|                     self._update_sub_header() |                     self._update_sub_header() | ||||||
| 
 | 
 | ||||||
|  |     def on_focus(self, event) -> None: | ||||||
|  |         """Captures focus changes to update footer""" | ||||||
|  |         # Check if focus changed to/from sidebar | ||||||
|  |         if hasattr(event.widget, 'id'): | ||||||
|  |             if event.widget.id == "photo_list": | ||||||
|  |                 self.sidebar_focused = True | ||||||
|  |                 self._update_footer_context() | ||||||
|  |             elif event.widget.id == "text_entry": | ||||||
|  |                 self.sidebar_focused = False | ||||||
|  |                 self._update_footer_context() | ||||||
|  | 
 | ||||||
|     def action_back_to_list(self) -> None: |     def action_back_to_list(self) -> None: | ||||||
|         """Goes back to the diary list""" |         """Goes back to the diary list""" | ||||||
|         if self.is_new_entry and not self.text_entry.text.strip() and not self.has_unsaved_changes: |         if self.is_new_entry and not self.text_entry.text.strip() and not self.has_unsaved_changes: | ||||||
|  | @ -375,8 +693,7 @@ class EditEntryScreen(Screen): | ||||||
|             service_manager = self.app.service_manager |             service_manager = self.app.service_manager | ||||||
|             entry_service = service_manager.get_entry_service() |             entry_service = service_manager.get_entry_service() | ||||||
| 
 | 
 | ||||||
|             # Get current date |             current_date = datetime.now() | ||||||
|             current_date = datetime.now().strftime("%d/%m/%Y") |  | ||||||
| 
 | 
 | ||||||
|             new_entry = entry_service.create( |             new_entry = entry_service.create( | ||||||
|                 travel_diary_id=self.diary_id, |                 travel_diary_id=self.diary_id, | ||||||
|  | @ -389,7 +706,6 @@ class EditEntryScreen(Screen): | ||||||
|                 self.entries.append(new_entry) |                 self.entries.append(new_entry) | ||||||
|                 self.entries.sort(key=lambda x: x.id) |                 self.entries.sort(key=lambda x: x.id) | ||||||
| 
 | 
 | ||||||
|                 # Find the new entry index |  | ||||||
|                 for i, entry in enumerate(self.entries): |                 for i, entry in enumerate(self.entries): | ||||||
|                     if entry.id == new_entry.id: |                     if entry.id == new_entry.id: | ||||||
|                         self.current_entry_index = i |                         self.current_entry_index = i | ||||||
|  | @ -419,7 +735,6 @@ class EditEntryScreen(Screen): | ||||||
|             current_entry = self.entries[self.current_entry_index] |             current_entry = self.entries[self.current_entry_index] | ||||||
|             updated_content = self.text_entry.text |             updated_content = self.text_entry.text | ||||||
| 
 | 
 | ||||||
|             # Create updated entry object |  | ||||||
|             updated_entry = Entry( |             updated_entry = Entry( | ||||||
|                 title=current_entry.title, |                 title=current_entry.title, | ||||||
|                 text=updated_content, |                 text=updated_content, | ||||||
|  | @ -445,8 +760,44 @@ class EditEntryScreen(Screen): | ||||||
|         except Exception as e: |         except Exception as e: | ||||||
|             self.notify(f"Error updating entry: {str(e)}") |             self.notify(f"Error updating entry: {str(e)}") | ||||||
| 
 | 
 | ||||||
|     def action_force_refresh(self): |     def on_key(self, event): | ||||||
|         """Forces manual refresh""" |         # Sidebar contextual shortcuts | ||||||
|         self.notify("Forcing refresh...") |         if self.sidebar_focused and self.sidebar_visible: | ||||||
|         self.refresh_entries() |             if event.key == "i": | ||||||
|         self.call_later(self.async_refresh_entries)  |                 self.action_insert_photo() | ||||||
|  |                 event.stop() | ||||||
|  |             elif event.key == "n": | ||||||
|  |                 self.action_ingest_new_photo() | ||||||
|  |                 event.stop() | ||||||
|  |             elif event.key == "d": | ||||||
|  |                 self.action_delete_photo() | ||||||
|  |                 event.stop() | ||||||
|  |             elif event.key == "e": | ||||||
|  |                 self.action_edit_photo() | ||||||
|  |                 event.stop() | ||||||
|  |         # Shift+Tab: remove indent | ||||||
|  |         elif self.focused is self.text_entry and event.key == "shift+tab": | ||||||
|  |             textarea = self.text_entry | ||||||
|  |             row, col = textarea.cursor_location | ||||||
|  |             lines = textarea.text.splitlines() | ||||||
|  |             if row < len(lines): | ||||||
|  |                 line = lines[row] | ||||||
|  |                 if line.startswith('\t'): | ||||||
|  |                     lines[row] = line[1:] | ||||||
|  |                     textarea.text = '\n'.join(lines) | ||||||
|  |                     textarea.cursor_location = (row, max(col - 1, 0)) | ||||||
|  |                 elif line.startswith('    '):  # 4 spaces | ||||||
|  |                     lines[row] = line[4:] | ||||||
|  |                     textarea.text = '\n'.join(lines) | ||||||
|  |                     textarea.cursor_location = (row, max(col - 4, 0)) | ||||||
|  |                 elif line.startswith(' '): | ||||||
|  |                     n = len(line) - len(line.lstrip(' ')) | ||||||
|  |                     to_remove = min(n, 4) | ||||||
|  |                     lines[row] = line[to_remove:] | ||||||
|  |                     textarea.text = '\n'.join(lines) | ||||||
|  |                     textarea.cursor_location = (row, max(col - to_remove, 0)) | ||||||
|  |             event.stop() | ||||||
|  |         # Tab: insert tab | ||||||
|  |         elif self.focused is self.text_entry and event.key == "tab": | ||||||
|  |             self.text_entry.insert('\t') | ||||||
|  |             event.stop()  | ||||||
|  | @ -0,0 +1,61 @@ | ||||||
|  | import os | ||||||
|  | from textual.app import ComposeResult | ||||||
|  | from textual.screen import Screen | ||||||
|  | from textual.widgets import Static, Input, Button | ||||||
|  | from textual.containers import Horizontal, Container | ||||||
|  | from .file_picker_modal import FilePickerModal | ||||||
|  | 
 | ||||||
|  | class AddPhotoModal(Screen): | ||||||
|  |     """Modal for adding a new photo""" | ||||||
|  |     def __init__(self, diary_id: int): | ||||||
|  |         super().__init__() | ||||||
|  |         self.diary_id = diary_id | ||||||
|  |         self.result = None | ||||||
|  | 
 | ||||||
|  |     def compose(self) -> ComposeResult: | ||||||
|  |         yield Container( | ||||||
|  |             Static("📷 Add New Photo", classes="AddPhotoModal-Title"), | ||||||
|  |             Static("File path:", classes="AddPhotoModal-Label"), | ||||||
|  |             Horizontal( | ||||||
|  |                 Input(placeholder="Enter file path...", id="filepath-input", classes="AddPhotoModal-Input"), | ||||||
|  |                 Button("Escolher arquivo...", id="choose-file-button", classes="AddPhotoModal-Button"), | ||||||
|  |                 classes="AddPhotoModal-FileRow" | ||||||
|  |             ), | ||||||
|  |             Static("Photo name:", classes="AddPhotoModal-Label"), | ||||||
|  |             Input(placeholder="Enter photo name...", id="name-input", classes="AddPhotoModal-Input"), | ||||||
|  |             Static("Caption (optional):", classes="AddPhotoModal-Label"), | ||||||
|  |             Input(placeholder="Enter caption...", id="caption-input", classes="AddPhotoModal-Input"), | ||||||
|  |             Horizontal( | ||||||
|  |                 Button("Add Photo", id="add-button", classes="AddPhotoModal-Button"), | ||||||
|  |                 Button("Cancel", id="cancel-button", classes="AddPhotoModal-Button"), | ||||||
|  |                 classes="AddPhotoModal-Buttons" | ||||||
|  |             ), | ||||||
|  |             classes="AddPhotoModal-Dialog" | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|  |     def on_button_pressed(self, event: Button.Pressed) -> None: | ||||||
|  |         if event.button.id == "choose-file-button": | ||||||
|  |             self.app.push_screen( | ||||||
|  |                 FilePickerModal(), | ||||||
|  |                 self.handle_file_picker_result | ||||||
|  |             ) | ||||||
|  |             return | ||||||
|  |         if event.button.id == "add-button": | ||||||
|  |             filepath = self.query_one("#filepath-input", Input).value | ||||||
|  |             name = self.query_one("#name-input", Input).value | ||||||
|  |             caption = self.query_one("#caption-input", Input).value | ||||||
|  |             if not filepath.strip() or not name.strip(): | ||||||
|  |                 self.notify("File path and name are required", severity="error") | ||||||
|  |                 return | ||||||
|  |             self.result = { | ||||||
|  |                 "filepath": filepath.strip(), | ||||||
|  |                 "name": name.strip(), | ||||||
|  |                 "caption": caption.strip() if caption.strip() else None | ||||||
|  |             } | ||||||
|  |             self.dismiss() | ||||||
|  |         elif event.button.id == "cancel-button": | ||||||
|  |             self.dismiss() | ||||||
|  | 
 | ||||||
|  |     def handle_file_picker_result(self, result: str | None) -> None: | ||||||
|  |         if result: | ||||||
|  |             self.query_one("#filepath-input", Input).value = result  | ||||||
|  | @ -0,0 +1,66 @@ | ||||||
|  | import os | ||||||
|  | from pathlib import Path | ||||||
|  | from typing import Iterable | ||||||
|  | from textual.app import ComposeResult | ||||||
|  | from textual.screen import Screen | ||||||
|  | from textual.widgets import Static, DirectoryTree, Button | ||||||
|  | from textual.containers import Horizontal, Container | ||||||
|  | 
 | ||||||
|  | class ImageDirectoryTree(DirectoryTree): | ||||||
|  |     """DirectoryTree that only shows image files""" | ||||||
|  |      | ||||||
|  |     def filter_paths(self, paths: Iterable[Path]) -> Iterable[Path]: | ||||||
|  |         """Filter to show only directories and image files""" | ||||||
|  |         image_extensions = {'.png', '.jpg', '.jpeg', '.gif', '.bmp', '.webp'} | ||||||
|  |         return [ | ||||||
|  |             path for path in paths  | ||||||
|  |             if path.is_dir() or path.suffix.lower() in image_extensions | ||||||
|  |         ] | ||||||
|  | 
 | ||||||
|  | class FilePickerModal(Screen): | ||||||
|  |     """Modal for picking an image file using DirectoryTree""" | ||||||
|  |      | ||||||
|  |     def __init__(self, start_path=None): | ||||||
|  |         super().__init__() | ||||||
|  |         self.start_path = Path(start_path or os.getcwd()) | ||||||
|  |         # Start one level up to make navigation easier | ||||||
|  |         self.current_path = self.start_path.parent | ||||||
|  |         self.result = None | ||||||
|  | 
 | ||||||
|  |     def compose(self) -> ComposeResult: | ||||||
|  |         yield Container( | ||||||
|  |             Static(f"Current: {self.current_path}", id="title", classes="FilePickerModal-Title"), | ||||||
|  |             ImageDirectoryTree(str(self.current_path), id="directory-tree"), | ||||||
|  |             Horizontal( | ||||||
|  |                 Button("Up", id="up-button", classes="FilePickerModal-Button"), | ||||||
|  |                 Button("Cancel", id="cancel-button", classes="FilePickerModal-Button"), | ||||||
|  |                 classes="FilePickerModal-Buttons" | ||||||
|  |             ), | ||||||
|  |             classes="FilePickerModal-Dialog" | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|  |     def on_directory_tree_file_selected(self, event: DirectoryTree.FileSelected) -> None: | ||||||
|  |         """Handle file selection""" | ||||||
|  |         file_path = event.path | ||||||
|  |         # Check if it's an image file | ||||||
|  |         image_extensions = {'.png', '.jpg', '.jpeg', '.gif', '.bmp', '.webp'} | ||||||
|  |         if file_path.suffix.lower() in image_extensions: | ||||||
|  |             self.result = str(file_path) | ||||||
|  |             self.dismiss() | ||||||
|  |         else: | ||||||
|  |             self.notify("Please select an image file", severity="warning") | ||||||
|  | 
 | ||||||
|  |     def on_button_pressed(self, event: Button.Pressed) -> None: | ||||||
|  |         """Handle button presses""" | ||||||
|  |         if event.button.id == "up-button": | ||||||
|  |             # Navigate to parent directory | ||||||
|  |             parent = self.current_path.parent | ||||||
|  |             if parent != self.current_path: | ||||||
|  |                 self.current_path = parent | ||||||
|  |                 self.query_one("#title", Static).update(f"Current: {self.current_path}") | ||||||
|  |                 # Reload the directory tree | ||||||
|  |                 tree = self.query_one("#directory-tree", ImageDirectoryTree) | ||||||
|  |                 tree.path = str(self.current_path) | ||||||
|  |                 tree.reload() | ||||||
|  |         elif event.button.id == "cancel-button": | ||||||
|  |             self.dismiss()  | ||||||
|  | @ -387,4 +387,205 @@ Screen.-modal { | ||||||
| .RenameEntryModal-cancel-button { | .RenameEntryModal-cancel-button { | ||||||
|     margin: 0 1; |     margin: 0 1; | ||||||
|     width: 1fr; |     width: 1fr; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .EditEntryScreen-sidebar { | ||||||
|  |     width: 40; | ||||||
|  |     min-height: 10; | ||||||
|  |     border-left: solid green; | ||||||
|  |     padding: 1; | ||||||
|  |     background: $surface-darken-2; | ||||||
|  |     color: $primary; | ||||||
|  |     text-style: bold; | ||||||
|  |     content-align: left top; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .EditEntryScreen-content-container { | ||||||
|  |     layout: horizontal; | ||||||
|  |     height: 1fr; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .EditEntryScreen-sidebar-title { | ||||||
|  |     text-align: center; | ||||||
|  |     text-style: bold; | ||||||
|  |     color: $accent; | ||||||
|  |     padding: 1; | ||||||
|  |     border-bottom: solid $accent; | ||||||
|  |     margin-bottom: 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .EditEntryScreen-sidebar-content { | ||||||
|  |     height: 1fr; | ||||||
|  |     layout: vertical; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .EditEntryScreen-sidebar-photo-list { | ||||||
|  |     height: 1fr; | ||||||
|  |     border: solid $accent; | ||||||
|  |     margin-bottom: 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .EditEntryScreen-sidebar-photo-info { | ||||||
|  |     height: auto; | ||||||
|  |     min-height: 3; | ||||||
|  |     border: solid $warning; | ||||||
|  |     padding: 1; | ||||||
|  |     margin-bottom: 1; | ||||||
|  |     background: $surface-darken-1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .EditEntryScreen-sidebar-help { | ||||||
|  |     height: auto; | ||||||
|  |     min-height: 8; | ||||||
|  |     border: solid $success; | ||||||
|  |     padding: 1; | ||||||
|  |     background: $surface-darken-1; | ||||||
|  |     text-style: italic; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Photo Modal Styles */ | ||||||
|  | .modal-dialog { | ||||||
|  |     layout: vertical; | ||||||
|  |     width: 60%; | ||||||
|  |     height: auto; | ||||||
|  |     background: $surface; | ||||||
|  |     border: thick $accent; | ||||||
|  |     padding: 2 4; | ||||||
|  |     align: center middle; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .modal-title { | ||||||
|  |     text-align: center; | ||||||
|  |     text-style: bold; | ||||||
|  |     color: $primary; | ||||||
|  |     margin-bottom: 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .modal-label { | ||||||
|  |     margin-bottom: 1; | ||||||
|  |     color: $text; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .modal-input { | ||||||
|  |     width: 1fr; | ||||||
|  |     margin-bottom: 2; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .modal-buttons { | ||||||
|  |     width: 1fr; | ||||||
|  |     height: auto; | ||||||
|  |     align: center middle; | ||||||
|  |     padding-top: 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .modal-button { | ||||||
|  |     margin: 0 1; | ||||||
|  |     width: 1fr; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* AddPhotoModal styles */ | ||||||
|  | .AddPhotoModal-Dialog { | ||||||
|  |     layout: vertical; | ||||||
|  |     width: 60%; | ||||||
|  |     height: auto; | ||||||
|  |     background: $surface; | ||||||
|  |     border: thick $accent; | ||||||
|  |     padding: 2 4; | ||||||
|  |     align: center middle; | ||||||
|  | } | ||||||
|  | .AddPhotoModal-Title { | ||||||
|  |     text-align: center; | ||||||
|  |     text-style: bold; | ||||||
|  |     color: $primary; | ||||||
|  |     margin-bottom: 1; | ||||||
|  | } | ||||||
|  | .AddPhotoModal-Label { | ||||||
|  |     margin-bottom: 1; | ||||||
|  |     color: $text; | ||||||
|  | } | ||||||
|  | .AddPhotoModal-Input { | ||||||
|  |     width: 1fr; | ||||||
|  |     margin-bottom: 2; | ||||||
|  | } | ||||||
|  | .AddPhotoModal-Buttons { | ||||||
|  |     width: 1fr; | ||||||
|  |     height: auto; | ||||||
|  |     align: center middle; | ||||||
|  |     padding-top: 1; | ||||||
|  | } | ||||||
|  | .AddPhotoModal-Button { | ||||||
|  |     margin: 0 1; | ||||||
|  |     width: 1fr; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* EditPhotoModal styles */ | ||||||
|  | .EditPhotoModal-Dialog { | ||||||
|  |     layout: vertical; | ||||||
|  |     width: 60%; | ||||||
|  |     height: auto; | ||||||
|  |     background: $surface; | ||||||
|  |     border: thick $accent; | ||||||
|  |     padding: 2 4; | ||||||
|  |     align: center middle; | ||||||
|  | } | ||||||
|  | .EditPhotoModal-Title { | ||||||
|  |     text-align: center; | ||||||
|  |     text-style: bold; | ||||||
|  |     color: $primary; | ||||||
|  |     margin-bottom: 1; | ||||||
|  | } | ||||||
|  | .EditPhotoModal-Label { | ||||||
|  |     margin-bottom: 1; | ||||||
|  |     color: $text; | ||||||
|  | } | ||||||
|  | .EditPhotoModal-Input { | ||||||
|  |     width: 1fr; | ||||||
|  |     margin-bottom: 2; | ||||||
|  | } | ||||||
|  | .EditPhotoModal-Buttons { | ||||||
|  |     width: 1fr; | ||||||
|  |     height: auto; | ||||||
|  |     align: center middle; | ||||||
|  |     padding-top: 1; | ||||||
|  | } | ||||||
|  | .EditPhotoModal-Button { | ||||||
|  |     margin: 0 1; | ||||||
|  |     width: 1fr; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* FilePickerModal styles */ | ||||||
|  | .FilePickerModal-Dialog { | ||||||
|  |     layout: vertical; | ||||||
|  |     width: 80%; | ||||||
|  |     height: 80%; | ||||||
|  |     background: $surface; | ||||||
|  |     border: thick $accent; | ||||||
|  |     padding: 2 4; | ||||||
|  |     align: center middle; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .FilePickerModal-Title { | ||||||
|  |     text-align: center; | ||||||
|  |     text-style: bold; | ||||||
|  |     color: $primary; | ||||||
|  |     margin-bottom: 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .FilePickerModal-Buttons { | ||||||
|  |     width: 1fr; | ||||||
|  |     height: auto; | ||||||
|  |     align: center middle; | ||||||
|  |     padding-top: 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .FilePickerModal-Button { | ||||||
|  |     margin: 0 1; | ||||||
|  |     width: 1fr; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* DirectoryTree specific styles */ | ||||||
|  | #directory-tree { | ||||||
|  |     height: 1fr; | ||||||
|  |     border: solid $accent; | ||||||
|  |     margin: 1; | ||||||
| } | } | ||||||
		Loading…
	
		Reference in New Issue