对齐中心CSS中的单个列表项

时间:2018-05-04 17:56:52

标签: html css reactjs flexbox

我在Reactjs中创建了一个带有四个元素的导航栏。这些项是一个简单的无序列表,其中一些css使用flexbox来对齐它们。

<ul>
    <li>One</li>
    <li>Two</li>
    <li>Three</li>
    <li>Four</li>
<ul/>

我想要实现的是:选择列表项后,将所选列表项对齐到中心。我添加了专业图片以便澄清。稍后将对此更改进行动画处理以实现平滑过渡,如旋转木马。

The selected item appear in the center row

以下是<ul>代码的css。

ul {
    list-style-type: none;
    display: flex;
    flex-direction: row;
    flex-wrap: nowrap;
    justify-content: center;
}

我尝试过的是在align-self: center项之一上使用<li>,但没有运气。

有没有人有做类似事情的经验?我对所有类型的解决方案都是开放的,即使是那些不使用flexbox的解决方案。

谢谢!

3 个答案:

答案 0 :(得分:2)

列表元素是否已固定,您知道有多少项吗?如果是这样,您可以计算列表的中心,项目偏移量,并添加CSS变换。

示例:

  • 您有四个项目的列表。
  • 每个项目的宽度等于100px。
  • 因此列表的总宽度为400px。
  • 列表的中心点距离左侧200px。
  • 项目编号2的中心点距离左侧150px。因此,我们必须从左侧移动列表200px - 150px = 50px。
  • 项目编号4的中心点距离左侧350px。因此,我们必须从左侧移动列表200px - 350px = -150px。

如果您的列表是动态的,无论是列表长度还是项目宽度,您都可以使用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>中心以获取leftwidth,然后将其传递给父级(<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>