1
1
from __future__ import annotations
2
2
3
3
import logging
4
+ import sys
4
5
from asyncio import Event , Task , create_task , gather
6
+ from contextvars import ContextVar , Token
5
7
from typing import Any , Callable , Protocol , TypeVar
6
8
7
9
from anyio import Semaphore
8
10
9
11
from reactpy .core ._thread_local import ThreadLocal
10
12
from reactpy .types import ComponentType , Context , ContextProviderType
13
+ from reactpy .utils import Singleton
11
14
12
15
T = TypeVar ("T" )
13
16
@@ -18,16 +21,39 @@ async def __call__(self, stop: Event) -> None: ...
18
21
19
22
logger = logging .getLogger (__name__ )
20
23
21
- _HOOK_STATE : ThreadLocal [list [LifeCycleHook ]] = ThreadLocal (list )
22
24
25
+ class _HookStack (Singleton ): # pragma: no cover
26
+ """A singleton object which manages the current component tree's hooks.
27
+ Life cycle hooks can be stored in a thread local or context variable depending
28
+ on the platform."""
23
29
24
- def current_hook () -> LifeCycleHook :
25
- """Get the current :class:`LifeCycleHook`"""
26
- hook_stack = _HOOK_STATE .get ()
27
- if not hook_stack :
28
- msg = "No life cycle hook is active. Are you rendering in a layout?"
29
- raise RuntimeError (msg )
30
- return hook_stack [- 1 ]
30
+ _state : ThreadLocal [list [LifeCycleHook ]] | ContextVar [list [LifeCycleHook ]] = (
31
+ ThreadLocal (list ) if sys .platform == "emscripten" else ContextVar ("hook_state" )
32
+ )
33
+
34
+ def get (self ) -> list [LifeCycleHook ]:
35
+ return self ._state .get ()
36
+
37
+ def initialize (self ) -> Token [list [LifeCycleHook ]] | None :
38
+ return None if isinstance (self ._state , ThreadLocal ) else self ._state .set ([])
39
+
40
+ def reset (self , token : Token [list [LifeCycleHook ]] | None ) -> None :
41
+ if isinstance (self ._state , ThreadLocal ):
42
+ self ._state .get ().clear ()
43
+ elif token :
44
+ self ._state .reset (token )
45
+ else :
46
+ raise RuntimeError ("Hook stack is an ContextVar but no token was provided" )
47
+
48
+ def current_hook (self ) -> LifeCycleHook :
49
+ hook_stack = self .get ()
50
+ if not hook_stack :
51
+ msg = "No life cycle hook is active. Are you rendering in a layout?"
52
+ raise RuntimeError (msg )
53
+ return hook_stack [- 1 ]
54
+
55
+
56
+ HOOK_STACK = _HookStack ()
31
57
32
58
33
59
class LifeCycleHook :
@@ -37,7 +63,7 @@ class LifeCycleHook:
37
63
a component is first rendered until it is removed from the layout. The life cycle
38
64
is ultimately driven by the layout itself, but components can "hook" into those
39
65
events to perform actions. Components gain access to their own life cycle hook
40
- by calling :func:`current_hook`. They can then perform actions such as:
66
+ by calling :func:`HOOK_STACK. current_hook`. They can then perform actions such as:
41
67
42
68
1. Adding state via :meth:`use_state`
43
69
2. Adding effects via :meth:`add_effect`
@@ -57,7 +83,7 @@ class LifeCycleHook:
57
83
.. testcode::
58
84
59
85
from reactpy.core._life_cycle_hook import LifeCycleHook
60
- from reactpy.core.hooks import current_hook
86
+ from reactpy.core.hooks import HOOK_STACK
61
87
62
88
# this function will come from a layout implementation
63
89
schedule_render = lambda: ...
@@ -75,15 +101,15 @@ class LifeCycleHook:
75
101
...
76
102
77
103
# the component may access the current hook
78
- assert current_hook() is hook
104
+ assert HOOK_STACK. current_hook() is hook
79
105
80
106
# and save state or add effects
81
- current_hook().use_state(lambda: ...)
107
+ HOOK_STACK. current_hook().use_state(lambda: ...)
82
108
83
109
async def my_effect(stop_event):
84
110
...
85
111
86
- current_hook().add_effect(my_effect)
112
+ HOOK_STACK. current_hook().add_effect(my_effect)
87
113
finally:
88
114
await hook.affect_component_did_render()
89
115
@@ -130,7 +156,7 @@ def __init__(
130
156
self ._scheduled_render = False
131
157
self ._rendered_atleast_once = False
132
158
self ._current_state_index = 0
133
- self ._state : tuple [ Any , ...] = ()
159
+ self ._state : list = []
134
160
self ._effect_funcs : list [EffectFunc ] = []
135
161
self ._effect_tasks : list [Task [None ]] = []
136
162
self ._effect_stops : list [Event ] = []
@@ -157,7 +183,7 @@ def use_state(self, function: Callable[[], T]) -> T:
157
183
if not self ._rendered_atleast_once :
158
184
# since we're not initialized yet we're just appending state
159
185
result = function ()
160
- self ._state += (result , )
186
+ self ._state . append (result )
161
187
else :
162
188
# once finalized we iterate over each succesively used piece of state
163
189
result = self ._state [self ._current_state_index ]
@@ -232,13 +258,13 @@ def set_current(self) -> None:
232
258
This method is called by a layout before entering the render method
233
259
of this hook's associated component.
234
260
"""
235
- hook_stack = _HOOK_STATE .get ()
261
+ hook_stack = HOOK_STACK .get ()
236
262
if hook_stack :
237
263
parent = hook_stack [- 1 ]
238
264
self ._context_providers .update (parent ._context_providers )
239
265
hook_stack .append (self )
240
266
241
267
def unset_current (self ) -> None :
242
268
"""Unset this hook as the active hook in this thread"""
243
- if _HOOK_STATE .get ().pop () is not self :
269
+ if HOOK_STACK .get ().pop () is not self :
244
270
raise RuntimeError ("Hook stack is in an invalid state" ) # nocov
0 commit comments