From 573d684b709e11e0e29f0af775e21a9ad4fbe590 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Loi=CC=88c=20Goyet?= Date: Wed, 5 Apr 2017 14:19:12 +0200 Subject: [PATCH] ribbons statements and theme --- dest/css/ribbons.css | 134 +++++++++++++++++++++++- dest/css/ribbons.min.css | 2 +- src/scss/ribbons.scss | 79 ++++++++++++-- src/scss/utils/_get-hypotenuse.scss | 28 +++++ src/scss/utils/_mixin-example.scss | 5 - src/scss/utils/_mixin-ribbons-band.scss | 26 +++++ src/scss/utils/_mixin-ribbons.scss | 19 ++++ src/scss/utils/_trowel-variables.scss | 24 +++-- src/twig/ribbon.html.twig | 13 ++- test/src/index.html.twig | 12 ++- 10 files changed, 313 insertions(+), 29 deletions(-) create mode 100644 src/scss/utils/_get-hypotenuse.scss delete mode 100644 src/scss/utils/_mixin-example.scss create mode 100644 src/scss/utils/_mixin-ribbons-band.scss create mode 100644 src/scss/utils/_mixin-ribbons.scss diff --git a/dest/css/ribbons.css b/dest/css/ribbons.css index 43dbadd..1ed58a7 100644 --- a/dest/css/ribbons.css +++ b/dest/css/ribbons.css @@ -2,6 +2,136 @@ * Ribbons */ .ribbon { - font-size: 1rem; } + font-size: 1rem; + color: white; + position: relative; + width: 7.5em; + height: 7.5em; + overflow: hidden; + pointer-events: none; + display: block; } -/*# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["../../src/scss/ribbons.scss","bower_components/trowel-core/src/trowel-api.scss","bower_components/trowel-core/src/core/_statement.scss","../../src/scss/utils/_trowel-variables.scss"],"names":[],"mappings":"AAAA;;GAEG;ACsCO;ICwCF,gBChFiB,EF4CtB","file":"style.css","sourcesContent":["/**\n * Ribbons\n */\n\n@import 'utils/enables';\n@import 'utils/syntaxes';\n@import 'utils/trowel-variables';\n@import 'utils/mixin-example';\n\n$ribbon--selector: selector(new 'ribbon' with $ribbons--syntax);\n\n@if $ribbons--enabled {\n  @include statement($ribbon--selector, (\n    ('@include ribbons--mixin-example', null),\n    ('font-size', $ribbons--font-size),\n    ('font-family', $ribbons--font-family),\n    ('font-weight', $ribbons--font-weight),\n    ('text-transform', $ribbons--text-transform),\n    ('letter-spacing', $ribbons--letter-spacing),\n    ('line-height', $ribbons--line-height),\n    ('text-align', $ribbons--text-align),\n    ('text-decoration', $ribbons--text-decoration),\n    ('margin', $ribbons--margin),\n    ('border-width', $ribbons--border-width),\n    ('border-style', $ribbons--border-style),\n    ('border-color', $ribbons--border-color),\n    ('border-radius', $ribbons--border-radius),\n    ('color', $ribbons--color),\n    ('box-shadow', $ribbons--box-shadow),\n    ('background', $ribbons--background),\n    ('background-color', $ribbons--background-color),\n  ));\n}\n","@import 'core/statement';\n@import 'core/breakpoints';\n@import 'core/selector';\n@import 'core/value';\n\n// statement\n@mixin statement($selector, $block, $flag: null) {\n  $block: block-sorted($block);\n  @include print-statement($selector, $block, $flag);\n}\n\n\n// media-query\n@mixin media-query ($fork, $philosophy: 'and-more') {\n  $breakpoints-config: map-get($trowel-config, 'breakpoints');\n\n  @if not $breakpoints-config or not $fork {\n    @content;\n  } @else {\n    $breakpoints: breakpoints--constructor($breakpoints-config);\n\n    @if not breakpoint--get-fork-query($breakpoints, $fork, $philosophy) {\n      @content;\n    } @else {\n      @media #{breakpoint--get-fork-query($breakpoints, $fork, $philosophy)} {\n        @content;\n      }\n    }\n  }\n}\n\n\n// selector\n@function print-selector($selector) {\n  @return selector--get-print-parents($selector) + selector--get-print($selector);\n}\n\n@mixin selector($selector) {\n  $breakpoint-fork: selector--get-breakpoint($selector);\n\n  @at-root #{print-selector($selector)} {\n    @include media-query($breakpoint-fork) {\n      @content;\n    }\n  }\n}\n\n@function selector($args) {\n  $args-length: length($args);\n  $directive: nth($args, 1);\n\n  @if $directive == 'new' {\n    @if not ($args-length == 2 or $args-length == 4) {\n      @error 'selector() function with `new` as first argument must have 2 or 4 arguments';\n    }\n\n    $block: nth($args, 2);\n    $syntax: if($args-length >= 4, nth($args, 4), null);\n    @return selector--constructor(nth($args, 2), $syntax);\n  }\n\n  @if not ($args-length == 4 or $args-length == 6) {\n    @error 'selector() function with `set` or `append` as first argument must have 4 or 6 arguments';\n  }\n\n  $selector: nth($args, 2);\n  $property: nth($args, 3);\n  $param-1: nth($args, 4);\n  $param-2: if($args-length >= 6, nth($args, 6), null);\n\n  @if $param-2 {\n    @return call('selector--#{$directive}-#{$property}', $selector, $param-1, $param-2);\n  }\n\n  @return call('selector--#{$directive}-#{$property}', $selector, $param-1);\n}\n\n\n// value\n\n@function value($value, $flags: null) {\n  @if type-of($value) != 'map' {\n    @return $value;\n  }\n\n  @if $flags == null {\n    @return map-get($value, 'default');\n  }\n\n  $values-correspondant: values-correspondant($value, $flags);\n\n  @if length($values-correspondant) == 0 {\n    @return map-get($value, 'default');\n  }\n\n  $value: nth($values-correspondant, 1);\n  @each $value-correspondant in $values-correspondant {\n    @if value-correspondant--get-flags-length($value-correspondant) >= value-correspondant--get-flags-length($value) {\n      $value: $value-correspondant;\n    }\n  }\n\n  @return value-correspondant--get-value($value);\n}\n","@import '../extensions/sort';\n@import '../extensions/map';\n@import 'selector';\n\n@function declaration($block, $property, $value, $flags: ()) {\n  @if type-of($value) == 'map' {\n    @each $flag, $value in $value {\n      $new-flags: $flags;\n\n      @if index((':', '<', '@', '-', '~', '['), str-slice($flag, 0, 1)) {\n        $new-flags: append($new-flags, $flag);\n        $new-flags: sort($new-flags);\n      }\n\n      $block: declaration($block, $property, $value, $new-flags);\n    }\n  } @else {\n    @if str-slice($property, 1, 9) == '@include ' {\n      $mixin: str-slice($property, 10);\n      $mixin-result: call($mixin, $value, $flags);\n      $block: block-sorted($mixin-result, $block);\n      @return $block;\n    } @else {\n      @if length($flags) >= 1 {\n        $previous-block: map-get-deep($block, append($flags, 'default'));\n\n        @if $previous-block {\n          $previous-block: map-smart-merge($previous-block, ($property: $value));\n          $block: map-set-deep($block, append($flags, 'default'), $previous-block);\n        } @else {\n          $block: map-set-deep($block, append($flags, 'default'), ($property: $value));\n        }\n      } @else {\n        $block: map-smart-merge($block, ('default': ($property: $value)));\n      }\n    }\n  }\n\n  @return $block;\n}\n\n@function block-sorted($declarations, $block: ()) {\n  @each $declaration in $declarations {\n    $property: nth($declaration, 1);\n    $value: nth($declaration, 2);\n    $flags: if(length($declaration) >= 3, nth($declaration, 3), ());\n\n    $block: declaration($block, $property, $value, $flags);\n  }\n\n  @return $block;\n}\n\n@mixin print-statement($selector, $block, $flag) {\n  @if $flag != 'default' {\n    @each $flag, $block in $block {\n      $new-selector: $selector;\n\n      $flag-type: str-slice($flag, 0, 1);\n      $flag-name: str-slice($flag, 2);\n\n      @if $flag-type == ':' {\n        $new-selector: selector--append-pseudo-class($new-selector, $flag-name);\n      } @elseif $flag-type == '<' {\n        $new-selector: selector--set-tag($new-selector, $flag-name);\n      } @elseif $flag-type == '@' {\n        $new-selector: selector--set-breakpoint($new-selector, $flag-name);\n      } @elseif $flag-type == '-' {\n        $new-selector: selector--append-modifier($new-selector, $flag-name);\n      } @elseif $flag-type == '~' {\n        $new-selector: selector--append-modifier-element($new-selector, $flag-name);\n      } @elseif $flag-type == '[' {\n        $new-selector: selector--append-attribute($new-selector, $flag-name);\n      }\n\n      @include print-statement($new-selector, $block, $flag);\n    }\n  } @else {\n    @include selector($selector) {\n      @each $property, $value in $block {\n        #{$property}: $value;\n      }\n    }\n  }\n}\n","$ribbons--font-size: 1rem !default;\n$ribbons--font-family: null !default;\n$ribbons--font-weight: null !default;\n$ribbons--text-transform: null !default;\n$ribbons--letter-spacing: null !default;\n$ribbons--line-height: null !default;\n$ribbons--text-align: null !default;\n$ribbons--text-decoration: null !default;\n$ribbons--margin: null !default;\n$ribbons--border-width: null !default;\n$ribbons--border-style: null !default;\n$ribbons--border-color: null !default;\n$ribbons--border-radius: null !default;\n$ribbons--color: null !default;\n$ribbons--box-shadow: null !default;\n$ribbons--background: null !default;\n$ribbons--background-color: null !default;\n"]} */ +.ribbon__band { + right: 100%; + left: auto; + margin-right: -3.75em; + margin-left: 0; + -webkit-transform-origin: top right; + transform-origin: top right; + -webkit-transform: rotate(-45deg); + transform: rotate(-45deg); + width: 5.303301em; + background-color: #9b9b9d; + position: absolute; + top: 0; + height: 1.875em; + text-align: center; + -webkit-user-select: none; + -moz-user-select: none; + -ms-user-select: none; + user-select: none; + display: block; } + +.ribbon--right .ribbon__band { + left: 100%; + right: auto; + margin-left: -3.75em; + margin-right: 0; + -webkit-transform-origin: top left; + transform-origin: top left; + -webkit-transform: rotate(45deg); + transform: rotate(45deg); } + +.ribbon--primary .ribbon__band { + background-color: #1f0d33; } + +.ribbon--success .ribbon__band { + background-color: #37d3bd; } + +.ribbon--warning .ribbon__band { + background-color: #ffb000; } + +.ribbon--danger .ribbon__band { + background-color: #fd4164; } + +.ribbon__band::before { + content: ""; + display: block; + position: absolute; + top: 0; + height: 100%; + width: 10000px; + margin: 0 -1px; + background-color: #9b9b9d; + left: 100%; } + +.ribbon--primary .ribbon__band::before { + background-color: #1f0d33; } + +.ribbon--success .ribbon__band::before { + background-color: #37d3bd; } + +.ribbon--warning .ribbon__band::before { + background-color: #ffb000; } + +.ribbon--danger .ribbon__band::before { + background-color: #fd4164; } + +.ribbon__band::after { + content: ""; + display: block; + position: absolute; + top: 0; + height: 100%; + width: 10000px; + margin: 0 -1px; + background-color: #9b9b9d; + right: 100%; } + +.ribbon--primary .ribbon__band::after { + background-color: #1f0d33; } + +.ribbon--success .ribbon__band::after { + background-color: #37d3bd; } + +.ribbon--warning .ribbon__band::after { + background-color: #ffb000; } + +.ribbon--danger .ribbon__band::after { + background-color: #fd4164; } + +.ribbon__content { + position: absolute; + top: 50%; + left: 50%; + -webkit-transform: translate(-50%, -50%); + transform: translate(-50%, -50%); + z-index: 1; + white-space: nowrap; + overflow: hidden; + display: block; } + +.ribbon__content::before { + content: ""; + background-color: #9b9b9d; + position: absolute; + top: -1px; + bottom: -1px; + left: -1px; + right: -1px; + z-index: -1; + overflow: hidden; } + +.ribbon--primary .ribbon__content::before { + background-color: #1f0d33; } + +.ribbon--success .ribbon__content::before { + background-color: #37d3bd; } + +.ribbon--warning .ribbon__content::before { + background-color: #ffb000; } + +.ribbon--danger .ribbon__content::before { + background-color: #fd4164; } + +/*# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["../../src/scss/ribbons.scss","bower_components/trowel-core/src/trowel-api.scss","bower_components/trowel-core/src/core/_statement.scss","bower_components/trowel-core/src/trowel-theme.scss","../../src/scss/utils/_trowel-variables.scss","../../src/scss/utils/_mixin-ribbons.scss","../../src/scss/utils/_get-hypotenuse.scss","../../src/scss/utils/_mixin-ribbons-band.scss"],"names":[],"mappings":"AAAA;;GAEG;ACsCO;ICwCF,gBChEM;IDgEN,aExE2B;IFwE3B,mBFtDiB;IEsDjB,aE9Da;IF8Db,cE9Da;IF8Db,iBFnDe;IEmDf,qBFlDmB;IEkDnB,eFjDa,ECalB;;AAJO;ICwCF,YGzEuB;IHyEvB,WGxEiB;IHwEjB,sBGvEmE;IHuEnE,eGtE0B;IHsE1B,oCGrE4C;YHqE5C,4BGrE4C;IHqE5C,kCGpE6C;YHoE7C,0BGpE6C;IHoE7C,kBIzD+D;IJyD/D,0BCzE4B;IDyE5B,mBFvCiB;IEuCjB,OFtCK;IEsCL,gBExDiB;IFwDjB,mBFpCiB;IEoCjB,0BFnCgB;OEmChB,uBFnCgB;QEmChB,sBFnCgB;YEmChB,kBFnCgB;IEmChB,eFlCa,ECFlB;;AAJO;ICwCF,WGzEuB;IHyEvB,YGxEiB;IHwEjB,qBGvEmE;IHuEnE,gBGtE0B;IHsE1B,mCGrE4C;YHqE5C,2BGrE4C;IHqE5C,iCGpE6C;YHoE7C,yBGpE6C,EJgClD;;AAJO;ICwCF,0BCxEqB,EFoC1B;;AAJO;ICwCF,0BCvEuB,EFmC5B;;AAJO;ICwCF,0BCtEsB,EFkC3B;;AAJO;ICwCF,0BCrEsB,EFiC3B;;AAJO;ICwCF,YK/DU;IL+DV,eK9Da;IL8Db,mBK7DiB;IL6DjB,OK5DK;IL4DL,aK3DW;IL2DX,eK1Da;IL0Db,eKzDa;ILyDb,0BCzE4B;IDyE5B,WFzBS,ECXd;;AAJO;ICwCF,0BCxEqB,EFoC1B;;AAJO;ICwCF,0BCvEuB,EFmC5B;;AAJO;ICwCF,0BCtEsB,EFkC3B;;AAJO;ICwCF,0BCrEsB,EFiC3B;;AAJO;ICwCF,YK/DU;IL+DV,eK9Da;IL8Db,mBK7DiB;IL6DjB,OK5DK;IL4DL,aK3DW;IL2DX,eK1Da;IL0Db,eKzDa;ILyDb,0BCzE4B;IDyE5B,YFhBU,ECpBf;;AAJO;ICwCF,0BCxEqB,EFoC1B;;AAJO;ICwCF,0BCvEuB,EFmC5B;;AAJO;ICwCF,0BCtEsB,EFkC3B;;AAJO;ICwCF,0BCrEsB,EFiC3B;;AAJO;ICwCF,mBFTiB;IESjB,SFRO;IEQP,UFPQ;IEOR,yCFN+B;YEM/B,iCFN+B;IEM/B,WFLS;IEKT,oBFJkB;IEIlB,iBFHe;IEGf,eFFa,EClClB;;AAJO;ICwCF,YFKU;IELV,0BCzE4B;IDyE5B,mBFOiB;IEPjB,UFQQ;IERR,aFSW;IETX,WFUS;IEVT,YFWU;IEXV,YFYU;IEZV,iBFae,ECjDpB;;AAJO;ICwCF,0BCxEqB,EFoC1B;;AAJO;ICwCF,0BCvEuB,EFmC5B;;AAJO;ICwCF,0BCtEsB,EFkC3B;;AAJO;ICwCF,0BCrEsB,EFiC3B","file":"style.css","sourcesContent":["/**\n * Ribbons\n */\n\n@import 'utils/enables';\n@import 'utils/syntaxes';\n@import 'utils/trowel-variables';\n@import 'utils/mixin-ribbons';\n@import 'utils/mixin-ribbons-band';\n\n\n@if $ribbons--enabled {\n  $ribbon--selector: selector(new 'ribbon' with $ribbons--syntax);\n\n  @include statement($ribbon--selector, (\n    ('font-size', $ribbons--font-size),\n    ('font-family', $ribbons--font-family),\n    ('font-weight', $ribbons--font-weight),\n    ('text-transform', $ribbons--text-transform),\n    ('letter-spacing', $ribbons--letter-spacing),\n    ('line-height', $ribbons--line-height),\n    ('text-decoration', $ribbons--text-decoration),\n    ('margin', $ribbons--margin),\n    ('color', $ribbons--color),\n    ('box-shadow', $ribbons--box-shadow),\n    ('background', $ribbons--background),\n    ('position', relative),\n    ('width', $ribbons--size),\n    ('height', $ribbons--size),\n    ('overflow', hidden),\n    ('pointer-events', none),\n    ('display', block),\n  ));\n\n\n  $ribbon-band--selector: selector(set $ribbon--selector element 'band');\n\n  @include statement($ribbon-band--selector, (\n    ('@include ribbons--position', $ribbons--position),\n    ('@include ribbons-band--width', $ribbons--gutter),\n    ('background-color', $ribbons--background-color),\n    ('position', absolute),\n    ('top', 0),\n    ('height', $ribbons--height),\n    ('text-align', center),\n    ('user-select', none),\n    ('display', block),\n  ));\n\n\n  $ribbon-band-before--selector: selector(set $ribbon-band--selector pseudo-element 'before');\n\n  @include statement($ribbon-band-before--selector, (\n    ('@include ribbons-band-pseudos--commons', ''),\n    ('background-color', $ribbons--background-color),\n    ('left', 100%),\n  ));\n\n\n  $ribbon-band-after--selector: selector(set $ribbon-band--selector pseudo-element 'after');\n\n  @include statement($ribbon-band-after--selector, (\n    ('@include ribbons-band-pseudos--commons', ''),\n    ('background-color', $ribbons--background-color),\n    ('right', 100%),\n  ));\n\n\n  $ribbon-content--selector: selector(set $ribbon--selector element 'content');\n\n  @include statement($ribbon-content--selector, (\n    ('position', absolute),\n    ('top', 50%),\n    ('left', 50%),\n    ('transform', translate(-50%, -50%)),\n    ('z-index', 1),\n    ('white-space', nowrap),\n    ('overflow', hidden),\n    ('display', block),\n  ));\n\n\n  $ribbon-content-before--selector: selector(set $ribbon-content--selector pseudo-element 'before');\n\n  @include statement($ribbon-content-before--selector, (\n    ('content', ''),\n    ('background-color', $ribbons--background-color),\n    ('position', absolute),\n    ('top', -1px),\n    ('bottom', -1px),\n    ('left', -1px),\n    ('right', -1px),\n    ('z-index', -1),\n    ('overflow', hidden),\n  ))\n}\n","@import 'core/statement';\n@import 'core/breakpoints';\n@import 'core/selector';\n@import 'core/value';\n\n// statement\n@mixin statement($selector, $block, $flag: null) {\n  $block: block-sorted($block);\n  @include print-statement($selector, $block, $flag);\n}\n\n\n// media-query\n@mixin media-query ($fork, $philosophy: 'and-more') {\n  $breakpoints-config: map-get($trowel-config, 'breakpoints');\n\n  @if not $breakpoints-config or not $fork {\n    @content;\n  } @else {\n    $breakpoints: breakpoints--constructor($breakpoints-config);\n\n    @if not breakpoint--get-fork-query($breakpoints, $fork, $philosophy) {\n      @content;\n    } @else {\n      @media #{breakpoint--get-fork-query($breakpoints, $fork, $philosophy)} {\n        @content;\n      }\n    }\n  }\n}\n\n\n// selector\n@function print-selector($selector) {\n  @return selector--get-print-parents($selector) + selector--get-print($selector);\n}\n\n@mixin selector($selector) {\n  $breakpoint-fork: selector--get-breakpoint($selector);\n\n  @at-root #{print-selector($selector)} {\n    @include media-query($breakpoint-fork) {\n      @content;\n    }\n  }\n}\n\n@function selector($args) {\n  $args-length: length($args);\n  $directive: nth($args, 1);\n\n  @if $directive == 'new' {\n    @if not ($args-length == 2 or $args-length == 4) {\n      @error 'selector() function with `new` as first argument must have 2 or 4 arguments';\n    }\n\n    $block: nth($args, 2);\n    $syntax: if($args-length >= 4, nth($args, 4), null);\n    @return selector--constructor(nth($args, 2), $syntax);\n  }\n\n  @if not ($args-length == 4 or $args-length == 6) {\n    @error 'selector() function with `set` or `append` as first argument must have 4 or 6 arguments';\n  }\n\n  $selector: nth($args, 2);\n  $property: nth($args, 3);\n  $param-1: nth($args, 4);\n  $param-2: if($args-length >= 6, nth($args, 6), null);\n\n  @if $param-2 {\n    @return call('selector--#{$directive}-#{$property}', $selector, $param-1, $param-2);\n  }\n\n  @return call('selector--#{$directive}-#{$property}', $selector, $param-1);\n}\n\n\n// value\n\n@function value($value, $flags: null) {\n  @if type-of($value) != 'map' {\n    @return $value;\n  }\n\n  @if $flags == null {\n    @return map-get($value, 'default');\n  }\n\n  $values-correspondant: values-correspondant($value, $flags);\n\n  @if length($values-correspondant) == 0 {\n    @return map-get($value, 'default');\n  }\n\n  $value: nth($values-correspondant, 1);\n  @each $value-correspondant in $values-correspondant {\n    @if value-correspondant--get-flags-length($value-correspondant) >= value-correspondant--get-flags-length($value) {\n      $value: $value-correspondant;\n    }\n  }\n\n  @return value-correspondant--get-value($value);\n}\n","@import '../extensions/sort';\n@import '../extensions/map';\n@import 'selector';\n\n@function declaration($block, $property, $value, $flags: ()) {\n  @if type-of($value) == 'map' {\n    @each $flag, $value in $value {\n      $new-flags: $flags;\n\n      @if index((':', '<', '@', '-', '~', '['), str-slice($flag, 0, 1)) {\n        $new-flags: append($new-flags, $flag);\n        $new-flags: sort($new-flags);\n      }\n\n      $block: declaration($block, $property, $value, $new-flags);\n    }\n  } @else {\n    @if str-slice($property, 1, 9) == '@include ' {\n      $mixin: str-slice($property, 10);\n      $mixin-result: call($mixin, $value, $flags);\n      $block: block-sorted($mixin-result, $block);\n      @return $block;\n    } @else {\n      @if length($flags) >= 1 {\n        $previous-block: map-get-deep($block, append($flags, 'default'));\n\n        @if $previous-block {\n          $previous-block: map-smart-merge($previous-block, ($property: $value));\n          $block: map-set-deep($block, append($flags, 'default'), $previous-block);\n        } @else {\n          $block: map-set-deep($block, append($flags, 'default'), ($property: $value));\n        }\n      } @else {\n        $block: map-smart-merge($block, ('default': ($property: $value)));\n      }\n    }\n  }\n\n  @return $block;\n}\n\n@function block-sorted($declarations, $block: ()) {\n  @each $declaration in $declarations {\n    $property: nth($declaration, 1);\n    $value: nth($declaration, 2);\n    $flags: if(length($declaration) >= 3, nth($declaration, 3), ());\n\n    $block: declaration($block, $property, $value, $flags);\n  }\n\n  @return $block;\n}\n\n@mixin print-statement($selector, $block, $flag) {\n  @if $flag != 'default' {\n    @each $flag, $block in $block {\n      $new-selector: $selector;\n\n      $flag-type: str-slice($flag, 0, 1);\n      $flag-name: str-slice($flag, 2);\n\n      @if $flag-type == ':' {\n        $new-selector: selector--append-pseudo-class($new-selector, $flag-name);\n      } @elseif $flag-type == '<' {\n        $new-selector: selector--set-tag($new-selector, $flag-name);\n      } @elseif $flag-type == '@' {\n        $new-selector: selector--set-breakpoint($new-selector, $flag-name);\n      } @elseif $flag-type == '-' {\n        $new-selector: selector--append-modifier($new-selector, $flag-name);\n      } @elseif $flag-type == '~' {\n        $new-selector: selector--append-modifier-element($new-selector, $flag-name);\n      } @elseif $flag-type == '[' {\n        $new-selector: selector--append-attribute($new-selector, $flag-name);\n      }\n\n      @include print-statement($new-selector, $block, $flag);\n    }\n  } @else {\n    @include selector($selector) {\n      @each $property, $value in $block {\n        #{$property}: $value;\n      }\n    }\n  }\n}\n","// scss-lint:disable ColorVariable\n\n$trwl-theme: (\n  'colors': (\n   'gray-light': rgb(248, 248, 250),\n   'gray-medium': rgb(232, 232, 236),\n   'gray-dark': rgb(207, 207, 213),\n   'gray-darker': rgb(155, 155, 157),\n   'primary': rgb(31, 13, 51),\n   'success': rgb(55, 211, 189),\n   'warning': rgb(255, 176, 0),\n   'danger': rgb(253, 65, 100),\n  ),\n  'font-sizes': (\n    'xs': 0.65rem,\n    'sm': 0.8rem,\n    'md': 1rem,\n    'lg': 1.2rem,\n  ),\n  'font-family': #{\"'Open Sans', verdana, sans-serif\"},\n  'line-height': 1.2,\n  'padding-y': 0.65em,\n  'padding-x': 1.65em,\n  'border-width': 0.0625em,\n  'border-style': solid,\n  'border-radius': 0.2em,\n);\n\n@function trwl-theme($property) {\n  @return map-get($trwl-theme, $property);\n}\n\n@function trwl-theme-color($color) {\n  @return map-get(map-get($trwl-theme, 'colors'), $color);\n}\n\n@function trwl-theme-size($size) {\n  @return map-get(map-get($trwl-theme, 'font-sizes'), $size);\n}\n","$ribbons--font-size: trwl-theme-size('md') !default;\n$ribbons--font-family: null !default;\n$ribbons--font-weight: null !default;\n$ribbons--text-transform: null !default;\n$ribbons--letter-spacing: null !default;\n$ribbons--line-height: null !default;\n$ribbons--text-decoration: null !default;\n$ribbons--margin: null !default;\n$ribbons--color: rgb(255, 255, 255) !default;\n$ribbons--box-shadow: null !default;\n$ribbons--background: null !default;\n$ribbons--background-color: (\n  'default': trwl-theme-color('gray-darker'),\n  '-primary': trwl-theme-color('primary'),\n  '-success': trwl-theme-color('success'),\n  '-warning': trwl-theme-color('warning'),\n  '-danger': trwl-theme-color('danger'),\n) !default;\n$ribbons--size: 7.5em !default;\n$ribbons--position: (\n  'default': 'left',\n  '-right': 'right',\n) !default; // 'left' or 'right'\n$ribbons--gutter: 3.75em !default;\n$ribbons--height: 1.875em !default;\n","@function ribbons--position($orientation, $flags) {\n  $property-affected: if($orientation == 'right', 'left', 'right');\n\n  @if $orientation == 'right' or $orientation == 'left' {\n    $rotate-direction: if($orientation == 'left', -1, 1);\n\n    @return (\n      ($property-affected, 100%, $flags),\n      ($orientation, auto, $flags),\n      ('margin-#{$property-affected}', value($ribbons--gutter, $flags) * -1, $flags),\n      ('margin-#{$orientation}', 0, $flags),\n      ('transform-origin', top #{$property-affected}, $flags),\n      ('transform', rotate(45deg * $rotate-direction), $flags),\n    );\n  } @else {\n    @error \"The parameter `$orientation` must be a string value that can only be 'right' or 'left'\";\n  }\n\n}\n","@function square-root($r) {\n  $x0: 1;\n  $x1: $x0;\n\n  @for $i from 1 through 10 {\n    $x1: $x0 - ($x0 * $x0 - abs($r)) / (2 * $x0);\n    $x0: $x1;\n  }\n\n  @return $x1;\n}\n\n@function strip-units($number) {\n  @return $number / ($number * 0 + 1);\n}\n\n@function square($number) {\n  $unitless: strip-units($number);\n  @return ($unitless * $unitless);\n}\n\n@function get-hypotenuse($ab, $ac) {\n  @if unit($ab) == unit($ac) {\n    @return square-root(square($ab) + square($ac)) + unquote(unit($ab));\n  } @else {\n    @error 'The two value into the get-hypotenuse() function must share the same unit type';\n  }\n}\n","@import './get-hypotenuse';\n\n@function ribbons-band--width($gutter, $flags) {\n  @return (\n    ('width', get-hypotenuse($gutter, $gutter), $flags),\n  );\n}\n\n\n// (1) Big static value in order to make sure that for any gutter value the\n//     before and after goes off the thumbnail element\n// (2) We set a negative margin on x axis in order to avoid a very thin band\n//     where the `::after` and `::before` pseudo elements are not perfectly\n//     sticked to the `.thumbnail__ribbon`.\n\n@function ribbons-band-pseudos--commons($nothing, $flags) {\n  @return (\n    ('content', ''),\n    ('display', block),\n    ('position', absolute),\n    ('top', 0),\n    ('height', 100%),\n    ('width', 10000px), // (1)\n    ('margin', 0 -1px), // (2)\n  );\n}\n"]} */ diff --git a/dest/css/ribbons.min.css b/dest/css/ribbons.min.css index e82a104..a46287a 100644 --- a/dest/css/ribbons.min.css +++ b/dest/css/ribbons.min.css @@ -1 +1 @@ -.ribbon{font-size:1rem} \ No newline at end of file +.ribbon{font-size:1rem;color:#fff;position:relative;width:7.5em;height:7.5em;overflow:hidden;pointer-events:none;display:block}.ribbon__band{right:100%;left:auto;margin-right:-3.75em;margin-left:0;-webkit-transform-origin:top right;transform-origin:top right;-webkit-transform:rotate(-45deg);transform:rotate(-45deg);width:5.303301em;background-color:#9b9b9d;position:absolute;top:0;height:1.875em;text-align:center;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;display:block}.ribbon--right .ribbon__band{left:100%;right:auto;margin-left:-3.75em;margin-right:0;-webkit-transform-origin:top left;transform-origin:top left;-webkit-transform:rotate(45deg);transform:rotate(45deg)}.ribbon__band::after,.ribbon__band::before{top:0;height:100%;width:10000px;margin:0 -1px;display:block;content:"";position:absolute}.ribbon--primary .ribbon__band{background-color:#1f0d33}.ribbon--success .ribbon__band{background-color:#37d3bd}.ribbon--warning .ribbon__band{background-color:#ffb000}.ribbon--danger .ribbon__band{background-color:#fd4164}.ribbon__band::before{background-color:#9b9b9d;left:100%}.ribbon--primary .ribbon__band::before{background-color:#1f0d33}.ribbon--success .ribbon__band::before{background-color:#37d3bd}.ribbon--warning .ribbon__band::before{background-color:#ffb000}.ribbon--danger .ribbon__band::before{background-color:#fd4164}.ribbon__band::after{background-color:#9b9b9d;right:100%}.ribbon--primary .ribbon__band::after{background-color:#1f0d33}.ribbon--success .ribbon__band::after{background-color:#37d3bd}.ribbon--warning .ribbon__band::after{background-color:#ffb000}.ribbon--danger .ribbon__band::after{background-color:#fd4164}.ribbon__content{position:absolute;top:50%;left:50%;-webkit-transform:translate(-50%,-50%);transform:translate(-50%,-50%);z-index:1;white-space:nowrap;overflow:hidden;display:block}.ribbon__content::before{content:"";background-color:#9b9b9d;position:absolute;top:-1px;bottom:-1px;left:-1px;right:-1px;z-index:-1;overflow:hidden}.ribbon--primary .ribbon__content::before{background-color:#1f0d33}.ribbon--success .ribbon__content::before{background-color:#37d3bd}.ribbon--warning .ribbon__content::before{background-color:#ffb000}.ribbon--danger .ribbon__content::before{background-color:#fd4164} \ No newline at end of file diff --git a/src/scss/ribbons.scss b/src/scss/ribbons.scss index 90d293b..a1e9ab8 100644 --- a/src/scss/ribbons.scss +++ b/src/scss/ribbons.scss @@ -5,29 +5,92 @@ @import 'utils/enables'; @import 'utils/syntaxes'; @import 'utils/trowel-variables'; -@import 'utils/mixin-example'; +@import 'utils/mixin-ribbons'; +@import 'utils/mixin-ribbons-band'; -$ribbon--selector: selector(new 'ribbon' with $ribbons--syntax); @if $ribbons--enabled { + $ribbon--selector: selector(new 'ribbon' with $ribbons--syntax); + @include statement($ribbon--selector, ( - ('@include ribbons--mixin-example', null), ('font-size', $ribbons--font-size), ('font-family', $ribbons--font-family), ('font-weight', $ribbons--font-weight), ('text-transform', $ribbons--text-transform), ('letter-spacing', $ribbons--letter-spacing), ('line-height', $ribbons--line-height), - ('text-align', $ribbons--text-align), ('text-decoration', $ribbons--text-decoration), ('margin', $ribbons--margin), - ('border-width', $ribbons--border-width), - ('border-style', $ribbons--border-style), - ('border-color', $ribbons--border-color), - ('border-radius', $ribbons--border-radius), ('color', $ribbons--color), ('box-shadow', $ribbons--box-shadow), ('background', $ribbons--background), + ('position', relative), + ('width', $ribbons--size), + ('height', $ribbons--size), + ('overflow', hidden), + ('pointer-events', none), + ('display', block), + )); + + + $ribbon-band--selector: selector(set $ribbon--selector element 'band'); + + @include statement($ribbon-band--selector, ( + ('@include ribbons--position', $ribbons--position), + ('@include ribbons-band--width', $ribbons--gutter), + ('background-color', $ribbons--background-color), + ('position', absolute), + ('top', 0), + ('height', $ribbons--height), + ('text-align', center), + ('user-select', none), + ('display', block), + )); + + + $ribbon-band-before--selector: selector(set $ribbon-band--selector pseudo-element 'before'); + + @include statement($ribbon-band-before--selector, ( + ('@include ribbons-band-pseudos--commons', ''), ('background-color', $ribbons--background-color), + ('left', 100%), )); + + + $ribbon-band-after--selector: selector(set $ribbon-band--selector pseudo-element 'after'); + + @include statement($ribbon-band-after--selector, ( + ('@include ribbons-band-pseudos--commons', ''), + ('background-color', $ribbons--background-color), + ('right', 100%), + )); + + + $ribbon-content--selector: selector(set $ribbon--selector element 'content'); + + @include statement($ribbon-content--selector, ( + ('position', absolute), + ('top', 50%), + ('left', 50%), + ('transform', translate(-50%, -50%)), + ('z-index', 1), + ('white-space', nowrap), + ('overflow', hidden), + ('display', block), + )); + + + $ribbon-content-before--selector: selector(set $ribbon-content--selector pseudo-element 'before'); + + @include statement($ribbon-content-before--selector, ( + ('content', ''), + ('background-color', $ribbons--background-color), + ('position', absolute), + ('top', -1px), + ('bottom', -1px), + ('left', -1px), + ('right', -1px), + ('z-index', -1), + ('overflow', hidden), + )) } diff --git a/src/scss/utils/_get-hypotenuse.scss b/src/scss/utils/_get-hypotenuse.scss new file mode 100644 index 0000000..e5e7025 --- /dev/null +++ b/src/scss/utils/_get-hypotenuse.scss @@ -0,0 +1,28 @@ +@function square-root($r) { + $x0: 1; + $x1: $x0; + + @for $i from 1 through 10 { + $x1: $x0 - ($x0 * $x0 - abs($r)) / (2 * $x0); + $x0: $x1; + } + + @return $x1; +} + +@function strip-units($number) { + @return $number / ($number * 0 + 1); +} + +@function square($number) { + $unitless: strip-units($number); + @return ($unitless * $unitless); +} + +@function get-hypotenuse($ab, $ac) { + @if unit($ab) == unit($ac) { + @return square-root(square($ab) + square($ac)) + unquote(unit($ab)); + } @else { + @error 'The two value into the get-hypotenuse() function must share the same unit type'; + } +} diff --git a/src/scss/utils/_mixin-example.scss b/src/scss/utils/_mixin-example.scss deleted file mode 100644 index 9ada614..0000000 --- a/src/scss/utils/_mixin-example.scss +++ /dev/null @@ -1,5 +0,0 @@ -@function ribbons--mixin-example($value, $flags) { - @return ( - ('content', $value, $flags), - ); -} diff --git a/src/scss/utils/_mixin-ribbons-band.scss b/src/scss/utils/_mixin-ribbons-band.scss new file mode 100644 index 0000000..a7bc119 --- /dev/null +++ b/src/scss/utils/_mixin-ribbons-band.scss @@ -0,0 +1,26 @@ +@import './get-hypotenuse'; + +@function ribbons-band--width($gutter, $flags) { + @return ( + ('width', get-hypotenuse($gutter, $gutter), $flags), + ); +} + + +// (1) Big static value in order to make sure that for any gutter value the +// before and after goes off the thumbnail element +// (2) We set a negative margin on x axis in order to avoid a very thin band +// where the `::after` and `::before` pseudo elements are not perfectly +// sticked to the `.thumbnail__ribbon`. + +@function ribbons-band-pseudos--commons($nothing, $flags) { + @return ( + ('content', ''), + ('display', block), + ('position', absolute), + ('top', 0), + ('height', 100%), + ('width', 10000px), // (1) + ('margin', 0 -1px), // (2) + ); +} diff --git a/src/scss/utils/_mixin-ribbons.scss b/src/scss/utils/_mixin-ribbons.scss new file mode 100644 index 0000000..14a5f05 --- /dev/null +++ b/src/scss/utils/_mixin-ribbons.scss @@ -0,0 +1,19 @@ +@function ribbons--position($orientation, $flags) { + $property-affected: if($orientation == 'right', 'left', 'right'); + + @if $orientation == 'right' or $orientation == 'left' { + $rotate-direction: if($orientation == 'left', -1, 1); + + @return ( + ($property-affected, 100%, $flags), + ($orientation, auto, $flags), + ('margin-#{$property-affected}', value($ribbons--gutter, $flags) * -1, $flags), + ('margin-#{$orientation}', 0, $flags), + ('transform-origin', top #{$property-affected}, $flags), + ('transform', rotate(45deg * $rotate-direction), $flags), + ); + } @else { + @error "The parameter `$orientation` must be a string value that can only be 'right' or 'left'"; + } + +} diff --git a/src/scss/utils/_trowel-variables.scss b/src/scss/utils/_trowel-variables.scss index db65b4c..3d9ed2f 100644 --- a/src/scss/utils/_trowel-variables.scss +++ b/src/scss/utils/_trowel-variables.scss @@ -1,17 +1,25 @@ -$ribbons--font-size: 1rem !default; +$ribbons--font-size: trwl-theme-size('md') !default; $ribbons--font-family: null !default; $ribbons--font-weight: null !default; $ribbons--text-transform: null !default; $ribbons--letter-spacing: null !default; $ribbons--line-height: null !default; -$ribbons--text-align: null !default; $ribbons--text-decoration: null !default; $ribbons--margin: null !default; -$ribbons--border-width: null !default; -$ribbons--border-style: null !default; -$ribbons--border-color: null !default; -$ribbons--border-radius: null !default; -$ribbons--color: null !default; +$ribbons--color: rgb(255, 255, 255) !default; $ribbons--box-shadow: null !default; $ribbons--background: null !default; -$ribbons--background-color: null !default; +$ribbons--background-color: ( + 'default': trwl-theme-color('gray-darker'), + '-primary': trwl-theme-color('primary'), + '-success': trwl-theme-color('success'), + '-warning': trwl-theme-color('warning'), + '-danger': trwl-theme-color('danger'), +) !default; +$ribbons--size: 7.5em !default; +$ribbons--position: ( + 'default': 'left', + '-right': 'right', +) !default; // 'left' or 'right' +$ribbons--gutter: 3.75em !default; +$ribbons--height: 1.875em !default; diff --git a/src/twig/ribbon.html.twig b/src/twig/ribbon.html.twig index 00788a4..2406ec3 100644 --- a/src/twig/ribbon.html.twig +++ b/src/twig/ribbon.html.twig @@ -9,6 +9,13 @@ {% set blockSelector = syntax.prefix|default(false) ? syntax.prefix ~ syntax.separator.prefix ~ 'ribbon' : 'ribbon' %} -
- {{ block('body') }} -
+{% set modifiersClass = '' %} +{% for modifier in modifiers|default([]) %} + {% set modifiersClass = modifiersClass ~ ' ' ~ blockSelector ~ syntax.separator.modifier ~ modifier %} +{% endfor %} + + + + {{ block('content') }} + + diff --git a/test/src/index.html.twig b/test/src/index.html.twig index a56d22e..4986b4b 100644 --- a/test/src/index.html.twig +++ b/test/src/index.html.twig @@ -16,9 +16,17 @@ {% embed '../../src/twig/ribbon.html.twig' %} - {% block body %}hello world !{% endblock %} + {% block content %}default{% endblock %} {% endembed %} - + {% embed '../../src/twig/ribbon.html.twig' with { modifiers: ['right'] } %} + {% block content %}on right{% endblock %} + {% endembed %} + + {% for theme in ['primary', 'success', 'warning', 'danger'] %} + {% embed '../../src/twig/ribbon.html.twig' with { modifiers: [theme] } %} + {% block content %}{{ theme }}{% endblock %} + {% endembed %} + {% endfor %}