Próbuję użyć zdarzenia event.stopPropagation () w komponencie ReactJS, aby zatrzymać propagację zdarzenia kliknięcia i wyzwolenie zdarzenia kliknięcia, które zostało dołączone do JQuery w starszym kodzie, ale wygląda na to, że stopPropagation () Reacta zatrzymuje tylko propagację do zdarzeń również dołączony w Reakcie, a stopPropagation () JQuery nie zatrzymuje propagacji do zdarzeń dołączonych do React.
Czy istnieje sposób, aby funkcja stopPropagation () działała w przypadku tych zdarzeń? Napisałem prosty JSFiddle, aby zademonstrować następujące zachowania:
/** @jsx React.DOM */
var Propagation = React.createClass({
alert: function(){
alert('React Alert');
},
stopPropagation: function(e){
e.stopPropagation();
},
render: function(){
return (
<div>
<div onClick={this.alert}>
<a href="#" onClick={this.stopPropagation}>React Stop Propagation on React Event</a>
</div>
<div className="alert">
<a href="#" onClick={this.stopPropagation}>React Stop Propagation on JQuery Event</a>
</div>
<div onClick={this.alert}>
<a href="#" className="stop-propagation">JQuery Stop Propagation on React Event</a>
</div>
<div className="alert">
<a href="#" className="stop-propagation">JQuery Stop Propagation on JQuery Event</a>
</div>
</div>
);
}
});
React.renderComponent(<Propagation />, document.body);
$(function(){
$(document).on('click', '.alert', function(e){
alert('Jquery Alert');
});
$(document).on('click', '.stop-propagation', function(e){
e.stopPropagation();
});
});
javascript
jquery
reactjs
lofiinterstate
źródło
źródło
event.nativeEvent.stopImmediatePropagation
aby zapobiec uruchamianiu innych detektorów zdarzeń, ale kolejność wykonywania nie jest gwarantowana.stopImmediatePropagation
detektory zdarzeń roszczeń będą wywoływane w kolejności, w jakiej zostały powiązane. Jeśli twój React JS został zainicjowany przed twoim jQuery (tak jak na twoich skrzypcach), zatrzymanie natychmiastowej propagacji zadziała.componentDidMount
, ale może to kolidować z innymi programami obsługi zdarzeń React w nieoczekiwany sposób..stop-propagation
nie zadziała. Twój przykład używa delegowania zdarzeń, ale próbuje zatrzymać propagację w elemencie. Słuchacz musi być związany z samego elementu:$('.stop-propagation').on('click', function(e) { e.stopPropagation(); });
. To skrzypce zapobiega wszelkiej propagacji, tak jak próbowałeś: jsfiddle.net/7LEDT/6Odpowiedzi:
React używa delegowania zdarzeń z włączonym jednym nasłuchiwaniem
document
dla zdarzeń, które są w dymku, jak na przykład „kliknięcie” w tym przykładzie, co oznacza, że zatrzymanie propagacji nie jest możliwe; rzeczywiste zdarzenie zostało już rozpropagowane do czasu interakcji z nim w Reakcie.stopPropagation
na syntetycznym zdarzeniu Reacta jest możliwe, ponieważ React obsługuje wewnętrzną propagację zdarzeń syntetycznych.Działa JSFiddle z poprawkami od dołu.
React Stop Propagation on jQuery Event
Służy
Event.stopImmediatePropagation
do zapobiegania wywoływaniu innych słuchaczy (w tym przypadku jQuery) w katalogu głównym. Jest obsługiwany w IE9 + i nowoczesnych przeglądarkach.jQuery Stop Propagation on React Event
Twój kod jQuery również używa delegowania zdarzeń, co oznacza, że wywołanie
stopPropagation
procedury obsługi niczego nie zatrzymuje; zdarzenie zostało już rozpropagowane dodocument
, a detektor Reacta zostanie uruchomiony.Aby zapobiec propagacji poza element, odbiornik musi być powiązany z samym elementem:
Edycja (2016/01/14): Wyjaśniono, że delegacja jest koniecznie używana tylko w przypadku wydarzeń w dymku. Aby uzyskać więcej informacji na temat obsługi zdarzeń, źródło Reacta zawiera opisowe komentarze: ReactBrowserEventEmitter.js .
źródło
document
lub główny komponent React? Strona, do której prowadzi link, mówi tylko „na najwyższym poziomie”, co rozumiem, że nie jest todocument
(ale nie mogę się dowiedzieć, czy jest to w ogóle istotne).window
zamiastdocument
: stackoverflow.com/a/52879137/438273Warto zauważyć (na podstawie tego wydania ), że jeśli dołączasz wydarzenia do
document
,e.stopPropagation()
nie pomoże. Aby obejść ten problem, możesz użyćwindow.addEventListener()
zamiastdocument.addEventListener
, a następnieevent.stopPropagation()
zatrzyma propagację zdarzenia do okna.źródło
To wciąż jeden interesujący moment:
Użyj tej konstrukcji, jeśli twoja funkcja jest opakowana tagiem
źródło
event.nativeEvent
to poprawna odpowiedź; Udało mi się skorzystaćcomposedPath()
za jego pośrednictwem:event.nativeEvent.composedPath()
wrapped by tag
? czy możesz podać przykład?<div onClick=(firstHandler)> <div onClick=(secHandler)>active text</div> </div>
Udało mi się rozwiązać ten problem, dodając następujące elementy do mojego komponentu:
źródło
Wczoraj napotkałem ten problem, więc stworzyłem rozwiązanie przyjazne Reactowi.
Wypróbuj reagować na natywne-nasłuchiwanie . Jak dotąd działa bardzo dobrze. Opinie mile widziane.
źródło
react-native-listener
zastosowania,findDomNode
które zostaną wycofane github.com/yannickcr/eslint-plugin-react/issues/…Z dokumentacji Reacta : poniższe programy obsługi zdarzeń są wyzwalane przez zdarzenie w fazie propagacji . Aby zarejestrować procedurę obsługi zdarzeń dla fazy przechwytywania, dołącz Capture . (podkreślenie dodane)
Jeśli masz odbiornik zdarzeń kliknięcia w swoim kodzie Reacta i nie chcesz, aby był widoczny, myślę, że
onClickCapture
zamiast tego chcesz go użyćonClick
. Następnie przekazałbyś zdarzenie do modułu obsługi i zrobiłbyś,event.nativeEvent.stopPropagation()
aby natywne zdarzenie nie przepływało do zwykłego nasłuchiwania zdarzeń JS (lub czegokolwiek, co nie reaguje).źródło
Aktualizacja: możesz teraz
<Elem onClick={ proxy => proxy.stopPropagation() } />
źródło
Szybkim obejściem jest użycie
window.addEventListener
zamiastdocument.addEventListener
.źródło