Source code for bauiv1lib.fileselector

# Released under the MIT License. See LICENSE for details.
#
"""UI functionality for selecting files."""

from __future__ import annotations

import os
import time
import logging
from threading import Thread
from typing import TYPE_CHECKING, override

import bauiv1 as bui

if TYPE_CHECKING:
    from typing import Any, Callable, Sequence


[docs] class FileSelectorWindow(bui.MainWindow): """Window for selecting files.""" def __init__( self, path: str, callback: Callable[[str | None], Any] | None = None, *, show_base_path: bool = True, valid_file_extensions: Sequence[str] | None = None, allow_folders: bool = False, transition: str | None = 'in_right', origin_widget: bui.Widget | None = None, ): if valid_file_extensions is None: valid_file_extensions = [] assert bui.app.classic is not None uiscale = bui.app.ui_v1.uiscale self._width = 850 if uiscale is bui.UIScale.SMALL else 600 self._x_inset = x_inset = 100 if uiscale is bui.UIScale.SMALL else 0 self._height = 365 if uiscale is bui.UIScale.SMALL else 418 self._callback = callback self._base_path = path self._path: str | None = None self._recent_paths: list[str] = [] self._show_base_path = show_base_path self._valid_file_extensions = [ '.' + ext for ext in valid_file_extensions ] self._allow_folders = allow_folders self._subcontainer: bui.Widget | None = None self._subcontainerheight: float | None = None self._scroll_width = self._width - (80 + 2 * x_inset) self._scroll_height = self._height - 170 self._r = 'fileSelectorWindow' super().__init__( root_widget=bui.containerwidget( size=(self._width, self._height), scale=( 1.93 if uiscale is bui.UIScale.SMALL else 1.4 if uiscale is bui.UIScale.MEDIUM else 1.0 ), stack_offset=( (0, -35) if uiscale is bui.UIScale.SMALL else (0, 0) ), ), transition=transition, origin_widget=origin_widget, ) bui.textwidget( parent=self._root_widget, position=(self._width * 0.5, self._height - 42), size=(0, 0), color=bui.app.ui_v1.title_color, h_align='center', v_align='center', text=( bui.Lstr(resource=f'{self._r}.titleFolderText') if (allow_folders and not valid_file_extensions) else ( bui.Lstr(resource=f'{self._r}.titleFileText') if not allow_folders else bui.Lstr(resource=f'{self._r}.titleFileFolderText') ) ), maxwidth=210, ) self._button_width = 146 self._cancel_button = bui.buttonwidget( parent=self._root_widget, position=(35 + x_inset, self._height - 67), autoselect=True, size=(self._button_width, 50), label=bui.Lstr(resource='cancelText'), on_activate_call=self._cancel, ) bui.widget(edit=self._cancel_button, left_widget=self._cancel_button) b_color = (0.6, 0.53, 0.63) self._back_button = bui.buttonwidget( parent=self._root_widget, button_type='square', position=(43 + x_inset, self._height - 113), color=b_color, textcolor=(0.75, 0.7, 0.8), enable_sound=False, size=(55, 35), label=bui.charstr(bui.SpecialChar.LEFT_ARROW), on_activate_call=self._on_back_press, ) self._folder_tex = bui.gettexture('folder') self._folder_color = (1.1, 0.8, 0.2) self._file_tex = bui.gettexture('file') self._file_color = (1, 1, 1) self._use_folder_button: bui.Widget | None = None self._folder_center = self._width * 0.5 + 15 self._folder_icon = bui.imagewidget( parent=self._root_widget, size=(40, 40), position=(40, self._height - 117), texture=self._folder_tex, color=self._folder_color, ) self._path_text = bui.textwidget( parent=self._root_widget, position=(self._folder_center, self._height - 98), size=(0, 0), color=bui.app.ui_v1.title_color, h_align='center', v_align='center', text=self._path, maxwidth=self._width * 0.9, ) self._scrollwidget: bui.Widget | None = None bui.containerwidget( edit=self._root_widget, cancel_button=self._cancel_button ) self._set_path(path)
[docs] @override def get_main_window_state(self) -> bui.MainWindowState: # Support recreating our window for back/refresh purposes. cls = type(self) # Pull everything out of self here. If we do it below in the lambda, # we'll keep self alive which is bad. path = self._base_path callback = self._callback show_base_path = self._show_base_path valid_file_extensions = self._valid_file_extensions allow_folders = self._allow_folders return bui.BasicMainWindowState( create_call=lambda transition, origin_widget: cls( transition=transition, origin_widget=origin_widget, path=path, callback=callback, show_base_path=show_base_path, valid_file_extensions=valid_file_extensions, allow_folders=allow_folders, ) )
def _on_up_press(self) -> None: self._on_entry_activated('..') def _on_back_press(self) -> None: if len(self._recent_paths) > 1: bui.getsound('swish').play() self._recent_paths.pop() self._set_path(self._recent_paths.pop()) else: bui.getsound('error').play() def _on_folder_entry_activated(self) -> None: if self._callback is not None: assert self._path is not None self._callback(self._path) def _on_entry_activated(self, entry: str) -> None: # pylint: disable=too-many-branches new_path = None try: assert self._path is not None if entry == '..': chunks = self._path.split('/') if len(chunks) > 1: new_path = '/'.join(chunks[:-1]) if new_path == '': new_path = '/' else: bui.getsound('error').play() else: if self._path == '/': test_path = self._path + entry else: test_path = self._path + '/' + entry if os.path.isdir(test_path): bui.getsound('swish').play() new_path = test_path elif os.path.isfile(test_path): if self._is_valid_file_path(test_path): bui.getsound('swish').play() if self._callback is not None: self._callback(test_path) else: bui.getsound('error').play() else: print( ( 'Error: FileSelectorWindow found non-file/dir:', test_path, ) ) except Exception: logging.exception( 'Error in FileSelectorWindow._on_entry_activated().' ) if new_path is not None: self._set_path(new_path) class _RefreshThread(Thread): def __init__( self, path: str, callback: Callable[[list[str], str | None], Any] ): super().__init__() self._callback = callback self._path = path @override def run(self) -> None: try: starttime = time.time() files = os.listdir(self._path) duration = time.time() - starttime min_time = 0.1 # Make sure this takes at least 1/10 second so the user # has time to see the selection highlight. if duration < min_time: time.sleep(min_time - duration) bui.pushcall( bui.Call(self._callback, files, None), from_other_thread=True, ) except Exception as exc: # Ignore permission-denied. if 'Errno 13' not in str(exc): logging.exception('Error in fileselector refresh thread.') nofiles: list[str] = [] bui.pushcall( bui.Call(self._callback, nofiles, str(exc)), from_other_thread=True, ) def _set_path(self, path: str, add_to_recent: bool = True) -> None: self._path = path if add_to_recent: self._recent_paths.append(path) self._RefreshThread(path, self._refresh).start() def _refresh(self, file_names: list[str], error: str | None) -> None: # pylint: disable=too-many-statements # pylint: disable=too-many-branches # pylint: disable=too-many-locals if not self._root_widget: return scrollwidget_selected = ( self._scrollwidget is None or self._root_widget.get_selected_child() == self._scrollwidget ) in_top_folder = self._path == self._base_path hide_top_folder = in_top_folder and self._show_base_path is False if hide_top_folder: folder_name = '' elif self._path == '/': folder_name = '/' else: assert self._path is not None folder_name = os.path.basename(self._path) b_color = (0.6, 0.53, 0.63) b_color_disabled = (0.65, 0.65, 0.65) if len(self._recent_paths) < 2: bui.buttonwidget( edit=self._back_button, color=b_color_disabled, textcolor=(0.5, 0.5, 0.5), ) else: bui.buttonwidget( edit=self._back_button, color=b_color, textcolor=(0.75, 0.7, 0.8), ) max_str_width = 300.0 str_width = min( max_str_width, bui.get_string_width(folder_name, suppress_warning=True), ) bui.textwidget( edit=self._path_text, text=folder_name, maxwidth=max_str_width ) bui.imagewidget( edit=self._folder_icon, position=( self._folder_center - str_width * 0.5 - 40, self._height - 117, ), opacity=0.0 if hide_top_folder else 1.0, ) if self._scrollwidget is not None: self._scrollwidget.delete() if self._use_folder_button is not None: self._use_folder_button.delete() bui.widget(edit=self._cancel_button, right_widget=self._back_button) self._scrollwidget = bui.scrollwidget( parent=self._root_widget, position=( (self._width - self._scroll_width) * 0.5, self._height - self._scroll_height - 119, ), size=(self._scroll_width, self._scroll_height), ) if scrollwidget_selected: bui.containerwidget( edit=self._root_widget, selected_child=self._scrollwidget ) # show error case.. if error is not None: self._subcontainer = bui.containerwidget( parent=self._scrollwidget, size=(self._scroll_width, self._scroll_height), background=False, ) bui.textwidget( parent=self._subcontainer, color=(1, 1, 0, 1), text=error, maxwidth=self._scroll_width * 0.9, position=( self._scroll_width * 0.48, self._scroll_height * 0.57, ), size=(0, 0), h_align='center', v_align='center', ) else: file_names = [f for f in file_names if not f.startswith('.')] file_names.sort(key=lambda x: x[0].lower()) entries = file_names entry_height = 35 folder_entry_height = 100 show_folder_entry = False show_use_folder_button = self._allow_folders and not in_top_folder self._subcontainerheight = entry_height * len(entries) + ( folder_entry_height if show_folder_entry else 0 ) v = self._subcontainerheight - ( folder_entry_height if show_folder_entry else 0 ) self._subcontainer = bui.containerwidget( parent=self._scrollwidget, size=(self._scroll_width, self._subcontainerheight), background=False, ) bui.containerwidget( edit=self._scrollwidget, claims_left_right=False, ) bui.containerwidget( edit=self._subcontainer, claims_left_right=False, selection_loops=False, print_list_exit_instructions=False, ) bui.widget(edit=self._subcontainer, up_widget=self._back_button) if show_use_folder_button: self._use_folder_button = btn = bui.buttonwidget( parent=self._root_widget, position=( self._width - self._button_width - 35 - self._x_inset, self._height - 67, ), size=(self._button_width, 50), label=bui.Lstr( resource=f'{self._r}.useThisFolderButtonText' ), on_activate_call=self._on_folder_entry_activated, ) bui.widget( edit=btn, left_widget=self._cancel_button, down_widget=self._scrollwidget, ) bui.widget(edit=self._cancel_button, right_widget=btn) bui.containerwidget(edit=self._root_widget, start_button=btn) folder_icon_size = 35 for num, entry in enumerate(entries): cnt = bui.containerwidget( parent=self._subcontainer, position=(0, v - entry_height), size=(self._scroll_width, entry_height), root_selectable=True, background=False, click_activate=True, on_activate_call=bui.Call(self._on_entry_activated, entry), ) if num == 0: bui.widget(edit=cnt, up_widget=self._back_button) is_valid_file_path = self._is_valid_file_path(entry) assert self._path is not None is_dir = os.path.isdir(self._path + '/' + entry) if is_dir: bui.imagewidget( parent=cnt, size=(folder_icon_size, folder_icon_size), position=( 10, 0.5 * entry_height - folder_icon_size * 0.5, ), draw_controller=cnt, texture=self._folder_tex, color=self._folder_color, ) else: bui.imagewidget( parent=cnt, size=(folder_icon_size, folder_icon_size), position=( 10, 0.5 * entry_height - folder_icon_size * 0.5, ), opacity=1.0 if is_valid_file_path else 0.5, draw_controller=cnt, texture=self._file_tex, color=self._file_color, ) bui.textwidget( parent=cnt, draw_controller=cnt, text=entry, h_align='left', v_align='center', position=(10 + folder_icon_size * 1.05, entry_height * 0.5), size=(0, 0), maxwidth=self._scroll_width * 0.93 - 50, color=( (1, 1, 1, 1) if (is_valid_file_path or is_dir) else (0.5, 0.5, 0.5, 1) ), ) v -= entry_height def _is_valid_file_path(self, path: str) -> bool: return any( path.lower().endswith(ext) for ext in self._valid_file_extensions ) def _cancel(self) -> None: self.main_window_back() if self._callback is not None: self._callback(None)