Vue может переиспользовать существующие DOM узлы при ререндере. Если на DOM узле, уже есть обработчик (el._vei), то Vue обновит его проставив ссылку на новую функцию, при этом addEventListener использоваться не будет.
В некоторых случаях vnode узлы генерируются с флагом 32 (HYDRATE_EVENTS), в этом случае Vue не добавит vnode как динамический:
// данный код выполняется внутри createBaseVNode
// track vnode for block tree
if (isBlockTreeEnabled > 0 &&
// avoid a block node from tracking itself
!isBlockNode &&
// has current parent block
currentBlock &&
// presence of a patch flag indicates this node needs patching on updates.
// component nodes also should always be patched, because even if the
// component doesn't need to update, it needs to persist the instance on to
// the next vnode so that it can be properly unmounted later.
(vnode.patchFlag > 0 || shapeFlag & 6 /* ShapeFlags.COMPONENT */) &&
// the EVENTS flag is only for hydration and if it is the only flag, the
// vnode should not be considered dynamic due to handler caching.
vnode.patchFlag !== 32 /* PatchFlags.HYDRATE_EVENTS */) {
currentBlock.push(vnode);
}
Мы со своей стороны для vnode задаем проп data-has-v-on-directives, и в patchFlag дописываем 8 (PROPS), однако, так как при создании vnode в createBaseVNode в currentBlock узел не добавляется, то узел все равно будет считаться не динамическим.
Потенциально проблема очень серьезная, так как если изначально у узла нет патч флагов и это не компонент, то такой узел отслеживаться не будет (при условии, что он внутри блока).
Vue может переиспользовать существующие DOM узлы при ререндере. Если на DOM узле, уже есть обработчик (el._vei), то Vue обновит его проставив ссылку на новую функцию, при этом addEventListener использоваться не будет.
В некоторых случаях vnode узлы генерируются с флагом 32 (HYDRATE_EVENTS), в этом случае Vue не добавит vnode как динамический:
Мы со своей стороны для vnode задаем проп data-has-v-on-directives, и в patchFlag дописываем 8 (PROPS), однако, так как при создании vnode в createBaseVNode в currentBlock узел не добавляется, то узел все равно будет считаться не динамическим.
Пример
Исходный код
Скомпилированный узел
Если же
@click.capture.stop
заменить на@click.stop
или@click
, то патч флаги не будут заданы вообще и блок все равно будет не динамическим.