/sass-mixin-flex-grid

A sass mixin based on fukol-grids

Primary LanguageCSS

SASS mixin flex grid

A sass mixin based on fukol-grids.

Works with over 4 cols, but 4 is already a good number: do we need other cols in a web layout?

Take a look on a DEMO

SCSS

// ↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧↧
// Flex grid
// ↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥↥

@mixin flex-grid($cols: 2, $gap: 0, $media-queries: true) {

  @if $media-queries {
    // grid works only from small viewport and beyond
    @media screen and (min-width: 48em) {
      display: flex;
      flex-wrap: wrap;

      @if $gap > 0 {
        overflow: hidden;
        margin-left: -($gap);
        margin-right: -($gap);
      }
    }
  }

  & > * {
    width: 100%; // for IE 10

    @if $gap > 0 {
      margin-bottom: $gap * 2; // vertical space between items, relative to horizontal space
		}

    @if $media-queries != true {
      // contextual var
      $item-width: if($gap > 0, calc((100% / #{$cols}) - #{$gap * 2}), (100% / $cols));
      // ---------------
      flex: 0 0 $item-width;
      flex-basis: $item-width; // repeat for IE 11

      @if $gap > 0 {
        margin-left: $gap;
        margin-right: $gap;
       }
    } @else {

      // medium viewport
      @media screen and (min-width: 60em) {
        // contextual var
        $item-width: if($gap > 0, calc(50% - #{$gap * 2}), 50%);
        // ---------------
        flex: 0 0 $item-width;

        @if $gap > 0 {
          margin-left: $gap;
          margin-right: $gap;
         }
      }

      // MORE THAN 2 COLS
      @if $cols >= 3 {
        // large viewport
        @media screen and (min-width: 80em) {
          // contextual var
          $item-width: if($gap > 0, calc((100% / #{$cols}) - #{$gap * 2}), (100% / $cols));
          // ---------------
          flex-basis: $item-width;
        }
      }

      // MORE THAN 3 COLS
      @if $cols >= 4 {
        // large viewport
        @media screen and (min-width: 80em) {
          // contextual var
          $item-width: if($gap > 0, calc((100% / 3) - #{$gap * 2}), (100% / 3)); // always 3 cols on large viewport
          // ---------------
          flex-basis: $item-width;
        }

        // extra-large viewport
        @media screen and (min-width: 100em) {
          // contextual var
          $item-width: if($gap > 0, calc((100% / #{$cols}) - #{$gap * 2}), (100% / $cols));
          // ---------------
          flex-basis: $item-width;
        }
      }

      // MORE THAN 4 COLS
      @if $cols >= 5 {
        // large viewport
        @media screen and (min-width: 80em) {
          // contextual var
          $item-width: if($gap > 0, calc((100% / 4) - #{$gap * 2}), (100% / 4)); // always 4 cols on large viewport
          // ---------------
          flex-basis: $item-width;
        }

        // extra-large viewport
        @media screen and (min-width: 100em) {
          // contextual var
          $item-width: if($gap > 0, calc((100% / #{$cols}) - #{$gap * 2}), (100% / $cols));
          // ---------------
          flex-basis: $item-width;
        }
      }
    }
  }
}

Usage

without space between columns:

.my-grid {
  @include flex-grid(3);
}

with space between columns:

.my-grid {
  @include flex-grid(3, 2vh);
}

I want my grid only with my precise media query (@media-query: false)

.my-grid {
  
  // vertical space for children
  & > * {
    margin-bottom: 1rem;
  }

  @media screen and @large {
    display: flex;
    flex-wrap: wrap;
    @include flex-grid(3, .5rem, false);
}