Codrops Playground

#cssref linear-gradient() by SaraSoueidan

HTML

19
 
1
<div class='outer-wrap round'>
2
  <ul class='quadrants fill'>
3
    <li class='q'>Q I</li>
4
    <li class='q'>Q II</li>
5
    <li class='q'>Q III</li>
6
    <li class='q'>Q IV</li>
7
  </ul>
8
  <div class='box centre-me'></div>
9
  <div class='axis angle-start-axis centre-me'></div>
10
  <div class='axis h-axis centre-me'></div>
11
  <div class='moving-line grad-line centre-me'></div>
12
  <div class='moving-line grad-start centre-me'></div>
13
  <div class='moving-line grad-end centre-me'></div>
14
  <ul class='angles fill round'></ul>
15
</div>
16
<code class='grad-code'></code>
17
<p>
18
  Demo by <a href="http://codepen.io/thebabydino">Ana Tudor</a>. Licensed under MIT.
19
</p>

CSS

219
 
1
html,
2
body {
3
  min-width: 240px;
4
  min-height: 240px;
5
  height: 100%;
6
  margin: 0;
7
}
8
9
html {
10
  font: .8em Verdana, sans-serif;
11
  text-align: center;
12
}
13
14
body {
15
  padding: 20px;
16
}
17
18
.centre-me {
19
  position: absolute;
20
  top: 50%;
21
  left: 50%;
22
}
23
24
.fill {
25
  position: absolute;
26
  top: 0;
27
  right: 0;
28
  bottom: 0;
29
  left: 0;
30
  margin: 0;
31
}
32
33
.round {
34
  border-radius: 50%;
35
}
36
37
.outer-wrap {
38
  position: relative;
39
  margin: 4em auto 0;
40
}
41
42
.outer-wrap ul {
43
  list-style: none;
44
  -webkit-backface-visibility: hidden;
45
  backface-visibility: hidden;
46
}
47
48
.q {
49
  position: absolute;
50
  font-weight: 700;
51
}
52
53
.q:first-child {
54
  top: -1em;
55
  right: -1em;
56
}
57
58
.q:nth-child(2) {
59
  right: -1em;
60
  bottom: -1em;
61
}
62
63
.q:nth-child(3) {
64
  bottom: -1em;
65
  left: -1em;
66
}
67
68
.q:nth-child(4) {
69
  top: -1em;
70
  left: -1em;
71
}
72
73
.angles {
74
  -webkit-transform: rotate(-90deg);
75
  -ms-transform: rotate(-90deg);
76
  transform: rotate(-90deg);
77
}
78
79
.angle {
80
  margin: -1px;
81
  border: solid 1px;
82
  cursor: pointer;
83
  -webkit-transition: box-shadow .3s;
84
  transition: box-shadow .3s;
85
}
86
87
.angle:after {
88
  position: absolute;
89
  top: 50%;
90
  left: 50%;
91
  margin: -7px;
92
  width: 14px;
93
  height: 14px;
94
  border-radius: 50%;
95
  content: '';
96
}
97
98
.angle:before {
99
  position: absolute;
100
  top: -10px;
101
  left: 11px;
102
  padding: 0 3px;
103
  border-radius: 4px;
104
  box-shadow: 0 0 2px;
105
  background: -webkit-linear-gradient(whitesmoke, gainsboro);
106
  background: linear-gradient(whitesmoke, gainsboro);
107
  color: black;
108
}
109
110
.sec:before {
111
  -webkit-transform: rotate(180deg);
112
  -ms-transform: rotate(180deg);
113
  transform: rotate(180deg);
114
}
115
116
.angle:hover,
117
.selected {
118
  box-shadow: 0 0 1px 3px crimson, 0 0 1px 5px white, 0 0 1px 7px crimson;
119
}
120
121
.angle:hover:before,
122
.selected:before {
123
  content: attr(data-angle) '°';
124
}
125
126
.major {
127
  margin: -2px;
128
  border: solid 2px;
129
  color: crimson;
130
}
131
132
.box {
133
  margin: -15% -30%;
134
  width: 60%;
135
  height: 30%;
136
  background: -webkit-linear-gradient(90deg, deeppink, yellow, springgreen);
137
  background: linear-gradient(0deg, deeppink, yellow, springgreen);
138
  -webkit-transition: background-image .75s linear;
139
  transition: background-image .75s linear;
140
}
141
142
.axis {
143
  top: -50px;
144
  bottom: -50px;
145
  width: 1px;
146
  background: dimgrey;
147
}
148
149
.angle-start-axis:before {
150
  display: block;
151
  min-width: 125px;
152
  text-align: left;
153
  text-indent: 2px;
154
  content: '← angle start axis';
155
}
156
157
.h-axis {
158
  -webkit-transform: rotate(90deg);
159
  -ms-transform: rotate(90deg);
160
  transform: rotate(90deg);
161
}
162
163
.grad-line {
164
  top: 12px;
165
  bottom: 12px;
166
  width: 1px;
167
  background: dodgerblue;
168
  -webkit-transition: -webkit-transform .75s linear;
169
  transition: -webkit-transform .75s linear;
170
  transition: transform .75s linear;
171
}
172
173
.grad-line:before {
174
  display: block;
175
  margin-top: 7px;
176
  min-width: 105px;
177
  text-align: left;
178
  text-indent: 2px;
179
  content: '← gradient line';
180
}
181
182
.rev:before {
183
  -webkit-transform: rotate(180deg);
184
  -ms-transform: rotate(180deg);
185
  transform: rotate(180deg);
186
  content: 'gradient line →';
187
}
188
189
.grad-start,
190
.grad-end {
191
  left: 20%;
192
  right: 20%;
193
  height: 1px;
194
  background: crimson;
195
  -webkit-transition: .75s linear;
196
  transition: .75s linear;
197
}
198
199
.grad-end {
200
  background: black;
201
}
202
203
.grad-code {
204
  display: inline-block;
205
  position: relative;
206
  z-index: 1;
207
  margin-top: 3em;
208
  padding: .5em;
209
  background: black;
210
  color: white;
211
  font: 1.25em monospace;
212
}
213
214
@media (min-width: 240px) {
215
  html {
216
    overflow-x: hidden;
217
  }
218
}
219

JS

171
 
1
(function() {
2
  var fullCircle = 360,
3
      Demo = function() {
4
        // utility class
5
        var EUtil = function() {
6
          this.getSize = function(m /* integer */) {
7
            var w = document.body.clientWidth, h = document.body.clientHeight, s;
8
            s = Math.max(Math.min(Math.min(w,h) - m - 96, 640), 199);
9
            return (s%2 !== 0)?s:(s-1);
10
          }
11
12
          this.setStyle = function(elStyles) {
13
            /*
14
        elStyles is an array
15
        [
16
          [element_1, {property_1_1: value_1_1, property_1_2: value_1_2}], 
17
          [element_2, {property_2_1: value_2_1}]
18
        ]
19
        */
20
            var t, len = elStyles.length,
21
                prefixed = {'property': ['transform'], 'value': ['linear-gradient']}, 
22
                prefixList = ['-webkit-', '-moz', '-o-'], l = prefixList.length;
23
            for(var i = 0; i < len; i++) {
24
              for(var p in elStyles[i][1]) {
25
                if(prefixed['property'][0].indexOf(p) != -1) { /* check if the property needs to be prefixed */
26
                  for(var j = 0; j < l; j++)
27
                    elStyles[i][0].style[prefixList[j] + p] = elStyles[i][1][p];
28
                }
29
                if(elStyles[i][1][p].indexOf(prefixed['value'][0]) != -1) { /* check if the value needs to be prefixed */
30
                  t = 90 - parseInt(elStyles[i][1][p].split(prefixed['value'][0]+'(')[1].split('deg')[0], 10);
31
                  t = prefixed['value'][0] + '(' + t + 'deg' + elStyles[i][1][p].split('deg')[1];
32
                  for(var j = 0; j < l; j++)
33
                    elStyles[i][0].style[p] = prefixList[j] + t;
34
                }
35
                elStyles[i][0].style[p] = elStyles[i][1][p]; /* just set styles, no prefixes */
36
              }
37
            }
38
          };
39
        };
40
41
        // angle class
42
        var Angle = function(angleValue, u, r) {
43
          var value = angleValue, 
44
              selected = (angleValue === 0)?true:false, 
45
              el = document.createElement('li'); /* this is a dot on the circle */
46
          el.setAttribute('class', 'angle centre-me round');
47
          this.isSelected = function()  { return selected; };
48
          this.select = function()      { selected = true; el.classList.add('selected'); };
49
          this.unselect = function()    { selected = false; el.classList.remove('selected'); };
50
          this.getValue = function()    { return value; };
51
          this.getElement = function()  { return el; };
52
          this.isMajor = function()     { return value%15 === 0; };
53
          this.isSec = function()       { return value > 180; };
54
          if(this.isMajor()) el.classList.add('major');
55
          if(this.isSec()) el.classList.add('sec');
56
          if(this.isSelected()) el.classList.add('selected');
57
          if(el.dataset) el.dataset.angle = value;
58
          else el.setAttribute('data-angle', value);
59
          u.setStyle([[el, {'transform': 'rotate(' + value + 'deg) translate(' + r + 'px)'}]]);
60
        };
61
62
        var AngleCircle = function() {
63
          var angles = [], currentAngle, 
64
              getSensitivity = function(size) {
65
                if(size < 240) return 15; /* every 15° for smallest screen */
66
                if(size < 320) return 10; /* every 10° */
67
                if(size < 480) return 6;
68
                if(size < 560) return 5;
69
                return 3;
70
              }
71
72
          this.createCircle = function(parent, u) {
73
            var parentStyle = window.getComputedStyle(parent, null), 
74
                margin = parseInt(parentStyle.marginTop.split('px')[0], 10) + 36, 
75
                size = u.getSize(margin), r = size/2, s = getSensitivity(size), 
76
                listWrap = parent.querySelector('.angles');
77
            u.setStyle([[parent, {'width': size + 'px', 'height': size + 'px'}]]);
78
            listWrap.innerHTML = '';
79
            listWrap = parent.removeChild(listWrap);
80
            for(var i = 0; i < fullCircle/s; i++) {
81
              currentAngle = new Angle(i*s, u, r);
82
              angles.push(currentAngle);
83
              listWrap.appendChild(currentAngle.getElement());
84
            }
85
            parent.appendChild(listWrap);
86
          };
87
88
          this.getSelected = function() {
89
            var len = angles.length;
90
            for(var i = 0; i < len; i++)
91
              if(angles[i].isSelected()) return angles[i];
92
            console.log('error, no angle selected :(');
93
            return null;
94
          };
95
96
          this.getAngle = function(angleValue) {
97
            var len = angles.length;
98
            for(var i = 0; i < len; i++) 
99
              if(angles[i].getValue() == angleValue) return angles[i];
100
            console.log('error, no angle with such value :(');
101
            return null;
102
          };
103
104
          this.select = function(angleValue) {
105
            var len = angles.length, selectedAngle = this.getSelected();
106
            if(angleValue == selectedAngle.getValue()) {
107
              console.log('already selected :)');
108
              return -1;
109
            }
110
            this.getAngle(angleValue).select();
111
            selectedAngle.unselect();
112
          };
113
        };
114
115
        this.init = function() {
116
          var u = new EUtil(), wrap = document.querySelector('.outer-wrap'), 
117
              box = wrap.querySelector('.box'), 
118
              codeEl = document.querySelector('.grad-code'), 
119
              codetxt = 'linear-gradient(0deg, deeppink, yellow, lime)', 
120
              gradLine = document.querySelector('.grad-line'), 
121
              gradStart = document.querySelector('.grad-start'), 
122
              gradEnd = document.querySelector('.grad-end'), 
123
              angleCircle = new AngleCircle(), currentAngle = 0;
124
125
          angleCircle.createCircle(wrap, u);
126
          u.setStyle([[gradLine, {'transform': 'rotate(0)'}], 
127
                      [gradStart, {'transform': 'rotate(0)', 'left': '-10%', 'right': '50%', 'top': '65%'}], 
128
                      [gradEnd, {'transform': 'rotate(0)', 'left': '50%', 'right': '-10%', 'top': '35%'}], 
129
                      [box, {'backgroundImage': codetxt}]]);
130
          codeEl.innerHTML = codetxt;
131
          if(gradLine.classList.contains('rev')) gradLine.classList.remove('rev');
132
133
          wrap.addEventListener('click', function(e) {
134
            var t = e.target, a, dif;
135
            if(t.classList.contains('angle')) {
136
              if(t.dataset) a = parseInt(t.dataset.angle, 10);
137
              else a = parseInt(t.getAttribute('data-angle'), 10);
138
              if(angleCircle.select(a) == -1) return;
139
              dif = a - currentAngle%360;
140
              currentAngle += (Math.abs(dif) > 180) ? (Math.abs(dif) - 360)*Math.abs(dif)/dif : dif;
141
              codetxt = 'linear-gradient(' + currentAngle + 'deg, deeppink, yellow, lime)';
142
              u.setStyle([[gradLine, {'transform': 'rotate(' + currentAngle + 'deg)'}], 
143
                          [gradStart, {'transform': 'rotate(' + currentAngle + 'deg)'}], 
144
                          [gradEnd, {'transform': 'rotate(' + currentAngle + 'deg)'}], 
145
                          [box, {'backgroundImage': codetxt}]]);
146
              codetxt = 'linear-gradient(' + ((currentAngle%360>=0)?(currentAngle%360):(360+currentAngle%360)) + 'deg, deeppink, yellow, lime)';
147
              codeEl.innerHTML = codetxt;
148
              if(a > 90 && a < 270) gradLine.classList.add('rev');
149
              else gradLine.classList.remove('rev');
150
              if(a >= 0 && a < 90)
151
                u.setStyle([[gradStart, {'left': '-10%', 'right': '50%', 'top': '65%'}], 
152
                            [gradEnd, {'left': '50%', 'right': '-10%', 'top': '35%'}]]);
153
              if(a >= 90 && a < 180)
154
                u.setStyle([[gradStart, {'left': '-10%', 'right': '50%', 'top': '35%'}], 
155
                            [gradEnd, {'left': '50%', 'right': '-10%', 'top': '65%'}]]);
156
              if(a >= 180 && a < 270)
157
                u.setStyle([[gradStart, {'left': '50%', 'right': '-10%', 'top': '35%'}], 
158
                            [gradEnd, {'left': '-10%', 'right': '50%', 'top': '65%'}]]);
159
              if(a >= 270 && a < 360)
160
                u.setStyle([[gradStart, {'left': '50%', 'right': '-10%', 'top': '65%'}], 
161
                            [gradEnd, {'left': '-10%', 'right': '50%', 'top': '35%'}]]);
162
163
            }
164
          }, false);
165
        };
166
      }
167
168
  var d = new Demo();
169
  d.init();
170
  window.addEventListener('resize', function(e) { d.init(); }, false);
171
}());

#cssref linear-gradient()

by SaraSoueidan on

Demo for the Codrops CSS Reference entry linear-gradient()

Codrops Logo Codrops Playground

Version 0.0.9 (alpha)

Codrops Playground is currently in development with many features to come, including mobile support. We are in alpha stage right now, so if you find a bug, please send a mail to playground@codrops.com or use the contact form on Codrops.

If you'd like to stay updated, follow us on Twitter @codrops. Thank you and stay tuned :)