我在Reactjs中创建了一个带有四个元素的导航栏。这些项是一个简单的无序列表,其中一些css使用flexbox来对齐它们。
<ul>
<li>One</li>
<li>Two</li>
<li>Three</li>
<li>Four</li>
<ul/>
我想要实现的是:选择列表项后,将所选列表项对齐到中心。我添加了专业图片以便澄清。稍后将对此更改进行动画处理以实现平滑过渡,如旋转木马。
以下是<ul>
代码的css。
ul {
list-style-type: none;
display: flex;
flex-direction: row;
flex-wrap: nowrap;
justify-content: center;
}
我尝试过的是在align-self: center
项之一上使用<li>
,但没有运气。
有没有人有做类似事情的经验?我对所有类型的解决方案都是开放的,即使是那些不使用flexbox的解决方案。
谢谢!
答案 0 :(得分:2)
列表元素是否已固定,您知道有多少项吗?如果是这样,您可以计算列表的中心,项目偏移量,并添加CSS变换。
示例:强>
如果您的列表是动态的,无论是列表长度还是项目宽度,您都可以使用Element.getBoundingClientRect()查找元素的尺寸,并使用与上面相同的计算方法。
Codepen: https://codepen.io/anon/pen/vjJMVL
<强> HTML:强>
<ul class="selected-2">
<li>1</li>
<li class="selected">2</li>
<li>3</li>
<li>4</li>
</ul>
<ul class="selected-4">
<li>1</li>
<li>2</li>
<li>3</li>
<li class="selected">4</li>
</ul>
<强> CSS:强>
ul {
display: flex;
justify-content: center;
list-style: none;
}
li {
width: 80px;
height: 80px;
background-color: #eee;
margin: 10px;
}
.selected {
background-color: #ccc;
}
.selected-2 {
transform: translateX(50px)
}
.selected-4 {
transform: translateX(-150px)
}
答案 1 :(得分:1)
使用Element.getBoundingClientRect()
计算点击的<MenuItem>
中心以获取left
和width
,然后将其传递给父级(<Menu>
)。在父级中使用<ul>
s参考获取left
和宽度为Element.getBoundingClientRect()
。计算moveTo
状态,并相应地更新<ul>
的样式transform: translateX()
:
const { Component } = React;
const items = ['One', 'Two', 'Three', 'Four'];
class MenuItem extends Component {
clickHandler = (e) => {
const { left, width } = e.target.getBoundingClientRect();
const itemCenter = left + width / 2;
this.props.updateCenter(itemCenter);
}
render() {
const { children } = this.props;
return (
<li onClick={this.clickHandler}>{children}</li>
);
}
}
class Menu extends Component {
state = {
moveTo: 0
};
updateCenter = (itemCenter) => {
const { left, width } = this.ul.getBoundingClientRect();
//this.ul.style.transform = `translateX(${center}px)`;
this.setState(() => ({
moveTo: left + width / 2 - itemCenter
}));
};
render() {
const { items } = this.props;
const { moveTo } = this.state;
return (
<nav>
<ul ref={(ul) => this.ul = ul} style={{
transform: `translateX(${moveTo}px)`
}}>
{
items.map((text) => (
<MenuItem key={text}
updateCenter={this.updateCenter}>
{text}
</MenuItem>
))
}
</ul>
</nav>
);
}
}
ReactDOM.render(
<Menu items={items} />,
demo
);
/** demo only - display the center **/
body::before {
position: absolute;
left: 50%;
height: 100vh;
border-right: 1px solid black;
content: '';
}
nav {
overflow: hidden;
}
ul {
list-style-type: none;
display: flex;
flex-direction: row;
flex-wrap: nowrap;
justify-content: center;
transition: transform 0.5s cubic-bezier(0.175, 0.885, 0.32, 1.275);
}
li {
padding: 1em;
border: 1px solid black;
cursor: pointer;
}
ul li:not(:last-child) {
margin: 0 1em 0 0;
}
<script crossorigin src="https://unpkg.com/react@16/umd/react.development.js"></script>
<script crossorigin src="https://unpkg.com/react-dom@16/umd/react-dom.development.js"></script>
<div id="demo"></div>
答案 2 :(得分:0)
由于我没有完全得到你想要的东西,这是我能想到的最好的,将绝对定位应用于所选择的并且使其与z-index重叠
document.querySelectorAll('li').forEach(function(li){
li.addEventListener('click',function(e){
document.querySelectorAll('li').forEach(function(obj){
obj.classList.remove('selected');
});
e.target.classList.add("selected");
});
});
ul {
list-style-type: none;
display: flex;
flex-direction: row;
flex-wrap: nowrap;
justify-content: center;
}
li{
border:1px solid;
padding:10px;
cursor:pointer;
position:relative;
transition:all ease 1s;
margin:0 20px;
}
.selected{
transform:scale(2.1);
background:white;
box-shadow:0px 0px 55px black;
position:absolute;
z-index:5;
}
<ul>
<li>One</li>
<li>Two</li>
<li>Three</li>
<li>Four</li>
</ul>