在進行內聯樣式設計時,如何在ReactJS中實現懸停事件或活動事件?
我發現onMouseEnter,onMouseLeave方法有問題,所以希望有另一種方法。
具體來說,如果您非常快速地將鼠標懸停在一個組件上,那么只會注冊onMouseEnter事件。onMouseLeave永遠不會觸發,因此無法更新狀態...使得組件看起來好像仍然被懸停在上方。我注意到同樣的事情,如果你試著模仿& quot:活動& quotcss偽類。如果你快速點擊,只有onMouseDown事件會被注冊。onMouseUp事件將被忽略...使組件看起來是活動的。
這是一個顯示問題的js fiddle:https://jsfiddle.net/y9swecyu/5/
代碼:
var Hover = React.createClass({
getInitialState: function() {
return {
hover: false
};
},
onMouseEnterHandler: function() {
this.setState({
hover: true
});
console.log('enter');
},
onMouseLeaveHandler: function() {
this.setState({
hover: false
});
console.log('leave');
},
render: function() {
var inner = normal;
if(this.state.hover) {
inner = hover;
}
return (
<div style={outer}>
<div style={inner}
onMouseEnter={this.onMouseEnterHandler}
onMouseLeave={this.onMouseLeaveHandler} >
{this.props.children}
</div>
</div>
);
}
});
var outer = {
height: '120px',
width: '200px',
margin: '100px',
backgroundColor: 'green',
cursor: 'pointer',
position: 'relative'
}
var normal = {
position: 'absolute',
top: 0,
bottom: 0,
left: 0,
right: 0,
backgroundColor: 'red',
opacity: 0
}
var hover = {
position: 'absolute',
top: 0,
bottom: 0,
left: 0,
right: 0,
backgroundColor: 'red',
opacity: 1
}
React.render(
<Hover></Hover>,
document.getElementById('container')
)
你試過這些嗎?
onMouseDown onMouseEnter on mouse leave onMouseMove on mouseout on mouseover on mouseup
合成事件
它還提到以下幾點:
React將事件規范化,使它們在不同的瀏覽器中具有一致的屬性。
下面的事件處理程序由冒泡階段的事件觸發。要注冊捕獲階段的事件處理程序,請將capture追加到事件名稱中。例如,不使用onClick,而是使用onClickCapture來處理捕獲階段的Click事件。
之前的回答相當混亂。你不需要一個反應狀態來解決這個問題,也不需要任何特殊的外部庫。這可以通過純css/sass來實現:
風格:
.hover {
position: relative;
&:hover &__no-hover {
opacity: 0;
}
&:hover &__hover {
opacity: 1;
}
&__hover {
position: absolute;
top: 0;
opacity: 0;
}
&__no-hover {
opacity: 1;
}
}
反應組件
一個簡單的懸停純渲染函數:
const Hover = ({ onHover, children }) => (
<div className="hover">
<div className="hover__no-hover">{children}</div>
<div className="hover__hover">{onHover}</div>
</div>
)
使用
然后像這樣使用它:
<Hover onHover={<div> Show this on hover </div>}>
<div> Show on no hover </div>
</Hover>
您可以使用onMouseOver = { this . ontoggleopen }和onMouseOut={this.onToggleOpen}反復思考組件
注意:這個答案是針對這個問題的前一個版本的,在這個版本中,提問者試圖使用JavaScript來應用css樣式…這可以簡單地用CSS來完成。
一個簡單的純css解決方案。 對于應用基本樣式,CSS在99%的情況下比JS解決方案更簡單、更高效。(盡管更現代的CSS-in-JS解決方案——例如React組件等——更容易維護。)
運行下面的代碼片段,看看它是如何工作的…
.hover-button .hover-button--on,
.hover-button:hover .hover-button--off {
display: none;
}
.hover-button:hover .hover-button--on {
display: inline;
}
<button class='hover-button'>
<span class='hover-button--off'>Default</span>
<span class='hover-button--on'>Hover!</span>
</button>
如果你能制作一個展示onMouseEnter / onMouseLeave或onMouseDown / onMouseUp bug的小演示,把它發布到ReactJS的問題頁面或郵件列表上是值得的,只是為了提出問題,聽聽開發人員對此有什么看法。
在您的用例中,您似乎暗示CSS :hover和:active狀態對于您的目的來說已經足夠了,所以我建議您使用它們。CSS比Javascript快幾個數量級,也更可靠,因為它直接在瀏覽器中實現。
但是,不能在內聯樣式中指定:hover和:active狀態。您所能做的就是為您的元素分配一個ID或類名,并在樣式表中編寫您的樣式,如果它們在您的應用程序中是常量的話,或者在動態生成的& ltstyle & gt標簽。
這里有一個后一種技術的例子:https://jsfiddle.net/ors1vos9/
我剛剛在一個禁用的按鈕上監聽onMouseLeave事件時遇到了同樣的問題。我通過偵聽包裝禁用按鈕的元素上的本機mou seleave事件來解決這個問題。
componentDidMount() {
this.watchForNativeMouseLeave();
},
componentDidUpdate() {
this.watchForNativeMouseLeave();
},
// onMouseLeave doesn't work well on disabled elements
// https://github.com/facebook/react/issues/4251
watchForNativeMouseLeave() {
this.refs.hoverElement.addEventListener('mouseleave', () => {
if (this.props.disabled) {
this.handleMouseOut();
}
});
},
render() {
return (
<span ref='hoverElement'
onMouseEnter={this.handleMouseEnter}
onMouseLeave={this.handleMouseLeave}
>
<button disabled={this.props.disabled}>Submit</button>
</span>
);
}
這里有一把小提琴,https://jsfiddle.net/qfLzkz5x/8/
我會用onMouseOver & amponMouseOut。反應中的原因:
onMouseEnter和onMouseLeave事件從左邊的元素傳播到正在進入的元素,而不是普通的冒泡,并且沒有捕獲階段。
這是鼠標事件的React文檔。
一個叫做styled-components的包可以很好地解決這個問題。
參考
Glen Maddern -使用樣式化組件對應用程序進行樣式化反應 例子
const styled = styled.default
const Square = styled.div`
height: 120px;
width: 200px;
margin: 100px;
background-color: green;
cursor: pointer;
position: relative;
&:hover {
background-color: red;
};
`
class Application extends React.Component {
render() {
return (
<Square>
</Square>
)
}
}
/*
* Render the above component into the div#app
*/
ReactDOM.render(<Application />, document.getElementById('app'));
<script src="https://cdnjs.cloudflare.com/ajax/libs/react/15.6.1/react.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/react/15.6.1/react-dom.min.js"></script>
<script src="https://unpkg.com/styled-components/dist/styled-components.min.js"></script>
<div id='app'></div>
你不能只使用內嵌樣式。不要建議在JavaScript中重新實現CSS特性,我們已經有了一種非常強大的語言,而且為這個用例構建的速度非常快——CSS。所以用吧!做了Style It來輔助。
npm安裝樣式-保存
函數語法
import React from 'react';
import Style from 'style-it';
class Intro extends React.Component {
render() {
return Style.it(`
.intro:hover {
color: red;
}
`,
<p className="intro">CSS-in-JS made simple -- just Style It.</p>
);
}
}
export default Intro;
JSX語法
import React from 'react';
import Style from 'style-it';
class Intro extends React.Component {
render() {
return (
<Style>
{`
.intro:hover {
color: red;
}
`}
<p className="intro">CSS-in-JS made simple -- just Style It.</p>
</Style>
}
}
export default Intro;
用鐳!
以下是他們網站上的一個例子:
var Radium = require('radium');
var React = require('react');
var color = require('color');
@Radium
class Button extends React.Component {
static propTypes = {
kind: React.PropTypes.oneOf(['primary', 'warning']).isRequired
};
render() {
// Radium extends the style attribute to accept an array. It will merge
// the styles in order. We use this feature here to apply the primary
// or warning styles depending on the value of the `kind` prop. Since its
// all just JavaScript, you can use whatever logic you want to decide which
// styles are applied (props, state, context, etc).
return (
<button
style={[
styles.base,
styles[this.props.kind]
]}>
{this.props.children}
</button>
);
}
}
// You can create your style objects dynamically or share them for
// every instance of the component.
var styles = {
base: {
color: '#fff',
// Adding interactive state couldn't be easier! Add a special key to your
// style object (:hover, :focus, :active, or @media) with the additional rules.
':hover': {
background: color('#0074d9').lighten(0.2).hexString()
}
},
primary: {
background: '#0074D9'
},
warning: {
background: '#FF4136'
}
};
當onMouseEnter被調用時,我也遇到了類似的問題,但是有時相應的onMouseLeave事件沒有被觸發,下面是一個對我很有效的解決方法(它部分依賴于jQuery):
var Hover = React.createClass({
getInitialState: function() {
return {
hover: false
};
},
onMouseEnterHandler: function(e) {
this.setState({
hover: true
});
console.log('enter');
$(e.currentTarget).one("mouseleave", function (e) {
this.onMouseLeaveHandler();
}.bind(this));
},
onMouseLeaveHandler: function() {
this.setState({
hover: false
});
console.log('leave');
},
render: function() {
var inner = normal;
if(this.state.hover) {
inner = hover;
}
return (
<div style={outer}>
<div style={inner}
onMouseEnter={this.onMouseEnterHandler} >
{this.props.children}
</div>
</div>
);
}
});
參見jsfiddle上的:http://jsfiddle.net/qtbr5cg6/1/
為什么會這樣(在我的例子中):我正在運行一個jQuery滾動動畫(通過$('#item ')。單擊項目時顯示動畫({ scrollTop: 0 })。所以光標不會“自然地”離開項目,而是在JavaScript驅動的動畫中...在這種情況下,onMouseLeave沒有被React正確觸發(React 15.3.0,Chrome 51,桌面)
我知道這個問題已經提出有一段時間了,但是我剛剛遇到了與onMouseLeave()不一致的問題 我所做的是對下拉列表使用onMouseOut()并對整個菜單使用鼠標左鍵,它是可靠的,并且每次我測試它時都有效。 我在文檔中看到了這些事件:https://Facebook . github . io/react/docs/events . html # mouse-events 下面是一個使用https://www . w3schools . com/bootstrap/bootstrap _ drop downs . ASP的示例:
handleHoverOff(event){
//do what ever, for example I use it to collapse the dropdown
let collapsing = true;
this.setState({dropDownCollapsed : collapsing });
}
render{
return(
<div class="dropdown" onMouseLeave={this.handleHoverOff.bind(this)}>
<button class="btn btn-primary dropdown-toggle" type="button" data-toggle="dropdown">Dropdown Example
<span class="caret"></span></button>
<ul class="dropdown-menu" onMouseOut={this.handleHoverOff.bind(this)}>
<li><a href="#">bla bla 1</a></li>
<li><a href="#">bla bla 2</a></li>
<li><a href="#">bla bla 3</a></li>
</ul>
</div>
)
}
我個人在React中使用Style It作為inline-style,或者在CSS或SASS文件中單獨保存我的樣式...
但是如果你真的有興趣做內聯,看看這個庫,我在下面分享了一些用法:
在組件中:
import React from 'react';
import Style from 'style-it';
class Intro extends React.Component {
render() {
return (
<Style>
{`
.intro {
font-size: 40px;
}
`}
<p className="intro">CSS-in-JS made simple -- just Style It.</p>
</Style>
);
}
}
export default Intro;
輸出:
<p class="intro _scoped-1">
<style type="text/css">
._scoped-1.intro {
font-size: 40px;
}
</style>
CSS-in-JS made simple -- just Style It.
</p>
你也可以在CSS中使用JavaScript變量,如下所示:
import React from 'react';
import Style from 'style-it';
class Intro extends React.Component {
render() {
const fontSize = 13;
return Style.it(`
.intro {
font-size: ${ fontSize }px; // ES2015 & ES6 Template Literal string interpolation
}
.package {
color: blue;
}
.package:hover {
color: aqua;
}
`,
<p className="intro">CSS-in-JS made simple -- just Style It.</p>
);
}
}
export default Intro;
結果如下:
<p class="intro _scoped-1">
<style type="text/css">
._scoped-1.intro {
font-size: 13px;
}
._scoped-1 .package {
color: blue;
}
._scoped-1 .package:hover {
color: aqua;
}
</style>
CSS-in-JS made simple -- just Style It.
</p>
懸停是CSS的一個特性;它附帶了應用程序的CSS部分,所以用正確的方式來做是非常有趣的。簡單地說,當我需要React應用程序中的一個元素上的懸停效果時,首先,我在我的CSS文件中創建一個類,然后我將創建的類添加到元素的className中。我遵循的步驟是:
創建一個CSS文件,除非你有index.css
創建一個啟用了懸停偽類的類
。懸停_ _效果:懸停{}
添加您需要的效果
。懸停_ _效果:懸停{ 背景色:rgb(255,255,255); 顏色:rgb(0,0,0); }
然后將hover _ _效果添加到該類的組件中,當鼠標指針懸停在該組件上時,該組件應該是不同的
盤旋
請檢查我的沙盒現場演示。