14
14
from collections .abc import Sequence
15
15
from contextlib import AsyncExitStack
16
16
from logging import getLogger
17
+ from types import TracebackType
17
18
from typing import (
18
19
Any ,
19
20
Callable ,
@@ -56,13 +57,13 @@ class Layout:
56
57
"""Responsible for "rendering" components. That is, turning them into VDOM."""
57
58
58
59
__slots__ : tuple [str , ...] = (
59
- "root" ,
60
60
"_event_handlers" ,
61
- "_rendering_queue " ,
61
+ "_model_states_by_life_cycle_state_id " ,
62
62
"_render_tasks" ,
63
63
"_render_tasks_ready" ,
64
+ "_rendering_queue" ,
64
65
"_root_life_cycle_state_id" ,
65
- "_model_states_by_life_cycle_state_id " ,
66
+ "root " ,
66
67
)
67
68
68
69
if not hasattr (abc .ABC , "__weakref__" ): # nocov
@@ -80,17 +81,17 @@ async def __aenter__(self) -> Layout:
80
81
self ._event_handlers : EventHandlerDict = {}
81
82
self ._render_tasks : set [Task [LayoutUpdateMessage ]] = set ()
82
83
self ._render_tasks_ready : Semaphore = Semaphore (0 )
83
-
84
84
self ._rendering_queue : _ThreadSafeQueue [_LifeCycleStateId ] = _ThreadSafeQueue ()
85
85
root_model_state = _new_root_model_state (self .root , self ._schedule_render_task )
86
-
87
86
self ._root_life_cycle_state_id = root_id = root_model_state .life_cycle_state .id
88
87
self ._model_states_by_life_cycle_state_id = {root_id : root_model_state }
89
88
self ._schedule_render_task (root_id )
90
89
91
90
return self
92
91
93
- async def __aexit__ (self , * exc : object ) -> None :
92
+ async def __aexit__ (
93
+ self , exc_type : type [Exception ], exc_value : Exception , traceback : TracebackType
94
+ ) -> None :
94
95
root_csid = self ._root_life_cycle_state_id
95
96
root_model_state = self ._model_states_by_life_cycle_state_id [root_csid ]
96
97
@@ -109,7 +110,7 @@ async def __aexit__(self, *exc: object) -> None:
109
110
del self ._root_life_cycle_state_id
110
111
del self ._model_states_by_life_cycle_state_id
111
112
112
- async def deliver (self , event : LayoutEventMessage ) -> None :
113
+ async def deliver (self , event : LayoutEventMessage | dict [ str , Any ] ) -> None :
113
114
"""Dispatch an event to the targeted handler"""
114
115
# It is possible for an element in the frontend to produce an event
115
116
# associated with a backend model that has been deleted. We only handle
@@ -217,7 +218,7 @@ async def _render_component(
217
218
parent .children_by_key [key ] = new_state
218
219
# need to add this model to parent's children without mutating parent model
219
220
old_parent_model = parent .model .current
220
- old_parent_children = old_parent_model [ "children" ]
221
+ old_parent_children = old_parent_model . setdefault ( "children" , [])
221
222
parent .model .current = {
222
223
** old_parent_model ,
223
224
"children" : [
@@ -318,8 +319,11 @@ async def _render_model_children(
318
319
new_state : _ModelState ,
319
320
raw_children : Any ,
320
321
) -> None :
321
- if not isinstance (raw_children , (list , tuple )):
322
- raw_children = [raw_children ]
322
+ if not isinstance (raw_children , list ):
323
+ if isinstance (raw_children , tuple ):
324
+ raw_children = list (raw_children )
325
+ else :
326
+ raw_children = [raw_children ]
323
327
324
328
if old_state is None :
325
329
if raw_children :
@@ -609,7 +613,7 @@ def __init__(
609
613
parent : _ModelState | None ,
610
614
index : int ,
611
615
key : Any ,
612
- model : Ref [VdomJson ],
616
+ model : Ref [VdomJson | dict [ str , Any ] ],
613
617
patch_path : str ,
614
618
children_by_key : dict [Key , _ModelState ],
615
619
targets_by_event : dict [str , str ],
@@ -656,7 +660,7 @@ def parent(self) -> _ModelState:
656
660
return parent
657
661
658
662
def append_child (self , child : Any ) -> None :
659
- self .model .current [ "children" ] .append (child )
663
+ self .model .current . setdefault ( "children" , []) .append (child )
660
664
661
665
def __repr__ (self ) -> str : # nocov
662
666
return f"ModelState({ {s : getattr (self , s , None ) for s in self .__slots__ } } )"
0 commit comments