Skip to content

Commit

Permalink
Convert to decorator
Browse files Browse the repository at this point in the history
  • Loading branch information
jwortmann committed Oct 16, 2024
1 parent 7a8b482 commit 0d5e561
Showing 1 changed file with 27 additions and 15 deletions.
42 changes: 27 additions & 15 deletions plugin/documents.py
Original file line number Diff line number Diff line change
Expand Up @@ -55,8 +55,10 @@
from .session_buffer import SessionBuffer
from .session_view import SessionView
from functools import partial
from typing import Any, Callable, Generator, Iterable
from functools import wraps
from typing import Any, Callable, Generator, Iterable, TypeVar
from typing import cast
from typing_extensions import ParamSpec
from weakref import WeakSet
from weakref import WeakValueDictionary
import itertools
Expand All @@ -68,6 +70,22 @@

SUBLIME_WORD_MASK = 515

P = ParamSpec('P')
R = TypeVar('R')


def requires_session(func: Callable[P, R]) -> Callable[P, R | None]:
"""
A decorator for the `DocumentSyncListener` event handlers, which immediately returns `None` if there are no
`SessionView`s.
"""
@wraps(func)
def wrapper(self: DocumentSyncListener, *args: P.args, **kwargs: P.kwargs) -> R | None:
if not self.session_views_async():
return None
return func(self, *args, **kwargs) # pyright: ignore[reportCallIssue]
return wrapper # pyright: ignore[reportReturnType]


def is_regular_view(v: sublime.View) -> bool:
# Not from the quick panel (CTRL+P), and not a special view like a console, output panel or find-in-files panels.
Expand Down Expand Up @@ -327,12 +345,10 @@ def session_buffers_async(self, capability: str | None = None) -> list[SessionBu
def session_views_async(self) -> list[SessionView]:
return list(self._session_views.values())

@requires_session
def on_text_changed_async(self, change_count: int, changes: Iterable[sublime.TextChange]) -> None:
session_views = self.session_views_async()
if not session_views:
return
if self.view.is_primary():
for sv in session_views:
for sv in self.session_views_async():
sv.on_text_changed_async(change_count, changes)
self._on_view_updated_async()

Expand Down Expand Up @@ -387,9 +403,9 @@ def on_activated_async(self) -> None:
sb.set_inlay_hints_pending_refresh(needs_refresh=False)
sb.do_inlay_hints_async(self.view)

@requires_session
def on_selection_modified_async(self) -> None:
if not self.session_views_async():
return
print("LSP: on_selection_modified_async")
first_region, _ = self._update_stored_selection_async()
if first_region is None:
return
Expand Down Expand Up @@ -490,9 +506,8 @@ def on_query_context(self, key: str, operator: int, operand: Any, match_all: boo
return operand == bool(session_view.session_buffer.get_document_link_at_point(self.view, position))
return None

@requires_session
def on_hover(self, point: int, hover_zone: int) -> None:
if not self.session_views_async():
return
if self.view.is_popup_visible():
return
window = self.view.window()
Expand Down Expand Up @@ -534,9 +549,8 @@ def _on_hover_gutter_async(self, point: int) -> None:
location=point,
on_navigate=lambda href: self._on_navigate(href, point))

@requires_session
def on_text_command(self, command_name: str, args: dict | None) -> tuple[str, dict] | None:
if not self.session_views_async():
return None
if command_name == "auto_complete":
self._auto_complete_triggered_manually = True
elif command_name == "show_scope_name" and userprefs().semantic_highlighting:
Expand All @@ -551,9 +565,8 @@ def on_text_command(self, command_name: str, args: dict | None) -> tuple[str, di
return ('paste', {})
return None

@requires_session
def on_post_text_command(self, command_name: str, args: dict[str, Any] | None) -> None:
if not self.session_views_async():
return
if command_name == 'paste':
format_on_paste = self.view.settings().get('lsp_format_on_paste', userprefs().lsp_format_on_paste)
if format_on_paste and self.session_async("documentRangeFormattingProvider"):
Expand All @@ -566,9 +579,8 @@ def on_post_text_command(self, command_name: str, args: dict[str, Any] | None) -
# hide the popup when `esc` or arrows are pressed pressed
self.view.hide_popup()

@requires_session
def on_query_completions(self, prefix: str, locations: list[int]) -> sublime.CompletionList | None:
if not self.session_views_async():
return None
completion_list = sublime.CompletionList()
triggered_manually = self._auto_complete_triggered_manually
self._auto_complete_triggered_manually = False # reset state for next completion popup
Expand Down

0 comments on commit 0d5e561

Please sign in to comment.