Browse Source

Avoid re-creating View children on edit

The older code attempted to be clever and sync component additions and
removals with what the message edit is doing. In some cases, this led
to the re-creation of those components causing lost attributes to be
dropped such as `_rendered_row` which would mess up handling of view
weights.

Instead of recreating the children list every time and keeping track
of additions and removals, this change just updates the old state with
the new state while ignoring any new or removed additions. This should
work fine in theory due to additions or removals already being present
before editing the View instance in the first place.

Closes #7231 #7511
pull/7939/head
Rapptz 3 years ago
parent
commit
e198a0e7e6
  1. 21
      discord/ui/view.py

21
discord/ui/view.py

@ -441,25 +441,26 @@ class View:
asyncio.create_task(self._scheduled_task(item, interaction), name=f'discord-ui-view-dispatch-{self.id}') asyncio.create_task(self._scheduled_task(item, interaction), name=f'discord-ui-view-dispatch-{self.id}')
def _refresh(self, components: List[Component]) -> None: def _refresh(self, components: List[Component]) -> None:
# This is pretty hacky at the moment
# fmt: off # fmt: off
old_state: Dict[Tuple[int, str], Item[Any]] = { old_state: Dict[str, Item[Any]] = {
(item.type.value, item.custom_id): item # type: ignore item.custom_id: item # type: ignore
for item in self._children for item in self._children
if item.is_dispatchable() if item.is_dispatchable()
} }
# fmt: on # fmt: on
children: List[Item[Any]] = []
for component in _walk_all_components(components): for component in _walk_all_components(components):
custom_id = getattr(component, 'custom_id', None)
if custom_id is None:
continue
try: try:
older = old_state[(component.type.value, component.custom_id)] # type: ignore older = old_state[custom_id]
except (KeyError, AttributeError): except KeyError:
children.append(_component_to_item(component)) _log.debug('View interaction referenced an unknown item custom_id %s. Discarding', custom_id)
continue
else: else:
older._refresh_component(component) older._refresh_component(component)
children.append(older)
self._children = children
def stop(self) -> None: def stop(self) -> None:
"""Stops listening to interaction events from this view. """Stops listening to interaction events from this view.

Loading…
Cancel
Save