Макет с 3 колонками разных размеров с использованием Flex - программирование
Подтвердить что ты не робот

Макет с 3 колонками разных размеров с использованием Flex

У меня есть раздел сайта, который должен содержать три элемента бок о бок. Каждый элемент имеет минимальную ширину. Если ширина экрана недостаточна, чтобы содержать все три элемента, те, которые не подходят, переходят в новую строку.

Чтобы построить этот макет, я использовал Flex. Код.

HTML:

<main id='main'>
    <div id='firstRow' class='row'>
        <div id='col1C' class='col'>col1C title
            <div id='col1Ccon'>col1Ccon content</div>
        </div>
        <div id='col2C' class='col'>col2C title
            <div id='col2Ccon'>col2Ccon content</div>
        </div>
        <div id='col3C' class='col'>col3C title
            <div id='col3Ccon'>col3Ccon content</div>
        </div>
    </div>
</main>

CSS:

:root {
    --w1Col: 478px;
    --w2Col: 370px;
    --w3Col: 350px;
    --wSum3: calc(var(--w1Col) + var(--w2Col) + var(--w3Col));
}

html {
    /*height: 100%;*/
}

body {
    margin: 0;
    font-size: 9pt;
    font-family: 'Roboto', sans-serif;
    font-weight: 300;
    background-color: whitesmoke;
    height: 100%;
    display: flex;
    flex-direction: column;
}

/***************************************************************
 * Layout first row
 */
#main {
    background-color: white;
    /*border: 4px solid red;*/
    color: black;
    height: 100%;
    flex-grow: 1; /* ability for a flex item to grow if necessary */
    flex-shrink: 0; /* ability for a flex item to shrink if necessary */
    flex-basis: auto; /* defines the default size of an element before the remaining space is distributed */
}

#firstRow {
    background-color: white;
    /*border: 1px solid black;*/
    margin: 10px;
    padding: 10px;
    display: flex;
    flex-direction: row; /* colums layout */
    flex-wrap: wrap; /* wrap in multiple lines if it necessary */
    /*min-width: var(--wSum3);*/
    /*justify-content: flex-end;  defines the alignment along the main axis */
}

#firstRow .col {
    /*border: 1px solid black;*/
    flex: 1;
    padding: 5px;
    text-align: center;
}

#col1C {
    background-color: green;
    width: var(--w1Col);
    min-width: var(--w1Col);
    order: 1; /* column order */
    flex-basis: 40%;/*var(--w1Col);  column width */
    justify-content: flex-end;

}
#col2C {
    background-color: blue;
    width: var(--w2Col);
    min-width: var(--w2Col);
    order: 2; /* column order */
    flex-basis: 35%; /* var(--w2Col);  column width */
    justify-content: flex-end;
}
#col3C {
    background-color:  red;
    width: var(--w3Col);
    min-width: var(--w3Col);
    order: 3; /* column order */
    flex-basis: 25%; /*var(--w3Col);  column width */
    justify-content: flex-end;
}

#col1Ccon, #col2Ccon, #col3Ccon {
    /*border: 1px solid black;*/
    margin: 0 auto; /* center content */
}

#col1Ccon {
    width: var(--w1Col);
    background-color: lightgreen;
    height: 200px;
}
#col2Ccon {
    width: var(--w2Col);
    background-color: lightblue;
    height: 150px;
}
#col3Ccon {
    width: var(--w3Col);
    background-color: salmon;
    height: 200px;
}

Код работает, но есть некоторые трюки, которые я хотел бы исправить.

  1. все три колонки имеют одинаковую ширину, и я хотел бы иметь возможность выбирать эту ширину. Это связано с тем, что первый элемент имеет минимальную ширину, большую, чем две другие, поэтому край меньше и эстетически уродлив. См. Тот же пример, я изменил только цвета

  2. оставшееся пространство теперь растет с обеих сторон контейнеров col*Ccon. Вместо этого я хотел бы расти только на левой стороне col1Ccon и на правой стороне col3Ccon. Поэтому мне хотелось бы, чтобы содержимое сайта (col1Ccon + col2Ccon + col3Ccon) всегда оставалось в центре страницы и какие изменения являются "границей", которые растут и уменьшаются.

Я застрял, и любые предложения очень ценятся. Спасибо :)

4b9b3361

Ответ 1

Причина, по которой ваши столбцы имеют одинаковую ширину, заключается в том, что вы дали flex: 1 классу col. В этом случае это означает, что для каждого столбца (который есть все) имеется одна треть экрана, доступного для них. Что вы можете сделать, так это удалить свойство flex из класса col и вместо этого использовать его в области селекторов идентификаторов столбцов. Поэтому в приведенном ниже коде я помещаю свойство flex в селектора идентификаторов с небольшой разницей, что второй столбец получает с ним свойство flex-grow 0 (первое значение в свойстве flex). Первая колонка немного больше, как вы сказали, поэтому занимает немного больше места.

Затем, чтобы сохранить контент в центре: первая часть - дать этому второму столбцу flex-grow 0, поэтому он остается посередине. Остальные два столбца растут, и с автоматическим margin с одной стороны столбцов содержимого он будет расти с одной стороны, а столбец содержимого будет выровнен как слева, так и справа, чтобы дать представление о центрированном контенте. Это только для большего экрана, для небольших экранов вы можете использовать медиа-запросы для правильного выравнивания содержимого.

Итак, в конце концов, ваш код может выглядеть так. Я не знаю, было ли это именно то, что вы имели в виду, но вы можете поиграть с ним и посмотреть, работает ли это.

:root {
    --w1Col: 478px;
    --w2Col: 370px;
    --w3Col: 350px;
    --wSum3: calc(var(--w1Col) + var(--w2Col) + var(--w3Col));
}

body {
    margin: 0;
    font-size: 9pt;
    font-family: 'Roboto', sans-serif;
    font-weight: 300;
    background-color: whitesmoke;
    height: 100%;
    display: flex;
    flex-direction: column;
}

/***************************************************************
 * Layout first row
 */
#main {
    background-color: white;
    color: black;
    height: 100%;
    flex: 1 0 auto; 
    width: 100%;
}

#firstRow {
    background-color: white;
    margin: 10px;
    padding: 10px;
    display: flex;
    flex-direction: row; /* colums layout */
    flex-wrap: wrap; /* wrap in multiple lines if it necessary */
}

#firstRow .col {
    padding: 5px;
    text-align: center;
}

#col1C {
    background-color: green;
    order: 1; /* column order */
    flex: 1 1 var(--w1Col);
}
#col2C {
    background-color: blue;
    order: 2; /* column order */
    flex: 0 1 var(--w2Col);
}
#col3C {
    background-color:  red;
    order: 3; /* column order */
    flex: 1 1 var(--w3Col);
}

#col1Ccon {
    max-width: var(--w1Col);
    background-color: lightgreen;
    height: 200px;
    margin: 0 0 0 auto;
}

#col2Ccon {
    max-width: var(--w2Col);
    background-color: lightblue;
    height: 150px;
    margin: 0 auto;
}
#col3Ccon {
    max-width: var(--w3Col);
    background-color: salmon;
    height: 200px;
    margin: 0 auto 0 0;
}

@media only screen and (max-width: 1268px) {
  #col2C {
    flex-grow: 1;
  }

  #col2Ccon {
    margin: 0 auto 0 0;
  }

  #col3Ccon {
    margin: 0 auto;
  }
}

@media only screen and (max-width: 908px) {
  #col1Ccon {
    margin: 0 auto;
  }

  #col2Ccon {
    margin: 0 0 0 auto;
  }

  #col3Ccon {
    margin: 0 auto 0 0;
  }
}

@media only screen and (max-width: 780px) {
  #col2Ccon, #col3Ccon {
    margin: 0 auto;
  }
}

Что касается второго вопроса