<div id="app">
<div id='header'>
 <h1>
      Box Shadow Generator
    </h1>
</div>
  <div id='info-box' :class="{ show: showInfo}">
    <h1>
      Info
    </h1>
    <span class='hide-info' @click="showInfo = false">×</span>
    <h3>Tips</h3>
    <div class='text'>
      <li> Click on the plus button to add a new shadow and change the settings as your like. </li>
      <li>Click on any of the boxes next to the add button to access and change particular shadow settings.  </li>
      <li> Use mousehweel on  any of the circular controls for more precise increase/decrease in value.</li>
    </div>
     <h3>Resources</h3>
    <div class='text'>
     <a href='https://github.com/xiaokaike/vue-color' target=_blank>Vue Color Picker</a>
     <br><a href='https://codepen.io/Brownsugar/pen/NaGPKy' target=_blank>Vue Color Implementation</a>
     <br><a href='https://www.npmjs.com/package/vue-circle-slider' target=_blank>Vue Circular Slider</a>
      <br>Split Vertical icon By <a href='https://thenounproject.com/term/split-vertical/1268414' target=_blank>Bhima</a>
      <br>Spread icon By <a href='https://thenounproject.com/term/spread/1863148' target=_blank>projecthayat</a>
      <br>Blur icon By <a href='https://thenounproject.com/term/blur/705680' target=_blank>abderraouf omara</a>
    </div>

    <h3>Projects</h3>
    <div class='text'>
      <li>Check my other projects at <a href='https://codepen.io/khr2003/' target=_blank>CodePen</a></li>

    </div>
  </div>
  <div id='controls'>
    <div id='reset' @click='resetControls'>
      RESET
    </div>
    <div id='info' @click="showInfo = !showInfo">
     🛈
    </div>

   
    <div id='box-shadows-container'>
      <span v-for='(shadow, index) in boxShadows' :style='{"background-color": shadow.color}' :class='{highlight: (currentShadow == index)}' @click='setShadow(index)'></span>
      <span class='add-transform' @click='addShadow'>+</span>
    </div> 

    <box-shadow-control :data='getData' v-if='boxShadows.length > 0' @delete-shadow='deleteShadow'></box-shadow-control>

  </div>

  <div class='show-effects'>
    <div class='transform-box' :style="style"></div>
  </div>
  <transition name="fade">
    <div id='css' v-show='style["box-shadow"] != ""'>
      <span v-if='style.transform != ""'>box-shadow: {{style["box-shadow"]}};</span>
    </div>
  </transition>

</div>

<template id='box-shadow-control'>
<div class='shadow-control'>
 <div class="control inset">
      <input id='inset-checkbox' type="checkbox" v-model='data.inset'>
      <label for='inset-checkbox'>
       <span class='switch'></span>
    </label>
      <span>Inset Shadow</span>
    </div>
  
  <div class='control color' ref="colorPicker" v-click-outside="hidePicker">
<input type='text' v-model='data.color' @focus='displayPicker = true' readonly/>
<span class='color-box' :style='{"background-color": data.color}'></span>
    <chrome-picker :value="data.color" @input="updatePicker" v-if="displayPicker"/>
  </div>
  
  <div class='control circle' @wheel='handleScroll("xoffset", $event)'>
    <circle-slider v-model="data.xoffset" :min="-50"
  :max="50">data.xoffset</circle-slider>
  <span class='icon xoffset'></span>
  <span class='amount'>{{data.xoffset}}px
    </span>
  </div>
  
  <div class='control circle'@wheel='handleScroll("yoffset", $event)'>
    <circle-slider v-model="data.yoffset" :min="-50"
  :max="50">data.yoffset</circle-slider>
  <span class='icon yoffset'></span>
  <span class='amount'>{{data.yoffset}}px
    </span>
  </div>
  
    <div class='control circle'@wheel='handleScroll("blur", $event)'>
    <circle-slider v-model="data.blur" :min="0"
  :max="100">data.blur</circle-slider>
  <span class='icon blur'></span>
  <span class='amount'>{{data.blur}}px
    </span>
  </div>
  
  <div class='control circle'@wheel='handleScroll("spread", $event)'>
    <circle-slider v-model="data.spread" :min="-50"
  :max="50">data.spread</circle-slider>
  <span class='icon spread'></span>
  <span class='amount'>{{data.spread}}px
    </span>
  </div>
  
  <div class='control delete' @click='$emit("delete-shadow")'>
  DELETE
  </div>

</div>
</template>
@import url("https://fonts.googleapis.com/css?family=Roboto:400,500,900");
@import url("https://fonts.googleapis.com/css?family=Handlee"); 

body {
  background-color: #fafafa;
  font-size: 16px;
  * {
    box-sizing: border-box;
    font-family: "Roboto", sans-serif;
  }
}

#app > #info-box {
  left: 0;
  transform: translateX(-100%);
  opacity: 0;
  background-color: white;
  z-index: 5;
  color: rgba(180, 180, 180, 1);
  font-size: 0.9em;
  
  &.show {
    transform: translateX(0);
    opacity: 1;
  }
  h1
  {
    position: relative;
    font-family: "Handlee";
    color: #3498db;
    
    &::after {
    content: "Info";
      position: absolute;
      top: 50%;
      left: 0%;
      font-size: 1.7em;
      line-height: 1em;
      transform: translate(0%, -50%);
      opacity: 0.1;
      font-weight: bold;
      color: transparentize(#3498db, 0.1);
      
  }
  }
  
  .hide-info
  {
    position: absolute;
    top: 1%;
    right: 1%;
    width: 1em;
    height: 1em;
    cursor: pointer;
    
  }
  h3 {
    font-weight: bold;
    color: rgba(180, 180, 180, 1);
  }
  
  .text {
    
    line-height: 1.5em;
    padding: 0 10px;
    li {
      padding: 10px 0;
      list-style: none;
    }
      a {
        font-weight: bold;
        color: rgba(180, 180, 180, 1);
      }
  }
}

#header
{
  position: fixed;
  top: 0;
  left: 0;
  height: 150px;
  width: 100vw;
  text-align: center;
  
   h1 {
    position: relative;
    padding: 40px 0 60px 0;
    color: darken(#3498db, 25%);
    font-size: 3em;
    font-weight: bold;
    line-height: 1.2em;
    font-family: "Handlee";
    z-index: -2;
    text-shadow: 10px 10px 5px rgba(139, 196, 234, 0.58);
  
  }
}
#controls,
#info-box {
  position: fixed;
  top: 0;
  left: 0;
  bottom: 0;
  width: 300px;
  padding: 0 30px;
  overflow-y: auto;
  overflow: hidden;
  display: flex;
  flex-direction: column;
  z-index: 2;
  transition: all 0.2s ease-in;
  /* iOS Safari */
  -webkit-touch-callout: none;
  /* Safari */
  -webkit-user-select: none;
  /* Konqueror HTML */
  -khtml-user-select: none;
  /* Firefox */
  -moz-user-select: none;
  /* Internet Explorer/Edge */
  -ms-user-select: none;
  /* Non-prefixed version, currently supported by Chrome and Opera */
  user-select: none;
  &:hover {
    opacity: 1;
  }
  &.transparent:not(:hover):not(.info):not(.add-transform) {
    opacity: 1;
  }
  #reset {
    position: absolute;
    bottom: 1%;
    left: 5%;
    padding: 5px 10px;
    font-size: 0.7em;
    color: #e74c3c;
    border: 1px solid darken(#e74c3c, 5%);
    border-radius: 5px;
    cursor: pointer;
    transition: all 0.2s;
    &:hover {
      background-color: #e74c3c;
      color: white;
    }
  }
  #info {
    position: absolute;
    top: 1%;
    left: 2%;
    font-size: 0.7em;
    color: #3498db;
    height: 1em;
    width: 1em;
    border-radius: 5px;
    cursor: pointer;
    transition: all 0.2s;
    font-size: 2em;
    display: flex;
    justify-content: center;
    align-items: center;
    &:hover {
      color: lighten(#3498db, 15%);
    }
  }
 
  h3 {
    width: 100%;
    text-align: left;
    text-transform: capitalize;
    font-size: 1em;
    color: rgba(180, 180, 180, 0.5);
    text-indent: 5%;
    margin-top: 15px;
    margin-bottom: 10px;
  }
  
  #box-shadows-container {
    display: flex;
    flex-wrap: wrap;
    margin-top: 25vh;
    
    span {
      border: 1px solid #3498db;
      width: 20%;
      display: flex;
      justify-content: center;
      align-content: center;
      color: darken(#3498db, 20%);
      margin: 3px;
      height: 25px;
      cursor: pointer;
      transition: all 0.3s;
      opacity: 0.5;
      
      &.highlight, &:hover
      {
        opacity: 1;
      }
      &.add-transform {
        opacity: 1;
      
        &:hover {
          background-color: lighten(#3498db, 10%);
        }
      }
    }
  }
  .shadow-control {
    margin: 30px 0;
    display: flex;
    flex-wrap: wrap;
    
    .control {
      margin: 10px 10px 20px 10px;
      width: 100px;
      height: 100px;
      position: relative;
    
      .amount {
         width: 20%;
        font-size: 0.7em;
        text-align: center;
        position: absolute;
        top: 100%;
        width: 100%;
        left: 0;

        &:hover {
          .delete {
            opacity: 1;
          }
        }
      }
      
      &.inset {
      display: flex;
      margin-bottom: 10px;
      color: rgba(180, 180, 180, 0.5);
      align-items: center;
      width: 100%;
      height: auto;
      
      input {
        display: none;
        &:checked+label {
          background-color: #3498db;
          .switch {
            left: 55%;
          }
        }
      }
      label {
        width: 30px;
        height: 15px;
        border-radius: 10px;
        background-color: gray;
        display: inline-block;
        margin-right: 10px;
        position: relative;
        transition: all 0.3s;
        cursor: pointer;
        span {
          background-color: white;
          height: 80%;
          width: 40%;
          position: absolute;
          border-radius: 50%;
          left: 5%;
          top: 9%;
          transition: all 0.3s; 
        }
      }
    }
      &.circle
      {
        height: 100px;
        width: 100px;
        z-index: 1;
        svg
        {
          width: 100%;
          z-index: 1;
        }
        
        .icon
      {
        position: absolute;
        top: 35%;
        left: 35%;
        width: 30%;
        height: 30%;
        display: block;
        z-index: 5;
        
        &::after
        {
          content: "";
          position: absolute;
          top: 0;
          left: 0;
          width: 100%;
          height: 100%;
          filter: sepia(1);
        }
        
        &.xoffset, &.yoffset
        {
        &::after
        {
      background-image: url('');
        }
        }
        
        &.xoffset::after
        {
          transform: rotate(90deg);
        }
        
          &.blur::after
        {
          background-image: url("");
        }
        &.spread::after
        {
          background-image: url("");
        }
      }
      
      }
      &.color {
        height: auto;
        position: relative;
        flex-grow: 1;
        width: 100%;
        
        >input {
          width: 100%;
          padding: 6px;
          border-radius: 5px;
          border: 1px solid rgba(0, 0, 0, 0.3);
          outline: 0;
        }
        >span {
          position: absolute;
          right: 1px;
          top: 1px;
          height: calc(100% - 2px);
          width: 15%;
          background: gray;
          border-radius: 0 5px 5px 0;
          pointer-events: none;
        }
        .vc-chrome {
          position: absolute;
          top: 35px;
          right: 0;
          z-index: 9;
          width: 100%;
        }
      }
        &.delete {
          margin-top: 30px;
        
          height: auto;
        width: 100%;
           padding: 5px 10px;
              font-size: 0.7em;
              color: #e74c3c;
              border: 1px solid darken(#e74c3c, 5%);
              border-radius: 5px;
              cursor: pointer;
              transition: all 0.2s;
              text-align: center;
              &:hover {
                background-color: #e74c3c;
                color: white;
              }
          }
    }
  }
}

.show-effects {
  position: fixed;
  left: 70%;
  z-index: 1;
  top: 50%;
  transform: translate(-50%, -50%);
  
  width: 30vw;
  height: 30vh;
  &.transform-3d {
    transform-style: preserve-3d;
  }
  .transform-box {
    width: 100%;
    left: 0%;
    top: 0%;
    height: 100%;
    background-color: #2980b9;
    position: absolute;
    color: white;
    display: flex;
    justify-content: center;
    align-items: center;
    font-weight: bold;
    font-size: 2em;
    transition: all 0.2s;
  }
}

#css {
  position: fixed;
  bottom: 1vh;
  width: calc(90vw - 235px);
  background-color: rgba(255, 255, 255, 0.8);
  border-radius: 1vw / 2vh;
  right: 5vw;
  display: flex;
  flex-direction: column;
  box-sizing: border-box;
  font-size: 1em;
  color: black;
  z-index: 1;
  opacity: 0.3;
  transition: all 0.2s;
  padding: 10px;
  &:hover {
    opacity: 1;
  }
  span {
    padding: 5px;
  }
}

// Transition
.fade-enter-active,
.fade-leave-active {
  transition: opacity 0.2s;
}

.fade-enter,
.fade-leave-to {
  opacity: 0;
}


/*
Split Vertical icon designed by Bhima on the @NounProject. https://thenounproject.com/term/split-vertical/1268414

spread icon designed by projecthayat on the @NounProject. https://thenounproject.com/term/spread/1863148

Blur icon designed by abderraouf omara on the @NounProject. https://thenounproject.com/term/blur/705680
*/
View Compiled
// https://stackoverflow.com/a/42389266/529024
Vue.directive('click-outside', {
  bind: function(el, binding, vnode) {
    this.event = function(event) {
      if (!(el == event.target || el.contains(event.target))) {
        vnode.context[binding.expression](event);
      }
    };
    document.body.addEventListener('click', this.event)
  },
  unbind: function(el) {
    document.body.removeEventListener('click', this.event)
  },
});

var chrome = VueColor.Chrome;
Vue.component("box-shadow-control", {
  template: "#box-shadow-control",
  props: ["data"],
  components: {
    'chrome-picker': chrome
  },
  data() { 
    return {
      displayPicker: false
    }
  },
  methods: {
    handleScroll: function(item, e) {
      var element = e.target;
      var max = (item == 'blur') ? 100 : 50;
      var min = (item == 'blur') ? 0 : -50;
      
      var current = this.data[item];

      // Scrolling up
      if (e.deltaY < 0) {
        if (current < max)
          this.data[item] = this.data[item] + 1;
      }
      // Scrolling down
      if (e.deltaY > 0) {
        if (current > min)
          this.data[item] = this.data[item] - 1;

      }
    },
    updatePicker: function(color) {
      if (color.rgba.a == 1) {
        this.data.color = color.hex;
      } else {
        this.data.color = "rgba(" + color.rgba.r + ", " + color.rgba.g + ", " + color.rgba.b + ", " + color.rgba.a + ")";
      }
    },
    hidePicker: function() {
      this.displayPicker = false;
    }
  }
});

new Vue({
  el: "#app",

  data() {
    return {
      style: {
        "box-shadow": ""
      },
      showInfo: false,
      currentShadow: 0,
      boxShadows: [{
        inset: false,
        xoffset: 10,
        yoffset: 10,
        blur: 0,
        spread: 0,
        color: "black"
      }],
      boxShadowTemplate: {
        inset: false,
        xoffset: 10,
        yoffset: 10,
        blur: 0,
        spread: 0,
        color: "black"
      }
    };
  },
  computed: {
    getData: function() {
      return this.boxShadows[this.currentShadow];
    }
  },
  methods: {
    resetControls: function(e) { 
      Object.assign(this.$data, this.$options.data.apply(this));
    },
    addShadow: function() {
      // this code >> JSON.parse(JSON.stringify(transform)) << helps copy object by value not by reference
      var newShadow = JSON.parse(JSON.stringify(this.boxShadowTemplate));

      //newShadow.id = this.uid();
      this.boxShadows.push(newShadow);
      this.setShadow(this.boxShadows.length - 1);
    },
    setShadow: function(shadowIndex) {
      this.currentShadow = shadowIndex;
    },
    deleteShadow: function() {
// First delete element
this.boxShadows.splice(this.currentShadow, 1);

  // Second set other elements (if any)
  var length = this.boxShadows.length;
  if(length > 0)
    this.setShadow(length - 1);
  
    },
    getStyle: function() {
        var final = [];
        // Loop through set box shadows and "flatten"
        this.boxShadows.forEach(shadow => {
          var shadowFinal = [];

          // Loop through each box shadow set and format accordingly
          for (const item in shadow) {

            // if inset is set add the word inset
            if (item == 'inset') {
              if (shadow[item] == true)
                shadowFinal.push("inset");
            } else if (item == 'color') {
              // Color does not require "px" affix so add as is
              shadowFinal.push(shadow[item]);
            } else {
              // For all other properites add "px"
              shadowFinal.push(shadow[item] + "px");
            }
          }

          // Join flatten properties
          final.push(
            shadowFinal.join(" ")
          );
        });

        // Join everything together and assign
        this.style["box-shadow"] = final.join(", ");
      }
  },
  watch: {
    boxShadows: {
      handler: function(){
        this.getStyle();
      },
      deep: true
    }
  },
   mounted() {
   this.getStyle();
  }
});

External CSS

This Pen doesn't use any external CSS resources.

External JavaScript

  1. https://cdnjs.cloudflare.com/ajax/libs/vue/2.5.17/vue.min.js
  2. https://cdnjs.cloudflare.com/ajax/libs/vue-color/2.7.0/vue-color.min.js
  3. https://unpkg.com/vue-circle-slider