如何为所有实例设置相同的组件 child 动态宽度?网络组件

How to set same dynamic width of component child for all instances? Web Components

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

一个非常简单的 React 实现(伪代码)可能如下所示:

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

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

它可能看起来像这样:

现在假设下面有多个这样的。变得凌乱:

为了使这个更清晰,所有图标都应该从左边开始有相同的填充。 padding不是固定的而是动态的,这样最长的title决定了所有图标的位置:

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

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

不存在

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

然后用 div 包裹你的 component/s,display: inline-block 占据里面最大元素的宽度。

这是一个例子:

.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>

您也可以尝试使用 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>

我认为,严格来说,考虑到您设置的所有限制,答案是 "no, it's not possible"。

但是因为我相信您无论如何都需要以某种方式对这些扩展器进行分组(或者页面上的每个扩展器都将水平对齐,对此也不存在完整的 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>

我个人会选择经典 <table>。它简单,语义正确,3 分钟内完成。但是我们也可以使用 display:inline-blockfloat:right,或者使用 display:tabledisplay:table-rowdisplay:table-cell 来模仿 <table>。在我看来 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>

根据您的代码片段,我能想到的最简单的示例如下:

 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>
);
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>

在 CSS 中,页面的不同元素不可能 "know" 彼此。最接近的是使用 display flex and/or grid。这样直接 parent 就可以控制它的 children 的显示方式,有点类似于 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,您需要从两个层面考虑 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 是 [=17 的直接 children =].请注意,出于用户体验的原因,我还移动了整个标题上的 onClick。

CSS Grid 非常有用和灵活,值得 learning it. The browser support 是最近的但是相当不错,请确保您使用 autoprefixer。

您还可以使用 display: inline-block; 的技巧,其大小基于其 children,但当 children 之一需要溢出时(如您的情况),将涉及解决方法。所以我不推荐它。我还建议不要使用 Javascript.

不要犹豫,告诉我你对这个答案的看法。

有可能。

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。内联弹性框:

如果容器位置对你很重要,你可以使用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...