如何为所有实例设置子组件的相同动态宽度? Web组件

时间:2018-12-05 17:27:36

标签: javascript html css reactjs web-component

假设我在Web应用程序中有一个小组件,代表扩展程序控件,即标题文本,展开/折叠的图标和一些内容。

一个非常简单的React实现(伪代码)可能看起来像这样:

const Expander = ({children, title, onExpandToggle, isExpanded}) => (
  <div>
    <div><span>{title}</span><img src={...} onClick={onExpandToggle} /></div>
    {isExpanded && children}
  </div>
);

此实现在标题后显示图标,因此图标的位置由标题的长度确定。

可能看起来像这样:
Example

现在假设彼此之间有多个类似的内容。变得凌乱:
Example messy

要使此清洁器更干净,所有图标的左侧应具有相同的填充。填充不应该是固定的,而是动态的,以便最长的标题确定所有图标的位置:
Example clean

假设我想将扩展器保留在其自己的组件中,是否有CSS方法可以实现我的目标?

到目前为止,我还没有尝试任何东西,因为我没有起点。在WPF中,我会使用SharedSizeGroup之类的东西,但是CSS不存在。

8 个答案:

答案 0 :(得分:11)

假设您的组件中有一个容器,则可以将display: flex设置为内部容器,将align-self: flex-end设置为图像。

然后用一个具有display: inline-block的div包裹您的组件,该div占用内部最大元素的宽度。

这是一个例子:

.container{
  display: inline-block;
  padding: 3px;
}

.item{
  display: flex;
  flex-direction: row; 
  justify-content: space-between;
}

.item .plus{
  width: 15px;
  height: 15px;
  background-image: url("https://cdn1.iconfinder.com/data/icons/mix-color-3/502/Untitled-43-512.png");
  background-size: cover;
  background-repeat: no-repeat;
  align-self: flex-end;
  margin-left: 10px;
}
<div class="container">
  <div class="item">
    <div>Synonyms</div>
    <div class="plus"></div>
  </div>
  <div class="item">
    <div>Concept</div>
    <div class="plus"></div>
  </div>
  <div class="item">
    <div>Term</div>
    <div class="plus"></div>
  </div>
</div>

由于没有办法使用CSS访问父元素,因此不可能在没有共享容器或固定宽度且仅靠CSS的情况下影响所有实例。因此,如果您有一个内部实例(最大的实例),它将无法将其宽度应用于自己的父代或任何祖先以及其他子代。

如果您要使用一种解决方案,将页面中的所有实例设置为相同大小,而无需共享容器,则可以使用JS实现。

计算每个实例的宽度,保存最大的宽度,然后为其余实例设置此宽度。在此示例中,我还将重点介绍最大的项目。这些项目都在页面周围,可以放在各种div和显示器中,也可以没有任何容器。

var biggestWidth = 0;
var biggestItem;

function setWidth() {
  $(".item").each(function() {
    var currentWidth = $(this).width();
    if (currentWidth > biggestWidth) {
      biggestWidth = Math.ceil(currentWidth);
      biggestItem = $(this);
    }
  });

  $(".item").width(biggestWidth);
  biggestItem.addClass("biggest");
}

$(setWidth());
section {
  width: 40%;
  float: left;
  border: 1px solid black;
  border-radius: 3px;
  margin: 10px;
  padding: 10px;
}

.s1 {
  background-color: #e5e5e5;
  display: table;
}

.item {
  display: inline-block;
  clear: both;
  float: left;
}

.txt {
  float: left;
  display: inline-block;
}

.plus {
  width: 15px;
  height: 15px;
  background-image: url("https://cdn1.iconfinder.com/data/icons/mix-color-3/502/Untitled-43-512.png");
  background-size: cover;
  margin-left: 10px;
  float: right;
}

.shift{
  margin-left: 30%;
}

.clear{
  clear: both;
}

.biggest{
  background-color: yellow;
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.3.1/jquery.min.js"></script>
<section class="s1">
  <div class="item">
    <div class="txt">Synonyms</div>
    <div class="plus"></div>
  </div>
  <div class="item">
    <div class="txt">Concept</div>
    <div class="plus"></div>
  </div>
  <div class="item">
    <div class="txt">Term</div>
    <div class="plus"></div>
  </div>
</section>
<section>
  <div class="item">
    <div class="txt">Synonyms</div>
    <div class="plus"></div>
  </div>
  <div class="item">
    <div class="txt">Concept</div>
    <div class="plus"></div>
  </div>
  <div class="item">
    <div class="txt">Term</div>
    <div class="plus"></div>
  </div>
</section>

<section>
  <div class="item">
    <div class="txt">Synonyms - Long</div>
    <div class="plus"></div>
  </div>
  <div class="item">
    <div class="txt">Concept</div>
    <div class="plus"></div>
  </div>
  <div class="item">
    <div class="txt">Term</div>
    <div class="plus"></div>
  </div>
</section>

<div class="item">
  <div class="txt">Synonyms</div>
  <div class="plus"></div>
</div>
<div class="item">
  <div class="txt">Concept</div>
  <div class="plus"></div>
</div>
<div class="item">
  <div class="txt">Term</div>
  <div class="plus"></div>
</div>

<div class="clear"></div>
<div class="shift">
<div class="item">
  <div class="txt">Synonyms</div>
  <div class="plus"></div>
</div>
<div class="item">
  <div class="txt">Concept</div>
  <div class="plus"></div>
</div>
<div class="item">
  <div class="txt">The bigget item is here</div>
  <div class="plus"></div>
</div>
</div>

答案 1 :(得分:1)

您还可以尝试使用css display属性"table-row""table-cell"

样式:

   .trow {
        display: table-row;
      }
      .tcell
     {
        display: table-cell;
        padding: 2px;
      }

您可以像下面那样设计组件。

const Expander = ({children, title, onExpandToggle, isExpanded}) => (
    <div className="trow">
     <div className="tcell">{title}</div>
      <div className="tcell"><img src={...} onClick={onExpandToggle} /></div>
      {isExpanded && children}
    </div>
  );

示例

      .trow {
        display: table-row;
      }
      .tcell
     {
        display: table-cell;
        padding: 2px;
      }
    <div>
      <div class="trow">
        <div class="tcell">Synonyms</div>
        <div class="tcell">X</div>
      </div>
      <div class="trow">
        <div class="tcell">Concept</div>
        <div class="tcell">X</div>
      </div>
      <div class="trow">
        <div class="tcell">Term</div>
        <div class="tcell">X</div>
      </div>
    </div>

答案 2 :(得分:1)

我认为严格来说,在您已设定所有限制的情况下,答案是“不,不可能”。

但是,由于我认为您无论如何都需要以某种方式对这些扩展器进行分组(或者,页面上的每个扩展器都将水平对齐,也不存在完整的CSS解决方案),因此您可以简单地依靠CSS盒子模型。

<div style="background-color: #efffef; display: inline-block;">
   <div style="position: relative; padding-right: 36px;">
      <span>Title 1</span>
      <span style="position: absolute; right: 0; top: 0;">[x]</span>
   </div>
   <div style="position: relative; padding-right: 36px;">
      <span>Long title 2</span>
      <span style="position: absolute; right: 0; top: 0;">[x]</span>
   </div>
   <div style="position: relative; padding-right: 36px;">
      <span>Title 3, at your service</span>
      <span style="position: absolute; right: 0; top: 0;">[x]</span>
   </div>
</div>

答案 3 :(得分:1)

我个人会使用经典的<table>来做到这一点。它简单易懂,语义正确,将在3分钟内完成。但是我们也可以使用display:inline-blockfloat:right,或者使用<table>display:tabledisplay:table-row进行display:table-cell模仿来做到这一点。我认为display: flex是相对较新的,并且不与跨浏览器完全兼容(例如IE10,IE11等)。因此,我想为您提供3种解决方案。

使用display:inline-blockfloat:right的解决方案

使用display:inline-block时,尺寸始终与内容对齐。

.container,
.item div,
.item b{display:inline-block}
.item b
{
    float:right;
    width:16px;
    height:16px;
    margin-left:5px;
    cursor:pointer;
    background:url("https://i.stack.imgur.com/bKWrw.png")
}
<div class="container">
    <div class="item">
        <div>Synonyms</div>
        <b></b>
    </div>
    <div class="item">
        <div>Concept</div>
        <b></b>
    </div>
    <div class="item">
        <div>Term</div>
        <b></b>
    </div>
</div>
<br style="clear:both">

具有<table>模仿的解决方案

.container{display:table}
.item{display:table-row}
.item div{display:table-cell; vertical-align:top}
.item b
{
    display:inline-block;
    width:16px;
    height:16px;
    cursor:pointer;
    margin-left:5px;
    background:url("https://i.stack.imgur.com/bKWrw.png")
}
<div class="container">
    <div class="item">
        <div>Synonyms</div>
        <div><b></b></div>
    </div>
    <div class="item">
        <div>Concept</div>
        <div><b></b></div>
    </div>
    <div class="item">
        <div>Term</div>
        <div><b></b></div>
    </div>
</div>

经典<table> (我的最爱)

.menu td{vertical-align:top}
.menu td b
{
    display:inline-block;
    width:16px;
    height:16px;
    cursor:pointer;
    margin-left:5px;
    background:url("https://i.stack.imgur.com/bKWrw.png")
}
<table class="menu" cellpadding="0" cellspacing="0">
<tr>
    <td>Synonyms</td>
    <td><b></b></td>
</tr>
<tr>
    <td>Concept</td>
    <td><b></b></td>
</tr>
<tr>
    <td>Term</td>
    <td><b></b></td>
</tr>
</table>

答案 4 :(得分:1)

根据您的代码片段,我能想到的最简单的例子是:

 const containerStyle = {
   width: `[whatever width you need for the container]`,
   display: `flex`,
   flexDirection: `column`    
 }

 const itemStyle = { 
  width: `100%`, 
  display: `flex`, 
  flexDirection: `row`, 
  justifyContent: `space-between` 
 }

const Expander = ({children, title, onExpandToggle, isExpanded}) => (
  <div style={containerStyle}>
    <div style={itemStyle}>
      <span>{title}</span>
      <img src={...} onClick={onExpandToggle} />
    </div>
    {isExpanded && children}
  </div>
);

答案 5 :(得分:1)

const Expander = (children, title, onExpandToggle, isExpanded) => (
   <div>
     {title}
     <img src={...} style={{marginLeft : '5px',float : 'right'}} onClick={onExpandToggle}/>
   </div>
   <div style={{width : '0px',overflowX : 'visible'}}>
     {isExpanded && <div>
       {children}
     </div>}
   </div>
);

并将其放入:

<div style={{display : 'inline-block'}}>
   <Expander />
   <Expander />
</div>

答案 6 :(得分:1)

在CSS中,页面的不同元素不可能彼此“了解”。您可以获得的最接近的结果是使用display flex和/或grid。通过这种方式,直接父级可以控制其子级的显示方式,类似于React。

flex和grid的主要区别是flex是1轴,grid是2轴。

使用grid和flex的工作示例:

<div class="container">
    <div class="summary"><span class="text">Synonyms</span> <span class="icon">+</span></div>
    <div class="details">
        <p>Lorem ipsum dolor sit amet consectetur adipisicing elit. Eveniet officiis sequi perspiciatis sapiente iure minus laudantium! Quidem iure consectetur quod odit iusto, labore sunt quae, enim nesciunt officiis, quia ad.</p>
    </div>
    <div class="summary"><span class="text">Concept</span> <span class="icon">+</span></div>
    <div class="details">
        <p>Lorem ipsum dolor sit amet consectetur adipisicing elit. Eveniet officiis sequi perspiciatis sapiente iure minus laudantium! Quidem iure consectetur quod odit iusto, labore sunt quae, enim nesciunt officiis, quia ad.</p>
    </div>
    <div class="summary"><span class="text">Terms</span> <span class="icon">+</span></div>
    <div class="details">
        <p>Lorem ipsum dolor sit amet consectetur adipisicing elit. Eveniet officiis sequi perspiciatis sapiente iure minus laudantium! Quidem iure consectetur quod odit iusto, labore sunt quae, enim nesciunt officiis, quia ad.</p>
    </div>
</div>
<style>
    .container {
        display: grid;
        /* summary's width needs to be relative to the longest text
           and details needs to span the whole grid */
        grid-template-columns: auto auto; 
    }
    .summary {
        grid-column: 1 / 2; /* start at the beginning of 1st column and end at the beginning of the 2nd column */
        display: flex; /* create horizontal layout */
        justify-content: space-between; /* place the available space between the children */
    }
    .details {
        grid-column: 1 / 3; /* start at the beginning of 1st column and end at the end of the 2nd column */
    }
</style>

如您所见,它可读性强,并且几乎不需要标记。

使用flex和grid,您需要考虑2个层次parent > child。这意味着您需要相应地更改React组件:

const Expander = ({children, title, onExpandToggle, isExpanded}) => (
    <>
        <div class="summary" onClick={onExpandToggle}><span class="title">{title}</span> <span class="icon">+</span></div>
        {isExpanded && (
            <div class="details">
                {children}
            </div>
        )}
    </>
);

在这里,我删除了包装器div,以便使用Fragment确保.summary.details.container的直接子代。请注意,出于UX的原因,我也将onClick移到了整个标题上。

CSS Grid非常有用且灵活,值得learning itbrowser support是最新的,但相当不错,只需确保使用自动前缀即可。

您还可以使用display: inline-block;使用技巧,该技巧的大小取决于其子元素,但是当其中一个子元素需要溢出时(如您的情况),将涉及变通方法。所以我不推荐它。我也建议不要使用Javascript。

请立即告诉我您对此答案的看法。

答案 7 :(得分:1)

有可能。

1。弹性框+赦免:

如果您不介意使用绝对定位的容器,则可以采用这种方式

.container {
  display: inline-flex;
  flex-direction: column;
  align-items: stretch;
  width: auto;
}

.item {
  margin: 5px 0;
  padding: 5px;
  border: 1px solid black;
}

.header {
  display: flex;
  align-items: center;
  justify-content: space-between;
}

.header img {
  width: 8px;
  height: 8px;
  margin-left: 20px;
  cursor: pointer;
}
<div class="container">
  <div class="item">
    <div class="header">
      <span>Synonyms</span>
      <img src="https://image.flaticon.com/icons/png/512/61/61155.png" />
    </div>
  </div>
  <div class="item">
    <div class="header">
      <span>Concept</span>
      <img src="https://image.flaticon.com/icons/png/512/61/61155.png" />
    </div>
  </div>
  <div class="item">
    <div class="header">
      <span>Term</span>
      <img src="https://image.flaticon.com/icons/png/512/61/61155.png" />
    </div>
  </div>
</div>

这里的主要问题是您不知道容器的高度,因此它可能会影响其他内容。

2。内联Flex-box:

如果容器的位置对您很重要,则可以使用 inline-flex

.wrapper {
  margin: 20px 10px;
}

.container {
  display: inline-flex;
  flex-direction: column;
  align-items: stretch;
  width: auto;
}

.item {
  margin: 5px 0;
  padding: 5px;
  border: 1px solid black;
}

.header {
  display: flex;
  align-items: center;
  justify-content: space-between;
}

.header img {
  width: 8px;
  height: 8px;
  margin-left: 20px;
  cursor: pointer;
}
Some content above the container...
<div class="wrapper">
  <div class="container">
    <div class="item">
      <div class="header">
        <span>Synonyms</span>
        <img src="https://image.flaticon.com/icons/png/512/61/61155.png" />
      </div>
    </div>
    <div class="item">
      <div class="header">
        <span>Concept</span>
        <img src="https://image.flaticon.com/icons/png/512/61/61155.png" />
      </div>
    </div>
    <div class="item">
      <div class="header">
        <span>Term</span>
        <img src="https://image.flaticon.com/icons/png/512/61/61155.png" />
      </div>
    </div>
  </div>
</div>
Some content below the container...