Как сделать так, чтобы высота div всегда была пропорциональна его ширине (без хаков)? - PullRequest
0 голосов
/ 11 января 2019

У меня есть Flexbox div с 12 изображениями внутри. Я хочу, чтобы width каждого изображения было 25% этого div. Это изображение должно иметь height = 133,33% из width. И если изображение имеет размеры, которые не равны моим 3 * 4, изображение должно быть масштабировано, чтобы соответствовать моим 3 * 4. Так как я могу этого достичь? Я думаю, что мне нужен инструмент, как calc(width*4/3).

Мои CSS и HTML:

.image-grid {
  display: flex;
  flex-wrap: wrap;
  justify-content: space-around;
  align-items: center;
  width: 100%; 
  }
  .image-grid .image-wrapper {
    width: 25%;
    height: 133.3333%; // nothing happens if I delete this line. Flexbox?
    }
  .image-grid .image {
    display: block;
    width: 100%;
    object-fit: cover; 
    }
<div class="image-grid">
    <div class="image-wrapper"><img src="assets/img/portfolio-1.jpg" alt="" class="image"></div>
    <div class="image-wrapper"><img src="assets/img/portfolio-2.jpg" alt="" class="image"></div>
    <div class="image-wrapper"><img src="assets/img/portfolio-3.jpg" alt="" class="image"></div>
    <div class="image-wrapper"><img src="assets/img/portfolio-4.jpg" alt="" class="image"></div>
    <div class="image-wrapper"><img src="assets/img/portfolio-5.jpg" alt="" class="image"></div>
    <div class="image-wrapper"><img src="assets/img/portfolio-6.jpg" alt="" class="image"></div>
    <div class="image-wrapper"><img src="assets/img/portfolio-7.jpg" alt="" class="image"></div>
    <div class="image-wrapper"><img src="assets/img/portfolio-8.jpg" alt="" class="image"></div>
    <div class="image-wrapper"><img src="assets/img/portfolio-9.jpg" alt="" class="image"></div>
    <div class="image-wrapper"><img src="assets/img/portfolio-10.jpg" alt="" class="image"></div>
    <div class="image-wrapper"><img src="assets/img/portfolio-11.jpg" alt="" class="image"></div>
    <div class="image-wrapper"><img src="assets/img/portfolio-12.jpg" alt="" class="image"></div>
</div>
Мой код прекрасно работает с изображениями точно 3 * 4, но если изображение имеет другие размеры, я получаю это bad image

Хорошо, решение найдено. Просто установите отступ обертки как 133,33% вместо высоты (и установите позицию моего img как абсолютную). Это то, что я бы назвал «хитростью» или «взломом», но это решает мою проблему довольно четко. Подробнее здесь: описание

Ответы [ 3 ]

0 голосов
/ 11 января 2019

Вы можете поддерживать соотношение сторон 3: 4 на обертке, поместив на нее padding-top: 133.33%. Таким образом, вместо установки высоты, вы можете установить отступы.

Также вам нужно установить position: relative для оболочки и position: absolute для дочернего элемента, чтобы при позиционировании дочернего элемента игнорировалось заполнение.

есть пример в следующем фрагменте:

.row {
  display: flex;
  flex-wrap: wrap;
}

.column {
  width: 25%;
}

.wrapper {
  padding-top: 133.33%;
  border: 1px solid red;
  position: relative;
}

.wrapper img {
  position: absolute;
  display: block;
  width: 100%;
  height: 100%;
  top: 0;
  left: 0;
  bottom: 0;
  right: 0;
  object-fit: cover;
}
<div class="row">
  <div class="column">
    <div class="wrapper">
      <img src="https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcREb_QCChITKrV0oIljZeP1irlF4UT74fQbJYNiawkZ9efEgREQoA">
    </div>
  </div>
  <div class="column">
    <div class="wrapper">
      <img src="https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcREb_QCChITKrV0oIljZeP1irlF4UT74fQbJYNiawkZ9efEgREQoA">
    </div>
  </div>
  <div class="column">
    <div class="wrapper">
      <img src="https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcREb_QCChITKrV0oIljZeP1irlF4UT74fQbJYNiawkZ9efEgREQoA">
    </div>
  </div>
  <div class="column">
    <div class="wrapper">
      <img src="https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcREb_QCChITKrV0oIljZeP1irlF4UT74fQbJYNiawkZ9efEgREQoA">
    </div>
  </div>
  <div class="column">
    <div class="wrapper">
      <img src="https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcREb_QCChITKrV0oIljZeP1irlF4UT74fQbJYNiawkZ9efEgREQoA">
    </div>
  </div>
  <div class="column">
    <div class="wrapper">
      <img src="https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcREb_QCChITKrV0oIljZeP1irlF4UT74fQbJYNiawkZ9efEgREQoA">
    </div>
  </div>
  <div class="column">
    <div class="wrapper">
      <img src="https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcREb_QCChITKrV0oIljZeP1irlF4UT74fQbJYNiawkZ9efEgREQoA">
    </div>
  </div>
  <div class="column">
    <div class="wrapper">
      <img src="https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcREb_QCChITKrV0oIljZeP1irlF4UT74fQbJYNiawkZ9efEgREQoA">
    </div>
  </div>
  <div class="column">
    <div class="wrapper">
      <img src="https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcREb_QCChITKrV0oIljZeP1irlF4UT74fQbJYNiawkZ9efEgREQoA">
    </div>
  </div>
  <div class="column">
    <div class="wrapper">
      <img src="https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcREb_QCChITKrV0oIljZeP1irlF4UT74fQbJYNiawkZ9efEgREQoA">
    </div>
  </div>
  <div class="column">
    <div class="wrapper">
      <img src="https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcREb_QCChITKrV0oIljZeP1irlF4UT74fQbJYNiawkZ9efEgREQoA">
    </div>
  </div>
  <div class="column">
    <div class="wrapper">
      <img src="https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcREb_QCChITKrV0oIljZeP1irlF4UT74fQbJYNiawkZ9efEgREQoA">
    </div>
  </div>
  <div class="column">
    <div class="wrapper">
      <img src="https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcREb_QCChITKrV0oIljZeP1irlF4UT74fQbJYNiawkZ9efEgREQoA">
    </div>
  </div>
</div>
0 голосов
/ 11 января 2019

Попробуйте это:

.image-grid {
  display: flex;
  flex-wrap: wrap;
}

.image-grid .image-wrapper {
  position: relative;
  width: 25%;
  height: 0;
  padding-bottom: 33.3333%;
}

.image-grid .image {
  position: absolute;
  width: 100%;
  height: 100%;
  object-fit: cover;
}
<div class="image-grid">
  <div class="image-wrapper"><img src="https://picsum.photos/200/300?1.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="https://picsum.photos/300/200?2.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="https://picsum.photos/200/200?3.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="https://picsum.photos/300/300?4.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="https://picsum.photos/300/400?5.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="https://picsum.photos/200/200?6.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="https://picsum.photos/200/200?7.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="https://picsum.photos/200/400?8.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="https://picsum.photos/400/200?9.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="https://picsum.photos/200/200?1.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="https://picsum.photos/200/200?2.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="https://picsum.photos/200/200?4.jpg" alt="" class="image"></div>
</div>
0 голосов
/ 11 января 2019

Вот идея, которая должна работать, если контейнер охватывает всю ширину окна:

Сделайте контейнер изображений шириной 25% и высотой 25 В * 4/3 (там у вас есть calc ...). Также используйте overflow: hidden на контейнере, чтобы сохранить фиксированное соотношение.

Сделайте изображения 100% высотой и установите их ширину auto. Это заставит их заполнить контейнер вертикально. Затем отцентрируйте их, применив transform: translate-x(-50%), position: relative и left: 50%;.

Это работает только для изображений, высота которых меньше, чем 4/3 ширины, но насколько я понял, у вас такая ситуация?

* {
  margin: 0;
}

.image-grid {
  display: flex;
  flex-wrap: wrap;
}

.image-grid .image-wrapper {
  flex-shrink: 1;
  flex-grow: 0;
  width: 25%;
  height: calc(25vw*4/3);
  overflow: hidden;
}

.image-grid .image {
  height: 100%;
  width: auto;
  display: block;
  position: relative;
  left: 50%;
  transform: translateX(-50%);
}
<div class="image-grid">
  <div class="image-wrapper"><img src="http://lorempixel.com/output/city-h-c-300-400-1.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="http://lorempixel.com/output/transport-h-c-300-400-5.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="http://lorempixel.com/output/cats-h-c-300-360-6.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="http://lorempixel.com/output/nature-h-c-300-335-10.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="http://lorempixel.com/output/transport-h-c-300-400-5.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="http://lorempixel.com/output/cats-h-c-300-360-6.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="http://lorempixel.com/output/nature-h-c-300-335-10.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="http://lorempixel.com/output/city-h-c-300-400-1.jpg" alt="" class="image"></div>

  <div class="image-wrapper"><img src="http://lorempixel.com/output/cats-h-c-300-360-6.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="http://lorempixel.com/output/city-h-c-300-400-1.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="http://lorempixel.com/output/transport-h-c-300-400-5.jpg" alt="" class="image"></div>
  <div class="image-wrapper"><img src="http://lorempixel.com/output/nature-h-c-300-335-10.jpg" alt="" class="image"></div>

</div>
...