2

First, some simplified code and expected properties. This:

<html>

<head>
  <title>Broken image in grid</title>
  <style>
    .App {
      display: flex;
      justify-content: center;
    }

    .Grid {
      height: 300px;
      width: 300px;
      display: grid;
      justify-items: center;
      align-items: center;
      background-color: blue;
      grid-template-columns: 0.25fr repeat(3, 1fr) 0.25fr;
      grid-template-rows: 0.25fr repeat(3, 1fr) 0.25fr;
    }

    .Circle {
      display: block;
      height: 90%;
      width: auto;
      overflow: hidden;
      grid-row: 3;
      grid-column: 3;
    }

  </style>
</head>

<body>
  <div class="App">
    <div class="Grid">
      <img
        src=""
        alt="circle" class="Circle" />
    </div>
  </div>
</body>

</html>

Should render like this (+ chrome dev tools overlay to see grid):

Circle with dev tools overlay

However, on some older Apple devices, e.g. iPhone 6 and iPad 6, it renders like this:

90 percent

I don't have access to an Apple computer, so I can't use Safari dev tools, but as best as I can tell, the image is ignoring the grid and rendering relative to the outer container instead. Weirdly, it isn't at 90% of the outer container, but rather somewhat less. Setting height to 100%, we see the following:

100 percent

which is clearly 100%, but at 50%, we get:

50 percent

which, without actually measuring, looks closer to 25% to me.

It's also worth noting that e.g. a div with background color renders correctly, i.e. this:

<html>

<head>
  <title>Working div in grid</title>
  <style>
    .App {
      display: flex;
      justify-content: center;
    }

    .Grid {
      height: 300px;
      width: 300px;
      display: grid;
      justify-items: center;
      align-items: center;
      background-color: blue;
      grid-template-columns: 0.25fr repeat(3, 1fr) 0.25fr;
      grid-template-rows: 0.25fr repeat(3, 1fr) 0.25fr;
    }

    .Square {
      background-color: black;
      height: 90%;
      width: 90%;
      grid-row: 3;
      grid-column: 3;
      z-index: 1;
    }

  </style>
</head>

<body>
  <div class="App">
    <div class="Grid">
      <div class="Square" />
    </div>
  </div>
</body>

</html>

renders correctly on all tested devices as this:

enter image description here

To summarize, here are the results on all devices that I have ready access to:

  • All: correct div
  • iPhone 6 (iOS 12.5.3, Software 12.5.2), Safari & Chrome: broken img
  • iPhone XR (iOS & Software 14.5.1), Safari: correct img
  • iPad 6th gen (iPadOS 14.5.1, Software 13.3.1), Safari: broken img
  • iPad 7th gen (iPadOS 14.5.1, Software 14.4.2), Safari: correct img

What is going on here? I have no real design experience, so css is admitedly a little bit black magicky to me at times. I'd be more than happy to learn that this is programmer error, but whatever the case, help would be much appreciated. Also helpful would be a description of how to help myself next time, e.g. is there some way I could have looked up device differences on caniuse.com or the like. Thanks in advance.

thisisrandy
  • 2,660
  • 2
  • 12
  • 25
  • 1
    I think it is related to this: https://stackoverflow.com/a/44773109/231316 – Chris Haas May 12 '21 at 22:33
  • @ChrisHaas that answer is very helpful and does indeed sound like the "why" of it. It's a bit surprising that Safari's interpretation of the height spec isn't consistent across devices that are still under support, but I'm willing to accept that it's nonetheless true. That said, the answer (use flexbox instead) doesn't work for me, because I actually require a 2d grid for my application. Anywhere you could point for help with that? – thisisrandy May 13 '21 at 22:00
  • Actually, I think nested flexboxes (per https://stackoverflow.com/a/33644245/12162258) might be the solution. It feels like the code will be significantly less clean, but if it works... I'll try and report back – thisisrandy May 13 '21 at 22:20

1 Answers1

0

As in the answer @ChrisHaas pointed out, this seems to have been due to Safari, at least on older devices, having a different interpretation of the height spec that doesn't consider grid height sufficient for a parent reference. As such, percentage height is meaningless vis-a-vis the grid, and the image scales based on the grid container instead (or something close to it. See the comparisons in my original question above).

This doesn't explain everything in my question, especially why the div scales correctly but the img does not, but it did lead to a working solution using nested flexboxes thanks to an answer linked from the answer noted above. There's a lot more markup doing it this way, and I think the code is much less intuitive, but it does work on the older devices mentioned in the question, so that's something.

NB for anyone trying to use this solution, I'm not certain that using height/width percentages as I have below is the best solution. I tried flex-grow at 1 for the cells and 0.25 for the borders, which constructs the correct empty "grid," but that solution also allows the image to grow the center box instead of being constrained by it, so I had to throw it out. Someone with deeper flexbox knowledge than me might be able to point out a way to make that angle work.

Anyhow, here's the working code:

<html>
  <head>
    <title>Working image in grid</title>
    <style>
      .App {
        display: flex;
        justify-content: center;
      }

      .Grid {
        height: 300px;
        width: 300px;
        display: flex;
        flex-direction: column;
        background-color: blue;
      }

      .Grid-Row {
        width: 100%;
        height: 100%;
        display: flex;
      }

      .Top-Bottom {
        height: 25%;
      }

      .Cell {
        width: 100%;
        display: flex;
        justify-content: center;
        align-items: center;
      }

      .Left-Right {
        width: 25%;
      }

      .Circle {
        width: 90%;
      }
    </style>
  </head>
  <body>
    <div id="root">
      <div class="App">
        <div class="Grid">
          <div class="Grid-Row Top-Bottom"></div>
          <div class="Grid-Row">
            <span class="Cell Left-Right"></span><span class="Cell"></span
            ><span class="Cell"></span><span class="Cell"></span
            ><span class="Cell Left-Right"></span>
          </div>
          <div class="Grid-Row">
            <span class="Cell Left-Right"></span><span class="Cell"></span
            ><span class="Cell"
              ><img
                src=""
                alt="circle"
                class="Circle"
              /> </span
            ><span class="Cell"></span><span class="Cell Left-Right"></span>
          </div>
          <div class="Grid-Row">
            <span class="Cell Left-Right"></span><span class="Cell"></span
            ><span class="Cell"></span><span class="Cell"></span
            ><span class="Cell Left-Right"></span>
          </div>
          <div class="Grid-Row Top-Bottom"></div>
        </div>
      </div>
    </div>
  </body>
</html>

UPDATE: It turns out that there's a better solution. I observed initially that a div in the grid would be correctly placed and sized. It turns out that using such a div to wrap the image also works, so with a small amount of extra markup we can still use CSS Grid.

.App {
  display: flex;
  justify-content: center;
}

.Grid {
  height: 300px;
  width: 300px;
  display: grid;
  background-color: blue;
  grid-template-columns: 0.25fr repeat(3, 1fr) 0.25fr;
  grid-template-rows: 0.25fr repeat(3, 1fr) 0.25fr;
}

.Cell {
  height: 100%;
  width: 100%;
  display: flex;
  justify-content: center;
  align-items: center;
  grid-row: 3;
  grid-column: 3;
}

.Circle {
  display: block;
  height: 90%;
  width: 90%;
}
<html>
  <head>
    <title>Working image in div in grid</title>
  </head>
  <body>
    <div id="root">
      <div class="App">
        <div class="Grid">
          <div class="Cell">
            <img
              src=""
              alt="circle"
              class="Circle"
            />
          </div>
        </div>
      </div>
    </div>
  </body>
</html>
thisisrandy
  • 2,660
  • 2
  • 12
  • 25