Use new PCN format for settings config file
This commit is contained in:
		
							
								
								
									
										1
									
								
								TODO.md
									
									
									
									
									
								
							
							
						
						
									
										1
									
								
								TODO.md
									
									
									
									
									
								
							| @@ -1,5 +1,6 @@ | |||||||
| # TODO | # TODO | ||||||
|  |  | ||||||
|  | - login page password spinner | ||||||
| - Encrypted rooms don't show invites in member list after Mirage restart | - Encrypted rooms don't show invites in member list after Mirage restart | ||||||
| - Room display name not updated when someone removes theirs | - Room display name not updated when someone removes theirs | ||||||
| - Fix right margin of own `<image url>\n<image url>` messages | - Fix right margin of own `<image url>\n<image url>` messages | ||||||
|   | |||||||
| @@ -109,7 +109,7 @@ class Backend: | |||||||
|         self.settings       = Settings(self) |         self.settings       = Settings(self) | ||||||
|         self.ui_state       = UIState(self) |         self.ui_state       = UIState(self) | ||||||
|         self.history        = History(self) |         self.history        = History(self) | ||||||
|         self.theme          = Theme(self, self.settings["theme"]) |         self.theme          = Theme(self, self.settings.General.theme) | ||||||
|  |  | ||||||
|         self.clients: Dict[str, MatrixClient] = {} |         self.clients: Dict[str, MatrixClient] = {} | ||||||
|  |  | ||||||
| @@ -427,13 +427,13 @@ class Backend: | |||||||
|         return path |         return path | ||||||
|  |  | ||||||
|  |  | ||||||
|     async def get_settings(self) -> Tuple[Settings, UIState, History, str]: |     async def get_settings(self) -> Tuple[dict, UIState, History, str]: | ||||||
|         """Return parsed user config files for QML.""" |         """Return parsed user config files for QML.""" | ||||||
|         return ( |         return ( | ||||||
|             self.settings.data, |             self.settings.qml_data, | ||||||
|             self.ui_state.data, |             self.ui_state.qml_data, | ||||||
|             self.history.data, |             self.history.qml_data, | ||||||
|             self.theme.data, |             self.theme.qml_data, | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -493,7 +493,7 @@ class MatrixClient(nio.AsyncClient): | |||||||
|  |  | ||||||
|             utils.dict_update_recursive(first, self.low_limit_filter) |             utils.dict_update_recursive(first, self.low_limit_filter) | ||||||
|  |  | ||||||
|             if self.backend.settings["hideUnknownEvents"]: |             if not self.backend.settings.Chat.show_unknown_events: | ||||||
|                 first["room"]["timeline"]["not_types"].extend( |                 first["room"]["timeline"]["not_types"].extend( | ||||||
|                     self.no_unknown_events_filter |                     self.no_unknown_events_filter | ||||||
|                     ["room"]["timeline"]["not_types"], |                     ["room"]["timeline"]["not_types"], | ||||||
| @@ -1146,14 +1146,22 @@ class MatrixClient(nio.AsyncClient): | |||||||
|         room = self.models[self.user_id, "rooms"][room_id] |         room = self.models[self.user_id, "rooms"][room_id] | ||||||
|         room.bookmarked = not room.bookmarked |         room.bookmarked = not room.bookmarked | ||||||
|  |  | ||||||
|         settings = self.backend.ui_settings |         settings       = self.backend.settings | ||||||
|         bookmarks = settings["roomBookmarkIDs"].setdefault(self.user_id, []) |         bookmarks      = settings.RoomList.bookmarks | ||||||
|         if room.bookmarked and room_id not in bookmarks: |         user_bookmarks = bookmarks.setdefault(self.user_id, []) | ||||||
|             bookmarks.append(room_id) |  | ||||||
|         elif not room.bookmarked and room_id in bookmarks: |  | ||||||
|             bookmarks.remove(room_id) |  | ||||||
|  |  | ||||||
|         await self.backend.ui_settings.write(self.backend.ui_settings._data) |         if room.bookmarked and room_id not in user_bookmarks: | ||||||
|  |             user_bookmarks.append(room_id) | ||||||
|  |  | ||||||
|  |         while not room.bookmarked and room_id in user_bookmarks: | ||||||
|  |             user_bookmarks.remove(room_id) | ||||||
|  |  | ||||||
|  |         # Changes inside dicts/lists aren't monitored, need to reassign | ||||||
|  |         settings.RoomList.bookmarks = { | ||||||
|  |             **bookmarks, self.user_id: user_bookmarks, | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         self.backend.settings.save() | ||||||
|  |  | ||||||
|     async def room_forget(self, room_id: str) -> None: |     async def room_forget(self, room_id: str) -> None: | ||||||
|         """Leave a joined room (or decline an invite) and forget its history. |         """Leave a joined room (or decline an invite) and forget its history. | ||||||
| @@ -1866,7 +1874,7 @@ class MatrixClient(nio.AsyncClient): | |||||||
|             ) |             ) | ||||||
|             unverified_devices = registered.unverified_devices |             unverified_devices = registered.unverified_devices | ||||||
|  |  | ||||||
|         bookmarks = self.backend.ui_settings["roomBookmarkIDs"] |         bookmarks = self.backend.settings.RoomList.bookmarks | ||||||
|         room_item = Room( |         room_item = Room( | ||||||
|             id             = room.room_id, |             id             = room.room_id, | ||||||
|             for_account    = self.user_id, |             for_account    = self.user_id, | ||||||
| @@ -1912,7 +1920,7 @@ class MatrixClient(nio.AsyncClient): | |||||||
|             local_unreads    = local_unreads, |             local_unreads    = local_unreads, | ||||||
|             local_highlights = local_highlights, |             local_highlights = local_highlights, | ||||||
|  |  | ||||||
|             lexical_sorting = self.backend.settings["lexicalRoomSorting"], |             lexical_sorting = self.backend.settings.RoomList.lexical_sort, | ||||||
|             bookmarked = room.room_id in bookmarks.get(self.user_id, {}), |             bookmarked = room.room_id in bookmarks.get(self.user_id, {}), | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -498,7 +498,7 @@ class NioCallbacks: | |||||||
|  |  | ||||||
|         # Membership changes |         # Membership changes | ||||||
|         if not prev or membership != prev_membership: |         if not prev or membership != prev_membership: | ||||||
|             if self.client.backend.settings["hideMembershipEvents"]: |             if not self.client.backend.settings.Chat.show_membership_events: | ||||||
|                 return None |                 return None | ||||||
|  |  | ||||||
|             reason = escape( |             reason = escape( | ||||||
| @@ -564,7 +564,7 @@ class NioCallbacks: | |||||||
|                         avatar_url      = now.get("avatar_url") or "", |                         avatar_url      = now.get("avatar_url") or "", | ||||||
|                     ) |                     ) | ||||||
|  |  | ||||||
|             if self.client.backend.settings["hideProfileChangeEvents"]: |             if not self.client.backend.settings.Chat.show_profile_changes: | ||||||
|                 return None |                 return None | ||||||
|  |  | ||||||
|             return ( |             return ( | ||||||
| @@ -682,7 +682,7 @@ class NioCallbacks: | |||||||
|     async def onUnknownEvent( |     async def onUnknownEvent( | ||||||
|         self, room: nio.MatrixRoom, ev: nio.UnknownEvent, |         self, room: nio.MatrixRoom, ev: nio.UnknownEvent, | ||||||
|     ) -> None: |     ) -> None: | ||||||
|         if self.client.backend.settings["hideUnknownEvents"]: |         if not self.client.backend.settings.Chat.show_unknown_events: | ||||||
|             await self.client.register_nio_room(room) |             await self.client.register_nio_room(room) | ||||||
|             return |             return | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								src/backend/pcn/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								src/backend/pcn/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | # Copyright Mirage authors & contributors <https://github.com/mirukana/mirage> | ||||||
|  | # SPDX-License-Identifier: LGPL-3.0-or-later | ||||||
|  |  | ||||||
|  | """Parse and operate on PCN (Python Config Notation) files.""" | ||||||
							
								
								
									
										37
									
								
								src/backend/pcn/globals_dict.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								src/backend/pcn/globals_dict.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | |||||||
|  | from collections import UserDict | ||||||
|  | from typing import TYPE_CHECKING, Any, Dict, Iterator | ||||||
|  |  | ||||||
|  | if TYPE_CHECKING: | ||||||
|  |     from .section import Section | ||||||
|  |  | ||||||
|  | PCN_GLOBALS: Dict[str, Any] = {} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class GlobalsDict(UserDict): | ||||||
|  |     def __init__(self, section: "Section") -> None: | ||||||
|  |         super().__init__() | ||||||
|  |         self.section = section | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def full_dict(self) -> Dict[str, Any]: | ||||||
|  |         return { | ||||||
|  |             **PCN_GLOBALS, | ||||||
|  |             **(self.section.root if self.section.root else {}), | ||||||
|  |             **(self.section.root.globals if self.section.root else {}), | ||||||
|  |             "self": self.section, | ||||||
|  |             "parent": self.section.parent, | ||||||
|  |             "root": self.section.parent, | ||||||
|  |             **self.data, | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     def __getitem__(self, key: str) -> Any: | ||||||
|  |         return self.full_dict[key] | ||||||
|  |  | ||||||
|  |     def __iter__(self) -> Iterator[str]: | ||||||
|  |         return iter(self.full_dict) | ||||||
|  |  | ||||||
|  |     def __len__(self) -> int: | ||||||
|  |         return len(self.full_dict) | ||||||
|  |  | ||||||
|  |     def __repr__(self) -> str: | ||||||
|  |         return repr(self.full_dict) | ||||||
							
								
								
									
										52
									
								
								src/backend/pcn/property.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								src/backend/pcn/property.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | |||||||
|  | import re | ||||||
|  | from dataclasses import dataclass, field | ||||||
|  | from typing import TYPE_CHECKING, Any, Callable, Dict, Type | ||||||
|  |  | ||||||
|  | if TYPE_CHECKING: | ||||||
|  |     from .section import Section | ||||||
|  |  | ||||||
|  | TYPE_PROCESSORS: Dict[str, Callable[[Any], Any]] = { | ||||||
|  |     "tuple": lambda v: tuple(v), | ||||||
|  |     "set": lambda v: set(v), | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Unset: | ||||||
|  |     pass | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass | ||||||
|  | class Property: | ||||||
|  |     name:           str       = field() | ||||||
|  |     annotation:     str       = field() | ||||||
|  |     expression:     str       = field() | ||||||
|  |     section:        "Section" = field() | ||||||
|  |     value_override: Any       = Unset | ||||||
|  |  | ||||||
|  |     def __get__(self, obj: "Section", objtype: Type["Section"]) -> Any: | ||||||
|  |         if not obj: | ||||||
|  |             return self | ||||||
|  |  | ||||||
|  |         if self.value_override is not Unset: | ||||||
|  |             return self.value_override | ||||||
|  |  | ||||||
|  |         env    = obj.globals | ||||||
|  |         result = eval(self.expression, dict(env), env)  # nosec | ||||||
|  |  | ||||||
|  |         return process_value(self.annotation, result) | ||||||
|  |  | ||||||
|  |     def __set__(self, obj: "Section", value: Any) -> None: | ||||||
|  |         self.value_override = value | ||||||
|  |         obj._edited[self.name] = value | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def process_value(annotation: str, value: Any) -> Any: | ||||||
|  |     annotation = re.sub(r"\[.*\]$", "", annotation) | ||||||
|  |  | ||||||
|  |     if annotation in TYPE_PROCESSORS: | ||||||
|  |         return TYPE_PROCESSORS[annotation](value) | ||||||
|  |  | ||||||
|  |     if annotation.lower() in TYPE_PROCESSORS: | ||||||
|  |         return TYPE_PROCESSORS[annotation.lower()](value) | ||||||
|  |  | ||||||
|  |     return value | ||||||
							
								
								
									
										395
									
								
								src/backend/pcn/section.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										395
									
								
								src/backend/pcn/section.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,395 @@ | |||||||
|  | import textwrap | ||||||
|  | from collections import OrderedDict | ||||||
|  | from collections.abc import MutableMapping | ||||||
|  | from dataclasses import dataclass, field | ||||||
|  | from operator import attrgetter | ||||||
|  | from pathlib import Path | ||||||
|  | from typing import ( | ||||||
|  |     Any, Callable, ClassVar, Dict, Generator, List, Optional, Set, Tuple, Type, | ||||||
|  |     Union, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | import redbaron as red | ||||||
|  |  | ||||||
|  | from .globals_dict import GlobalsDict | ||||||
|  | from .property import Property, process_value | ||||||
|  |  | ||||||
|  | # TODO: docstrings, error handling, support @property, non-section classes | ||||||
|  | BUILTINS_DIR: Path = Path(__file__).parent.parent.parent.resolve()  # src dir | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass(repr=False, eq=False) | ||||||
|  | class Section(MutableMapping): | ||||||
|  |     sections:   ClassVar[Set[str]]        = set() | ||||||
|  |     methods:    ClassVar[Set[str]]        = set() | ||||||
|  |     properties: ClassVar[Set[str]]        = set() | ||||||
|  |     order:      ClassVar[Dict[str, None]] = OrderedDict() | ||||||
|  |  | ||||||
|  |     source_path:   Optional[Path]      = None | ||||||
|  |     root:          Optional["Section"] = None | ||||||
|  |     parent:        Optional["Section"] = None | ||||||
|  |     builtins_path: Path                = BUILTINS_DIR | ||||||
|  |     globals:       GlobalsDict         = field(init=False) | ||||||
|  |  | ||||||
|  |     _edited: Dict[str, Any] = field(init=False, default_factory=dict) | ||||||
|  |  | ||||||
|  |     def __init_subclass__(cls, **kwargs) -> None: | ||||||
|  |         # Make these attributes not shared between Section and its subclasses | ||||||
|  |         cls.sections   = set() | ||||||
|  |         cls.methods    = set() | ||||||
|  |         cls.properties = set() | ||||||
|  |         cls.order      = OrderedDict() | ||||||
|  |  | ||||||
|  |         for parent_class in cls.__bases__: | ||||||
|  |             if not issubclass(parent_class, Section): | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |             cls.sections   |= parent_class.sections  # union operator | ||||||
|  |             cls.methods    |= parent_class.methods | ||||||
|  |             cls.properties |= parent_class.properties | ||||||
|  |             cls.order.update(parent_class.order) | ||||||
|  |  | ||||||
|  |         super().__init_subclass__(**kwargs)  # type: ignore | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def __post_init__(self) -> None: | ||||||
|  |         self.globals = GlobalsDict(self) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def __getattr__(self, name: str) -> Union["Section", Any]: | ||||||
|  |         # This method signature tells mypy about the dynamic attribute types | ||||||
|  |         # we can access. The body is run for attributes that aren't found. | ||||||
|  |  | ||||||
|  |         return super().__getattribute__(name) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def __setattr__(self, name: str, value: Any) -> None: | ||||||
|  |         # This method tells mypy about the dynamic attribute types we can set. | ||||||
|  |         # The body is also run when setting an existing or new attribute. | ||||||
|  |  | ||||||
|  |         if name in self.__dataclass_fields__: | ||||||
|  |             super().__setattr__(name, value) | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         if name in self.properties: | ||||||
|  |             value = process_value(getattr(type(self), name).annotation, value) | ||||||
|  |  | ||||||
|  |             if self[name] == value: | ||||||
|  |                 return | ||||||
|  |  | ||||||
|  |             getattr(type(self), name).value_override = value | ||||||
|  |             self._edited[name]                       = value | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         if name in self.properties: | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         if name in self.sections: | ||||||
|  |             raise NotImplementedError(f"cannot overwrite section {name!r}") | ||||||
|  |  | ||||||
|  |         if name in self.methods: | ||||||
|  |             raise NotImplementedError(f"cannot overwrite method {name!r}") | ||||||
|  |  | ||||||
|  |         raise NotImplementedError(f"cannot add new attribute {name!r}") | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def __delattr__(self, name: str) -> None: | ||||||
|  |         raise NotImplementedError(f"cannot delete existing attribute {name!r}") | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def __getitem__(self, key: str) -> Any: | ||||||
|  |         try: | ||||||
|  |             return getattr(self, key) | ||||||
|  |         except AttributeError as err: | ||||||
|  |             raise KeyError(str(err)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def __setitem__(self, key: str, value: Union["Section", str]) -> None: | ||||||
|  |         setattr(self, key, value) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def __delitem__(self, key: str) -> None: | ||||||
|  |         delattr(self, key) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def __iter__(self) -> Generator[str, None, None]: | ||||||
|  |         for attr_name in self.order: | ||||||
|  |             yield attr_name | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def __len__(self) -> int: | ||||||
|  |         return len(self.order) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def __eq__(self, obj: Any) -> bool: | ||||||
|  |         if not isinstance(obj, Section): | ||||||
|  |             return False | ||||||
|  |  | ||||||
|  |         if self.globals.data != obj.globals.data or self.order != obj.order: | ||||||
|  |             return False | ||||||
|  |  | ||||||
|  |         return not any(self[attr] != obj[attr] for attr in self.order) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def __repr__(self) -> str: | ||||||
|  |         name:     str       = type(self).__name__ | ||||||
|  |         children: List[str] = [] | ||||||
|  |         content:  str       = "" | ||||||
|  |         newline:  bool      = False | ||||||
|  |  | ||||||
|  |         for attr_name in self.order: | ||||||
|  |             value = getattr(self, attr_name) | ||||||
|  |  | ||||||
|  |             if attr_name in self.sections: | ||||||
|  |                 before  = "\n" if children else "" | ||||||
|  |                 newline = True | ||||||
|  |  | ||||||
|  |                 try: | ||||||
|  |                     children.append(f"{before}{value!r},") | ||||||
|  |                 except RecursionError as err: | ||||||
|  |                     name = type(value).__name__ | ||||||
|  |                     children.append(f"{before}{name}(\n    {err!r}\n),") | ||||||
|  |                     pass | ||||||
|  |  | ||||||
|  |             elif attr_name in self.methods: | ||||||
|  |                 before  = "\n" if children else "" | ||||||
|  |                 newline = True | ||||||
|  |                 children.append(f"{before}def {value.__name__}(…),") | ||||||
|  |  | ||||||
|  |             elif attr_name in self.properties: | ||||||
|  |                 before  = "\n" if newline else "" | ||||||
|  |                 newline = False | ||||||
|  |  | ||||||
|  |                 try: | ||||||
|  |                     children.append(f"{before}{attr_name} = {value!r},") | ||||||
|  |                 except RecursionError as err: | ||||||
|  |                     children.append(f"{before}{attr_name} = {err!r},") | ||||||
|  |  | ||||||
|  |             else: | ||||||
|  |                 newline = False | ||||||
|  |  | ||||||
|  |         if children: | ||||||
|  |             content = "\n%s\n" % textwrap.indent("\n".join(children), " " * 4) | ||||||
|  |  | ||||||
|  |         return f"{name}({content})" | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _register_set_attr(cls, name: str, add_to_set_name: str) -> None: | ||||||
|  |         cls.methods.discard(name) | ||||||
|  |         cls.properties.discard(name) | ||||||
|  |         cls.sections.discard(name) | ||||||
|  |         getattr(cls, add_to_set_name).add(name) | ||||||
|  |         cls.order[name] = None | ||||||
|  |  | ||||||
|  |         for subclass in cls.__subclasses__(): | ||||||
|  |             subclass._register_set_attr(name, add_to_set_name) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def _set_section(self, section: "Section") -> None: | ||||||
|  |         name = type(section).__name__ | ||||||
|  |  | ||||||
|  |         if hasattr(self, name) and name not in self.order: | ||||||
|  |             raise AttributeError(f"{name!r}: forbidden name") | ||||||
|  |  | ||||||
|  |         if name in self.sections: | ||||||
|  |             self[name].deep_merge(section) | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         self._register_set_attr(name, "sections") | ||||||
|  |         setattr(type(self), name, section) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def _set_method(self, name: str, method: Callable) -> None: | ||||||
|  |         if hasattr(self, name) and name not in self.order: | ||||||
|  |             raise AttributeError(f"{name!r}: forbidden name") | ||||||
|  |  | ||||||
|  |         self._register_set_attr(name, "methods") | ||||||
|  |         setattr(type(self), name, method) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def _set_property( | ||||||
|  |         self, name: str, annotation: str, expression: str, | ||||||
|  |     ) -> None: | ||||||
|  |         if hasattr(self, name) and name not in self.order: | ||||||
|  |             raise AttributeError(f"{name!r}: forbidden name") | ||||||
|  |  | ||||||
|  |         prop = Property(name, annotation, expression, self) | ||||||
|  |         self._register_set_attr(name, "properties") | ||||||
|  |         setattr(type(self), name, prop) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def deep_merge(self, section2: "Section") -> None: | ||||||
|  |         for key in section2: | ||||||
|  |             if key in self.sections and key in section2.sections: | ||||||
|  |                 self.globals.data.update(section2.globals.data) | ||||||
|  |                 self[key].deep_merge(section2[key]) | ||||||
|  |  | ||||||
|  |             elif key in section2.sections: | ||||||
|  |                 self.globals.data.update(section2.globals.data) | ||||||
|  |                 new_type = type(key, (Section,), {}) | ||||||
|  |                 instance = new_type( | ||||||
|  |                     source_path   = self.source_path, | ||||||
|  |                     root          = self.root or self, | ||||||
|  |                     parent        = self, | ||||||
|  |                     builtins_path = self.builtins_path, | ||||||
|  |                 ) | ||||||
|  |                 self._set_section(instance) | ||||||
|  |                 instance.deep_merge(section2[key]) | ||||||
|  |  | ||||||
|  |             elif key in section2.methods: | ||||||
|  |                 self._set_method(key, section2[key]) | ||||||
|  |  | ||||||
|  |             else: | ||||||
|  |                 prop2 = getattr(type(section2), key) | ||||||
|  |                 self._set_property(key, prop2.annotation, prop2.expression) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def include_file(self, path: Union[Path, str]) -> None: | ||||||
|  |         if not Path(path).is_absolute() and self.source_path: | ||||||
|  |             path = self.source_path.parent / path | ||||||
|  |  | ||||||
|  |         self.deep_merge(Section.from_file(path)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def include_builtin(self, relative_path: Union[Path, str]) -> None: | ||||||
|  |         self.deep_merge(Section.from_file(self.builtins_path / relative_path)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def as_dict(self, _section: Optional["Section"] = None) -> Dict[str, Any]: | ||||||
|  |         dct     = {} | ||||||
|  |         section = self if _section is None else _section | ||||||
|  |  | ||||||
|  |         for key, value in section.items(): | ||||||
|  |             if isinstance(value, Section): | ||||||
|  |                 dct[key] = self.as_dict(value) | ||||||
|  |             else: | ||||||
|  |                 dct[key] = value | ||||||
|  |  | ||||||
|  |         return dct | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def edits_as_dict( | ||||||
|  |         self, _section: Optional["Section"] = None, | ||||||
|  |     ) -> Dict[str, Any]: | ||||||
|  |  | ||||||
|  |         warning = ( | ||||||
|  |             "This file is generated when settings are changed from the GUI, " | ||||||
|  |             "and properties in it override the ones in the corresponding " | ||||||
|  |             "PCN user config file. " | ||||||
|  |             "If a property is gets changed in the PCN file, any corresponding " | ||||||
|  |             "property override here is removed." | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         if _section is None: | ||||||
|  |             section = self | ||||||
|  |             dct     = {"__comment": warning, "set": section._edited.copy()} | ||||||
|  |             add_to  = dct["set"] | ||||||
|  |         else: | ||||||
|  |             section = _section | ||||||
|  |             dct     = { | ||||||
|  |                 prop_name: ( | ||||||
|  |                     getattr(type(section), prop_name).expression, | ||||||
|  |                     value_override, | ||||||
|  |                 ) | ||||||
|  |                 for prop_name, value_override in section._edited.items() | ||||||
|  |             } | ||||||
|  |             add_to  = dct | ||||||
|  |  | ||||||
|  |         for name in section.sections: | ||||||
|  |             edits = section.edits_as_dict(section[name]) | ||||||
|  |  | ||||||
|  |             if edits: | ||||||
|  |                 add_to[name] = edits  # type: ignore | ||||||
|  |  | ||||||
|  |         return dct | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def deep_merge_edits( | ||||||
|  |         self, edits: Dict[str, Any], has_expressions: bool = True, | ||||||
|  |     ) -> None: | ||||||
|  |         if not self.parent:  # this is Root | ||||||
|  |             edits = edits.get("set", {}) | ||||||
|  |  | ||||||
|  |         for name, value in edits.items(): | ||||||
|  |             if name not in self: | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |             if isinstance(self[name], Section) and isinstance(value, dict): | ||||||
|  |                 self[name].deep_merge_edits(value, has_expressions) | ||||||
|  |  | ||||||
|  |             elif not has_expressions: | ||||||
|  |                 self[name] = value | ||||||
|  |  | ||||||
|  |             elif isinstance(value, (tuple, list)): | ||||||
|  |                 user_expression, gui_value = value | ||||||
|  |  | ||||||
|  |                 if getattr(type(self), name).expression == user_expression: | ||||||
|  |                     self[name] = gui_value | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def from_source_code( | ||||||
|  |         cls, | ||||||
|  |         code:     str, | ||||||
|  |         path:     Optional[Path] = None, | ||||||
|  |         builtins: Optional[Path] = None, | ||||||
|  |         *, | ||||||
|  |         inherit:  Tuple[Type["Section"], ...]              = (), | ||||||
|  |         node:     Union[None, red.RedBaron, red.ClassNode] = None, | ||||||
|  |         name:     str                                      = "Root", | ||||||
|  |         root:     Optional["Section"]                      = None, | ||||||
|  |         parent:   Optional["Section"]                      = None, | ||||||
|  |     ) -> "Section": | ||||||
|  |  | ||||||
|  |         builtins                  = builtins or BUILTINS_DIR | ||||||
|  |         section:  Type["Section"] = type(name, inherit or (Section,), {}) | ||||||
|  |         instance: Section         = section(path, root, parent, builtins) | ||||||
|  |  | ||||||
|  |         node = node or red.RedBaron(code) | ||||||
|  |  | ||||||
|  |         for child in node.node_list: | ||||||
|  |             if isinstance(child, red.ClassNode): | ||||||
|  |                 root_arg      = instance if root is None else root | ||||||
|  |                 child_inherit = [] | ||||||
|  |  | ||||||
|  |                 for name in child.inherit_from.dumps().split(","): | ||||||
|  |                     name = name.strip() | ||||||
|  |  | ||||||
|  |                     if root_arg is not None and name: | ||||||
|  |                         child_inherit.append(type(attrgetter(name)(root_arg))) | ||||||
|  |  | ||||||
|  |                 instance._set_section(section.from_source_code( | ||||||
|  |                     code     = code, | ||||||
|  |                     path     = path, | ||||||
|  |                     builtins = builtins, | ||||||
|  |                     inherit  = tuple(child_inherit), | ||||||
|  |                     node     = child, | ||||||
|  |                     name     = child.name, | ||||||
|  |                     root     = root_arg, | ||||||
|  |                     parent   = instance, | ||||||
|  |                 )) | ||||||
|  |  | ||||||
|  |             elif isinstance(child, red.AssignmentNode): | ||||||
|  |                 instance._set_property( | ||||||
|  |                     child.target.value, | ||||||
|  |                     child.annotation.dumps() if child.annotation else "", | ||||||
|  |                     child.value.dumps(), | ||||||
|  |                 ) | ||||||
|  |  | ||||||
|  |             else: | ||||||
|  |                 env = instance.globals | ||||||
|  |                 exec(child.dumps(), dict(env), env)  # nosec | ||||||
|  |  | ||||||
|  |                 if isinstance(child, red.DefNode): | ||||||
|  |                     instance._set_method(child.name, env[child.name]) | ||||||
|  |  | ||||||
|  |         return instance | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def from_file( | ||||||
|  |         cls, path: Union[str, Path], builtins: Union[str, Path] = BUILTINS_DIR, | ||||||
|  |     ) -> "Section": | ||||||
|  |         path = Path(path) | ||||||
|  |         return Section.from_source_code(path.read_text(), path, Path(builtins)) | ||||||
| @@ -6,20 +6,23 @@ | |||||||
| import asyncio | import asyncio | ||||||
| import json | import json | ||||||
| import os | import os | ||||||
| import platform | import traceback | ||||||
| from collections.abc import MutableMapping | from collections.abc import MutableMapping | ||||||
| from dataclasses import dataclass, field | from dataclasses import dataclass, field | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
| from typing import TYPE_CHECKING, Any, ClassVar, Iterator, Optional, Tuple | from typing import ( | ||||||
|  |     TYPE_CHECKING, Any, ClassVar, Dict, Iterator, Optional, Tuple, | ||||||
|  | ) | ||||||
|  |  | ||||||
| import aiofiles | import aiofiles | ||||||
| from watchgod import Change, awatch | from watchgod import Change, awatch | ||||||
|  |  | ||||||
| import pyotherside | import pyotherside | ||||||
|  |  | ||||||
| from .pyotherside_events import UserFileChanged | from .pcn.section import Section | ||||||
|  | from .pyotherside_events import LoopException, UserFileChanged | ||||||
| from .theme_parser import convert_to_qml | from .theme_parser import convert_to_qml | ||||||
| from .utils import atomic_write, dict_update_recursive | from .utils import atomic_write, deep_serialize_for_qml, dict_update_recursive | ||||||
|  |  | ||||||
| if TYPE_CHECKING: | if TYPE_CHECKING: | ||||||
|     from .backend import Backend |     from .backend import Backend | ||||||
| @@ -36,18 +39,19 @@ class UserFile: | |||||||
|  |  | ||||||
|     data:        Any             = field(init=False, default_factory=dict) |     data:        Any             = field(init=False, default_factory=dict) | ||||||
|     _need_write: bool            = field(init=False, default=False) |     _need_write: bool            = field(init=False, default=False) | ||||||
|     _wrote:      bool = field(init=False, default=False) |     _mtime:      Optional[float] = field(init=False, default=None) | ||||||
|  |  | ||||||
|     _reader:     Optional[asyncio.Future] = field(init=False, default=None) |     _reader:     Optional[asyncio.Future] = field(init=False, default=None) | ||||||
|     _writer:     Optional[asyncio.Future] = field(init=False, default=None) |     _writer:     Optional[asyncio.Future] = field(init=False, default=None) | ||||||
|  |  | ||||||
|     def __post_init__(self) -> None: |     def __post_init__(self) -> None: | ||||||
|         try: |         try: | ||||||
|             self.data, save = self.deserialized(self.path.read_text()) |             text_data = self.path.read_text() | ||||||
|         except FileNotFoundError: |         except FileNotFoundError: | ||||||
|             self.data        = self.default_data |             self.data        = self.default_data | ||||||
|             self._need_write = self.create_missing |             self._need_write = self.create_missing | ||||||
|         else: |         else: | ||||||
|  |             self.data, save = self.deserialized(text_data) | ||||||
|             if save: |             if save: | ||||||
|                 self.save() |                 self.save() | ||||||
|  |  | ||||||
| @@ -56,14 +60,24 @@ class UserFile: | |||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def path(self) -> Path: |     def path(self) -> Path: | ||||||
|         """Full path of the file, can exist or not exist.""" |         """Full path of the file to read, can exist or not exist.""" | ||||||
|         raise NotImplementedError() |         raise NotImplementedError() | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def write_path(self) -> Path: | ||||||
|  |         """Full path of the file to write, can exist or not exist.""" | ||||||
|  |         return self.path | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def default_data(self) -> Any: |     def default_data(self) -> Any: | ||||||
|         """Default deserialized content to use if the file doesn't exist.""" |         """Default deserialized content to use if the file doesn't exist.""" | ||||||
|         raise NotImplementedError() |         raise NotImplementedError() | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def qml_data(self) -> Any: | ||||||
|  |         """Data converted for usage in QML.""" | ||||||
|  |         return self.data | ||||||
|  |  | ||||||
|     def deserialized(self, data: str) -> Tuple[Any, bool]: |     def deserialized(self, data: str) -> Tuple[Any, bool]: | ||||||
|         """Return parsed data from file text and whether to call `save()`.""" |         """Return parsed data from file text and whether to call `save()`.""" | ||||||
|         return (data, False) |         return (data, False) | ||||||
| @@ -76,6 +90,15 @@ class UserFile: | |||||||
|         """Inform the disk writer coroutine that the data has changed.""" |         """Inform the disk writer coroutine that the data has changed.""" | ||||||
|         self._need_write = True |         self._need_write = True | ||||||
|  |  | ||||||
|  |     def stop_watching(self) -> None: | ||||||
|  |         """Stop watching the on-disk file for changes.""" | ||||||
|  |         if self._reader: | ||||||
|  |             self._reader.cancel() | ||||||
|  |  | ||||||
|  |         if self._writer: | ||||||
|  |             self._writer.cancel() | ||||||
|  |  | ||||||
|  |  | ||||||
|     async def set_data(self, data: Any) -> None: |     async def set_data(self, data: Any) -> None: | ||||||
|         """Set `data` and call `save()`, conveniance method for QML.""" |         """Set `data` and call `save()`, conveniance method for QML.""" | ||||||
|         self.data = data |         self.data = data | ||||||
| @@ -88,45 +111,57 @@ class UserFile: | |||||||
|             await asyncio.sleep(1) |             await asyncio.sleep(1) | ||||||
|  |  | ||||||
|         async for changes in awatch(self.path): |         async for changes in awatch(self.path): | ||||||
|  |             try: | ||||||
|                 ignored = 0 |                 ignored = 0 | ||||||
|  |  | ||||||
|                 for change in changes: |                 for change in changes: | ||||||
|  |                     mtime = self.path.stat().st_mtime | ||||||
|  |  | ||||||
|                     if change[0] in (Change.added, Change.modified): |                     if change[0] in (Change.added, Change.modified): | ||||||
|                     if self._need_write or self._wrote: |                         if mtime == self._mtime: | ||||||
|                         self._wrote = False |  | ||||||
|                             ignored += 1 |                             ignored += 1 | ||||||
|                             continue |                             continue | ||||||
|  |  | ||||||
|                         async with aiofiles.open(self.path) as file: |                         async with aiofiles.open(self.path) as file: | ||||||
|                         self.data, save = self.deserialized(await file.read()) |                             text            = await file.read() | ||||||
|  |                             self.data, save = self.deserialized(text) | ||||||
|  |  | ||||||
|                             if save: |                             if save: | ||||||
|                                 self.save() |                                 self.save() | ||||||
|  |  | ||||||
|                     elif change[0] == Change.deleted: |                     elif change[0] == Change.deleted: | ||||||
|                     self._wrote      = False |  | ||||||
|                         self.data        = self.default_data |                         self.data        = self.default_data | ||||||
|                         self._need_write = self.create_missing |                         self._need_write = self.create_missing | ||||||
|  |  | ||||||
|  |                     self._mtime = mtime | ||||||
|  |  | ||||||
|                 if changes and ignored < len(changes): |                 if changes and ignored < len(changes): | ||||||
|                 UserFileChanged(type(self), self.data) |                     UserFileChanged(type(self), self.qml_data) | ||||||
|  |  | ||||||
|  |             except Exception as err: | ||||||
|  |                 LoopException(str(err), err, traceback.format_exc().rstrip()) | ||||||
|  |  | ||||||
|  |  | ||||||
|     async def _start_writer(self) -> None: |     async def _start_writer(self) -> None: | ||||||
|         """Disk writer coroutine, update the file with a 1 second cooldown.""" |         """Disk writer coroutine, update the file with a 1 second cooldown.""" | ||||||
|  |  | ||||||
|         self.path.parent.mkdir(parents=True, exist_ok=True) |         self.write_path.parent.mkdir(parents=True, exist_ok=True) | ||||||
|  |  | ||||||
|         while True: |         while True: | ||||||
|             await asyncio.sleep(1) |             await asyncio.sleep(1) | ||||||
|  |  | ||||||
|  |             try: | ||||||
|                 if self._need_write: |                 if self._need_write: | ||||||
|                 async with atomic_write(self.path) as (new, done): |                     async with atomic_write(self.write_path) as (new, done): | ||||||
|                         await new.write(self.serialized()) |                         await new.write(self.serialized()) | ||||||
|                         done() |                         done() | ||||||
|  |  | ||||||
|                     self._need_write = False |                     self._need_write = False | ||||||
|                 self._wrote      = True |                     self._mtime      = self.write_path.stat().st_mtime | ||||||
|  |  | ||||||
|  |             except Exception as err: | ||||||
|  |                 self._need_write = False | ||||||
|  |                 LoopException(str(err), err, traceback.format_exc().rstrip()) | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass | @dataclass | ||||||
| @@ -156,6 +191,7 @@ class UserDataFile(UserFile): | |||||||
| @dataclass | @dataclass | ||||||
| class MappingFile(MutableMapping, UserFile): | class MappingFile(MutableMapping, UserFile): | ||||||
|     """A file manipulable like a dict. `data` must be a mutable mapping.""" |     """A file manipulable like a dict. `data` must be a mutable mapping.""" | ||||||
|  |  | ||||||
|     def __getitem__(self, key: Any) -> Any: |     def __getitem__(self, key: Any) -> Any: | ||||||
|         return self.data[key] |         return self.data[key] | ||||||
|  |  | ||||||
| @@ -171,6 +207,22 @@ class MappingFile(MutableMapping, UserFile): | |||||||
|     def __len__(self) -> int: |     def __len__(self) -> int: | ||||||
|         return len(self.data) |         return len(self.data) | ||||||
|  |  | ||||||
|  |     def __getattr__(self, key: Any) -> Any: | ||||||
|  |         try: | ||||||
|  |             return self.data[key] | ||||||
|  |         except KeyError: | ||||||
|  |             return super().__getattribute__(key) | ||||||
|  |  | ||||||
|  |     def __setattr__(self, key: Any, value: Any) -> None: | ||||||
|  |         if key in self.__dataclass_fields__: | ||||||
|  |             super().__setattr__(key, value) | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         self.data[key] = value | ||||||
|  |  | ||||||
|  |     def __delattr__(self, key: Any) -> None: | ||||||
|  |         del self.data[key] | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass | @dataclass | ||||||
| class JSONFile(MappingFile): | class JSONFile(MappingFile): | ||||||
| @@ -180,7 +232,6 @@ class JSONFile(MappingFile): | |||||||
|     def default_data(self) -> dict: |     def default_data(self) -> dict: | ||||||
|         return {} |         return {} | ||||||
|  |  | ||||||
|  |  | ||||||
|     def deserialized(self, data: str) -> Tuple[dict, bool]: |     def deserialized(self, data: str) -> Tuple[dict, bool]: | ||||||
|         """Return parsed data from file text and whether to call `save()`. |         """Return parsed data from file text and whether to call `save()`. | ||||||
|  |  | ||||||
| @@ -193,12 +244,41 @@ class JSONFile(MappingFile): | |||||||
|         dict_update_recursive(all_data, loaded) |         dict_update_recursive(all_data, loaded) | ||||||
|         return (all_data, loaded != all_data) |         return (all_data, loaded != all_data) | ||||||
|  |  | ||||||
|  |  | ||||||
|     def serialized(self) -> str: |     def serialized(self) -> str: | ||||||
|         data = self.data |         data = self.data | ||||||
|         return json.dumps(data, indent=4, ensure_ascii=False, sort_keys=True) |         return json.dumps(data, indent=4, ensure_ascii=False, sort_keys=True) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass | ||||||
|  | class PCNFile(MappingFile): | ||||||
|  |     """File stored in the PCN format, with machine edits in a separate JSON.""" | ||||||
|  |  | ||||||
|  |     create_missing = False | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def write_path(self) -> Path: | ||||||
|  |         """Full path of file where programatically-done edits are stored.""" | ||||||
|  |         return self.path.with_suffix(".gui.json") | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def qml_data(self) -> Dict[str, Any]: | ||||||
|  |         return deep_serialize_for_qml(self.data.as_dict())  # type: ignore | ||||||
|  |  | ||||||
|  |     def deserialized(self, data: str) -> Tuple[Section, bool]: | ||||||
|  |         root  = Section.from_source_code(data, self.path) | ||||||
|  |         edits = self.write_path.read_text() if self.write_path.exists() else "" | ||||||
|  |         root.deep_merge_edits(json.loads(edits)) | ||||||
|  |         return (root, False) | ||||||
|  |  | ||||||
|  |     def serialized(self) -> str: | ||||||
|  |         edits = self.data.edits_as_dict() | ||||||
|  |         return json.dumps(edits, indent=4, ensure_ascii=False) | ||||||
|  |  | ||||||
|  |     async def set_data(self, data: Dict[str, Any]) -> None: | ||||||
|  |         self.data.deep_merge_edits({"set": data}, has_expressions=False) | ||||||
|  |         self.save() | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass | @dataclass | ||||||
| class Accounts(ConfigFile, JSONFile): | class Accounts(ConfigFile, JSONFile): | ||||||
|     """Config file for saved matrix accounts: user ID, access tokens, etc""" |     """Config file for saved matrix accounts: user ID, access tokens, etc""" | ||||||
| @@ -209,7 +289,6 @@ class Accounts(ConfigFile, JSONFile): | |||||||
|         """Return for QML whether there are any accounts saved on disk.""" |         """Return for QML whether there are any accounts saved on disk.""" | ||||||
|         return bool(self.data) |         return bool(self.data) | ||||||
|  |  | ||||||
|  |  | ||||||
|     async def add(self, user_id: str) -> None: |     async def add(self, user_id: str) -> None: | ||||||
|         """Add an account to the config and write it on disk. |         """Add an account to the config and write it on disk. | ||||||
|  |  | ||||||
| @@ -233,7 +312,6 @@ class Accounts(ConfigFile, JSONFile): | |||||||
|         }) |         }) | ||||||
|         self.save() |         self.save() | ||||||
|  |  | ||||||
|  |  | ||||||
|     async def set( |     async def set( | ||||||
|         self, |         self, | ||||||
|         user_id:    str, |         user_id:    str, | ||||||
| @@ -261,7 +339,6 @@ class Accounts(ConfigFile, JSONFile): | |||||||
|  |  | ||||||
|         self.save() |         self.save() | ||||||
|  |  | ||||||
|  |  | ||||||
|     async def forget(self, user_id: str) -> None: |     async def forget(self, user_id: str) -> None: | ||||||
|         """Delete an account from the config and write it on disk.""" |         """Delete an account from the config and write it on disk.""" | ||||||
|  |  | ||||||
| @@ -270,187 +347,33 @@ class Accounts(ConfigFile, JSONFile): | |||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass | @dataclass | ||||||
| class Settings(ConfigFile, JSONFile): | class Settings(ConfigFile, PCNFile): | ||||||
|     """General config file for UI and backend settings""" |     """General config file for UI and backend settings""" | ||||||
|  |  | ||||||
|     filename: str = "settings.json" |     filename: str = "settings.py" | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def default_data(self) -> dict: |     def default_data(self) -> Section: | ||||||
|         def ctrl_or_osx_ctrl() -> str: |         root  = Section.from_file("src/config/settings.py") | ||||||
|             # Meta in Qt corresponds to Ctrl on OSX |         edits = "{}" | ||||||
|             return "Meta" if platform.system() == "Darwin" else "Ctrl" |  | ||||||
|  |  | ||||||
|         def alt_or_cmd() -> str: |         if self.write_path.exists(): | ||||||
|             # Ctrl in Qt corresponds to Cmd on OSX |             edits = self.write_path.read_text() | ||||||
|             return "Ctrl" if platform.system() == "Darwin" else "Alt" |  | ||||||
|  |  | ||||||
|         return { |         root.deep_merge_edits(json.loads(edits)) | ||||||
|             "alertOnMentionForMsec": -1, |         return root | ||||||
|             "alertOnMessageForMsec": 0, |  | ||||||
|             "alwaysCenterRoomHeader": False, |  | ||||||
|             # "autoHideScrollBarsAfterMsec": 2000, |  | ||||||
|             "beUnavailableAfterSecondsIdle": 60 * 10, |  | ||||||
|             "centerRoomListOnClick": False, |  | ||||||
|             "compactMode": False, |  | ||||||
|             "clearRoomFilterOnEnter": True, |  | ||||||
|             "clearRoomFilterOnEscape": True, |  | ||||||
|             "clearMemberFilterOnEscape": True, |  | ||||||
|             "closeMinimizesToTray": False, |  | ||||||
|             "collapseSidePanesUnderWindowWidth": 450, |  | ||||||
|             "enableKineticScrolling": True, |  | ||||||
|             "hideProfileChangeEvents": True, |  | ||||||
|             "hideMembershipEvents": False, |  | ||||||
|             "hideUnknownEvents": True, |  | ||||||
|             "kineticScrollingMaxSpeed": 2500, |  | ||||||
|             "kineticScrollingDeceleration": 1500, |  | ||||||
|             "lexicalRoomSorting": False, |  | ||||||
|             "markRoomReadMsecDelay": 200, |  | ||||||
|             "maxMessageCharactersPerLine": 65, |  | ||||||
|             "nonKineticScrollingSpeed": 1.0, |  | ||||||
|             "ownMessagesOnLeftAboveWidth": 895, |  | ||||||
|             "theme": "Midnight.qpl", |  | ||||||
|             "writeAliases": {}, |  | ||||||
|             "zoom": 1.0, |  | ||||||
|             "roomBookmarkIDs": {}, |  | ||||||
|  |  | ||||||
|             "media": { |     def deserialized(self, data: str) -> Tuple[Section, bool]: | ||||||
|                 "autoLoad": True, |         section, save = super().deserialized(data) | ||||||
|                 "autoPlay": False, |  | ||||||
|                 "autoPlayGIF": True, |  | ||||||
|                 "autoHideOSDAfterMsec": 3000, |  | ||||||
|                 "defaultVolume": 100, |  | ||||||
|                 "openExternallyOnClick": False, |  | ||||||
|                 "startMuted": False, |  | ||||||
|             }, |  | ||||||
|             "keys": { |  | ||||||
|                 "startPythonDebugger": ["Alt+Shift+D"], |  | ||||||
|                 "toggleDebugConsole":  ["Alt+Shift+C", "F1"], |  | ||||||
|  |  | ||||||
|                 "zoomIn":             ["Ctrl++"], |         if self and self.General.theme != section.General.theme: | ||||||
|                 "zoomOut":            ["Ctrl+-"], |             self.backend.theme.stop_watching() | ||||||
|                 "zoomReset":          ["Ctrl+="], |             self.backend.theme = Theme( | ||||||
|                 "toggleCompactMode":  ["Ctrl+Alt+C"], |                 self.backend, section.General.theme,  # type: ignore | ||||||
|                 "toggleHideRoomPane": ["Ctrl+Alt+R"], |             ) | ||||||
|  |  | ||||||
|                 "scrollUp":       ["Alt+Up", "Alt+K"], |  | ||||||
|                 "scrollDown":     ["Alt+Down", "Alt+J"], |  | ||||||
|                 "scrollPageUp":   ["Alt+Ctrl+Up", "Alt+Ctrl+K", "PgUp"], |  | ||||||
|                 "scrollPageDown": ["Alt+Ctrl+Down", "Alt+Ctrl+J", "PgDown"], |  | ||||||
|                 "scrollToTop": |  | ||||||
|                     ["Alt+Ctrl+Shift+Up", "Alt+Ctrl+Shift+K", "Home"], |  | ||||||
|                 "scrollToBottom": |  | ||||||
|                     ["Alt+Ctrl+Shift+Down", "Alt+Ctrl+Shift+J", "End"], |  | ||||||
|  |  | ||||||
|                 "previousTab": ["Alt+Shift+Left", "Alt+Shift+H"], |  | ||||||
|                 "nextTab":     ["Alt+Shift+Right", "Alt+Shift+L"], |  | ||||||
|  |  | ||||||
|                 "addNewAccount":         ["Alt+Shift+A"], |  | ||||||
|                 "accountSettings":       ["Alt+A"], |  | ||||||
|                 "addNewChat":            ["Alt+C"], |  | ||||||
|                 "toggleFocusMainPane":   ["Alt+F"], |  | ||||||
|                 "clearRoomFilter":       ["Alt+Shift+F"], |  | ||||||
|                 "toggleCollapseAccount": ["Alt+O"], |  | ||||||
|  |  | ||||||
|                 "openPresenceMenu":          ["Alt+P"], |  | ||||||
|                 "togglePresenceUnavailable": ["Alt+Ctrl+U", "Alt+Ctrl+A"], |  | ||||||
|                 "togglePresenceInvisible":   ["Alt+Ctrl+I"], |  | ||||||
|                 "togglePresenceOffline":     ["Alt+Ctrl+O"], |  | ||||||
|  |  | ||||||
|                 "goToLastPage":              ["Ctrl+Tab"], |  | ||||||
|                 "goToPreviousAccount":       ["Alt+Shift+N"], |  | ||||||
|                 "goToNextAccount":           ["Alt+N"], |  | ||||||
|                 "goToPreviousRoom":          ["Alt+Shift+Up", "Alt+Shift+K"], |  | ||||||
|                 "goToNextRoom":              ["Alt+Shift+Down", "Alt+Shift+J"], |  | ||||||
|                 "goToPreviousUnreadRoom":    ["Alt+Shift+U"], |  | ||||||
|                 "goToNextUnreadRoom":        ["Alt+U"], |  | ||||||
|                 "goToPreviousMentionedRoom": ["Alt+Shift+M"], |  | ||||||
|                 "goToNextMentionedRoom":     ["Alt+M"], |  | ||||||
|  |  | ||||||
|                 "focusAccountAtIndex": { |  | ||||||
|                     "01": f"{ctrl_or_osx_ctrl()}+1", |  | ||||||
|                     "02": f"{ctrl_or_osx_ctrl()}+2", |  | ||||||
|                     "03": f"{ctrl_or_osx_ctrl()}+3", |  | ||||||
|                     "04": f"{ctrl_or_osx_ctrl()}+4", |  | ||||||
|                     "05": f"{ctrl_or_osx_ctrl()}+5", |  | ||||||
|                     "06": f"{ctrl_or_osx_ctrl()}+6", |  | ||||||
|                     "07": f"{ctrl_or_osx_ctrl()}+7", |  | ||||||
|                     "08": f"{ctrl_or_osx_ctrl()}+8", |  | ||||||
|                     "09": f"{ctrl_or_osx_ctrl()}+9", |  | ||||||
|                     "10": f"{ctrl_or_osx_ctrl()}+0", |  | ||||||
|                 }, |  | ||||||
|                 # On OSX, alt+numbers if used for symbols, use cmd instead |  | ||||||
|                 "focusRoomAtIndex": { |  | ||||||
|                     "01": f"{alt_or_cmd()}+1", |  | ||||||
|                     "02": f"{alt_or_cmd()}+2", |  | ||||||
|                     "03": f"{alt_or_cmd()}+3", |  | ||||||
|                     "04": f"{alt_or_cmd()}+4", |  | ||||||
|                     "05": f"{alt_or_cmd()}+5", |  | ||||||
|                     "06": f"{alt_or_cmd()}+6", |  | ||||||
|                     "07": f"{alt_or_cmd()}+7", |  | ||||||
|                     "08": f"{alt_or_cmd()}+8", |  | ||||||
|                     "09": f"{alt_or_cmd()}+9", |  | ||||||
|                     "10": f"{alt_or_cmd()}+0", |  | ||||||
|                 }, |  | ||||||
|  |  | ||||||
|                 "unfocusOrDeselectAllMessages":       ["Ctrl+D"], |  | ||||||
|                 "focusPreviousMessage":               ["Ctrl+Up", "Ctrl+K"], |  | ||||||
|                 "focusNextMessage":                   ["Ctrl+Down", "Ctrl+J"], |  | ||||||
|                 "toggleSelectMessage":                ["Ctrl+Space"], |  | ||||||
|                 "selectMessagesUntilHere":            ["Ctrl+Shift+Space"], |  | ||||||
|                 "removeFocusedOrSelectedMessages":    ["Ctrl+R", "Alt+Del"], |  | ||||||
|                 "replyToFocusedOrLastMessage":        ["Ctrl+Q"],  # Q → Quote |  | ||||||
|                 "debugFocusedMessage":                ["Ctrl+Shift+D"], |  | ||||||
|                 "openMessagesLinksOrFiles":           ["Ctrl+O"], |  | ||||||
|                 "openMessagesLinksOrFilesExternally": ["Ctrl+Shift+O"], |  | ||||||
|                 "copyFilesLocalPath":                 ["Ctrl+Shift+C"], |  | ||||||
|                 "clearRoomMessages":                  ["Ctrl+L"], |  | ||||||
|  |  | ||||||
|                 "sendFile":                    ["Alt+S"], |  | ||||||
|                 "sendFileFromPathInClipboard": ["Alt+Shift+S"], |  | ||||||
|                 "inviteToRoom":                ["Alt+I"], |  | ||||||
|                 "leaveRoom":                   ["Alt+Escape"], |  | ||||||
|                 "forgetRoom":                  ["Alt+Shift+Escape"], |  | ||||||
|  |  | ||||||
|                 "toggleFocusRoomPane": ["Alt+R"], |  | ||||||
|  |  | ||||||
|                 "refreshDevices":             ["Alt+R", "F5"], |  | ||||||
|                 "signOutCheckedOrAllDevices": ["Alt+S", "Delete"], |  | ||||||
|  |  | ||||||
|                 "imageViewer": { |  | ||||||
|                     "panLeft":  ["H", "Left", "Alt+H", "Alt+Left"], |  | ||||||
|                     "panDown":  ["J", "Down", "Alt+J", "Alt+Down"], |  | ||||||
|                     "panUp":    ["K", "Up", "Alt+K", "Alt+Up"], |  | ||||||
|                     "panRight": ["L", "Right", "Alt+L", "Alt+Right"], |  | ||||||
|  |  | ||||||
|                     "zoomReset": ["Alt+Z", "=", "Ctrl+="], |  | ||||||
|                     "zoomOut":   ["Shift+Z", "-", "Ctrl+-"], |  | ||||||
|                     "zoomIn":    ["Z", "+", "Ctrl++"], |  | ||||||
|  |  | ||||||
|                     "rotateReset": ["Alt+R"], |  | ||||||
|                     "rotateLeft":  ["Shift+R"], |  | ||||||
|                     "rotateRight": ["R"], |  | ||||||
|  |  | ||||||
|                     "resetSpeed":    ["Alt+S"], |  | ||||||
|                     "previousSpeed": ["Shift+S"], |  | ||||||
|                     "nextSpeed":     ["S"], |  | ||||||
|  |  | ||||||
|                     "pause":      ["Space"], |  | ||||||
|                     "expand":     ["E"], |  | ||||||
|                     "fullScreen": ["F", "F11", "Alt+Return", "Alt+Enter"], |  | ||||||
|                     "close":      ["X", "Q"], |  | ||||||
|                 }, |  | ||||||
|             }, |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|     def deserialized(self, data: str) -> Tuple[dict, bool]: |  | ||||||
|         dict_data, save = super().deserialized(data) |  | ||||||
|  |  | ||||||
|         if "theme" in self and self["theme"] != dict_data["theme"]: |  | ||||||
|             self.backend.theme = Theme(self.backend, dict_data["theme"]) |  | ||||||
|             UserFileChanged(Theme, self.backend.theme.data) |             UserFileChanged(Theme, self.backend.theme.data) | ||||||
|  |  | ||||||
|         return (dict_data, save) |         return (section, save) | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass | @dataclass | ||||||
|   | |||||||
| @@ -12,14 +12,16 @@ import json | |||||||
| import sys | import sys | ||||||
| import xml.etree.cElementTree as xml_etree  # FIXME: bandit warning | import xml.etree.cElementTree as xml_etree  # FIXME: bandit warning | ||||||
| from concurrent.futures import ProcessPoolExecutor | from concurrent.futures import ProcessPoolExecutor | ||||||
| from datetime import datetime, timedelta | from contextlib import suppress | ||||||
|  | from datetime import date, datetime, time, timedelta | ||||||
| from enum import Enum | from enum import Enum | ||||||
| from enum import auto as autostr | from enum import auto as autostr | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
| from tempfile import NamedTemporaryFile | from tempfile import NamedTemporaryFile | ||||||
| from types import ModuleType | from types import ModuleType | ||||||
| from typing import ( | from typing import ( | ||||||
|     Any, AsyncIterator, Callable, Dict, Mapping, Sequence, Tuple, Type, Union, |     Any, AsyncIterator, Callable, Dict, Iterable, Mapping, Sequence, Tuple, | ||||||
|  |     Type, Union, | ||||||
| ) | ) | ||||||
| from uuid import UUID | from uuid import UUID | ||||||
|  |  | ||||||
| @@ -27,10 +29,9 @@ import aiofiles | |||||||
| import filetype | import filetype | ||||||
| from aiofiles.threadpool.binary import AsyncBufferedReader | from aiofiles.threadpool.binary import AsyncBufferedReader | ||||||
| from aiofiles.threadpool.text import AsyncTextIOWrapper | from aiofiles.threadpool.text import AsyncTextIOWrapper | ||||||
| from PIL import Image as PILImage |  | ||||||
|  |  | ||||||
| from nio.crypto import AsyncDataT as File | from nio.crypto import AsyncDataT as File | ||||||
| from nio.crypto import async_generator_from_data | from nio.crypto import async_generator_from_data | ||||||
|  | from PIL import Image as PILImage | ||||||
|  |  | ||||||
| if sys.version_info >= (3, 7): | if sys.version_info >= (3, 7): | ||||||
|     from contextlib import asynccontextmanager |     from contextlib import asynccontextmanager | ||||||
| @@ -145,14 +146,17 @@ def plain2html(text: str) -> str: | |||||||
|                .replace("\t", " " * 4) |                .replace("\t", " " * 4) | ||||||
|  |  | ||||||
|  |  | ||||||
| def serialize_value_for_qml(value: Any, json_list_dicts: bool = False) -> Any: | def serialize_value_for_qml( | ||||||
|  |     value: Any, json_list_dicts: bool = False, reject_unknown: bool = False, | ||||||
|  | ) -> Any: | ||||||
|     """Convert a value to make it easier to use from QML. |     """Convert a value to make it easier to use from QML. | ||||||
|  |  | ||||||
|     Returns: |     Returns: | ||||||
|  |  | ||||||
|     - For `int`, `float`, `bool`, `str` and `datetime`: the unchanged value |     - For `bool`, `int`, `float`, `bytes`, `str`, `datetime`, `date`, `time`: | ||||||
|  |       the unchanged value (PyOtherSide handles these) | ||||||
|  |  | ||||||
|     - For `Sequence` and `Mapping` subclasses (includes `list` and `dict`): |     - For `Iterable` objects (includes `list` and `dict`): | ||||||
|       a JSON dump if `json_list_dicts` is `True`, else the unchanged value |       a JSON dump if `json_list_dicts` is `True`, else the unchanged value | ||||||
|  |  | ||||||
|     - If the value is an instancied object and has a `serialized` attribute or |     - If the value is an instancied object and has a `serialized` attribute or | ||||||
| @@ -168,10 +172,11 @@ def serialize_value_for_qml(value: Any, json_list_dicts: bool = False) -> Any: | |||||||
|  |  | ||||||
|     - For class types: the class `__name__` |     - For class types: the class `__name__` | ||||||
|  |  | ||||||
|     - For anything else: the unchanged value |     - For anything else: raise a `TypeError` if `reject_unknown` is `True`, | ||||||
|  |       else return the unchanged value. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     if isinstance(value, (int, float, bool, str, datetime)): |     if isinstance(value, (bool, int, float, bytes, str, datetime, date, time)): | ||||||
|         return value |         return value | ||||||
|  |  | ||||||
|     if json_list_dicts and isinstance(value, (Sequence, Mapping)): |     if json_list_dicts and isinstance(value, (Sequence, Mapping)): | ||||||
| @@ -180,6 +185,9 @@ def serialize_value_for_qml(value: Any, json_list_dicts: bool = False) -> Any: | |||||||
|     if not inspect.isclass(value) and hasattr(value, "serialized"): |     if not inspect.isclass(value) and hasattr(value, "serialized"): | ||||||
|         return value.serialized |         return value.serialized | ||||||
|  |  | ||||||
|  |     if isinstance(value, Iterable): | ||||||
|  |         return value | ||||||
|  |  | ||||||
|     if hasattr(value, "__class__") and issubclass(value.__class__, Enum): |     if hasattr(value, "__class__") and issubclass(value.__class__, Enum): | ||||||
|         return value.value |         return value.value | ||||||
|  |  | ||||||
| @@ -195,9 +203,43 @@ def serialize_value_for_qml(value: Any, json_list_dicts: bool = False) -> Any: | |||||||
|     if inspect.isclass(value): |     if inspect.isclass(value): | ||||||
|         return value.__name__ |         return value.__name__ | ||||||
|  |  | ||||||
|  |     if reject_unknown: | ||||||
|  |         raise TypeError("Unknown type reject") | ||||||
|  |  | ||||||
|     return value |     return value | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def deep_serialize_for_qml(obj: Iterable) -> Union[list, dict]: | ||||||
|  |     """Recursively serialize lists and dict values for QML.""" | ||||||
|  |  | ||||||
|  |     if isinstance(obj, Mapping): | ||||||
|  |         dct = {} | ||||||
|  |  | ||||||
|  |         for key, value in obj.items(): | ||||||
|  |             if isinstance(value, Iterable) and not isinstance(value, str): | ||||||
|  |                 # PyOtherSide only accept dicts with string keys | ||||||
|  |                 dct[str(key)] = deep_serialize_for_qml(value) | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |             with suppress(TypeError): | ||||||
|  |                 dct[str(key)] = \ | ||||||
|  |                     serialize_value_for_qml(value, reject_unknown=True) | ||||||
|  |  | ||||||
|  |         return dct | ||||||
|  |  | ||||||
|  |     lst = [] | ||||||
|  |  | ||||||
|  |     for value in obj: | ||||||
|  |         if isinstance(value, Iterable) and not isinstance(value, str): | ||||||
|  |             lst.append(deep_serialize_for_qml(value)) | ||||||
|  |             continue | ||||||
|  |  | ||||||
|  |         with suppress(TypeError): | ||||||
|  |             lst.append(serialize_value_for_qml(value, reject_unknown=True)) | ||||||
|  |  | ||||||
|  |     return lst | ||||||
|  |  | ||||||
|  |  | ||||||
| def classes_defined_in(module: ModuleType) -> Dict[str, Type]: | def classes_defined_in(module: ModuleType) -> Dict[str, Type]: | ||||||
|     """Return a `{name: class}` dict of all the classes a module defines.""" |     """Return a `{name: class}` dict of all the classes a module defines.""" | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										430
									
								
								src/config/settings.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										430
									
								
								src/config/settings.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,430 @@ | |||||||
|  | # pylint: skip-file | ||||||
|  | # flake8: noqa | ||||||
|  | # mypy: ignore-errors | ||||||
|  |  | ||||||
|  | class General: | ||||||
|  |     # When closing the window, minimize the application to system tray instead | ||||||
|  |     # of quitting the application. | ||||||
|  |     # A click on the tray icon reveals the window, middle click fully quits it | ||||||
|  |     # and right click opens a menu with these options. | ||||||
|  |     close_to_tray: bool = False | ||||||
|  |  | ||||||
|  |     # Show rooms, members and messages in way that takes less vertical space. | ||||||
|  |     compact: bool = False | ||||||
|  |  | ||||||
|  |     # When the window width is less than this number of pixels, switch to a | ||||||
|  |     # mobile-like mode where only the left main pane, center page/chat or | ||||||
|  |     # right room pane is visible at a time. | ||||||
|  |     hide_side_panes_under: int = 450 | ||||||
|  |  | ||||||
|  |     # How many seconds the cursor must hover on buttons and other elements | ||||||
|  |     # to show tooltips. | ||||||
|  |     tooltips_delay: float = 0.5 | ||||||
|  |  | ||||||
|  |     # Application theme to use. | ||||||
|  |     # Can be the name of a built-in theme (Mirage.qpl or Glass.qpl), or | ||||||
|  |     # the name (including extension) of a file in the user theme folder, which | ||||||
|  |     # is "$XDG_DATA_HOME/mirage/themes" or "~/.local/share/mirage/themes". | ||||||
|  |     # For Flatpak, it is | ||||||
|  |     # "~/.var/app/io.github.mirukana.mirage/data/mirage/themes". | ||||||
|  |     theme: str = "Midnight.qpl" | ||||||
|  |  | ||||||
|  |     # Interface scale multiplier, e.g. 0.5 makes everything half-size. | ||||||
|  |     zoom: float = 1.0 | ||||||
|  |  | ||||||
|  | class Presence: | ||||||
|  |     # Automatically set your presence to unavailable after this number of | ||||||
|  |     # seconds without any mouse or keyboard activity. | ||||||
|  |     # This currently only works on Linux X11. | ||||||
|  |     auto_away_after: int = 60 * 10 | ||||||
|  |  | ||||||
|  | class Notifications: | ||||||
|  |     # How long in seconds window alerts will last when a new message | ||||||
|  |     # is posted in a room. On most desktops, this highlights or flashes the | ||||||
|  |     # application in the taskbar or dock. | ||||||
|  |     # Can be set to 0 for no alerts. | ||||||
|  |     # Can be set to -1 for alerts that last until the window is focused. | ||||||
|  |     alert_time: float = 0 | ||||||
|  |  | ||||||
|  |     # Same as alert_time for urgent messages, e.g. when you are mentioned, | ||||||
|  |     # replied to, or the message contains a keyword. | ||||||
|  |     urgent_alert_time: float = -1 | ||||||
|  |  | ||||||
|  | class Scrolling: | ||||||
|  |     # Use velocity-based kinetic scrolling. | ||||||
|  |     # Can cause problems on laptop touchpads and some special mouse wheels. | ||||||
|  |     kinetic: bool = True | ||||||
|  |  | ||||||
|  |     # Maximum allowed velocity when kinetic scrolling is used. | ||||||
|  |     kinetic_max_speed: int = 2500 | ||||||
|  |  | ||||||
|  |     # When kinetic scrolling is used, how fast the view slows down when you | ||||||
|  |     # stop physically scrolling. | ||||||
|  |     kinetic_deceleration: int = 1500 | ||||||
|  |  | ||||||
|  |     # Multiplier for the scrolling speed when kinetic scrolling is | ||||||
|  |     # disabled, e.g. 1.5 is 1.5x faster than the default speed. | ||||||
|  |     non_kinetic_speed: float = 1.0 | ||||||
|  |  | ||||||
|  | class RoomList: | ||||||
|  |     # Prevent resizing the pane below this width in pixels. | ||||||
|  |     min_width: int = 144 | ||||||
|  |  | ||||||
|  |     # Sort rooms in alphabetical order instead of recent activity. | ||||||
|  |     # The application must be restarted to apply changes to this setting. | ||||||
|  |     lexical_sort: bool = False | ||||||
|  |  | ||||||
|  |     # Mapping of account user ID to list of room ID to always keep on top. | ||||||
|  |     # You can copy a room's ID by right clicking on it in the room list. | ||||||
|  |     # Example: {"@alice:example.org": ["!aBc@example.org", "!123:example.org"]} | ||||||
|  |     bookmarks: Dict[str, List[str]] = {} | ||||||
|  |  | ||||||
|  |     # When clicking on a room, recenter the room list on that room. | ||||||
|  |     click_centers: bool = False | ||||||
|  |  | ||||||
|  |     # When pressing enter in the room filter field, clear the field's text, | ||||||
|  |     # in addition to activating the keyboard-focused room. | ||||||
|  |     enter_clears_filter: bool = True | ||||||
|  |  | ||||||
|  |     # When pressing escape in the room filter field, clear the field's text. | ||||||
|  |     # in addition to focusing the current page or chat composer. | ||||||
|  |     escape_clears_filter: bool = True | ||||||
|  |  | ||||||
|  | class Chat: | ||||||
|  |     # Center the chat header (room avatar, name and topic) even when sidepanes | ||||||
|  |     # aren't hidden (see comment for the hide_sidepanes_under setting). | ||||||
|  |     always_center_header: bool = False | ||||||
|  |  | ||||||
|  |     # When the chat timeline is larger than this pixel number, | ||||||
|  |     # Align your own messages to the left of the timeline instead of right. | ||||||
|  |     # Can be 0 to always show your messages on the left. | ||||||
|  |     own_messages_on_left_above: int = 895 | ||||||
|  |  | ||||||
|  |     # Maximum number of characters in a message line before wrapping the text | ||||||
|  |     # to a new line. Ignores messages containing code blocks or tables. | ||||||
|  |     max_messages_line_length: int = 65 | ||||||
|  |  | ||||||
|  |     # Show membership events in the timeline: when someone is invited to the | ||||||
|  |     # room, joins, leaves, is kicked, banned or unbanned. | ||||||
|  |     show_membership_events: bool = True | ||||||
|  |  | ||||||
|  |     # Show room member display name and avatar change events in the timeline. | ||||||
|  |     show_profile_changes: bool = False | ||||||
|  |  | ||||||
|  |     # Show a notice in the timeline for events types that aren't recognized. | ||||||
|  |     show_unknown_events: bool = False | ||||||
|  |  | ||||||
|  |     # In a chat with unread messages, the messages will be marked as read | ||||||
|  |     # after this number of seconds. | ||||||
|  |     # Focusing another window or chat resets the timer. | ||||||
|  |     mark_read_delay: float = 0.2 | ||||||
|  |  | ||||||
|  |     class Composer: | ||||||
|  |         # Mapping of account user ID to alias. | ||||||
|  |         # From any chat, start a message with an alias followed by a space | ||||||
|  |         # to type and send as this associated account. | ||||||
|  |         # The account must have permission to talk in the room. | ||||||
|  |         # To ignore an alias when typing, prepend it with a space. | ||||||
|  |         # Example: {"@alice:example.org": "al", "@bob:example.org": "b"} | ||||||
|  |         aliases: Dict[str, str] = {} | ||||||
|  |  | ||||||
|  |     class Files: | ||||||
|  |         # Minimum width of the file name/size box for files without previews. | ||||||
|  |         min_file_width: int = 256 | ||||||
|  |  | ||||||
|  |         # Minimum (width, height) for image thumbnails. | ||||||
|  |         min_thumbnail_size: Tuple[int, int] = (256, 256) | ||||||
|  |  | ||||||
|  |         # How much of the chat height image thumbnails can take at most, | ||||||
|  |         # e.g. 0.4 for 40% of the chat or 1 for 100%. | ||||||
|  |         max_thumbnail_height_ratio: float = 0.4 | ||||||
|  |  | ||||||
|  |         # Automatically play animated GIF images in the timeline. | ||||||
|  |         auto_play_gif: bool = True | ||||||
|  |  | ||||||
|  |         # When clicking on a file in the timeline, open it in an external | ||||||
|  |         # programing instead of displaying it using Mirage's interface. | ||||||
|  |         # On Linux, the xdg-open command is used. | ||||||
|  |         click_opens_externally: bool = False | ||||||
|  |  | ||||||
|  |         # In the full image viewer, if the image is large enough to cover the | ||||||
|  |         # info bar or buttons, they will automatically hide after this number | ||||||
|  |         # of seconds. | ||||||
|  |         # Hovering on the top/bottom with a mouse or tapping on a touch screen | ||||||
|  |         # reveals the hidden controls. | ||||||
|  |         autohide_image_controls_after: float = 2.0 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Keys: | ||||||
|  |     # All keybind settings, unless their comment says otherwise, are list of | ||||||
|  |     # the possible shortcuts for an action, e.g. ["Ctrl+A", "Alt+Shift+A"]. | ||||||
|  |     # | ||||||
|  |     # The available modifiers are Ctrl, Shift, Alt and Meta. | ||||||
|  |     # On macOS, Ctrl corresponds to Cmd and Meta corresponds to Control. | ||||||
|  |     # On other systems, Meta corresponds to the Windows/Super/mod4 key. | ||||||
|  |     # | ||||||
|  |     # https://doc.qt.io/qt-5/qt.html#Key-enum lists the names of special | ||||||
|  |     # keys, e.g. for "Qt::Key_Space", you would use "Space" in this config. | ||||||
|  |     # | ||||||
|  |     # The Escape key by itself should not be bound, as it would conflict with | ||||||
|  |     # closing popups and various other actions. | ||||||
|  |     # | ||||||
|  |     # Key chords can be defined by having up to four shortcuts | ||||||
|  |     # separated by commas in a string, e.g. for ["Ctrl+A,B"], Ctrl+A then B | ||||||
|  |     # would need to be pressed. | ||||||
|  |  | ||||||
|  |     # Helper functions | ||||||
|  |  | ||||||
|  |     import platform | ||||||
|  |  | ||||||
|  |     def os_ctrl(self) -> str: | ||||||
|  |         # Return Meta on macOS, which corresponds to Ctrl, and Ctrl on others. | ||||||
|  |         return "Meta" if platform.system() == "Darwin" else "Ctrl" | ||||||
|  |  | ||||||
|  |     def alt_or_cmd(self) -> str: | ||||||
|  |         # Return Ctrl on macOS, which corresponds to Cmd, and Alt on others. | ||||||
|  |         return "Ctrl" if platform.system() == "Darwin" else "Alt" | ||||||
|  |  | ||||||
|  |     # Toggle compact interface mode. See the compact setting comment. | ||||||
|  |     compact = ["Alt+Ctrl+C"] | ||||||
|  |  | ||||||
|  |     # Control the interface scale. | ||||||
|  |     zoom_in    = ["Ctrl++"] | ||||||
|  |     zoom_out   = ["Ctrl+-"] | ||||||
|  |     reset_zoom = ["Ctrl+="] | ||||||
|  |  | ||||||
|  |     # Switch to the previous/next tab in pages. In chats, this controls what | ||||||
|  |     # the right room pane shows, e.g. member list or room settings. | ||||||
|  |     previous_tab = ["Alt+Shift+Left", "Alt+Shift+H"] | ||||||
|  |     next_tab     = ["Alt+Shift+Right", "Alt+Shift+L"] | ||||||
|  |  | ||||||
|  |     # Switch to the last opened page/chat, similar to Alt+Tab on most desktops. | ||||||
|  |     last_page = ["Ctrl+Tab"] | ||||||
|  |  | ||||||
|  |     # Toggle the QML developer console. Type ". help" in it for more info. | ||||||
|  |     qml_console = ["F1"] | ||||||
|  |  | ||||||
|  |     # Start the Python backend debugger. Unless the "remote-pdb" pip package is | ||||||
|  |     # installed, Mirage must be started from a terminal for this to work. | ||||||
|  |     python_debugger = ["Shift+F1"] | ||||||
|  |  | ||||||
|  |     class Scrolling: | ||||||
|  |         # Pages and chat timeline scrolling | ||||||
|  |         up        = ["Alt+Up", "Alt+K"] | ||||||
|  |         down      = ["Alt+Down", "Alt+J"] | ||||||
|  |         page_up   = ["Alt+Ctrl+Up", "Alt+Ctrl+K", "PgUp"] | ||||||
|  |         page_down = ["Alt+Ctrl+Down", "Alt+Ctrl+J", "PgDown"] | ||||||
|  |         top       = ["Alt+Ctrl+Shift+Up", "Alt+Ctrl+Shift+K", "Home"] | ||||||
|  |         bottom    = ["Alt+Ctrl+Shift+Down", "Alt+Ctrl+Shift+J", "End"] | ||||||
|  |  | ||||||
|  |     class Accounts: | ||||||
|  |         # The current account is the account under which a page or chat is | ||||||
|  |         # opened, or the keyboard-focused one when using the room filter field. | ||||||
|  |  | ||||||
|  |         # Add a new account | ||||||
|  |         add = ["Alt+Shift+A"] | ||||||
|  |  | ||||||
|  |         # Collapse the current account | ||||||
|  |         collapse = ["Alt+O"] | ||||||
|  |  | ||||||
|  |         # Open the current account settings | ||||||
|  |         settings = ["Alt+A"] | ||||||
|  |  | ||||||
|  |         # Open the current account context menu | ||||||
|  |         menu = ["Alt+P"] | ||||||
|  |  | ||||||
|  |         # Toggle current account presence between this status and online | ||||||
|  |         unavailable = ["Alt+Ctrl+U", "Alt+Ctrl+A"] | ||||||
|  |         invisible   = ["Alt+Ctrl+I"] | ||||||
|  |         offline     = ["Alt+Ctrl+O"] | ||||||
|  |  | ||||||
|  |         # Switch to first room of the previous/next account in the room list. | ||||||
|  |         previous = ["Alt+Shift+N"] | ||||||
|  |         next     = ["Alt+N"] | ||||||
|  |  | ||||||
|  |         # Switch to the first room of the account number X in the list. | ||||||
|  |         # This is a mapping of account number to keybind, e.g. | ||||||
|  |         # {1: "Ctrl+1"} would bind Ctrl+1 to the switch to the first account. | ||||||
|  |         at_index: Dict[int, str] = { | ||||||
|  |             "1": f"{parent.os_ctrl()}+1", | ||||||
|  |             "2": f"{parent.os_ctrl()}+2", | ||||||
|  |             "3": f"{parent.os_ctrl()}+3", | ||||||
|  |             "4": f"{parent.os_ctrl()}+4", | ||||||
|  |             "5": f"{parent.os_ctrl()}+5", | ||||||
|  |             "6": f"{parent.os_ctrl()}+6", | ||||||
|  |             "7": f"{parent.os_ctrl()}+7", | ||||||
|  |             "8": f"{parent.os_ctrl()}+8", | ||||||
|  |             "9": f"{parent.os_ctrl()}+9", | ||||||
|  |             "10": f"{parent.os_ctrl()}+0", | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     class Rooms: | ||||||
|  |         # Add a new room (direct chat, join or create a group). | ||||||
|  |         add = ["Alt+C"] | ||||||
|  |  | ||||||
|  |         # Focus or clear the text of the left main pane's room filter field. | ||||||
|  |         # When focusing the field, use Tab/Shift+Tab or the arrows to navigate | ||||||
|  |         # the list, Enter to switch to focused account/room, Escape to cancel, | ||||||
|  |         # Menu to open the context menu. | ||||||
|  |         focus_filter = ["Alt+F"] | ||||||
|  |         clear_filter = ["Alt+Shift+F"] | ||||||
|  |  | ||||||
|  |         # Switch to the previous/next room in the list. | ||||||
|  |         previous = ["Alt+Shift+Up", "Alt+Shift+K"] | ||||||
|  |         next     = ["Alt+Shift+Down", "Alt+Shift+J"] | ||||||
|  |  | ||||||
|  |         # Switch to the previous/next room with unread messages in the list. | ||||||
|  |         previous_unread = ["Alt+Shift+U"] | ||||||
|  |         next_unread     = ["Alt+U"] | ||||||
|  |  | ||||||
|  |         # Switch to the previous/next room with urgent messages in the list, | ||||||
|  |         # e.g. messages mentioning your name, replies to you or keywords. | ||||||
|  |         previous_urgent = ["Alt+Shift+M"] | ||||||
|  |         next_urgent     = ["Alt+M"] | ||||||
|  |  | ||||||
|  |         # Switch to room number X in the current account. | ||||||
|  |         # This is a mapping of room number to keybind, e.g. | ||||||
|  |         # {1: "Alt+1"} would bind Alt+1 to switch to the first room. | ||||||
|  |         at_index: Dict[int, str] = { | ||||||
|  |             "1": f"{parent.alt_or_cmd()}+1", | ||||||
|  |             "2": f"{parent.alt_or_cmd()}+2", | ||||||
|  |             "3": f"{parent.alt_or_cmd()}+3", | ||||||
|  |             "4": f"{parent.alt_or_cmd()}+4", | ||||||
|  |             "5": f"{parent.alt_or_cmd()}+5", | ||||||
|  |             "6": f"{parent.alt_or_cmd()}+6", | ||||||
|  |             "7": f"{parent.alt_or_cmd()}+7", | ||||||
|  |             "8": f"{parent.alt_or_cmd()}+8", | ||||||
|  |             "9": f"{parent.alt_or_cmd()}+9", | ||||||
|  |             "10": f"{parent.alt_or_cmd()}+0", | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     class Chat: | ||||||
|  |         # Keybinds specific to the current chat page. | ||||||
|  |  | ||||||
|  |         # Focus the right room pane. If the pane is currently showing the | ||||||
|  |         # room member list, the corresponding filter field is focused. | ||||||
|  |         # When focusing the field, use Tab/Shift+Tab or the arrows to navigate | ||||||
|  |         # the list, Enter to see the focused member's profile, Escape to cancel, | ||||||
|  |         # Menu to open the context menu. | ||||||
|  |         focus_room_pane = ["Alt+R"] | ||||||
|  |  | ||||||
|  |         # Toggle hiding the right pane. | ||||||
|  |         # Can also be done by clicking on current tab button at the top right. | ||||||
|  |         hide_room_pane = ["Alt+Ctrl+R"] | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         # Invite new members, leave or forget the current chat. | ||||||
|  |         invite = ["Alt+I"] | ||||||
|  |         leave  = ["Alt+Escape"] | ||||||
|  |         forget = ["Alt+Shift+Escape"] | ||||||
|  |  | ||||||
|  |         # Open the file picker to upload files in the current chat. | ||||||
|  |         send_file = ["Alt+S"] | ||||||
|  |  | ||||||
|  |         # If your clipboard contains a file path, upload that file. | ||||||
|  |         send_clipboard_path = ["Alt+Shift+S"] | ||||||
|  |  | ||||||
|  |     class Messages: | ||||||
|  |         # Focus the previous/next message in the timeline. | ||||||
|  |         # Keybinds defined below in this section affect the focused message. | ||||||
|  |         # The Menu key can open the context menu for a focused message. | ||||||
|  |         previous = ["Ctrl+Up", "Ctrl+K"] | ||||||
|  |         next     = ["Ctrl+Down", "Ctrl+J"] | ||||||
|  |  | ||||||
|  |         # Select the currently focused message, same as clicking on it. | ||||||
|  |         # When there are selected messages, some right click menu options | ||||||
|  |         # and keybinds defined below will affect these messages instead of | ||||||
|  |         # the focused (for keybinds) or mouse-targeted (right click menu) one. | ||||||
|  |         # The Menu key can open the context menu for selected messages. | ||||||
|  |         select = ["Ctrl+Space"] | ||||||
|  |  | ||||||
|  |         # Select all messages from point A to point B. | ||||||
|  |         # If used when no messages are already selected, all the messages | ||||||
|  |         # from the most recent in the timeline to the focused one are selected. | ||||||
|  |         # Otherwise, messages from the last selected to focused are selected. | ||||||
|  |         select_until_here = ["Ctrl+Shift+Space"] | ||||||
|  |  | ||||||
|  |         # Clear the message keyboard focus. | ||||||
|  |         # If no message is focused but some are selected, clear the selection. | ||||||
|  |         unfocus_or_deselect = ["Ctrl+D"] | ||||||
|  |  | ||||||
|  |         # Remove the selected messages if any, else the focused message if any, | ||||||
|  |         # else the last message you posted. | ||||||
|  |         remove = ["Ctrl+R", "Alt+Del"] | ||||||
|  |  | ||||||
|  |         # Reply/cancel reply to the focused message if any, | ||||||
|  |         # else the last message posted by someone else. | ||||||
|  |         # Replying can also be cancelled by pressing Escape. | ||||||
|  |         reply = ["Ctrl+Q"] | ||||||
|  |  | ||||||
|  |         # Open the QML developer console for the focused message if any, | ||||||
|  |         # and display the event source. | ||||||
|  |         debug = ["Ctrl+Shift+D"] | ||||||
|  |  | ||||||
|  |         # Open the files and links in selected messages if any, else the | ||||||
|  |         # file/links of the focused message if any, else the last | ||||||
|  |         # files/link in the timeline. | ||||||
|  |         open_links_files = ["Ctrl+O"] | ||||||
|  |  | ||||||
|  |         # Like open_links_files, but files open in external programs instead. | ||||||
|  |         # On Linux, this uses the xdg-open command. | ||||||
|  |         open_links_files_externally = ["Ctrl+Shift+O"] | ||||||
|  |  | ||||||
|  |         # Copy the downloaded files path in selected messages if any, | ||||||
|  |         # else the file path for the focused message if any, else the | ||||||
|  |         # path for the last downloaded file in the timeline. | ||||||
|  |         copy_files_path = ["Ctrl+Shift+C"] | ||||||
|  |  | ||||||
|  |         # Clear all messages from the chat. | ||||||
|  |         # This does not remove anything for other users. | ||||||
|  |         clear_all = ["Ctrl+L"] | ||||||
|  |  | ||||||
|  |     class ImageViewer: | ||||||
|  |         # Close the image viewer | ||||||
|  |         close = ["X", "Q"] | ||||||
|  |  | ||||||
|  |         # Toggle alternate image scaling mode: if the original image size is | ||||||
|  |         # smaller than the window, upscale it to fit the window. | ||||||
|  |         # If it is bigger than the window, show it as its real size. | ||||||
|  |         expand = ["E"] | ||||||
|  |  | ||||||
|  |         # Toggle fullscreen mode. | ||||||
|  |         fullscreen = ["F", "F11", "Alt+Return", "Alt+Enter"] | ||||||
|  |  | ||||||
|  |         # Pan/scroll the image. | ||||||
|  |         pan_left  = ["H", "Left", "Alt+H", "Alt+Left"] | ||||||
|  |         pan_down  = ["J", "Down", "Alt+J", "Alt+Down"] | ||||||
|  |         pan_up    = ["K", "Up", "Alt+K", "Alt+Up"] | ||||||
|  |         pan_right = ["L", "Right", "Alt+L", "Alt+Right"] | ||||||
|  |  | ||||||
|  |         # Control the image's zoom. Ctrl+wheel can also be used. | ||||||
|  |         zoom_in    = ["Z", "+", "Ctrl++"] | ||||||
|  |         zoom_out   = ["Shift+Z", "-", "Ctrl+-"] | ||||||
|  |         reset_zoom = ["Alt+Z", "=", "Ctrl+="] | ||||||
|  |  | ||||||
|  |         # Control the image's rotation. | ||||||
|  |         rotate_right   = ["R"] | ||||||
|  |         rotate_left    = ["Shift+R"] | ||||||
|  |         reset_rotation = ["Alt+R"] | ||||||
|  |  | ||||||
|  |         # Control the speed for animated GIF images. | ||||||
|  |         speed_up    = ["S"] | ||||||
|  |         slow_down   = ["Shift+S"] | ||||||
|  |         reset_speed = ["Alt+S"] | ||||||
|  |  | ||||||
|  |         # Toggle pausing for animated GIF images. | ||||||
|  |         pause = ["Space"] | ||||||
|  |  | ||||||
|  |     class Sessions: | ||||||
|  |         # These keybinds affect the session list in your account settings. | ||||||
|  |         # | ||||||
|  |         # Currently unchangable keys: | ||||||
|  |         # Tab/Shift+Tab or the arrow keys to navigate the list, | ||||||
|  |         # Space to check/uncheck focused session, | ||||||
|  |         # Menu to open the focused session's context menu. | ||||||
|  |  | ||||||
|  |         # Refresh the list of sessions. | ||||||
|  |         refresh = ["Alt+R", "F5"] | ||||||
|  |  | ||||||
|  |         # Sign out checked sessions if any, else sign out all sessions. | ||||||
|  |         sign_out_checked_or_all = ["Alt+S", "Delete"] | ||||||
| @@ -37,7 +37,7 @@ Drawer { | |||||||
|  |  | ||||||
|     property bool collapse: |     property bool collapse: | ||||||
|         (horizontal ? window.width : window.height) < |         (horizontal ? window.width : window.height) < | ||||||
|         window.settings.collapseSidePanesUnderWindowWidth * theme.uiScale |         window.settings.General.hide_side_panes_under * theme.uiScale | ||||||
|  |  | ||||||
|     property int peekSizeWhileCollapsed: |     property int peekSizeWhileCollapsed: | ||||||
|         (horizontal ? referenceSizeParent.width : referenceSizeParent.height) * |         (horizontal ? referenceSizeParent.width : referenceSizeParent.height) * | ||||||
|   | |||||||
| @@ -6,8 +6,8 @@ import QtQuick.Controls 2.12 | |||||||
|  |  | ||||||
| Flickable { | Flickable { | ||||||
|     id: flickable |     id: flickable | ||||||
|     maximumFlickVelocity: window.settings.kineticScrollingMaxSpeed |     maximumFlickVelocity: window.settings.Scrolling.kinetic_max_speed | ||||||
|     flickDeceleration: window.settings.kineticScrollingDeceleration |     flickDeceleration: window.settings.Scrolling.kinetic_deceleration | ||||||
|  |  | ||||||
|     ScrollBar.vertical: HScrollBar { |     ScrollBar.vertical: HScrollBar { | ||||||
|         visible: parent.interactive |         visible: parent.interactive | ||||||
|   | |||||||
| @@ -81,8 +81,8 @@ GridView { | |||||||
|     preferredHighlightBegin: height / 2 - currentItemHeight / 2 |     preferredHighlightBegin: height / 2 - currentItemHeight / 2 | ||||||
|     preferredHighlightEnd: height / 2 + currentItemHeight / 2 |     preferredHighlightEnd: height / 2 + currentItemHeight / 2 | ||||||
|  |  | ||||||
|     maximumFlickVelocity: window.settings.kineticScrollingMaxSpeed |     maximumFlickVelocity: window.settings.Scrolling.kinetic_max_speed | ||||||
|     flickDeceleration: window.settings.kineticScrollingDeceleration |     flickDeceleration: window.settings.Scrolling.kinetic_deceleration | ||||||
|  |  | ||||||
|  |  | ||||||
|     highlight: Rectangle { |     highlight: Rectangle { | ||||||
|   | |||||||
| @@ -18,7 +18,7 @@ Image { | |||||||
|     property alias radius: roundMask.radius |     property alias radius: roundMask.radius | ||||||
|     property alias showProgressBar: progressBarLoader.active |     property alias showProgressBar: progressBarLoader.active | ||||||
|     property bool showPauseButton: true |     property bool showPauseButton: true | ||||||
|     property bool pause: ! window.settings.media.autoPlayGIF |     property bool pause: ! window.settings.Chat.Files.auto_play_gif | ||||||
|     property bool forcePause: false |     property bool forcePause: false | ||||||
|     property real speed: 1 |     property real speed: 1 | ||||||
|  |  | ||||||
|   | |||||||
| @@ -16,7 +16,7 @@ MouseArea { | |||||||
|  |  | ||||||
|         const speedMultiply = |         const speedMultiply = | ||||||
|             Qt.styleHints.wheelScrollLines * |             Qt.styleHints.wheelScrollLines * | ||||||
|             window.settings.nonKineticScrollingSpeed |             window.settings.Scrolling.non_kinetic_speed | ||||||
|  |  | ||||||
|         const pixelDelta = { |         const pixelDelta = { | ||||||
|             x: wheel.pixelDelta.x || wheel.angleDelta.x / 8 * speedMultiply, |             x: wheel.pixelDelta.x || wheel.angleDelta.x / 8 * speedMultiply, | ||||||
| @@ -64,7 +64,7 @@ MouseArea { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|     enabled: ! window.settings.enableKineticScrolling |     enabled: ! window.settings.Scrolling.kinetic | ||||||
|     propagateComposedEvents: true |     propagateComposedEvents: true | ||||||
|     acceptedButtons: Qt.NoButton |     acceptedButtons: Qt.NoButton | ||||||
|  |  | ||||||
| @@ -84,7 +84,8 @@ MouseArea { | |||||||
|     Binding { |     Binding { | ||||||
|         target: flickable |         target: flickable | ||||||
|         property: "maximumFlickVelocity" |         property: "maximumFlickVelocity" | ||||||
|         value: mouseArea.enabled ? 0 : window.settings.kineticScrollingMaxSpeed |         value: | ||||||
|  |             mouseArea.enabled ? 0 : window.settings.Scrolling.kinetic_max_speed | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     Binding { |     Binding { | ||||||
| @@ -93,6 +94,6 @@ MouseArea { | |||||||
|         value: |         value: | ||||||
|             mouseArea.enabled ? |             mouseArea.enabled ? | ||||||
|             0 : |             0 : | ||||||
|             window.settings.kineticScrollingDeceleration |             window.settings.Scrolling.kinetic_deceleration | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -89,8 +89,8 @@ ListView { | |||||||
|     preferredHighlightBegin: height / 2 - currentItemHeight / 2 |     preferredHighlightBegin: height / 2 - currentItemHeight / 2 | ||||||
|     preferredHighlightEnd: height / 2 + currentItemHeight / 2 |     preferredHighlightEnd: height / 2 + currentItemHeight / 2 | ||||||
|  |  | ||||||
|     maximumFlickVelocity: window.settings.kineticScrollingMaxSpeed |     maximumFlickVelocity: window.settings.Scrolling.kinetic_max_speed | ||||||
|     flickDeceleration: window.settings.kineticScrollingDeceleration |     flickDeceleration: window.settings.Scrolling.kinetic_deceleration | ||||||
|  |  | ||||||
|     highlight: Rectangle { |     highlight: Rectangle { | ||||||
|         color: theme.controls.listView.highlight |         color: theme.controls.listView.highlight | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ import ".." | |||||||
| HButton { | HButton { | ||||||
|     id: tile |     id: tile | ||||||
|  |  | ||||||
|     property bool compact: window.settings.compactMode |     property bool compact: window.settings.General.compact | ||||||
|     property real contentOpacity: 1 |     property real contentOpacity: 1 | ||||||
|     property Component contextMenu: null |     property Component contextMenu: null | ||||||
|     property HMenu openedMenu: null |     property HMenu openedMenu: null | ||||||
|   | |||||||
| @@ -26,7 +26,7 @@ ToolTip { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|     delay: instant ? 0 : theme.controls.toolTip.delay |     delay: instant ? 0 : window.settings.General.tooltips_delay * 1000 | ||||||
|     padding: background.border.width |     padding: background.border.width | ||||||
|  |  | ||||||
|     background: Rectangle { |     background: Rectangle { | ||||||
|   | |||||||
| @@ -55,7 +55,7 @@ HColumnLayout { | |||||||
|  |  | ||||||
|     Timer { |     Timer { | ||||||
|         id: osdHideTimer |         id: osdHideTimer | ||||||
|         interval: window.settings.media.autoHideOSDAfterMsec |         interval: window.settings.Chat.Files.autohide_image_controls_after | ||||||
|         onTriggered: osd.showup = false |         onTriggered: osd.showup = false | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ Rectangle { | |||||||
|  |  | ||||||
|  |  | ||||||
|     implicitWidth: |     implicitWidth: | ||||||
|         window.settings.compactMode ? |         window.settings.General.compact ? | ||||||
|         theme.controls.presence.radius * 2 : |         theme.controls.presence.radius * 2 : | ||||||
|         theme.controls.presence.radius * 2.5 |         theme.controls.presence.radius * 2.5 | ||||||
|  |  | ||||||
|   | |||||||
| @@ -25,7 +25,7 @@ HDrawer { | |||||||
|     property string selectedOutputText: "" |     property string selectedOutputText: "" | ||||||
|  |  | ||||||
|     property string pythonDebugKeybind: |     property string pythonDebugKeybind: | ||||||
|         window.settings.keys.startPythonDebugger[0] |         window.settings.Keys.python_debugger[0] | ||||||
|  |  | ||||||
|     property string help: qsTr( |     property string help: qsTr( | ||||||
|         `Interact with the QML code using JavaScript ES6 syntax. |         `Interact with the QML code using JavaScript ES6 syntax. | ||||||
| @@ -71,7 +71,7 @@ HDrawer { | |||||||
|         if (addToHistory && history.slice(-1)[0] !== input) { |         if (addToHistory && history.slice(-1)[0] !== input) { | ||||||
|             history.push(input) |             history.push(input) | ||||||
|             while (history.length > maxHistoryLength) history.shift() |             while (history.length > maxHistoryLength) history.shift() | ||||||
|             window.historyChanged() |             window.saveHistory() | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         let output = "" |         let output = "" | ||||||
| @@ -154,7 +154,7 @@ HDrawer { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: settings.keys.toggleDebugConsole |         sequences: settings.Keys.qml_console | ||||||
|         onActivated: debugConsole.toggle() |         onActivated: debugConsole.toggle() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -17,7 +17,7 @@ Timer { | |||||||
|     interval: 1000 |     interval: 1000 | ||||||
|     repeat: true |     repeat: true | ||||||
|     running: |     running: | ||||||
|         window.settings.beUnavailableAfterSecondsIdle > 0 && |         window.settings.Presence.auto_away_after > 0 && | ||||||
|         CppUtils.idleMilliseconds() !== -1 |         CppUtils.idleMilliseconds() !== -1 | ||||||
|  |  | ||||||
|     onTriggered: { |     onTriggered: { | ||||||
| @@ -25,7 +25,7 @@ Timer { | |||||||
|  |  | ||||||
|         const beUnavailable = |         const beUnavailable = | ||||||
|             CppUtils.idleMilliseconds() / 1000 >= |             CppUtils.idleMilliseconds() / 1000 >= | ||||||
|             window.settings.beUnavailableAfterSecondsIdle |             window.settings.Presence.auto_away_after | ||||||
|  |  | ||||||
|         for (let i = 0; i < accounts.count; i++) { |         for (let i = 0; i < accounts.count; i++) { | ||||||
|             const account = accounts.get(i) |             const account = accounts.get(i) | ||||||
|   | |||||||
| @@ -81,7 +81,7 @@ Rectangle { | |||||||
|  |  | ||||||
|  |  | ||||||
|         HShortcut { |         HShortcut { | ||||||
|             sequences: window.settings.keys.goToPreviousAccount |             sequences: window.settings.Keys.Accounts.previous | ||||||
|             onActivated: { |             onActivated: { | ||||||
|                 accountList.moveCurrentIndexLeft() |                 accountList.moveCurrentIndexLeft() | ||||||
|                 accountList.currentItem.leftClicked() |                 accountList.currentItem.leftClicked() | ||||||
| @@ -89,7 +89,7 @@ Rectangle { | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         HShortcut { |         HShortcut { | ||||||
|             sequences: window.settings.keys.goToNextAccount |             sequences: window.settings.Keys.Accounts.next | ||||||
|             onActivated: { |             onActivated: { | ||||||
|                 accountList.moveCurrentIndexRight() |                 accountList.moveCurrentIndexRight() | ||||||
|                 accountList.currentItem.leftClicked() |                 accountList.currentItem.leftClicked() | ||||||
|   | |||||||
| @@ -26,7 +26,7 @@ HTile { | |||||||
|  |  | ||||||
|     function setCollapse(collapse) { |     function setCollapse(collapse) { | ||||||
|         window.uiState.collapseAccounts[model.id] = collapse |         window.uiState.collapseAccounts[model.id] = collapse | ||||||
|         window.uiStateChanged() |         window.saveUIState() | ||||||
|  |  | ||||||
|         py.callCoro("set_account_collapse", [model.id, collapse]) |         py.callCoro("set_account_collapse", [model.id, collapse]) | ||||||
|     } |     } | ||||||
| @@ -162,7 +162,7 @@ HTile { | |||||||
|  |  | ||||||
|             HShortcut { |             HShortcut { | ||||||
|                 enabled: enableKeybinds |                 enabled: enableKeybinds | ||||||
|                 sequences: window.settings.keys.addNewChat |                 sequences: window.settings.Keys.Rooms.add | ||||||
|                 onActivated: addChat.clicked() |                 onActivated: addChat.clicked() | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @@ -210,37 +210,37 @@ HTile { | |||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         enabled: enableKeybinds |         enabled: enableKeybinds | ||||||
|         sequences: window.settings.keys.accountSettings |         sequences: window.settings.Keys.Accounts.settings | ||||||
|         onActivated: leftClicked() |         onActivated: leftClicked() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         enabled: enableKeybinds |         enabled: enableKeybinds | ||||||
|         sequences: window.settings.keys.toggleCollapseAccount |         sequences: window.settings.Keys.Accounts.collapse | ||||||
|         onActivated: toggleCollapse() |         onActivated: toggleCollapse() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         enabled: enableKeybinds |         enabled: enableKeybinds | ||||||
|         sequences: window.settings.keys.openPresenceMenu |         sequences: window.settings.Keys.Accounts.menu | ||||||
|         onActivated: account.doRightClick(false) |         onActivated: account.doRightClick(false) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         enabled: enableKeybinds |         enabled: enableKeybinds | ||||||
|         sequences: window.settings.keys.togglePresenceUnavailable |         sequences: window.settings.Keys.Accounts.unavailable | ||||||
|         onActivated: account.togglePresence("unavailable") |         onActivated: account.togglePresence("unavailable") | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         enabled: enableKeybinds |         enabled: enableKeybinds | ||||||
|         sequences: window.settings.keys.togglePresenceInvisible |         sequences: window.settings.Keys.Accounts.invisible | ||||||
|         onActivated: account.togglePresence("invisible") |         onActivated: account.togglePresence("invisible") | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         enabled: enableKeybinds |         enabled: enableKeybinds | ||||||
|         sequences: window.settings.keys.togglePresenceOffline |         sequences: window.settings.Keys.Accounts.offline | ||||||
|         onActivated: account.togglePresence("offline") |         onActivated: account.togglePresence("offline") | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -32,7 +32,7 @@ Rectangle { | |||||||
|             Layout.fillHeight: true |             Layout.fillHeight: true | ||||||
|  |  | ||||||
|             HShortcut { |             HShortcut { | ||||||
|                 sequences: window.settings.keys.addNewAccount |                 sequences: window.settings.Keys.Accounts.add | ||||||
|                 onActivated: addAccountButton.clicked() |                 onActivated: addAccountButton.clicked() | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @@ -57,7 +57,7 @@ Rectangle { | |||||||
|             Keys.onEnterPressed: Keys.onReturnPressed(event) |             Keys.onEnterPressed: Keys.onReturnPressed(event) | ||||||
|             Keys.onReturnPressed: { |             Keys.onReturnPressed: { | ||||||
|                 roomList.showItemAtIndex() |                 roomList.showItemAtIndex() | ||||||
|                 if (window.settings.clearRoomFilterOnEnter) text = "" |                 if (window.settings.RoomList.enter_clears_filter) text = "" | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             Keys.onMenuPressed: |             Keys.onMenuPressed: | ||||||
| @@ -66,19 +66,19 @@ Rectangle { | |||||||
|  |  | ||||||
|             Keys.onEscapePressed: { |             Keys.onEscapePressed: { | ||||||
|                 mainPane.toggleFocus() |                 mainPane.toggleFocus() | ||||||
|                 if (window.settings.clearRoomFilterOnEscape) text = "" |                 if (window.settings.RoomList.escape_clears_filter) text = "" | ||||||
|             } |             } | ||||||
|  |  | ||||||
|  |  | ||||||
|             Behavior on opacity { HNumberAnimation {} } |             Behavior on opacity { HNumberAnimation {} } | ||||||
|  |  | ||||||
|             HShortcut { |             HShortcut { | ||||||
|                 sequences: window.settings.keys.clearRoomFilter |                 sequences: window.settings.Keys.Rooms.clear_filter | ||||||
|                 onActivated: filterField.text = "" |                 onActivated: filterField.text = "" | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             HShortcut { |             HShortcut { | ||||||
|                 sequences: window.settings.keys.toggleFocusMainPane |                 sequences: window.settings.Keys.Rooms.focus_filter | ||||||
|                 onActivated: mainPane.toggleFocus() |                 onActivated: mainPane.toggleFocus() | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -25,8 +25,9 @@ HDrawer { | |||||||
|  |  | ||||||
|     saveName: "mainPane" |     saveName: "mainPane" | ||||||
|     background: Rectangle { color: theme.mainPane.background } |     background: Rectangle { color: theme.mainPane.background } | ||||||
|     minimumSize: theme.mainPane.minimumSize |  | ||||||
|     requireDefaultSize: bottomBar.filterField.activeFocus |     requireDefaultSize: bottomBar.filterField.activeFocus | ||||||
|  |     minimumSize: | ||||||
|  |         window.settings.RoomList.min_width * window.settings.General.zoom | ||||||
|  |  | ||||||
|     Behavior on opacity { HNumberAnimation {} } |     Behavior on opacity { HNumberAnimation {} } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -73,12 +73,12 @@ HListView { | |||||||
|         ) : |         ) : | ||||||
|         pageLoader.showRoom(item.for_account, item.id) |         pageLoader.showRoom(item.for_account, item.id) | ||||||
|  |  | ||||||
|         if (fromClick && ! window.settings.centerRoomListOnClick) |         if (fromClick && ! window.settings.RoomList.click_centers) | ||||||
|             keepListCentered = false |             keepListCentered = false | ||||||
|  |  | ||||||
|         currentIndex = index |         currentIndex = index | ||||||
|  |  | ||||||
|         if (fromClick && ! window.settings.centerRoomListOnClick) |         if (fromClick && ! window.settings.RoomList.click_centers) | ||||||
|             keepListCentered = true |             keepListCentered = true | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -245,54 +245,53 @@ HListView { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.goToPreviousRoom |         sequences: window.settings.Keys.Rooms.previous | ||||||
|         onActivated: { decrementCurrentIndex(); showItemLimiter.restart() } |         onActivated: { decrementCurrentIndex(); showItemLimiter.restart() } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.goToNextRoom |         sequences: window.settings.Keys.Rooms.next | ||||||
|         onActivated: { incrementCurrentIndex(); showItemLimiter.restart() } |         onActivated: { incrementCurrentIndex(); showItemLimiter.restart() } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.goToPreviousUnreadRoom |         sequences: window.settings.Keys.Rooms.previous_unread | ||||||
|         onActivated: { cycleUnreadRooms(false) && showItemLimiter.restart() } |         onActivated: { cycleUnreadRooms(false) && showItemLimiter.restart() } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.goToNextUnreadRoom |         sequences: window.settings.Keys.Rooms.next_unread | ||||||
|         onActivated: { cycleUnreadRooms(true) && showItemLimiter.restart() } |         onActivated: { cycleUnreadRooms(true) && showItemLimiter.restart() } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.goToPreviousMentionedRoom |         sequences: window.settings.Keys.Rooms.previous_urgent | ||||||
|         onActivated: cycleUnreadRooms(false, true) && showItemLimiter.restart() |         onActivated: cycleUnreadRooms(false, true) && showItemLimiter.restart() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.goToNextMentionedRoom |         sequences: window.settings.Keys.Rooms.next_urgent | ||||||
|         onActivated: cycleUnreadRooms(true, true) && showItemLimiter.restart() |         onActivated: cycleUnreadRooms(true, true) && showItemLimiter.restart() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     Repeater { |     Repeater { | ||||||
|         model: Object.keys(window.settings.keys.focusAccountAtIndex) |         model: Object.keys(window.settings.Keys.Accounts.at_index) | ||||||
|  |  | ||||||
|         Item { |         Item { | ||||||
|             HShortcut { |             HShortcut { | ||||||
|                 sequence: window.settings.keys.focusAccountAtIndex[modelData] |                 sequence: window.settings.Keys.Accounts.at_index[modelData] | ||||||
|                 onActivated: goToAccountNumber(parseInt(modelData - 1, 10)) |                 onActivated: goToAccountNumber(parseInt(modelData, 10) - 1) | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     Repeater { |     Repeater { | ||||||
|         model: Object.keys(window.settings.keys.focusRoomAtIndex) |         model: Object.keys(window.settings.Keys.Rooms.at_index) | ||||||
|  |  | ||||||
|         Item { |         Item { | ||||||
|             HShortcut { |             HShortcut { | ||||||
|                 sequence: window.settings.keys.focusRoomAtIndex[modelData] |                 sequence: window.settings.Keys.Rooms.at_index[modelData] | ||||||
|                 onActivated: |                 onActivated: showAccountRoomAtIndex(parseInt(modelData,10) - 1) | ||||||
|                     showAccountRoomAtIndex(parseInt(modelData - 1, 10)) |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -44,7 +44,7 @@ HLoader { | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         window.uiState.pageProperties = properties |         window.uiState.pageProperties = properties | ||||||
|         window.uiStateChanged() |         window.saveUIState() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     function showRoom(userId, roomId) { |     function showRoom(userId, roomId) { | ||||||
| @@ -92,7 +92,7 @@ HLoader { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.goToLastPage |         sequences: window.settings.Keys.last_page | ||||||
|         onActivated: showPrevious() |         onActivated: showPrevious() | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -34,8 +34,10 @@ HFlickableColumnPage { | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (aliasFieldItem.changed) { |         if (aliasFieldItem.changed) { | ||||||
|             window.settings.writeAliases[userId] = aliasFieldItem.text |             window.settings.Chat.Composer.aliases[userId] = | ||||||
|             window.settingsChanged() |                 aliasFieldItem.text | ||||||
|  |  | ||||||
|  |             window.saveSettings() | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (avatar.changed) { |         if (avatar.changed) { | ||||||
| @@ -249,7 +251,7 @@ HFlickableColumnPage { | |||||||
|     HLabeledItem { |     HLabeledItem { | ||||||
|         id: aliasField |         id: aliasField | ||||||
|  |  | ||||||
|         readonly property var aliases: window.settings.writeAliases |         readonly property var aliases: window.settings.Chat.Composer.aliases | ||||||
|         readonly property string currentAlias: aliases[userId] || "" |         readonly property string currentAlias: aliases[userId] || "" | ||||||
|  |  | ||||||
|         readonly property bool hasWhiteSpace: /\s/.test(item.text) |         readonly property bool hasWhiteSpace: /\s/.test(item.text) | ||||||
|   | |||||||
| @@ -164,12 +164,12 @@ HColumnPage { | |||||||
|         Keys.onMenuPressed: Keys.onEnterPressed(event) |         Keys.onMenuPressed: Keys.onEnterPressed(event) | ||||||
|  |  | ||||||
|         HShortcut { |         HShortcut { | ||||||
|             sequences: window.settings.keys.refreshDevices |             sequences: window.settings.Keys.Sessions.refresh | ||||||
|             onActivated: refreshButton.clicked() |             onActivated: refreshButton.clicked() | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         HShortcut { |         HShortcut { | ||||||
|             sequences: window.settings.keys.signOutCheckedOrAllDevices |             sequences: window.settings.Keys.Sessions.sign_out_checked_or_all | ||||||
|             onActivated: signOutCheckedButton.clicked() |             onActivated: signOutCheckedButton.clicked() | ||||||
|         } |         } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -51,7 +51,7 @@ Item { | |||||||
|     onReadyChanged: longLoading = false |     onReadyChanged: longLoading = false | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.leaveRoom |         sequences: window.settings.Keys.Chat.leave | ||||||
|         active: userInfo && userInfo.presence !== "offline" |         active: userInfo && userInfo.presence !== "offline" | ||||||
|         onActivated: window.makePopup( |         onActivated: window.makePopup( | ||||||
|             "Popups/LeaveRoomPopup.qml", |             "Popups/LeaveRoomPopup.qml", | ||||||
| @@ -60,7 +60,7 @@ Item { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.forgetRoom |         sequences: window.settings.Keys.Chat.forget | ||||||
|         active: userInfo && userInfo.presence !== "offline" |         active: userInfo && userInfo.presence !== "offline" | ||||||
|         onActivated: window.makePopup( |         onActivated: window.makePopup( | ||||||
|             "Popups/ForgetRoomPopup.qml", |             "Popups/ForgetRoomPopup.qml", | ||||||
|   | |||||||
| @@ -21,7 +21,7 @@ HTextArea { | |||||||
|  |  | ||||||
|     readonly property var usableAliases: { |     readonly property var usableAliases: { | ||||||
|         const obj     = {} |         const obj     = {} | ||||||
|         const aliases = window.settings.writeAliases |         const aliases = window.settings.Chat.Composer.aliases | ||||||
|  |  | ||||||
|         // Get accounts that are members of this room with permission to talk |         // Get accounts that are members of this room with permission to talk | ||||||
|         for (const [id, alias] of Object.entries(aliases)) { |         for (const [id, alias] of Object.entries(aliases)) { | ||||||
|   | |||||||
| @@ -21,7 +21,7 @@ HButton { | |||||||
|     onClicked: sendFilePicker.dialog.open() |     onClicked: sendFilePicker.dialog.open() | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.sendFileFromPathInClipboard |         sequences: window.settings.Keys.Chat.send_clipboard_path | ||||||
|         onActivated: window.makePopup( |         onActivated: window.makePopup( | ||||||
|             "Popups/ConfirmUploadPopup.qml", |             "Popups/ConfirmUploadPopup.qml", | ||||||
|             { |             { | ||||||
| @@ -43,7 +43,7 @@ HButton { | |||||||
|         onReplied: chat.clearReplyTo() |         onReplied: chat.clearReplyTo() | ||||||
|  |  | ||||||
|         HShortcut { |         HShortcut { | ||||||
|             sequences: window.settings.keys.sendFile |             sequences: window.settings.Keys.Chat.send_file | ||||||
|             onActivated: sendFilePicker.dialog.open() |             onActivated: sendFilePicker.dialog.open() | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -14,7 +14,7 @@ Rectangle { | |||||||
|         (chat.roomPane.collapse || chat.roomPane.forceCollapse) |         (chat.roomPane.collapse || chat.roomPane.forceCollapse) | ||||||
|  |  | ||||||
|     readonly property bool center: |     readonly property bool center: | ||||||
|         showLeftButton || window.settings.alwaysCenterRoomHeader |         showLeftButton || window.settings.Chat.always_center_header | ||||||
|  |  | ||||||
|  |  | ||||||
|     implicitHeight: theme.baseElementsHeight |     implicitHeight: theme.baseElementsHeight | ||||||
|   | |||||||
| @@ -139,7 +139,9 @@ HColumnLayout { | |||||||
|                         stackView.currentItem.currentIndex = -1 |                         stackView.currentItem.currentIndex = -1 | ||||||
|  |  | ||||||
|                     roomPane.toggleFocus() |                     roomPane.toggleFocus() | ||||||
|                     if (window.settings.clearMemberFilterOnEscape) text = "" |  | ||||||
|  |                     if (window.settings.RoomList.escape_clears_filter) | ||||||
|  |                         text = "" | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 Behavior on opacity { HNumberAnimation {} } |                 Behavior on opacity { HNumberAnimation {} } | ||||||
| @@ -174,7 +176,7 @@ HColumnLayout { | |||||||
|                     Layout.preferredHeight: filterField.implicitHeight |                     Layout.preferredHeight: filterField.implicitHeight | ||||||
|  |  | ||||||
|                     HShortcut { |                     HShortcut { | ||||||
|                         sequences: window.settings.keys.inviteToRoom |                         sequences: window.settings.Keys.Chat.invite | ||||||
|                         onActivated: |                         onActivated: | ||||||
|                             if (inviteButton.enabled) inviteButton.clicked() |                             if (inviteButton.enabled) inviteButton.clicked() | ||||||
|                     } |                     } | ||||||
|   | |||||||
| @@ -105,12 +105,12 @@ MultiviewPane { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.toggleFocusRoomPane |         sequences: window.settings.Keys.Chat.focus_room_pane | ||||||
|         onActivated: roomPane.toggleFocus() |         onActivated: roomPane.toggleFocus() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.toggleHideRoomPane |         sequences: window.settings.Keys.Chat.hide_room_pane | ||||||
|         onActivated: roomPane.forceCollapse = ! roomPane.forceCollapse |         onActivated: roomPane.forceCollapse = ! roomPane.forceCollapse | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -75,11 +75,11 @@ HRowLayout { | |||||||
|     readonly property int maxMessageWidth: |     readonly property int maxMessageWidth: | ||||||
|         contentText.includes("<pre>") || contentText.includes("<table>") ? |         contentText.includes("<pre>") || contentText.includes("<table>") ? | ||||||
|         -1 : |         -1 : | ||||||
|         window.settings.maxMessageCharactersPerLine < 0 ? |         window.settings.Chat.max_messages_line_length < 0 ? | ||||||
|         -1 : |         -1 : | ||||||
|         Math.ceil( |         Math.ceil( | ||||||
|             mainUI.fontMetrics.averageCharacterWidth * |             mainUI.fontMetrics.averageCharacterWidth * | ||||||
|             window.settings.maxMessageCharactersPerLine |             window.settings.Chat.max_messages_line_length | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     readonly property alias selectedText: contentLabel.selectedPlainText |     readonly property alias selectedText: contentLabel.selectedPlainText | ||||||
|   | |||||||
| @@ -17,7 +17,7 @@ HColumnLayout { | |||||||
|     readonly property var nextModel: eventList.model.get(model.index - 1) |     readonly property var nextModel: eventList.model.get(model.index - 1) | ||||||
|     readonly property QtObject currentModel: model |     readonly property QtObject currentModel: model | ||||||
|  |  | ||||||
|     readonly property bool compact: window.settings.compactMode |     readonly property bool compact: window.settings.General.compact | ||||||
|     readonly property bool checked: model.id in eventList.checked |     readonly property bool checked: model.id in eventList.checked | ||||||
|     readonly property bool isOwn: chat.userId === model.sender_id |     readonly property bool isOwn: chat.userId === model.sender_id | ||||||
|     readonly property bool isRedacted: model.event_type === "RedactedEvent" |     readonly property bool isRedacted: model.event_type === "RedactedEvent" | ||||||
|   | |||||||
| @@ -17,7 +17,12 @@ HTile { | |||||||
|     width: Math.min( |     width: Math.min( | ||||||
|         eventDelegate.width, |         eventDelegate.width, | ||||||
|         eventContent.maxMessageWidth, |         eventContent.maxMessageWidth, | ||||||
|         Math.max(theme.chat.message.fileMinWidth, implicitWidth), |         Math.max( | ||||||
|  |             window.settings.Chat.Files.min_file_width * | ||||||
|  |             window.settings.General.zoom, | ||||||
|  |  | ||||||
|  |             implicitWidth, | ||||||
|  |         ), | ||||||
|     ) |     ) | ||||||
|     height: Math.max(theme.chat.message.avatarSize, implicitHeight) |     height: Math.max(theme.chat.message.avatarSize, implicitHeight) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -10,13 +10,16 @@ HMxcImage { | |||||||
|  |  | ||||||
|     property EventMediaLoader loader |     property EventMediaLoader loader | ||||||
|  |  | ||||||
|  |     readonly property real zoom: window.settings.General.zoom | ||||||
|  |  | ||||||
|     readonly property real maxHeight: |     readonly property real maxHeight: | ||||||
|         eventList.height * theme.chat.message.thumbnailMaxHeightRatio |         eventList.height * | ||||||
|  |         window.settings.Chat.Files.max_thumbnail_height_ratio * zoom | ||||||
|  |  | ||||||
|     readonly property size fitSize: utils.fitSize( |     readonly property size fitSize: utils.fitSize( | ||||||
|         // Minimum display size |         // Minimum display size | ||||||
|         theme.chat.message.thumbnailMinSize.width, |         window.settings.Chat.Files.min_thumbnail_size[0] * zoom, | ||||||
|         theme.chat.message.thumbnailMinSize.height, |         window.settings.Chat.Files.min_thumbnail_size[1] * zoom, | ||||||
|  |  | ||||||
|         // Real size |         // Real size | ||||||
|         ( |         ( | ||||||
| @@ -35,12 +38,18 @@ HMxcImage { | |||||||
|  |  | ||||||
|         // Maximum display size |         // Maximum display size | ||||||
|         Math.min( |         Math.min( | ||||||
|             Math.max(maxHeight, theme.chat.message.thumbnailMinSize.width), |             Math.max( | ||||||
|  |                 maxHeight, | ||||||
|  |                 window.settings.Chat.Files.min_thumbnail_size[0] * zoom, | ||||||
|  |             ), | ||||||
|             pureMedia ? Infinity : eventContent.maxMessageWidth, |             pureMedia ? Infinity : eventContent.maxMessageWidth, | ||||||
|             eventDelegate.width - eventContent.spacing - avatarWrapper.width - |             eventDelegate.width - eventContent.spacing - avatarWrapper.width - | ||||||
|             eventContent.spacing * 2,  // padding |             eventContent.spacing * 2,  // padding | ||||||
|         ), |         ), | ||||||
|         Math.max(maxHeight, theme.chat.message.thumbnailMinSize.height), |         Math.max( | ||||||
|  |             maxHeight, | ||||||
|  |             window.settings.Chat.Files.min_thumbnail_size[1] * zoom, | ||||||
|  |         ), | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     readonly property bool hovered: hover.hovered |     readonly property bool hovered: hover.hovered | ||||||
| @@ -92,7 +101,7 @@ HMxcImage { | |||||||
|                 return |                 return | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             window.settings.media.openExternallyOnClick ? |             window.settings.Chat.Files.click_opens_externally ? | ||||||
|             image.openExternally() : |             image.openExternally() : | ||||||
|             image.openInternally() |             image.openInternally() | ||||||
|         } |         } | ||||||
| @@ -103,7 +112,7 @@ HMxcImage { | |||||||
|         acceptedModifiers: Qt.NoModifier |         acceptedModifiers: Qt.NoModifier | ||||||
|         gesturePolicy: TapHandler.ReleaseWithinBounds |         gesturePolicy: TapHandler.ReleaseWithinBounds | ||||||
|         onTapped: |         onTapped: | ||||||
|             window.settings.media.openExternallyOnClick ? |             window.settings.Chat.Files.click_opens_externally ? | ||||||
|             image.openInternally() : |             image.openInternally() : | ||||||
|             image.openExternally() |             image.openExternally() | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -21,7 +21,7 @@ Rectangle { | |||||||
|     color: theme.chat.eventList.background |     color: theme.chat.eventList.background | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.unfocusOrDeselectAllMessages |         sequences: window.settings.Keys.Messages.unfocus_or_deselect | ||||||
|         onActivated: { |         onActivated: { | ||||||
|             eventList.selectedCount ? |             eventList.selectedCount ? | ||||||
|             eventList.checked = {} : |             eventList.checked = {} : | ||||||
| @@ -30,24 +30,24 @@ Rectangle { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.focusPreviousMessage |         sequences: window.settings.Keys.Messages.previous | ||||||
|         onActivated: eventList.focusPreviousMessage() |         onActivated: eventList.focusPreviousMessage() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.focusNextMessage |         sequences: window.settings.Keys.Messages.next | ||||||
|         onActivated: eventList.focusNextMessage() |         onActivated: eventList.focusNextMessage() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         active: eventList.currentItem |         active: eventList.currentItem | ||||||
|         sequences: window.settings.keys.toggleSelectMessage |         sequences: window.settings.Keys.Messages.select | ||||||
|         onActivated: eventList.toggleCheck(eventList.currentIndex) |         onActivated: eventList.toggleCheck(eventList.currentIndex) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         active: eventList.currentItem |         active: eventList.currentItem | ||||||
|         sequences: window.settings.keys.selectMessagesUntilHere |         sequences: window.settings.Keys.Messages.select_until_here | ||||||
|         onActivated: |         onActivated: | ||||||
|             eventList.checkFromLastToHere(eventList.currentIndex) |             eventList.checkFromLastToHere(eventList.currentIndex) | ||||||
|     } |     } | ||||||
| @@ -75,7 +75,7 @@ Rectangle { | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         enabled: (events && events.length > 0) || events === null |         enabled: (events && events.length > 0) || events === null | ||||||
|         sequences: window.settings.keys.removeFocusedOrSelectedMessages |         sequences: window.settings.Keys.Messages.remove | ||||||
|         onActivated: window.makePopup( |         onActivated: window.makePopup( | ||||||
|             "Popups/RedactPopup.qml", |             "Popups/RedactPopup.qml", | ||||||
|             { |             { | ||||||
| @@ -98,7 +98,7 @@ Rectangle { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.replyToFocusedOrLastMessage |         sequences: window.settings.Keys.Messages.reply | ||||||
|         onActivated: { |         onActivated: { | ||||||
|             let event = eventList.model.get(0) |             let event = eventList.model.get(0) | ||||||
|  |  | ||||||
| @@ -132,7 +132,7 @@ Rectangle { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.openMessagesLinksOrFiles |         sequences: window.settings.Keys.Messages.open_links_files | ||||||
|         onActivated: { |         onActivated: { | ||||||
|             const indice = |             const indice = | ||||||
|                 eventList.getFocusedOrSelectedOrLastMediaEvents(true) |                 eventList.getFocusedOrSelectedOrLastMediaEvents(true) | ||||||
| @@ -158,7 +158,7 @@ Rectangle { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.openMessagesLinksOrFilesExternally |         sequences: window.settings.Keys.Messages.open_links_files_externally | ||||||
|         onActivated: { |         onActivated: { | ||||||
|             const indice = |             const indice = | ||||||
|                 eventList.getFocusedOrSelectedOrLastMediaEvents(true) |                 eventList.getFocusedOrSelectedOrLastMediaEvents(true) | ||||||
| @@ -178,7 +178,7 @@ Rectangle { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.copyFilesLocalPath |         sequences: window.settings.Keys.Messages.copy_files_path | ||||||
|         onActivated: { |         onActivated: { | ||||||
|             const paths  = [] |             const paths  = [] | ||||||
|             const indice = |             const indice = | ||||||
| @@ -199,14 +199,14 @@ Rectangle { | |||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         active: eventList.currentItem |         active: eventList.currentItem | ||||||
|         sequences: window.settings.keys.debugFocusedMessage |         sequences: window.settings.Keys.Messages.debug | ||||||
|         onActivated: mainUI.debugConsole.toggle( |         onActivated: mainUI.debugConsole.toggle( | ||||||
|             eventList.currentItem.eventContent, "t.parent.json()", |             eventList.currentItem.eventContent, "t.parent.json()", | ||||||
|         ) |         ) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.clearRoomMessages |         sequences: window.settings.Keys.Messages.clear_all | ||||||
|         onActivated: window.makePopup( |         onActivated: window.makePopup( | ||||||
|             "Popups/ClearMessagesPopup.qml", |             "Popups/ClearMessagesPopup.qml", | ||||||
|             { |             { | ||||||
| @@ -231,9 +231,10 @@ Rectangle { | |||||||
|         property bool moreToLoad: true |         property bool moreToLoad: true | ||||||
|  |  | ||||||
|         property bool ownEventsOnLeft: |         property bool ownEventsOnLeft: | ||||||
|             window.settings.ownMessagesOnLeftAboveWidth < 0 ? |             window.settings.Chat.own_messages_on_left_above < 0 ? | ||||||
|             false : |             false : | ||||||
|             width > window.settings.ownMessagesOnLeftAboveWidth * theme.uiScale |             width > | ||||||
|  |             window.settings.Chat.own_messages_on_left_above * theme.uiScale | ||||||
|  |  | ||||||
|         property string delegateWithSelectedText: "" |         property string delegateWithSelectedText: "" | ||||||
|         property string selectedText: "" |         property string selectedText: "" | ||||||
| @@ -616,7 +617,7 @@ Rectangle { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     Timer { |     Timer { | ||||||
|         interval: Math.max(100, window.settings.markRoomReadMsecDelay) |         interval: Math.max(100, window.settings.Chat.mark_read_delay * 1000) | ||||||
|  |  | ||||||
|         running: |         running: | ||||||
|             ! eventList.updateMarkerFutureId && |             ! eventList.updateMarkerFutureId && | ||||||
|   | |||||||
| @@ -123,7 +123,7 @@ HPopup { | |||||||
|  |  | ||||||
|         Timer { |         Timer { | ||||||
|             id: autoHideTimer |             id: autoHideTimer | ||||||
|             interval: window.settings.media.autoHideOSDAfterMsec |             interval: window.settings.Chat.Files.autohide_image_controls_after | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         ViewerInfo { |         ViewerInfo { | ||||||
|   | |||||||
| @@ -29,7 +29,7 @@ HFlow { | |||||||
|         visible: viewer.isAnimated |         visible: viewer.isAnimated | ||||||
|  |  | ||||||
|         HPopupShortcut { |         HPopupShortcut { | ||||||
|             sequences: window.settings.keys.imageViewer.pause |             sequences: window.settings.Keys.ImageViewer.pause | ||||||
|             onActivated: pause.clicked() |             onActivated: pause.clicked() | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -46,7 +46,7 @@ HFlow { | |||||||
|         visible: viewer.isAnimated |         visible: viewer.isAnimated | ||||||
|  |  | ||||||
|         HPopupShortcut { |         HPopupShortcut { | ||||||
|             sequences: window.settings.keys.imageViewer.previousSpeed |             sequences: window.settings.Keys.ImageViewer.slow_down | ||||||
|             onActivated: viewer.imagesSpeed = viewer.availableSpeeds[Math.min( |             onActivated: viewer.imagesSpeed = viewer.availableSpeeds[Math.min( | ||||||
|                 viewer.availableSpeeds.indexOf(viewer.imagesSpeed) + 1, |                 viewer.availableSpeeds.indexOf(viewer.imagesSpeed) + 1, | ||||||
|                 viewer.availableSpeeds.length - 1, |                 viewer.availableSpeeds.length - 1, | ||||||
| @@ -54,14 +54,14 @@ HFlow { | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         HPopupShortcut { |         HPopupShortcut { | ||||||
|             sequences: window.settings.keys.imageViewer.nextSpeed |             sequences: window.settings.Keys.ImageViewer.speed_up | ||||||
|             onActivated: viewer.imagesSpeed = viewer.availableSpeeds[Math.max( |             onActivated: viewer.imagesSpeed = viewer.availableSpeeds[Math.max( | ||||||
|                 viewer.availableSpeeds.indexOf(viewer.imagesSpeed) - 1, 0, |                 viewer.availableSpeeds.indexOf(viewer.imagesSpeed) - 1, 0, | ||||||
|             )] |             )] | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         HPopupShortcut { |         HPopupShortcut { | ||||||
|             sequences: window.settings.keys.imageViewer.resetSpeed |             sequences: window.settings.Keys.ImageViewer.reset_speed | ||||||
|             onActivated: viewer.imagesSpeed = 1 |             onActivated: viewer.imagesSpeed = 1 | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -77,7 +77,7 @@ HFlow { | |||||||
|         onPressed: viewer.animatedRotationTarget -= 45 |         onPressed: viewer.animatedRotationTarget -= 45 | ||||||
|  |  | ||||||
|         HPopupShortcut { |         HPopupShortcut { | ||||||
|             sequences: window.settings.keys.imageViewer.rotateLeft |             sequences: window.settings.Keys.ImageViewer.rotate_left | ||||||
|             onActivated: viewer.animatedRotationTarget -= 45 |             onActivated: viewer.animatedRotationTarget -= 45 | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -93,13 +93,13 @@ HFlow { | |||||||
|         onPressed: viewer.animatedRotationTarget += 45 |         onPressed: viewer.animatedRotationTarget += 45 | ||||||
|  |  | ||||||
|         HPopupShortcut { |         HPopupShortcut { | ||||||
|             sequences: window.settings.keys.imageViewer.rotateRight |             sequences: window.settings.Keys.ImageViewer.rotate_right | ||||||
|             onActivated: viewer.animatedRotationTarget += 45 |             onActivated: viewer.animatedRotationTarget += 45 | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HPopupShortcut { |     HPopupShortcut { | ||||||
|         sequences: window.settings.keys.imageViewer.rotateReset |         sequences: window.settings.Keys.ImageViewer.reset_rotation | ||||||
|         onActivated: viewer.animatedRotationTarget = 0 |         onActivated: viewer.animatedRotationTarget = 0 | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -116,7 +116,7 @@ HFlow { | |||||||
|         onClicked: viewer.alternateScaling = ! viewer.alternateScaling |         onClicked: viewer.alternateScaling = ! viewer.alternateScaling | ||||||
|  |  | ||||||
|         HPopupShortcut { |         HPopupShortcut { | ||||||
|             sequences: window.settings.keys.imageViewer.expand |             sequences: window.settings.Keys.ImageViewer.expand | ||||||
|             onActivated: expand.clicked() |             onActivated: expand.clicked() | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -131,7 +131,7 @@ HFlow { | |||||||
|         visible: Qt.application.supportsMultipleWindows |         visible: Qt.application.supportsMultipleWindows | ||||||
|  |  | ||||||
|         HPopupShortcut { |         HPopupShortcut { | ||||||
|             sequences: window.settings.keys.imageViewer.fullScreen |             sequences: window.settings.Keys.ImageViewer.fullscreen | ||||||
|             onActivated: fullScreen.clicked() |             onActivated: fullScreen.clicked() | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -144,7 +144,7 @@ HFlow { | |||||||
|         onClicked: viewer.close() |         onClicked: viewer.close() | ||||||
|  |  | ||||||
|         HPopupShortcut { |         HPopupShortcut { | ||||||
|             sequences: window.settings.keys.imageViewer.close |             sequences: window.settings.Keys.ImageViewer.close | ||||||
|             onActivated: close.clicked() |             onActivated: close.clicked() | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -53,37 +53,37 @@ HFlickable { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     HPopupShortcut { |     HPopupShortcut { | ||||||
|         sequences: window.settings.keys.imageViewer.panLeft |         sequences: window.settings.Keys.ImageViewer.pan_left | ||||||
|         onActivated: utils.flickPages(flickable, -0.2, true, 5) |         onActivated: utils.flickPages(flickable, -0.2, true, 5) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HPopupShortcut { |     HPopupShortcut { | ||||||
|         sequences: window.settings.keys.imageViewer.panRight |         sequences: window.settings.Keys.ImageViewer.pan_right | ||||||
|         onActivated: utils.flickPages(flickable, 0.2, true, 5) |         onActivated: utils.flickPages(flickable, 0.2, true, 5) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HPopupShortcut { |     HPopupShortcut { | ||||||
|         sequences: window.settings.keys.imageViewer.panUp |         sequences: window.settings.Keys.ImageViewer.pan_up | ||||||
|         onActivated: utils.flickPages(flickable, -0.2, false, 5) |         onActivated: utils.flickPages(flickable, -0.2, false, 5) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HPopupShortcut { |     HPopupShortcut { | ||||||
|         sequences: window.settings.keys.imageViewer.panDown |         sequences: window.settings.Keys.ImageViewer.pan_down | ||||||
|         onActivated: utils.flickPages(flickable, 0.2, false, 5) |         onActivated: utils.flickPages(flickable, 0.2, false, 5) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HPopupShortcut { |     HPopupShortcut { | ||||||
|         sequences: window.settings.keys.imageViewer.zoomOut |         sequences: window.settings.Keys.ImageViewer.zoom_out | ||||||
|         onActivated: thumbnail.scale = Math.max(0.1, thumbnail.scale - 0.2) |         onActivated: thumbnail.scale = Math.max(0.1, thumbnail.scale - 0.2) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HPopupShortcut { |     HPopupShortcut { | ||||||
|         sequences: window.settings.keys.imageViewer.zoomIn |         sequences: window.settings.Keys.ImageViewer.zoom_in | ||||||
|         onActivated: thumbnail.scale = Math.min(10, thumbnail.scale + 0.2) |         onActivated: thumbnail.scale = Math.min(10, thumbnail.scale + 0.2) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HPopupShortcut { |     HPopupShortcut { | ||||||
|         sequences: window.settings.keys.imageViewer.zoomReset |         sequences: window.settings.Keys.ImageViewer.reset_zoom | ||||||
|         onActivated: resetScaleAnimation.start() |         onActivated: resetScaleAnimation.start() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -28,8 +28,8 @@ QtObject { | |||||||
|  |  | ||||||
|         const msec = |         const msec = | ||||||
|             highImportance ? |             highImportance ? | ||||||
|             window.settings.alertOnMentionForMsec : |             window.settings.Notifications.urgent_alert_time * 1000 : | ||||||
|             window.settings.alertOnMessageForMsec |             window.settings.Notifications.alert_time * 1000 | ||||||
|  |  | ||||||
|         if (msec) window.alert(msec === -1 ? 0 : msec)  // -1 → 0 = no time out |         if (msec) window.alert(msec === -1 ? 0 : msec)  // -1 → 0 = no time out | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -13,37 +13,37 @@ HQtObject { | |||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         active: root.active |         active: root.active | ||||||
|         sequences: window.settings.keys.scrollUp |         sequences: window.settings.Keys.Scrolling.up | ||||||
|         onActivated: utils.flickPages(flickable, -1 / 10) |         onActivated: utils.flickPages(flickable, -1 / 10) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         active: root.active |         active: root.active | ||||||
|         sequences: window.settings.keys.scrollDown |         sequences: window.settings.Keys.Scrolling.down | ||||||
|         onActivated: utils.flickPages(flickable, 1 / 10) |         onActivated: utils.flickPages(flickable, 1 / 10) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         active: root.active |         active: root.active | ||||||
|         sequences: window.settings.keys.scrollPageUp |         sequences: window.settings.Keys.Scrolling.page_up | ||||||
|         onActivated: utils.flickPages(flickable, -1) |         onActivated: utils.flickPages(flickable, -1) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         active: root.active |         active: root.active | ||||||
|         sequences: window.settings.keys.scrollPageDown |         sequences: window.settings.Keys.Scrolling.page_down | ||||||
|         onActivated: utils.flickPages(flickable, 1) |         onActivated: utils.flickPages(flickable, 1) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         active: root.active |         active: root.active | ||||||
|         sequences: window.settings.keys.scrollToTop |         sequences: window.settings.Keys.Scrolling.top | ||||||
|         onActivated: utils.flickToTop(flickable) |         onActivated: utils.flickToTop(flickable) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         active: root.active |         active: root.active | ||||||
|         sequences: window.settings.keys.scrollToBottom |         sequences: window.settings.Keys.Scrolling.bottom | ||||||
|         onActivated: utils.flickToBottom(flickable) |         onActivated: utils.flickToBottom(flickable) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -13,7 +13,7 @@ HQtObject { | |||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         active: root.active |         active: root.active | ||||||
|         sequences: window.settings.keys.previousTab |         sequences: window.settings.Keys.previous_tab | ||||||
|         onActivated: container.setCurrentIndex( |         onActivated: container.setCurrentIndex( | ||||||
|             utils.numberWrapAt(container.currentIndex - 1, container.count), |             utils.numberWrapAt(container.currentIndex - 1, container.count), | ||||||
|         ) |         ) | ||||||
| @@ -21,7 +21,7 @@ HQtObject { | |||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         active: root.active |         active: root.active | ||||||
|         sequences: window.settings.keys.nextTab |         sequences: window.settings.Keys.next_tab | ||||||
|         onActivated: container.setCurrentIndex( |         onActivated: container.setCurrentIndex( | ||||||
|             utils.numberWrapAt(container.currentIndex + 1, container.count), |             utils.numberWrapAt(container.currentIndex + 1, container.count), | ||||||
|         ) |         ) | ||||||
|   | |||||||
| @@ -36,39 +36,41 @@ Item { | |||||||
|     Component.onCompleted: window.mainUI = mainUI |     Component.onCompleted: window.mainUI = mainUI | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.startPythonDebugger |         sequences: window.settings.Keys.python_debugger | ||||||
|         onActivated: py.call("BRIDGE.pdb") |         onActivated: py.call("BRIDGE.pdb") | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.zoomIn |         sequences: window.settings.Keys.zoom_in | ||||||
|         onActivated: { |         onActivated: { | ||||||
|             window.settings.zoom += 0.1 |             window.settings.General.zoom += 0.1 | ||||||
|             window.settingsChanged() |             window.saveSettings() | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.zoomOut |         sequences: window.settings.Keys.zoom_out | ||||||
|         onActivated: { |         onActivated: { | ||||||
|             window.settings.zoom = Math.max(0.1, window.settings.zoom - 0.1) |             window.settings.General.zoom = | ||||||
|             window.settingsChanged() |                 Math.max(0.1, window.settings.General.zoom - 0.1) | ||||||
|  |  | ||||||
|  |             window.saveSettings() | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.zoomReset |         sequences: window.settings.Keys.reset_zoom | ||||||
|         onActivated: { |         onActivated: { | ||||||
|             window.settings.zoom = 1 |             window.settings.General.zoom = 1 | ||||||
|             window.settingsChanged() |             window.saveSettings() | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     HShortcut { |     HShortcut { | ||||||
|         sequences: window.settings.keys.toggleCompactMode |         sequences: window.settings.Keys.compact | ||||||
|         onActivated: { |         onActivated: { | ||||||
|             settings.compactMode = ! settings.compactMode |             window.settings.General.compact = ! window.settings.General.compact | ||||||
|             settingsChanged() |             windowsaveSettings() | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -36,6 +36,20 @@ ApplicationWindow { | |||||||
|     readonly property bool anyPopup: Object.keys(visiblePopups).length > 0 |     readonly property bool anyPopup: Object.keys(visiblePopups).length > 0 | ||||||
|     readonly property bool anyPopupOrMenu: anyMenu || anyPopup |     readonly property bool anyPopupOrMenu: anyMenu || anyPopup | ||||||
|  |  | ||||||
|  |     function saveSettings() { | ||||||
|  |         settingsChanged() | ||||||
|  |         py.saveConfig("settings", settings) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function saveUIState() { | ||||||
|  |         uiStateChanged() | ||||||
|  |         py.saveConfig("ui_state", uiState) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function saveHistory() { | ||||||
|  |         historyChanged() | ||||||
|  |         py.saveConfig("history", history) | ||||||
|  |     } | ||||||
|  |  | ||||||
|     function saveState(obj) { |     function saveState(obj) { | ||||||
|         if (! obj.saveName || ! obj.saveProperties || |         if (! obj.saveName || ! obj.saveProperties || | ||||||
| @@ -51,7 +65,7 @@ ApplicationWindow { | |||||||
|             [obj.saveName]: { [obj.saveId || "ALL"]: propertyValues }, |             [obj.saveName]: { [obj.saveId || "ALL"]: propertyValues }, | ||||||
|         }) |         }) | ||||||
|  |  | ||||||
|         uiStateChanged() |         saveUIState() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     function getState(obj, property, defaultValue=undefined) { |     function getState(obj, property, defaultValue=undefined) { | ||||||
| @@ -86,15 +100,9 @@ ApplicationWindow { | |||||||
|     visible: true |     visible: true | ||||||
|     color: "transparent" |     color: "transparent" | ||||||
|  |  | ||||||
|     // NOTE: For JS object variables, the corresponding method to notify |  | ||||||
|     // key/value changes must be called manually, e.g. settingsChanged(). |  | ||||||
|     onSettingsChanged: py.saveConfig("settings", settings) |  | ||||||
|     onUiStateChanged: py.saveConfig("ui_state", uiState) |  | ||||||
|     onHistoryChanged: py.saveConfig("history", history) |  | ||||||
|  |  | ||||||
|     onClosing: { |     onClosing: { | ||||||
|         close.accepted = ! settings.closeMinimizesToTray |         close.accepted = ! settings.General.close_to_tray | ||||||
|         settings.closeMinimizesToTray ? hide() : Qt.quit() |         settings.General.close_to_tray ? hide() : Qt.quit() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     PythonRootBridge { id: py } |     PythonRootBridge { id: py } | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ | |||||||
|  |  | ||||||
| // Base variables | // Base variables | ||||||
|  |  | ||||||
| real uiScale: window.settings.zoom | real uiScale: window.settings.General.zoom | ||||||
|  |  | ||||||
| int minimumSupportedWidth:  240 * uiScale | int minimumSupportedWidth:  240 * uiScale | ||||||
| int minimumSupportedHeight: 120 * uiScale | int minimumSupportedHeight: 120 * uiScale | ||||||
| @@ -211,7 +211,6 @@ controls: | |||||||
|         color placeholderText: controls.textField.placeholderText |         color placeholderText: controls.textField.placeholderText | ||||||
|  |  | ||||||
|     toolTip: |     toolTip: | ||||||
|         int delay:        500 |  | ||||||
|         color background: colors.strongBackground |         color background: colors.strongBackground | ||||||
|         color text:       colors.text |         color text:       colors.text | ||||||
|         color border:     "black" |         color border:     "black" | ||||||
| @@ -296,7 +295,6 @@ ui: | |||||||
|  |  | ||||||
|  |  | ||||||
| mainPane: | mainPane: | ||||||
|     int minimumSize: 144 * uiScale |  | ||||||
|     color background: "transparent" |     color background: "transparent" | ||||||
|  |  | ||||||
|     topBar: |     topBar: | ||||||
| @@ -465,17 +463,6 @@ chat: | |||||||
|         string styleInclude: |         string styleInclude: | ||||||
|             '<style type"text/css">\n' + styleSheet + '\n</style>\n' |             '<style type"text/css">\n' + styleSheet + '\n</style>\n' | ||||||
|  |  | ||||||
|         // Prefered minimum width of file messages |  | ||||||
|         int fileMinWidth: 256 * uiScale |  | ||||||
|  |  | ||||||
|         // Don't scale down thumbnails below this size in pixels, if |  | ||||||
|         // the becomes too small to show it at normal size. |  | ||||||
|         size thumbnailMinSize: Qt.size(256 * uiScale, 256 * uiScale) |  | ||||||
|  |  | ||||||
|         // How much of the chat height thumbnails can take at most, |  | ||||||
|         // by default 0.4 for 40%. |  | ||||||
|         real thumbnailMaxHeightRatio: 0.4 * Math.min(1, uiScale) |  | ||||||
|  |  | ||||||
|         real thumbnailCheckedOverlayOpacity: 0.4 |         real thumbnailCheckedOverlayOpacity: 0.4 | ||||||
|  |  | ||||||
|     daybreak: |     daybreak: | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ | |||||||
|  |  | ||||||
| // Base variables | // Base variables | ||||||
|  |  | ||||||
| real uiScale: window.settings.zoom | real uiScale: window.settings.General.zoom | ||||||
|  |  | ||||||
| int minimumSupportedWidth:  240 * uiScale | int minimumSupportedWidth:  240 * uiScale | ||||||
| int minimumSupportedHeight: 120 * uiScale | int minimumSupportedHeight: 120 * uiScale | ||||||
| @@ -217,7 +217,6 @@ controls: | |||||||
|         color placeholderText: controls.textField.placeholderText |         color placeholderText: controls.textField.placeholderText | ||||||
|  |  | ||||||
|     toolTip: |     toolTip: | ||||||
|         int delay:        500 |  | ||||||
|         color background: colors.strongBackground |         color background: colors.strongBackground | ||||||
|         color text:       colors.text |         color text:       colors.text | ||||||
|         color border:     "black" |         color border:     "black" | ||||||
| @@ -309,7 +308,6 @@ ui: | |||||||
|  |  | ||||||
|  |  | ||||||
| mainPane: | mainPane: | ||||||
|     int minimumSize: 144 * uiScale |  | ||||||
|     color background: "transparent" |     color background: "transparent" | ||||||
|  |  | ||||||
|     topBar: |     topBar: | ||||||
| @@ -474,17 +472,6 @@ chat: | |||||||
|         string styleInclude: |         string styleInclude: | ||||||
|             '<style type"text/css">\n' + styleSheet + '\n</style>\n' |             '<style type"text/css">\n' + styleSheet + '\n</style>\n' | ||||||
|  |  | ||||||
|         // Prefered minimum width of file messages |  | ||||||
|         int fileMinWidth: 256 * uiScale |  | ||||||
|  |  | ||||||
|         // Don't scale down thumbnails below this size in pixels, if |  | ||||||
|         // the becomes too small to show it at normal size. |  | ||||||
|         size thumbnailMinSize: Qt.size(256 * uiScale, 256 * uiScale) |  | ||||||
|  |  | ||||||
|         // How much of the chat height thumbnails can take at most, |  | ||||||
|         // by default 0.4 for 40%. |  | ||||||
|         real thumbnailMaxHeightRatio: 0.4 * Math.min(1, uiScale) |  | ||||||
|  |  | ||||||
|         real thumbnailCheckedOverlayOpacity: 0.4 |         real thumbnailCheckedOverlayOpacity: 0.4 | ||||||
|  |  | ||||||
|     daybreak: |     daybreak: | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	