diff --git a/.eslintrc b/.eslintrc
new file mode 100755
index 0000000..a79e0c6
--- /dev/null
+++ b/.eslintrc
@@ -0,0 +1,22 @@
+{
+ "rules": {
+ "no-console": "off",
+ "indent": [ "error", 2 ],
+ "quotes": [ "error", "single" ],
+ "semi": ["error", "always"],
+ "linebreak-style": [ "error", "unix" ],
+ # "linebreak-style": [ "error", "windows"]
+ },
+ "env": {
+ "es6": true,
+ "node": true,
+ "mocha": true,
+ "jasmine": true
+ },
+ "ecmaFeatures": {
+ "modules": true,
+ "experimentalObjectRestSpread": true,
+ "impliedStrict": true
+ },
+ "extends": "eslint:recommended"
+}
diff --git a/.gitignore b/.gitignore
new file mode 100755
index 0000000..c31d550
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,125 @@
+
+# Created by https://www.gitignore.io/api/monodevelop,vim,node,windows,archlinuxpackages
+
+### ArchLinuxPackages ###
+*.tar
+*.tar.*
+*.jar
+*.exe
+*.msi
+*.zip
+*.tgz
+*.log
+*.log.*
+*.sig
+
+pkg/
+src/
+
+### MonoDevelop ###
+#User Specific
+*.userprefs
+*.usertasks
+
+#Mono Project Files
+*.pidb
+*.resources
+test-results/
+
+### Node ###
+# Logs
+logs
+npm-debug.log*
+yarn-debug.log*
+yarn-error.log*
+
+# Runtime data
+pids
+*.pid
+*.seed
+*.pid.lock
+
+# Directory for instrumented libs generated by jscoverage/JSCover
+lib-cov
+
+# Coverage directory used by tools like istanbul
+coverage
+
+# nyc test coverage
+.nyc_output
+
+# Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
+.grunt
+
+# Bower dependency directory (https://bower.io/)
+bower_components
+
+# node-waf configuration
+.lock-wscript
+
+# Compiled binary addons (http://nodejs.org/api/addons.html)
+build/Release
+
+# Dependency directories
+node_modules/
+jspm_packages/
+
+# Typescript v1 declaration files
+typings/
+
+# Optional npm cache directory
+.npm
+
+# Optional eslint cache
+.eslintcache
+
+# Optional REPL history
+.node_repl_history
+
+# Output of 'npm pack'
+
+# Yarn Integrity file
+.yarn-integrity
+
+# dotenv environment variables file
+.env
+
+
+### Vim ###
+# swap
+[._]*.s[a-v][a-z]
+[._]*.sw[a-p]
+[._]s[a-v][a-z]
+[._]sw[a-p]
+# session
+Session.vim
+# temporary
+.netrwhist
+*~
+# auto-generated tag files
+tags
+
+### Windows ###
+# Windows thumbnail cache files
+Thumbs.db
+ehthumbs.db
+ehthumbs_vista.db
+
+# Folder config file
+Desktop.ini
+
+# Recycle Bin used on file shares
+$RECYCLE.BIN/
+
+# Windows Installer files
+*.cab
+*.msm
+*.msp
+
+# Windows shortcuts
+*.lnk
+
+#temp hide of public folder
+public/
+
+# End of https://www.gitignore.io/api/monodevelop,vim,node,windows,archlinuxpackages
diff --git a/app/entry.js b/app/entry.js
new file mode 100644
index 0000000..5848c8a
--- /dev/null
+++ b/app/entry.js
@@ -0,0 +1,41 @@
+'use strict';
+
+require('./scss/reset.scss');
+require('./scss/main.scss');
+
+const angular = require('angular');
+const cowsay = require('cowsay-browser');
+
+const cowsayApp = angular.module('cowsayApp', []);
+
+cowsayApp.controller('CowsayController', ['$log', '$scope', CowsayController]);
+
+function CowsayController($log, $scope) {
+ $log.debug('CowsayController');
+
+ let cowsayCtrl = $scope.cowsayCtrl = {}
+ let previous = [];
+
+ cowsayCtrl.title = 'Welcome to the farm!';
+ cowsayCtrl.title2 = 'We eat people around here.'
+
+ cowsayCtrl.speak = function(input) {
+ $log.debug('cowsayCtrl.speak');
+ return cowsay.say({ text: input || 'mooo!!! bitches', f: 'tux'});
+ };
+
+ cowsayCtrl.speak2 = function(input) {
+ $log.debug('cowsayCtrl.speak2');
+ return cowsay.say({ text: input || 'My name is Squeek, the Dragon', f: 'dragon'});
+ };
+
+ cowsayCtrl.remember = function(input) {
+ $log.debug('cowsayCtrl.remember');
+
+ }
+
+ cowsayCtrl.logger = function(input) {
+ $log.debug('cowsayCtrl.logger()');
+ $log.log(input);
+ };
+};
diff --git a/app/index.html b/app/index.html
new file mode 100644
index 0000000..769c23b
--- /dev/null
+++ b/app/index.html
@@ -0,0 +1,32 @@
+
+
+
+
+ Welcome to Cowville or is it
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/app/scss/main.scss b/app/scss/main.scss
new file mode 100644
index 0000000..ff10920
--- /dev/null
+++ b/app/scss/main.scss
@@ -0,0 +1,83 @@
+
+// -- $variables ----
+
+$brand-primary: #364;
+$btn-primary: $brand-primary / 2;
+$btn-hover: #31b1c1;
+$black: #000;
+$white: #fff;
+$box-std: 30%;
+$gutter-std: 5%;
+$gutter-sm: $gutter-std / 2;
+
+// -- global styles ----
+
+body {
+ font-family: helvetic;
+ font-size: 2vw;
+ background: $brand-primary * 3;
+}
+
+h2 {
+ font-size: 3vw;
+ text-align: center;
+ margin-bottom: $gutter-std;
+}
+
+button {
+ float: right;
+ width: $box-std;
+ padding: 2% 0;
+ background: $btn-primary / 3;
+ color: $white;
+ border: none;
+ border-radius: 3px;
+ cursor: pointer;
+ transition: 350ms all;
+
+ &:hover {
+ background: $btn-hover;
+ }
+}
+
+input[type="text"] {
+ float: left;
+ width: $box-std + 10%;
+ margin-right: $gutter-std / 2;
+ padding: 2%;
+ border: none;
+ border-radius: 3px;
+}
+
+pre {
+ font-family: monospace;
+ margin: $gutter-std;
+}
+
+// -- app styles ----
+
+.container {
+ float: left;
+ width: $box-std;
+ margin: 5%;
+}
+
+.bigguy {
+ clear: left;
+ width: $box-std * 2;
+ margin: 5%;
+}
+
+.text-binder {
+ height: 15vw;
+ font-size: 3vw;
+ margin: $gutter-std;
+ padding: 1vw;
+ color: $white;
+ background: $brand-primary / 3;
+ border: dotted .25vw $brand-primary;
+
+ + input {
+ width: 100%;
+ }
+}
diff --git a/app/scss/reset.scss b/app/scss/reset.scss
new file mode 100644
index 0000000..ed11813
--- /dev/null
+++ b/app/scss/reset.scss
@@ -0,0 +1,48 @@
+/* http://meyerweb.com/eric/tools/css/reset/
+ v2.0 | 20110126
+ License: none (public domain)
+*/
+
+html, body, div, span, applet, object, iframe,
+h1, h2, h3, h4, h5, h6, p, blockquote, pre,
+a, abbr, acronym, address, big, cite, code,
+del, dfn, em, img, ins, kbd, q, s, samp,
+small, strike, strong, sub, sup, tt, var,
+b, u, i, center,
+dl, dt, dd, ol, ul, li,
+fieldset, form, label, legend,
+table, caption, tbody, tfoot, thead, tr, th, td,
+article, aside, canvas, details, embed,
+figure, figcaption, footer, header, hgroup,
+menu, nav, output, ruby, section, summary,
+time, mark, audio, video {
+ margin: 0;
+ padding: 0;
+ border: 0;
+ font-size: 100%;
+ font: inherit;
+ vertical-align: baseline;
+}
+/* HTML5 display-role reset for older browsers */
+article, aside, details, figcaption, figure,
+footer, header, hgroup, menu, nav, section {
+ display: block;
+}
+body {
+ line-height: 1;
+}
+ol, ul {
+ list-style: none;
+}
+blockquote, q {
+ quotes: none;
+}
+blockquote:before, blockquote:after,
+q:before, q:after {
+ content: '';
+ content: none;
+}
+table {
+ border-collapse: collapse;
+ border-spacing: 0;
+}
diff --git a/gulpfile.js b/gulpfile.js
new file mode 100755
index 0000000..24190d6
--- /dev/null
+++ b/gulpfile.js
@@ -0,0 +1,23 @@
+'use strict';
+
+const gulp = require('gulp');
+const eslint = require('gulp-eslint');
+const mocha = require('gulp-mocha');
+
+gulp.task('test', function () {
+ gulp.src('./test/*-test.js', { read: false})
+ .pipe(mocha({reporter: 'spec'}));
+});
+
+gulp.task('lint', function() {
+ return gulp.src(['**/*.js', '!node_modules'])
+ .pipe(eslint())
+ .pipe(eslint.format())
+ .pipe(eslint.failAfterError());
+});
+
+gulp.task('dev', function() {
+ gulp.watch(['**/*.js', '!node_modules'], ['lint', 'test']);
+});
+
+gulp.task('default', ['dev']);
diff --git a/package.json b/package.json
new file mode 100644
index 0000000..bf9cd31
--- /dev/null
+++ b/package.json
@@ -0,0 +1,37 @@
+{
+ "name": "20-intro_to_angular_and_webpack",
+ "version": "1.0.0",
+ "description": " Lab 20 - Intro to Angular and Webpack ======",
+ "main": "index.js",
+ "scripts": {
+ "test": "echo \"Error: no test specified\" && exit 1",
+ "build": "./node_modules/webpack/bin/webpack.js",
+ "watch": "./node_modules/webpack-dev-server/bin/webpack-dev-server.js --progress --colors --inline --hot",
+ "monitor": "./node_modules/webpack-dev-server/bin/webpack-dev-server.js --progress --colors",
+ "lint": "./eslint.format()"
+ },
+ "repository": {
+ "type": "git",
+ "url": "git+https://github.com/GLSea1979/20-intro_to_angular_and_webpack.git"
+ },
+ "keywords": [],
+ "author": "",
+ "license": "ISC",
+ "bugs": {
+ "url": "https://github.com/GLSea1979/20-intro_to_angular_and_webpack/issues"
+ },
+ "homepage": "https://github.com/GLSea1979/20-intro_to_angular_and_webpack#readme",
+ "dependencies": {
+ "angular": "^1.6.3",
+ "cowsay-browser": "^1.1.8",
+ "css-loader": "^0.27.3",
+ "html-webpack-plugin": "^2.28.0",
+ "node-sass": "^4.5.0",
+ "sass-loader": "^6.0.3",
+ "style-loader": "^0.14.1",
+ "webpack": "^1.14.0"
+ },
+ "devDependencies": {
+ "webpack-dev-server": "^1.16.2"
+ }
+}
diff --git a/webpack.config.js b/webpack.config.js
new file mode 100644
index 0000000..279dac8
--- /dev/null
+++ b/webpack.config.js
@@ -0,0 +1,24 @@
+'use strict';
+
+const HTMLPlugin = require('html-webpack-plugin');
+
+module.exports = {
+ entry: `${__dirname}/app/entry.js`,
+ output: {
+ filename: 'bundle.js',
+ path: 'build'
+ },
+ plugins: [
+ new HTMLPlugin({
+ template: `${__dirname}/app/index.html`
+ })
+ ],
+ module: {
+ loaders: [
+ {
+ test: /\.scss$/,
+ loader: 'style!css!sass!'
+ }
+ ]
+ }
+};