├── .eslintrc.json ├── .gitignore ├── README.md ├── assets ├── css │ └── index.css └── js │ └── index.js ├── gulp.config.js ├── gulpfile.js ├── index.html ├── license.txt ├── package-lock.json ├── package.json ├── sass ├── _reset.scss └── index.scss └── scripts └── index.js /.eslintrc.json: -------------------------------------------------------------------------------- 1 | { 2 | "env": { "browser": true }, 3 | "extends": "standard", 4 | "globals": { 5 | "Vue": false, 6 | "Sine": false, 7 | "TweenLite": false 8 | }, 9 | "rules": { 10 | "space-before-function-paren": 0, 11 | "object-property-newline": 0, 12 | "arrow-parens": 0, 13 | "new-cap": 0, 14 | "no-eval": 0, 15 | "no-new": 0, 16 | "semi": 0 17 | } 18 | } 19 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules/ 2 | .DS_Store 3 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # SVG Animations with VueJS and TweenLite 2 | 3 |  4 | 5 | This repo contains the code for a demo from our blog post "Creating Vue.js Transitions & Animation: Live Examples". 6 | 7 | "An animation is a more complex, usually multi-step and sometimes continuous change in appearance. Animations will often call on JavaScript to pick up where CSS' lack of logic drops off. It could be to add a class and trigger the animation or to make complex state transitions that reflect onto the DOM. 8 | 9 | The possibilities and approaches are virtually endless, so I've chosen one of my favourite techniques to showcase how you could animate your data with Vue.js. 10 | 11 | Essentially we're going to use GSAP's TweenLite library to apply easing functions to our state's changes and let Vue's lightning fast reactivity reflect this on the DOM. Vue is just as comfortable working with inline SVG as it is with HTML." 12 | 13 | >[Read full tutorial](https://snipcart.com/blog/vuejs-transitions-animations) 14 | 15 | >[Try it on CodePen](https://codepen.io/udyux/pen/aLVXOg) 16 | 17 | Enjoy folks! 18 | -------------------------------------------------------------------------------- /assets/css/index.css: -------------------------------------------------------------------------------- 1 | @import "https://fonts.googleapis.com/css?family=Lato:300,900|Libre+Franklin:100,200,300,600,900|Prompt:100,200,300,600,900";html{-webkit-box-sizing:border-box;box-sizing:border-box;font-family:Libre Franklin,sans-serif;font-size:100%;height:100%}html.lato{font-family:Lato,sans-serif}html.prompt{font-family:Prompt,sans-serif}body{-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;height:100%;line-height:1;position:relative}*{-webkit-margin-after:0;-webkit-margin-before:0;-webkit-margin-end:0;-webkit-margin-start:0;-webkit-padding-after:0;-webkit-padding-before:0;-webkit-padding-end:0;-webkit-padding-start:0;background:none;border:none;border-radius:0;margin:0;outline:none;padding:0}*,:after,:before{-webkit-box-sizing:inherit;box-sizing:inherit}:active,:hover{outline:0}address,b,button,del,em,h1,h2,h3,h4,h5,h6,i,input,ins,pre,select,strong,td,textarea,th{font-family:inherit;font-size:inherit;font-style:normal;font-weight:400;letter-spacing:inherit;text-transform:inherit}input,textarea{-moz-appearance:none;-webkit-appearance:none;appearance:none;background-clip:padding-box}a,button,del,ins{color:inherit;text-decoration:none}menu,ol,ul{list-style:none}table{border-collapse:separate;border-spacing:0;width:100%}pre,textarea{max-width:100%;overflow:auto}img{display:block;height:auto;width:100%}svg:not(:root){overflow:hidden}form{width:100%}button{cursor:pointer;overflow:visible}textarea{resize:none}input[type=range]{-webkit-appearance:none;background-color:transparent}input[type=range]::-ms-track{background:transparent;border-color:transparent;color:transparent;cursor:pointer}input[type=range]::-webkit-slider-thumb{-webkit-appearance:none}::moz-focus-inner{border:none;padding:0}.chart{-ms-flex-align:center;-ms-flex-direction:column;-ms-flex-pack:center;-webkit-box-align:center;-webkit-box-direction:normal;-webkit-box-orient:vertical;-webkit-box-pack:center;align-items:center;background:-webkit-gradient(linear,left top,left bottom,from(#c0f195),to(#4c9b95));background:linear-gradient(#c0f195,#4c9b95);color:#36424b;display:-webkit-box;display:-ms-flexbox;display:flex;flex-direction:column;height:100vh;justify-content:center}.chart__content{max-width:775px;padding-right:1.5rem;position:relative;width:95%}.chart__content:after,.chart__content:before{bottom:0;content:"";left:0;position:absolute}.chart__content:before{background:-webkit-gradient(linear,left top,right top,from(rgba(0,0,0,.3)),color-stop(110%,transparent));background:linear-gradient(90deg,rgba(0,0,0,.3),transparent 110%);height:1px;width:100%}.chart__content:after{background:-webkit-gradient(linear,left bottom,left top,from(rgba(0,0,0,.3)),color-stop(110%,transparent));background:linear-gradient(0deg,rgba(0,0,0,.3),transparent 110%);height:100%;width:1px}.chart__content svg{display:block}.chart__caption{-webkit-transform:translate(-50%,100%);left:50%;position:absolute;text-align:center;top:100%;transform:translate(-50%,100%);width:100%}.chart__title{font-size:1.3rem}.chart__subtitle{font-size:1.15rem;margin-top:.5em}.modal{-ms-flex-align:center;-ms-flex-pack:center;-webkit-box-align:center;-webkit-box-pack:center;align-items:center;background-color:hsla(0,0%,100%,.6);display:-webkit-box;display:-ms-flexbox;display:flex;height:100vh;justify-content:center;left:0;position:fixed;top:0;width:100vw;z-index:1}.modal-enter-active,.modal-leave-active{-webkit-transition:opacity .35s;transition:opacity .35s}.modal-enter,.modal-leave-to{opacity:0}.modal-enter-to,.modal-leave{opacity:1}.modal__open{font-size:1.1rem;font-weight:200;position:fixed;right:2.5rem;top:2em}.modal__content{-ms-flex-align:center;-ms-flex-direction:column;-ms-flex-pack:center;-webkit-box-align:center;-webkit-box-direction:normal;-webkit-box-orient:vertical;-webkit-box-pack:center;-webkit-box-shadow:0 .5rem 1.75rem -.25rem rgba(61,83,88,.4);align-items:center;background-color:#fff;border:1px solid #c5d0d1;border-radius:12px;box-shadow:0 .5rem 1.75rem -.25rem rgba(61,83,88,.4);display:-webkit-box;display:-ms-flexbox;display:flex;flex-direction:column;justify-content:center;max-width:500px;min-height:250px;padding:1.5rem 1rem;position:relative;text-align:center;width:90%}.modal__title{font-size:1.5rem}.modal__link{font-size:1.2rem;font-weight:300;margin-top:1.5rem;position:relative;z-index:0}.modal__link:after{-webkit-transition:background-color 225ms ease-out;background-color:currentColor;bottom:0;content:"";height:1px;left:0;position:absolute;transition:background-color 225ms ease-out;width:100%;z-index:-1}.modal__link:hover:after{background-color:#4c9b95}.modal__close{-webkit-transition:opacity .15s ease-out;font-size:2.5rem;line-height:1;opacity:.5;position:absolute;right:1rem;top:.25rem;transition:opacity .15s ease-out}.modal__close:hover{opacity:1} 2 | /*# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["_reset.scss","index.scss"],"names":[],"mappings":"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","file":"index.css","sourcesContent":["@import 'https://fonts.googleapis.com/css?family=Lato:300,900|Libre+Franklin:100,200,300,600,900|Prompt:100,200,300,600,900';\n\nhtml {\n  box-sizing: border-box;\n  font-size: 100%;\n  height: 100%;\n  font-family: 'Libre Franklin', sans-serif;\n  \n  &.lato { font-family: 'Lato', sans-serif }\n  &.prompt { font-family: 'Prompt', sans-serif }\n}\n\nbody {\n  position: relative;\n  height: 100%;\n  line-height: 1;\n  -webkit-font-smoothing: antialiased;\n  -moz-osx-font-smoothing: grayscale;\n}\n\n/* reset */\n* {\n  box-sizing: inherit;\n  margin: 0;\n  padding: 0;\n  border-radius: 0;\n  border: none;\n  outline: none;\n  background: none;\n  -webkit-margin-before: 0;\n  -webkit-margin-after: 0;\n  -webkit-margin-start: 0;\n  -webkit-margin-end: 0;\n  -webkit-padding-before: 0;\n  -webkit-padding-start: 0;\n  -webkit-padding-end: 0;\n  -webkit-padding-after: 0;\n}\n\n*::before,\n*::after {\n  box-sizing: inherit;\n}\n\n*:active,\n*:hover {\n  outline: 0;\n}\n\nb, i, em, strong,\nh1, h2, h3, h4, h5, h6,\nth, td, pre, ins, del, address,\ninput, select, button, textarea {\n  text-transform: inherit;\n  font-family: inherit;\n  font-size: inherit;\n  font-weight: normal;\n  font-style: normal;\n  letter-spacing: inherit;\n}\n\ntextarea,\ninput {\n  appearance: none;\n  background-clip: padding-box;\n}\n\na, ins, del, button {\n  color: inherit;\n  text-decoration: none;\n}\n\nul, ol,\nmenu {\n  list-style: none;\n}\n\ntable {\n  width: 100%;\n  border-collapse: separate;\n  border-spacing: 0;\n}\n\npre,\ntextarea {\n  overflow: auto;\n  max-width: 100%;\n}\n\nimg {\n  width: 100%;\n  height: auto;\n  display: block;\n}\n\nsvg:not(:root) {\n  overflow: hidden;\n}\n\nform {\n  width: 100%;\n}\n\nbutton {\n  cursor: pointer;\n  overflow: visible;\n}\n\ntextarea {\n  resize: none;\n}\n\ninput[type='range'] {\n  -webkit-appearance: none;\n  background-color: transparent;\n  \n  &::-ms-track {\n    cursor: pointer;\n    background: transparent; \n    border-color: transparent;\n    color: transparent;\n  }\n\n  &::-webkit-slider-thumb {\n    -webkit-appearance: none;\n  }\n}\n\n::moz-focus-inner {\n  padding: 0;\n  border: none;\n}\n","@import 'reset';\n\n.chart {\n  height: 100vh;\n  display: flex;\n  color: #36424b;\n  align-items: center;\n  flex-direction: column;\n  justify-content: center;\n  background: linear-gradient(rgb(192, 241, 149), rgb(76, 155, 149));\n\n  &__content {\n    width: 95%;\n    max-width: 775px;\n    padding-right: 1.5rem;\n    position: relative;\n\n    &::before,\n    &::after {\n      content: '';\n      position: absolute;\n      bottom: 0;\n      left: 0;\n    }\n\n    &::before {\n      width: 100%;\n      height: 1px;\n      background: linear-gradient(to right, rgba(black, 0.3), rgba(black, 0) 110%);\n    }\n\n    &::after {\n      width: 1px;\n      height: 100%;\n      background: linear-gradient(to top, rgba(black, 0.3), rgba(black, 0) 110%);\n    }\n\n    svg {\n      display: block;\n    }\n  }\n\n  &__caption {\n    left: 50%;\n    top: 100%;\n    width: 100%;\n    position: absolute;\n    text-align: center;\n    transform: translate(-50%, 100%);\n  }\n\n  &__title {\n    font-size: 1.3rem;\n  }\n\n  &__subtitle {\n    font-size: 1.15rem;\n    margin-top: 0.5em;\n  }\n}\n\n.modal {\n  width: 100vw;\n  height: 100vh;\n  position: fixed;\n  top: 0;\n  left: 0;\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  background-color: rgba(white, 0.6);\n  z-index: 1;\n\n  &-enter-active,\n  &-leave-active { transition: opacity 350ms }\n\n  &-enter,\n  &-leave-to { opacity: 0 }\n\n  &-leave,\n  &-enter-to { opacity: 1 }\n\n  &__open {\n    position: fixed;\n    top: 2em;\n    right: 2.5rem;\n    font-weight: 200;\n    font-size: 1.1rem;\n  }\n\n  &__content {\n    position: relative;\n    display: flex;\n    flex-direction: column;\n    justify-content: center;\n    align-items: center;\n    width: 90%;\n    max-width: 500px;\n    min-height: 250px;\n    padding: 1.5rem 1rem;\n    background-color: white;\n    border: 1px solid #c5d0d1;\n    border-radius: 12px;\n    text-align: center;\n    box-shadow: 0 0.5rem 1.75rem -0.25rem rgba(#3d5358, 0.4);\n  }\n\n  &__title {\n    font-size: 1.5rem;\n  }\n\n  &__link {\n    margin-top: 1.5rem;\n    position: relative;\n    font-size: 1.2rem;\n    font-weight: 300;\n    z-index: 0;\n\n    &::after {\n      content: '';\n      width: 100%;\n      height: 1px;\n      position: absolute;\n      bottom: 0;\n      left: 0;\n      background-color: currentColor;\n      z-index: -1;\n      transition: background-color 225ms ease-out;\n    }\n\n    &:hover::after {\n      background-color: rgb(76, 155, 149);\n    }\n  }\n\n  &__close {\n    position: absolute;\n    top: 0.25rem;\n    right: 1rem;\n    font-size: 2.5rem;\n    opacity: 0.5;\n    line-height: 1;\n    transition: opacity 150ms ease-out;\n\n    &:hover {\n      opacity: 1;\n    }\n  }\n}\n"]} */ 3 | -------------------------------------------------------------------------------- /assets/js/index.js: -------------------------------------------------------------------------------- 1 | !function(){"use strict";var t=function(t,n,e){return n in t?Object.defineProperty(t,n,{value:e,enumerable:!0,configurable:!0,writable:!0}):t[n]=e,t};new Vue({el:"#app",data:function(){return{modal:!1,points:{a:-1,b:-1,c:-1,d:-1,e:-1}}},computed:{path:function(){var t=this;return Object.keys(this.points).filter(function(t){return~"abcde".indexOf(t)}).map(function(n,e){return[100*e,100-t.points[n]]})}},methods:{setPoint:function(t){var n=this.random(3,5),e=this.random(0,100);this.animatePoint({key:t,duration:n,destination:e})},animatePoint:function(n){var e,i=n.key,o=n.duration,a=n.destination;TweenLite.to(this.points,o,(e={},t(e,i,a),t(e,"ease",Sine.easeInOut),t(e,"onComplete",this.setPoint),t(e,"onCompleteParams",[i]),e))},random:function(t,n){return(Math.random()*(n-t)+t).toFixed(2)}},mounted:function(){Object.keys(this.points).forEach(this.setPoint)}})}(); 2 | //# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbInNjcmlwdHMvaW5kZXguanMiXSwibmFtZXMiOlsiVnVlIiwiYSIsImIiLCJjIiwiZCIsImUiLCJPYmplY3QiLCJrZXlzIiwidGhpcyIsInBvaW50cyIsImZpbHRlciIsImtleSIsImluZGV4T2YiLCJtYXAiLCJpIiwiX3RoaXMiLCJkdXJhdGlvbiIsInJhbmRvbSIsImRlc3RpbmF0aW9uIiwiYW5pbWF0ZVBvaW50IiwiX3JlZiIsInRvIiwiX1R3ZWVuTGl0ZSR0byIsImRlZmluZVByb3BlcnR5IiwiU2luZSIsImVhc2VJbk91dCIsInNldFBvaW50IiwibWluIiwibWF4IiwiTWF0aCIsInRvRml4ZWQiLCJmb3JFYWNoIl0sIm1hcHBpbmdzIjoic0pBQUEsSUFBSUEsUUFDRSxZQURFLHlCQUlLLFVBQ0dDLEdBQUksRUFBR0MsR0FBSSxFQUFHQyxHQUFJLEVBQUdDLEdBQUksRUFBR0MsR0FBSSxvQkFJcEMsNkJBRUNDLE9BQU9DLEtBQUtDLEtBQUtDLFFBQ3JCQyxPQUFPLFNBQUFDLFVBQVEsUUFBUUMsUUFBUUQsS0FDL0JFLElBQUksU0FBQ0YsRUFBS0csVUFBVyxJQUFKQSxFQUFTLElBQU1DLEVBQUtOLE9BQU9FLDBCQUkxQyxTQWpCSEEsT0FDRkssRUFERVIsS0FBQVMsT0FBQSxFQUFBLEdBQUFDLEVBRUNWLEtBQUFTLE9BQUEsRUFBQSxVQUNFRSxjQUFBUixJQUFBQSxFQUFBSyxTQUFBQSxFQUFBRSxZQUFBQSxrQkFjQSxTQUFBRSxTQWRQVCxFQUFBUyxFQUFBVCxJQUFBSyxFQUFBSSxFQUFBSixTQUFBRSxFQUFBRSxFQUFBRixzQkFISUcsR0FBQWIsS0FBQUMsT0FBQU8sR0FBQU0sS0FBQUMsRUFBQUQsRUEwQkNYLEVBQU1PLEdBMUJQSyxFQUFBRCxFQUFBLE9BMkJNRSxLQUFLQyxXQTNCWEYsRUFBQUQsRUFBQSxhQVNJZCxLQUFBa0IsVUFUSkgsRUFBQUQsRUFBQSxvQkFVR1gsSUFWSFcsWUFpQkcsU0FnQkFLLEVBckJhQyxVQUNYQyxLQUFBWixVQUFBVyxFQUFBRCxHQUFBQSxHQUFBRyxRQUFBLGFBYkwsa0JBdUNHdkIsS0FBS0MsS0FBS0MsUUFBUXNCLFFBQVF2QixLQUFLa0IiLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VzQ29udGVudCI6WyJuZXcgVnVlKHtcbiAgZWw6ICcjYXBwJyxcbiAgZGF0YSgpIHtcbiAgICByZXR1cm4ge1xuICAgICAgbW9kYWw6IGZhbHNlLFxuICAgICAgcG9pbnRzOiB7IGE6IC0xLCBiOiAtMSwgYzogLTEsIGQ6IC0xLCBlOiAtMSB9XG4gICAgfVxuICB9LFxuXG4gIGNvbXB1dGVkOiB7XG4gICAgcGF0aCgpIHtcbiAgICAgIHJldHVybiBPYmplY3Qua2V5cyh0aGlzLnBvaW50cylcbiAgICAgICAgLmZpbHRlcihrZXkgPT4gfidhYmNkZScuaW5kZXhPZihrZXkpKVxuICAgICAgICAubWFwKChrZXksIGkpID0+IFtpICogMTAwLCAxMDAgLSB0aGlzLnBvaW50c1trZXldXSlcbiAgICB9XG4gIH0sXG5cbiAgbWV0aG9kczoge1xuICAgIHNldFBvaW50KGtleSkge1xuICAgICAgbGV0IGR1cmF0aW9uID0gdGhpcy5yYW5kb20oMywgNSlcbiAgICAgIGxldCBkZXN0aW5hdGlvbiA9IHRoaXMucmFuZG9tKDAsIDEwMClcbiAgICAgIHRoaXMuYW5pbWF0ZVBvaW50KHsga2V5LCBkdXJhdGlvbiwgZGVzdGluYXRpb24gfSlcbiAgICB9LFxuXG4gICAgYW5pbWF0ZVBvaW50KHsga2V5LCBkdXJhdGlvbiwgZGVzdGluYXRpb24gfSkge1xuICAgICAgVHdlZW5MaXRlLnRvKHRoaXMucG9pbnRzLCBkdXJhdGlvbiwge1xuICAgICAgICBba2V5XTogZGVzdGluYXRpb24sXG4gICAgICAgIGVhc2U6IFNpbmUuZWFzZUluT3V0LFxuICAgICAgICBvbkNvbXBsZXRlOiB0aGlzLnNldFBvaW50LFxuICAgICAgICBvbkNvbXBsZXRlUGFyYW1zOiBba2V5XVxuICAgICAgfSlcbiAgICB9LFxuXG4gICAgcmFuZG9tKG1pbiwgbWF4KSB7XG4gICAgICByZXR1cm4gKChNYXRoLnJhbmRvbSgpICogKG1heCAtIG1pbikpICsgbWluKS50b0ZpeGVkKDIpXG4gICAgfVxuICB9LFxuXG4gIG1vdW50ZWQoKSB7XG4gICAgT2JqZWN0LmtleXModGhpcy5wb2ludHMpLmZvckVhY2godGhpcy5zZXRQb2ludClcbiAgfVxufSlcbiJdfQ== 3 | -------------------------------------------------------------------------------- /gulp.config.js: -------------------------------------------------------------------------------- 1 | /* config ———————————————————————————————————————————————————————————————————*/ 2 | const config = { 3 | browsersync: { 4 | localhost: undefined, 5 | port: 8080, 6 | notify: true, 7 | sync: true 8 | }, 9 | 10 | sass: { 11 | src: ['./sass/index.scss'], 12 | dest: './assets/css', 13 | options: { indentedSyntax: false }, 14 | autoprefixer: { 15 | browsers: ['last 2 versions', 'not ie <= 9'] 16 | } 17 | }, 18 | 19 | js: { 20 | src: './scripts/*.js', 21 | dest: './assets/js', 22 | eslint: { fix: true }, 23 | babel: { 24 | presets: ['es2015-rollup'] 25 | } 26 | }, 27 | 28 | watch: { 29 | sass: 'sass/**/*.scss', 30 | js: 'scripts/**/*.js', 31 | markup: '**/*.html' 32 | } 33 | } 34 | 35 | /* helpers ——————————————————————————————————————————————————————————————————*/ 36 | const notifier = require('node-notifier') 37 | const symbols = require('log-symbols') 38 | const chalk = require('chalk') 39 | 40 | module.exports = { 41 | get config() { 42 | config.sass.dest = config.sass.dest.replace(/\/$/, '') 43 | config.js.dest = config.js.dest.replace(/\/$/, '') 44 | return config 45 | }, 46 | 47 | get bsConfig() { 48 | let conf = { 49 | port: config.browsersync.port, 50 | ui: { port: config.browsersync.port + 1 }, 51 | notify: config.browsersync.notify, 52 | proxy: config.browsersync.localhost, 53 | server: (!config.browsersync.localhost) ? './' : false 54 | } 55 | 56 | if (!config.browsersync.sync) conf.ghostMode = false 57 | return conf 58 | }, 59 | 60 | sassReporter(e) { 61 | let title = `${e.relativePath}:${e.line}` 62 | let message = e.messageOriginal.replace(/\s{4}/,'') 63 | let count = chalk.bold(chalk.red(`${symbols.error} 1 problem (1 error, 0 warnings)`)) 64 | console.log(chalk.underline(title), '\n ', message, `\n\n${count}`, '\n') 65 | notifier.notify({ title, message }) 66 | this.emit('end') 67 | } 68 | } 69 | -------------------------------------------------------------------------------- /gulpfile.js: -------------------------------------------------------------------------------- 1 | const gulp = require('gulp') 2 | const sass = require('gulp-sass') 3 | const cssnano = require('cssnano') 4 | const uglify = require('gulp-uglify') 5 | const eslint = require('gulp-eslint') 6 | const rename = require('gulp-rename') 7 | const postcss = require('gulp-postcss') 8 | const plumber = require('gulp-plumber') 9 | const babel = require('rollup-plugin-babel') 10 | const rollup = require('gulp-better-rollup') 11 | const autoprefixer = require('autoprefixer') 12 | const sourcemaps = require('gulp-sourcemaps') 13 | const browserSync = require('browser-sync').create() 14 | const cjsResolve = require('rollup-plugin-commonjs') 15 | const nodeResolve = require('rollup-plugin-node-resolve') 16 | 17 | const utils = require('./gulp.config') 18 | const config = utils.config 19 | 20 | // sass 21 | gulp.task('sass', () => { 22 | return gulp.src(config.sass.src) 23 | .pipe(plumber(utils.sassReporter)) 24 | .pipe(sourcemaps.init()) 25 | .pipe(sass(config.sass.options)) 26 | .pipe(postcss([ 27 | autoprefixer(config.sass.autoprefixer), 28 | cssnano() 29 | ])) 30 | .pipe(sourcemaps.write()) 31 | .pipe(rename({ dirname: '' })) 32 | .pipe(gulp.dest(config.sass.dest)) 33 | .pipe(browserSync.stream({ match: '**/*.css' })) 34 | }) 35 | 36 | // js 37 | gulp.task('js', () => { 38 | return gulp.src(config.js.src) 39 | .pipe(sourcemaps.init()) 40 | .pipe(eslint()) 41 | .pipe(eslint.format()) 42 | .pipe(rollup({ 43 | plugins: [ 44 | cjsResolve(), 45 | nodeResolve(), 46 | babel(config.js.babel) 47 | ] 48 | }, 'iife')) 49 | .pipe(uglify()) 50 | .pipe(sourcemaps.write()) 51 | .pipe(rename({ dirname: '' })) 52 | .pipe(gulp.dest(config.js.dest)) 53 | }) 54 | 55 | // builds 56 | gulp.task('build', ['sass', 'js']) 57 | 58 | // dev 59 | gulp.task('watch', ['build'], () => { 60 | gulp.watch(config.watch.sass, ['sass']) 61 | gulp.watch(config.watch.js, ['js']) 62 | }) 63 | 64 | // browser sync 65 | gulp.task('dev', ['watch'], () => { 66 | browserSync.init(utils.bsConfig) 67 | gulp.watch(`${config.js.dest}/*.js`).on('change', browserSync.reload) 68 | gulp.watch(config.watch.markup).on('change', browserSync.reload) 69 | }) 70 | -------------------------------------------------------------------------------- /index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 |
4 | 5 |