Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feature debug #31

Open
wants to merge 15 commits into
base: master
Choose a base branch
from
4 changes: 4 additions & 0 deletions README.MD
Original file line number Diff line number Diff line change
Expand Up @@ -148,6 +148,10 @@ Avalanche includes a media query mixin that makes use of the breakpoint aliases
}
}
```
## Debug

**`$av-enable-debug`**
Enable debug mode to view the current breakpoint in the bottom right corner of your browser.

## Browser support

Expand Down
7 changes: 7 additions & 0 deletions _avalanche.scss
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@ $av-enable-grid-large: false !default;
$av-enable-grid-huge: false !default;
$av-enable-grid-auto: false !default;
$av-enable-grid-rev: false !default;
$av-enable-debug: false !default;

/**
* LOGIC aka THE MAGIC
Expand Down Expand Up @@ -326,3 +327,9 @@ $av-enable-grid-rev: false !default;
}
}
}

/**
* DEBUG
*/

@import 'debug/breakpoints';
37 changes: 37 additions & 0 deletions debug/_breakpoints.scss
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
// Derived vars
$av-number-of-breakpoints: length($av-breakpoints);
$av-breakpoint-keys : map-keys($av-breakpoints);
$av-breakpoint-values : map-values($av-breakpoints);

@import './breakpoints/helpers/string';
@import './breakpoints/helpers/list';
@import './breakpoints/helpers/mediaquery';
@import './breakpoints/label';
@import './breakpoints/mixins';

@if $av-enable-debug {

// Objective: create combinations of media queries.
// Desired combination sequence for example of 4 queries.
// More complex queries come last, because they need to overrule the simpler ones.
// Difficulty: 'and' and 'or' mediaqueries needs to be combined accordingly
// Some combinations won't make any sence, but let the browser handle this

// 1
// 2
// 3
// 4
// 1 2
// 1 3
// 1 4
// 2 3
// 2 4
// 3 4
// 1 2 3
// 1 2 4
// 1 3 4
// 2 3 4
// 1 2 3 4

@include breakpointCombinations;
}
23 changes: 23 additions & 0 deletions debug/breakpoints/_label.scss
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
@if $av-enable-debug {
html:after {
position: fixed;
right: 0;
bottom: 0;
z-index: 100000;
display: block;
padding: 5px 5px 5px 28px;
width: auto;
height: auto;
border-width: 1px 0 0 1px;
border-style: solid;
border-color: #ddd;
background: #fbfbfb url('') no-repeat;
background-position: 4px center;
background-size: 16px 16px;
color: #494949;
content: 'No breakpoints';
font-weight: 300;
font-size: 10px;
font-family: monospace;
}
}
104 changes: 104 additions & 0 deletions debug/breakpoints/_mixins.scss
Original file line number Diff line number Diff line change
@@ -0,0 +1,104 @@
@mixin breakpointCombinations() {

// Increase media query complexity
@for $complexity from 1 through $av-number-of-breakpoints{
// Loop through all queries and create combinations for this complexity level
@for $index from 1 through $av-number-of-breakpoints {
$combination: $index;
@include breakpointCombinationsLoop($index, $combination, $complexity);
}
}
}

@mixin breakpointCombinationsLoop($index, $combination, $complexity) {

// The hard work: recursive loop to create a combination of breakpoint indexes for a given complexity
@if length($combination) < $complexity {
@if( $index < $av-number-of-breakpoints){
@for $next from $index + 1 through $av-number-of-breakpoints {
@include breakpointCombinationsLoop($next, append($combination, $next), $complexity);
}
}
}
@else {
@include renderBreakpointCombinations($combination);
}
}

@mixin renderBreakpointCombinations($indexes) {

// Given 4 mediaqueries with and / or
// a , b (=> or)
// c , d (=> or)
// e and f
// g and h

// Desired output for...
// eg. queries 1 & 2
// => a and c , a and d, b and c, b and d
//
// eg. queries 2 & 4
// => c and e and f , d and e and f

$queries: ();
$outputs: ();

// Loop through all query indexes to add them to the final queries
@each $index in $indexes {

$query: getQuery($index);

// $queries has OR
@if queryHasOrCondition($queries) {
@if queryHasOrCondition($query) {
$queries: combineOrWithOr($queries, $query);
}
@else {
$queries: combineOrWithAnd($queries, $query);
}
}
// $queries has AND or is empty the first time
@else {
@if queryHasOrCondition($query) {
$queries: combineAndWithOr($queries, $query);
}
@else {
$queries: combineAndWithAnd($queries, $query);
}
}
}

@each $query in str-explode($queries, ',') {

$media-types: ();
$expressions: ();

// Strip double media types
$expressions-or-types: list-remove-duplicates(str-explode($query, 'and'));

@each $expression-or-type in $expressions-or-types {
@if queryIsMediaType($expression-or-type) {
$media-types: append($media-types, $expression-or-type);
}
@else {
$expressions: append($expressions, $expression-or-type, comma);
}
}

// Cannot have more than 1 media type in 1 query
@if length($media-types) < 2 {
$output: join($media-types, $expressions);
$output: list-implode($output, ' and ');
$outputs: append($outputs, $output, comma);
}
}

// That's all folks
@if length($outputs) > 0 {
@media #{list-implode($outputs, ', ')} {
html:after {
content: getQueryKeys($indexes);
}
}
}
}
34 changes: 34 additions & 0 deletions debug/breakpoints/helpers/_list.scss
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
@function list-remove-duplicates($list, $recursive: false) {
$result: ();

@each $item in $list {

@if not index($result, $item) {

@if length($item) > 1 and $recursive {
$result: append($result, remove-duplicates($item, $recursive));
}
@else {
$result: append($result, $item);
}
}
}

@return $result;
}

@function list-implode($list, $delimiter) {
$string: null;

@each $item in $list {

@if $string and $item {
$string: $string + $delimiter + $item ;
}
@else {
$string: $item ;
}
}

@return $string;
}
70 changes: 70 additions & 0 deletions debug/breakpoints/helpers/_mediaquery.scss
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
@function queryHasOrCondition($query) {
@if type-of($query) == string and str-index($query, ',') {
@return true;
}

@return false;
}

@function queryHasAndCondition($query) {
@return not queryHasOrCondition($query);
}

@function queryIsMediaType($query) {
// !! Only consider screen and print for now
@if( $query == screen or $query == print) {
@return true;
}

@return false;
}

@function getQuery($index) {
@return unquote(nth($av-breakpoint-values, $index));
}

@function getQueryKeys($indexes) {

$results: ();

@each $index in $indexes {
$results: append($results, nth($av-breakpoint-keys, $index), comma);
}

@return list-implode($results, ', ');
}

@function combineAndWithOr($and-query, $or-query) {

$return : ();

@each $or-condition in str-explode($or-query, ',') {
$combination: combineAndWithAnd($and-query, $or-condition);
$return: append($return, $combination, comma);
}

@return inspect($return); // render as string
}

@function combineOrWithAnd($or-query, $and-query) {
@return combineAndWithOr($and-query, $or-query);
}

@function combineAndWithAnd($and-query, $and-query2) {
$join: join($and-query, $and-query2);
@return list-implode($join, ' and ');
}

@function combineOrWithOr($or-query, $or-query2) {

$return : ();

@each $or-condition in str-explode($or-query, ',') {
@each $or-condition2 in str-explode($or-query2, ',') {
$combination: combineAndWithAnd($or-condition, $or-condition2);
$return: append($return, $combination, comma);
}
}

@return inspect($return);
}
42 changes: 42 additions & 0 deletions debug/breakpoints/helpers/_string.scss
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
@function str-explode($string, $delimiter) {

@if type-of($string) == string {
$list: ();
$sum: str-length($string);
$delimiter-length: str-length($delimiter);

// whitespace around delimiter
$string: str-replace($string, $delimiter + ' ', $delimiter);
$string: str-replace($string, ' ' + $delimiter , $delimiter);

@for $i from 1 through $sum {
$str: str-index($string, $delimiter);

@if str-length($string) >= 1 and $str == null {
$list: append($list, $string, comma);
$string: '';
}

@if type-of($str) == number {
$each: str-slice($string, 0, ($str - 1));
$list: append($list, $each, comma);
$string: str-slice($string, ($str + $delimiter-length), $sum);
}
}

@return $list;
}
@else {
@return null;
}
}

@function str-replace($string, $search, $replace: '') {
$index: str-index($string, $search);

@if $index {
@return str-slice($string, 1, $index - 1) + $replace + str-replace(str-slice($string, $index + str-length($search)), $search, $replace);
}

@return $string;
}
11 changes: 11 additions & 0 deletions demo/index.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
<html>
<head>
<title>Avalanche Debug Demo</title>
<link rel="stylesheet" href="style.css">
</head>
<body>
<p class=help>
See the media queries in effect (bottom right)
</p>
</body>
</html>
19 changes: 19 additions & 0 deletions demo/style.scss
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
$av-enable-debug: true;

@import "../_avalanche";


body {
display: flex;
align-items: center;
justify-content: center;
margin: 0;
}

.help {
padding: 10px;
color: #ddd;
text-align: center;
font-size: 21px;
font-family: sans-serif;
}
Loading