├── .bowerrc ├── .editorconfig ├── .gitignore ├── .jshintrc ├── .travis.yml ├── .yo-rc.json ├── LICENSE ├── README.md ├── bower.json ├── dist ├── vg-src.js └── vg-src.min.js ├── gulpfile.js ├── karma-dist-concatenated.conf.js ├── karma-dist-minified.conf.js ├── karma-src.conf.js ├── package.json ├── sample ├── index.html └── index.js ├── src ├── _end.js ├── _start.js ├── vgSrc.config.js ├── vgSrc.directive.js ├── vgSrc.ie.js └── vgSrc.js └── test └── unit └── vg-src └── vgSrcSpec.js /.bowerrc: -------------------------------------------------------------------------------- 1 | { 2 | "directory": "bower" 3 | } -------------------------------------------------------------------------------- /.editorconfig: -------------------------------------------------------------------------------- 1 | # http://editorconfig.org 2 | root = true 3 | 4 | [*] 5 | indent_style = space 6 | indent_size = 2 7 | charset = utf-8 8 | trim_trailing_whitespace = true 9 | insert_final_newline = true 10 | 11 | [*.md] 12 | trim_trailing_whitespace = false 13 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | ._* 2 | .~lock.* 3 | .buildpath 4 | .DS_Store 5 | .idea 6 | .project 7 | .settings 8 | *.log 9 | 10 | # Ignore node stuff 11 | node_modules/ 12 | npm-debug.log 13 | libpeerconnection.log 14 | 15 | # OS-specific 16 | .DS_Store 17 | 18 | # Bower components 19 | bower 20 | -------------------------------------------------------------------------------- /.jshintrc: -------------------------------------------------------------------------------- 1 | { 2 | "bitwise": true, 3 | "camelcase": true, 4 | "curly": true, 5 | "eqeqeq": true, 6 | "es3": false, 7 | "forin": true, 8 | "freeze": true, 9 | "immed": true, 10 | "indent": 2, 11 | "latedef": "nofunc", 12 | "newcap": true, 13 | "noarg": true, 14 | "noempty": true, 15 | "nonbsp": true, 16 | "nonew": true, 17 | "plusplus": false, 18 | "quotmark": "single", 19 | "undef": true, 20 | "unused": false, 21 | "strict": false, 22 | "maxparams": 10, 23 | "maxdepth": 5, 24 | "maxstatements": 40, 25 | "maxcomplexity": 8, 26 | "maxlen": 120, 27 | 28 | "asi": false, 29 | "boss": false, 30 | "debug": false, 31 | "eqnull": true, 32 | "esnext": false, 33 | "evil": false, 34 | "expr": false, 35 | "funcscope": false, 36 | "globalstrict": false, 37 | "iterator": false, 38 | "lastsemic": false, 39 | "laxbreak": false, 40 | "laxcomma": false, 41 | "loopfunc": true, 42 | "maxerr": false, 43 | "moz": false, 44 | "multistr": false, 45 | "notypeof": false, 46 | "proto": false, 47 | "scripturl": false, 48 | "shadow": false, 49 | "sub": true, 50 | "supernew": false, 51 | "validthis": false, 52 | "noyield": false, 53 | 54 | "browser": true, 55 | "node": true, 56 | 57 | "globals": { 58 | "angular": false, 59 | "$": false 60 | } 61 | } 62 | -------------------------------------------------------------------------------- /.travis.yml: -------------------------------------------------------------------------------- 1 | language: node_js 2 | node_js: ["0.10"] 3 | before_script: 4 | - npm install -g bower 5 | - bower install 6 | -------------------------------------------------------------------------------- /.yo-rc.json: -------------------------------------------------------------------------------- 1 | { 2 | "generator-angularjs-library": { 3 | "props": { 4 | "author": { 5 | "name": "van", 6 | "email": "1321907687@qq.com" 7 | }, 8 | "libraryName": { 9 | "original": "vgSrc", 10 | "camelized": "vgSrc", 11 | "dasherized": "vg-src", 12 | "slugified": "vgsrc", 13 | "parts": [ 14 | "vgsrc" 15 | ] 16 | }, 17 | "includeModuleDirectives": true, 18 | "includeModuleFilters": false, 19 | "includeModuleServices": false, 20 | "includeAngularModuleResource": false, 21 | "includeAngularModuleCookies": false, 22 | "includeAngularModuleSanitize": false, 23 | "librarySrcDirectory": "src/vg-src", 24 | "libraryUnitTestDirectory": "test/unit/vg-src", 25 | "libraryUnitE2eDirectory": "test/e2e/vg-src" 26 | } 27 | } 28 | } -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Copyright (c) 2015 van 2 | 3 | Permission is hereby granted, free of charge, to any person obtaining 4 | a copy of this software and associated documentation files (the 5 | "Software"), to deal in the Software without restriction, including 6 | without limitation the rights to use, copy, modify, merge, publish, 7 | distribute, sublicense, and/or sell copies of the Software, and to 8 | permit persons to whom the Software is furnished to do so, subject to 9 | the following conditions: 10 | 11 | The above copyright notice and this permission notice shall be 12 | included in all copies or substantial portions of the Software. 13 | 14 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 15 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 16 | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 17 | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE 18 | LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION 19 | OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION 20 | WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 21 | 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # vgSrc 2 | ## 简介 3 | 一个简单的 Angular 图片加载插件,插件根据图片资源的不同加载状态,显示不同图片。 4 | 5 | ## 使用 6 | 1. 推荐使用 bower 加载: 7 | ```bash 8 | bower install vgSrc --save 9 | ``` 10 | 并引入: 11 | ```html 12 | 13 | ``` 14 | 15 | 1. 也可下载源码,在页面引入: 16 | ```html 17 | } 18 | ``` 19 | 20 | ## example 21 | 1. 简单实例 22 | ```html 23 | 24 | ``` 25 | 26 | 1. 添加样式 27 | ```html 28 | 29 | ``` 30 | 31 | 1. 监听事件 32 | ```html 33 | 34 | ``` 35 | 36 | 更多实例,请查阅 **[sample/index.html](https://github.com/VanMess/vgSrc/blob/master/sample/index.html)** 文件 37 | 38 | ## API 39 | #### vgSrcConfigProvider 40 | 配置接口: 41 | ```javascript 42 | vgSrcConfigProvider.$set(config) 43 | ``` 44 | 45 | example: 46 | ```javascript 47 | ng.module('vgSrc.sample', ['vgSrc']).config([ 48 | 'vgSrcConfigProvider', 49 | function(vgSrcConfigProvider) { 50 | vgSrcConfigProvider.$set({ 51 | debug: false, 52 | error: 'http://ico.ooopic.com/iconset01/status-icons/gif/99589.gif', 53 | onBegin: function($e) { 54 | // console.log('start load:' + $e.src); 55 | }, 56 | onError: function($e) { 57 | // console.log('failure load:' + $e.src); 58 | }, 59 | onLoad: function($e) { 60 | // console.log('complete load:' + $e.src); 61 | } 62 | }); 63 | } 64 | ]); 65 | ``` 66 | 67 | #### vgSrc (directive) 68 | vgSrc 指令用法与 ngSrc 指令类型。指令支持 angular 表达式,如. 69 | ```html 70 | 71 | 72 | ``` 73 | 74 | ## 配置项 75 | #### 替换图片 76 | vgSrc 支持 loading、error、empty 状态下的图片替换: 77 | 78 | 1. vgSrc 指令求值结果为空时(null、undefined、空字符串),将显示 empty 值指定的图片 79 | 1. 开始加载时,将显示 loading 值指定的图片 80 | 1. 加载出错时,将显示 error 值指定的图片 81 | 1. 加载成功后,正常显示图片 82 | 83 | #### 事件 84 | vgSrc 支持 onBegin、onError、onLoad 事件,可通过 vgSrcConfigProvider 、 vgSrc 两种方式注册不同类型的事件处理器: 85 | 86 | 1. 通过 vgSrcConfigProvider 方式注册的监听器将做为默认的事件监听器,事件参数为:`$e{src:''}`,用法如: 87 | ```javascript 88 | onBegin:function($e){ 89 | console.log($e.src); 90 | } 91 | ``` 92 | 93 | 1. 通过 vgSrc 方式注册的监听器将覆盖默认的事件监听器,事件参数为:`$e{src:''}`,用法如: 94 | ```html 95 | 96 | ``` 97 | 98 | #### 样式class 99 | vgSrc 支持 loadingCls、loadedCls、errorCls、emptyCls 样式,可通过 vgSrcConfigProvider 、 vgSrc 两种方式注册 class 值: 100 | 101 | 1. 通过 vgSrcConfigProvider 方式注册的 class 将做为默认的 class ,用法如: 102 | ```javascript 103 | errorCls:'errorClass' 104 | ``` 105 | 106 | 1. 通过 vgSrc 方式注册的 class 将做为此image特定的 class,用法如: 107 | ```html 108 | 109 | ``` 110 | ** 注意,class 属性不支持angular表达式 —— 你只能传递简单的字符串 ** 111 | 112 | #### 配置项汇总 113 | ```javascript 114 | { 115 | // 启动调试模式 116 | debug: false, 117 | 118 | // 图片加载中的替换显示图片 119 | loading: '/loading.jpg', 120 | 121 | // 图片加载中的样式 class 122 | loadingCls: '', 123 | 124 | // 图片加载完成的样式 class 125 | loadedCls: '', 126 | 127 | // 图片加载失败的替换显示图片 128 | error: '/error.jpg', 129 | 130 | // 图片加载失败的样式 class 131 | errorCls: '', 132 | 133 | // 图片值为空时的替换显示图片 134 | empty: '', 135 | 136 | // 图片值为空时的样式 class 137 | emptyCls: '', 138 | 139 | // 资源开始加载事件 140 | 'onBegin': ng.noop, 141 | 142 | // 资源加载出错事件 143 | 'onError': ng.noop, 144 | 145 | // 资源加载完毕事件 146 | 'onLoad': ng.noop 147 | } 148 | ``` 149 | 150 | ## 兼容性 151 | 目前兼容主流浏览器及ie8 152 | -------------------------------------------------------------------------------- /bower.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "vgSrc", 3 | "version": "0.1.0", 4 | "authors": [ 5 | { 6 | "name": "van", 7 | "email": "1321907687@qq.com" 8 | } 9 | ], 10 | "main": ["dist/vg-src.js"], 11 | "ignore": [ 12 | "src", 13 | "test", 14 | "gulpfile.js", 15 | "karma-*.conf.js", 16 | "**/.*", 17 | "sample" 18 | ], 19 | "dependencies": {}, 20 | "devDependencies": { 21 | "angular": ">=1.2.0" 22 | } 23 | } 24 | -------------------------------------------------------------------------------- /dist/vg-src.js: -------------------------------------------------------------------------------- 1 | /* 2 | * angular-image-lazyload - v0.0.1 - 2015-10-14 3 | * https://github.com/VanMess/angular-image-lazyload 4 | * Copyright (c) 2014 Van (http://vanmess.github.io/) 5 | */ 6 | !(function(factory) { 7 | if (typeof define === 'function' && define.amd) { 8 | // AMD 9 | define(['angular'], factory); 10 | } else { 11 | // Global Variables 12 | factory(window.angular); 13 | } 14 | })(function(ng) { 15 | 'use strict'; 16 | 17 | var DONT_ENUM = "propertyIsEnumerable,isPrototypeOf,hasOwnProperty,toLocaleString,toString,valueOf,constructor".split(","), 18 | hasOwn = ({}).hasOwnProperty; 19 | for (var i in { 20 | toString: 1 21 | }) { 22 | DONT_ENUM = false; 23 | } 24 | Object.keys = Object.keys || (function(obj) { //ecma262v5 15.2.3.14 25 | return function(obj) { 26 | var result = [], 27 | key; 28 | for (key in obj) { 29 | if (hasOwn.call(obj, key)) { 30 | result.push(key); 31 | } 32 | } 33 | if (DONT_ENUM && obj) { 34 | for (var i = 0; i < DONT_ENUM.length; i++) { 35 | key = DONT_ENUM[i++]; 36 | if (hasOwn.call(obj, key)) { 37 | result.push(key); 38 | } 39 | } 40 | } 41 | return result; 42 | }; 43 | })(); 44 | 45 | if (!Function.prototype.bind) { 46 | Function.prototype.bind = function(oThis) { 47 | if (typeof this !== "function") { 48 | // closest thing possible to the ECMAScript 5 internal IsCallable function 49 | throw new TypeError("Function.prototype.bind - what is trying to be bound is not callable"); 50 | } 51 | var aArgs = Array.prototype.slice.call(arguments, 1), 52 | fToBind = this, 53 | fNOP = function() {}, 54 | fBound = function() { 55 | return fToBind.apply(this instanceof fNOP && oThis ? this : oThis, 56 | aArgs.concat(Array.prototype.slice.call(arguments))); 57 | }; 58 | fNOP.prototype = this.prototype; 59 | fBound.prototype = new fNOP(); 60 | return fBound; 61 | }; 62 | } 63 | 64 | // Create all modules and define dependencies to make sure they exist 65 | // and are loaded in the correct order to satisfy dependency injection 66 | // before all nested files are concatenated by Gulp 67 | 68 | angular.module('vgSrc', []); 69 | 70 | /* 71 | * The provider of this module, just for setting default config. 72 | */ 73 | var 74 | 75 | // default setting 76 | defaults = { 77 | debug: false, 78 | // place image content when it begin to load real image resource. 79 | loading: '', 80 | 81 | // element would applied this class when it is loading. 82 | loadingCls: '', 83 | 84 | // element would applied this class when it is load success. 85 | loadedCls: '', 86 | 87 | // place image content when image resource fail to load. 88 | error: '', 89 | 90 | // element would applied this class when it faild to load. 91 | errorCls: '', 92 | 93 | // place image content when element's "src" attribute is empty(null,undefine). 94 | empty: '', 95 | 96 | // element would applied this class when src attribute is empty(null,undefine). 97 | emptyCls: '', 98 | 99 | // event occurs when it start to load something. 100 | 'onBegin': ng.noop, 101 | 102 | // event occurs when it fail load something. 103 | 'onError': ng.noop, 104 | 105 | // event occurs when it finish to load something. 106 | 'onLoad': ng.noop 107 | }, 108 | 109 | provider = function() { 110 | var moduleConfig = ng.copy(defaults), 111 | result = { 112 | $set: _setConfig, 113 | $get: function() { 114 | return moduleConfig; 115 | } 116 | }; 117 | 118 | return result; 119 | 120 | function _setConfig(cfg) { 121 | 122 | if (!ng.isDefined(cfg)) return; 123 | 124 | ng.forEach(Object.keys(moduleConfig), function(key) { 125 | if (cfg.hasOwnProperty(key) && key !== 'debug') { 126 | moduleConfig[key] = cfg[key] + ''; 127 | } 128 | }); 129 | 130 | // init debug model 131 | if (cfg.hasOwnProperty('debug')) { 132 | moduleConfig.debug = !!cfg.debug; 133 | } 134 | 135 | // init events 136 | var events = ['onBegin', 'onError', 'onLoad'], 137 | tmp; 138 | ng.forEach(events, function(e) { 139 | tmp = ng.isFunction(cfg[e]) ? cfg[e] : ng.noop; 140 | moduleConfig[e] = (function(_old) { 141 | return function($scope, $e) { 142 | _old($e); 143 | }; 144 | })(tmp); 145 | }); 146 | } 147 | }; 148 | 149 | angular.module('vgSrc').provider('vgSrcConfig', [provider]); 150 | 151 | /* 152 | * An angular directive for image loading status present. 153 | */ 154 | var directive = function($parse, defaults) { 155 | var status = ['loading', 'empty', 'error'], 156 | statusCls = ['loadingCls', 'emptyCls', 'errorCls', 'loadedCls'], 157 | events = ['onBegin', 'onError', 'onLoad'], 158 | // angular's directive define object. 159 | defineObj = { 160 | priority: 99, 161 | restrict: 'A', 162 | name: 'vgSrc', 163 | compile: function(element, attrs) { 164 | var attrName = attrs.$normalize(defineObj.name), 165 | srcParser = $parse(attrs[attrName]); 166 | 167 | return function _link($scope, element, attrs) { 168 | var opt = ng.copy(defaults), 169 | $log = opt.debug ? console.log.bind(console) : ng.noop; 170 | // parse everything in status lists. 171 | ng.forEach(status, function(att) { 172 | if (ng.isString(attrs[att])) { 173 | // parse element's setting attribute use ng's '$parse' 174 | // so that users can define the configuration by ng's 'expression'. 175 | opt[att] = $parse(attrs[att])($scope); 176 | } 177 | }); 178 | 179 | // simply copy everything in statusCls lists. 180 | ng.forEach(statusCls, function(att) { 181 | if (ng.isString(attrs[att])) { 182 | opt[att] = attrs[att]; 183 | } 184 | }); 185 | 186 | // parse event handlers 187 | // so that we can occu 188 | ng.forEach(events, function(att) { 189 | if (ng.isString(attrs[att])) { 190 | opt[att] = $parse(attrs[att]); 191 | } 192 | }); 193 | 194 | // watching vgSrc attribute 195 | // so that we could dynamicly fresh element's image when each time user change the value 196 | $scope.$watch(function() { 197 | return srcParser($scope); 198 | }, function _bindImg(newVal, oldVal) { 199 | var $e = { 200 | src: newVal 201 | }; 202 | if (ng.isString(newVal) && newVal.length > 0) { 203 | attrs.$set('src', opt.loading); 204 | _refreshCls(opt['loadingCls']); 205 | opt['onBegin'].call($scope, $scope, $e); 206 | $log('start loading resource:' + $e.src); 207 | 208 | _lazyLoad(newVal, function() { 209 | attrs.$set('src', newVal); 210 | _refreshCls(opt['loadedCls']); 211 | opt['onLoad'].call($scope, $scope, $e); 212 | $log('success load resource:' + $e.src); 213 | }, function() { 214 | attrs.$set('src', opt.error); 215 | _refreshCls(opt['errorCls']); 216 | opt['onError'].call($scope, $scope, $e); 217 | $log('failure load resource:' + $e.src); 218 | }) 219 | } else { 220 | attrs.$set('src', opt.empty); 221 | _refreshCls(opt['emptyCls']); 222 | opt['onError'].call($scope, $scope, $e); 223 | $log('current img is empty'); 224 | } 225 | }); 226 | 227 | // clear element's status class 228 | // and add the new class 229 | function _refreshCls(cls) { 230 | ng.forEach(statusCls, function(cls) { 231 | element.removeClass(opt[cls]); 232 | }); 233 | element.addClass(cls); 234 | } 235 | }; 236 | 237 | } 238 | }; 239 | 240 | return defineObj; 241 | }; 242 | 243 | angular.module('vgSrc').directive('vgSrc', ['$parse', 'vgSrcConfig', directive]); 244 | 245 | /* 246 | * load function to excute a shadow load 247 | */ 248 | function _lazyLoad(src, loadCallback, errorCallback) { 249 | var $imgDom = ng.element(new Image()); 250 | loadCallback = ng.isFunction(loadCallback) ? loadCallback : ng.noop; 251 | errorCallback = ng.isFunction(errorCallback) ? errorCallback : ng.noop; 252 | 253 | $imgDom.bind('error', errorCallback.bind(this)).bind('load', loadCallback.bind(this)).attr('src', src); 254 | } 255 | 256 | }); 257 | -------------------------------------------------------------------------------- /dist/vg-src.min.js: -------------------------------------------------------------------------------- 1 | !function(A){"function"==typeof define&&define.amd?define(["angular"],A):A(window.angular)}(function(A){"use strict";function C(C,E,Q){var I=A.element(new Image);E=A.isFunction(E)?E:A.noop,Q=A.isFunction(Q)?Q:A.noop,I.bind("error",Q.bind(this)).bind("load",E.bind(this)).attr("src",C)}var E="propertyIsEnumerable,isPrototypeOf,hasOwnProperty,toLocaleString,toString,valueOf,constructor".split(","),Q={}.hasOwnProperty;for(var I in{toString:1})E=!1;Object.keys=Object.keys||function(){return function(A){var C,I=[];for(C in A)Q.call(A,C)&&I.push(C);if(E&&A)for(var g=0;g0?(B.$set("src",M.loading),D(M.loadingCls),M.onBegin.call(w,w,Q),R("start loading resource:"+Q.src),C(E,function(){B.$set("src",E),D(M.loadedCls),M.onLoad.call(w,w,Q),R("success load resource:"+Q.src)},function(){B.$set("src",M.error),D(M.errorCls),M.onError.call(w,w,Q),R("failure load resource:"+Q.src)})):(B.$set("src",M.empty),D(M.emptyCls),M.onError.call(w,w,Q),R("current img is empty"))})}}};return w};angular.module("vgSrc").directive("vgSrc",["$parse","vgSrcConfig",w])}); -------------------------------------------------------------------------------- /gulpfile.js: -------------------------------------------------------------------------------- 1 | var gulp = require('gulp'); 2 | var karma = require('karma').server; 3 | var concat = require('gulp-concat'); 4 | var uglify = require('gulp-uglify'); 5 | var rename = require('gulp-rename'); 6 | var path = require('path'); 7 | var plumber = require('gulp-plumber'); 8 | var runSequence = require('run-sequence'); 9 | var jshint = require('gulp-jshint'); 10 | 11 | /** 12 | * File patterns 13 | **/ 14 | 15 | // Root directory 16 | var rootDirectory = path.resolve('./'); 17 | 18 | // Source directory for build process 19 | var sourceDirectory = path.join(rootDirectory, './src'); 20 | 21 | var sourceFiles = [ 22 | 'vgSrc.ie.js', 23 | 'vgSrc.js', 24 | 'vgSrc.config.js', 25 | 'vgSrc.directive.js' 26 | ]; 27 | 28 | var lintFiles = [ 29 | 'gulpfile.js', 30 | // Karma configuration 31 | 'karma-*.conf.js' 32 | ].concat(sourceFiles); 33 | 34 | gulp.task('build', function() { 35 | var srcFiles = ['_start.js'].concat(sourceFiles); 36 | srcFiles.push('_end.js'); 37 | 38 | for (var i = 0; i < srcFiles.length; i++) { 39 | srcFiles[i] = path.join(sourceDirectory, srcFiles[i]); 40 | } 41 | gulp.src(srcFiles) 42 | .pipe(plumber()) 43 | .pipe(concat('vg-src.js')) 44 | .pipe(gulp.dest('./dist/')) 45 | .pipe(uglify()) 46 | .pipe(rename('vg-src.min.js')) 47 | .pipe(gulp.dest('./dist')); 48 | }); 49 | 50 | /** 51 | * Process 52 | */ 53 | gulp.task('process-all', function(done) { 54 | runSequence('jshint', 'test-src', 'build', done); 55 | }); 56 | 57 | /** 58 | * Watch task 59 | */ 60 | gulp.task('watch', function() { 61 | 62 | // Watch JavaScript files 63 | gulp.watch(sourceFiles, ['process-all']); 64 | }); 65 | 66 | /** 67 | * Validate source JavaScript 68 | */ 69 | gulp.task('jshint', function() { 70 | return gulp.src(lintFiles) 71 | .pipe(plumber()) 72 | .pipe(jshint()) 73 | .pipe(jshint.reporter('jshint-stylish')) 74 | .pipe(jshint.reporter('fail')); 75 | }); 76 | 77 | /** 78 | * Run test once and exit 79 | */ 80 | gulp.task('test-src', function(done) { 81 | karma.start({ 82 | configFile: __dirname + '/karma-src.conf.js', 83 | singleRun: true 84 | }, done); 85 | }); 86 | 87 | /** 88 | * Run test once and exit 89 | */ 90 | gulp.task('test-dist-concatenated', function(done) { 91 | karma.start({ 92 | configFile: __dirname + '/karma-dist-concatenated.conf.js', 93 | singleRun: true 94 | }, done); 95 | }); 96 | 97 | /** 98 | * Run test once and exit 99 | */ 100 | gulp.task('test-dist-minified', function(done) { 101 | karma.start({ 102 | configFile: __dirname + '/karma-dist-minified.conf.js', 103 | singleRun: true 104 | }, done); 105 | }); 106 | 107 | gulp.task('default', function() { 108 | runSequence('process-all', 'watch'); 109 | }); 110 | -------------------------------------------------------------------------------- /karma-dist-concatenated.conf.js: -------------------------------------------------------------------------------- 1 | // Karma configuration 2 | // Generated on Thu Aug 21 2014 10:24:39 GMT+0200 (CEST) 3 | 4 | module.exports = function(config) { 5 | config.set({ 6 | 7 | // base path that will be used to resolve all patterns (eg. files, exclude) 8 | basePath: '', 9 | 10 | 11 | // frameworks to use 12 | // available frameworks: https://npmjs.org/browse/keyword/karma-adapter 13 | frameworks: ['mocha', 'chai-jquery', 'jquery-1.8.3', 'sinon-chai'], 14 | 15 | plugins: [ 16 | 'karma-mocha', 17 | 'karma-chai', 18 | 'karma-sinon-chai', 19 | 'karma-chrome-launcher', 20 | 'karma-phantomjs-launcher', 21 | 'karma-jquery', 22 | 'karma-chai-jquery' 23 | ], 24 | 25 | // list of files / patterns to load in the browser 26 | files: [ 27 | 'bower/angular/angular.js', 28 | 'bower/angular-mocks/angular-mocks.js', 29 | 'dist/vg-src.js', 30 | 'test/unit/**/*.js' 31 | ], 32 | 33 | 34 | // list of files to exclude 35 | exclude: [ 36 | ], 37 | 38 | 39 | // preprocess matching files before serving them to the browser 40 | // available preprocessors: https://npmjs.org/browse/keyword/karma-preprocessor 41 | preprocessors: { 42 | }, 43 | 44 | 45 | // test results reporter to use 46 | // possible values: 'dots', 'progress' 47 | // available reporters: https://npmjs.org/browse/keyword/karma-reporter 48 | reporters: ['progress'], 49 | 50 | 51 | // web server port 52 | port: 9876, 53 | 54 | 55 | // enable / disable colors in the output (reporters and logs) 56 | colors: true, 57 | 58 | 59 | // level of logging 60 | // possible values: config.LOG_DISABLE || config.LOG_ERROR || config.LOG_WARN || config.LOG_INFO || config.LOG_DEBUG 61 | logLevel: config.LOG_INFO, 62 | 63 | 64 | // enable / disable watching file and executing tests whenever any file changes 65 | autoWatch: true, 66 | 67 | 68 | // start these browsers 69 | // available browser launchers: https://npmjs.org/browse/keyword/karma-launcher 70 | browsers: ['PhantomJS'], 71 | 72 | 73 | // Continuous Integration mode 74 | // if true, Karma captures browsers, runs the tests and exits 75 | singleRun: false 76 | }); 77 | }; 78 | -------------------------------------------------------------------------------- /karma-dist-minified.conf.js: -------------------------------------------------------------------------------- 1 | // Karma configuration 2 | // Generated on Thu Aug 21 2014 10:24:39 GMT+0200 (CEST) 3 | 4 | module.exports = function(config) { 5 | config.set({ 6 | 7 | // base path that will be used to resolve all patterns (eg. files, exclude) 8 | basePath: '', 9 | 10 | 11 | // frameworks to use 12 | // available frameworks: https://npmjs.org/browse/keyword/karma-adapter 13 | frameworks: ['mocha', 'chai-jquery', 'jquery-1.8.3', 'sinon-chai'], 14 | 15 | plugins: [ 16 | 'karma-mocha', 17 | 'karma-chai', 18 | 'karma-sinon-chai', 19 | 'karma-chrome-launcher', 20 | 'karma-phantomjs-launcher', 21 | 'karma-jquery', 22 | 'karma-chai-jquery' 23 | ], 24 | 25 | // list of files / patterns to load in the browser 26 | files: [ 27 | 'bower/angular/angular.js', 28 | 'bower/angular-mocks/angular-mocks.js', 29 | 'dist/vg-src.min.js', 30 | 'test/unit/**/*.js' 31 | ], 32 | 33 | 34 | // list of files to exclude 35 | exclude: [ 36 | ], 37 | 38 | 39 | // preprocess matching files before serving them to the browser 40 | // available preprocessors: https://npmjs.org/browse/keyword/karma-preprocessor 41 | preprocessors: { 42 | }, 43 | 44 | 45 | // test results reporter to use 46 | // possible values: 'dots', 'progress' 47 | // available reporters: https://npmjs.org/browse/keyword/karma-reporter 48 | reporters: ['progress'], 49 | 50 | 51 | // web server port 52 | port: 9876, 53 | 54 | 55 | // enable / disable colors in the output (reporters and logs) 56 | colors: true, 57 | 58 | 59 | // level of logging 60 | // possible values: config.LOG_DISABLE || config.LOG_ERROR || config.LOG_WARN || config.LOG_INFO || config.LOG_DEBUG 61 | logLevel: config.LOG_INFO, 62 | 63 | 64 | // enable / disable watching file and executing tests whenever any file changes 65 | autoWatch: true, 66 | 67 | 68 | // start these browsers 69 | // available browser launchers: https://npmjs.org/browse/keyword/karma-launcher 70 | browsers: ['PhantomJS'], 71 | 72 | 73 | // Continuous Integration mode 74 | // if true, Karma captures browsers, runs the tests and exits 75 | singleRun: false 76 | }); 77 | }; 78 | -------------------------------------------------------------------------------- /karma-src.conf.js: -------------------------------------------------------------------------------- 1 | // Karma configuration 2 | // Generated on Thu Aug 21 2014 10:24:39 GMT+0200 (CEST) 3 | 4 | module.exports = function(config) { 5 | config.set({ 6 | 7 | // base path that will be used to resolve all patterns (eg. files, exclude) 8 | basePath: '', 9 | 10 | 11 | // frameworks to use 12 | // available frameworks: https://npmjs.org/browse/keyword/karma-adapter 13 | frameworks: ['mocha', 'chai-jquery', 'jquery-1.8.3', 'sinon-chai'], 14 | 15 | plugins: [ 16 | 'karma-mocha', 17 | 'karma-chai', 18 | 'karma-sinon-chai', 19 | 'karma-chrome-launcher', 20 | 'karma-phantomjs-launcher', 21 | 'karma-jquery', 22 | 'karma-chai-jquery', 23 | 'karma-spec-reporter' 24 | ], 25 | 26 | // list of files / patterns to load in the browser 27 | files: [ 28 | 'bower/angular/angular.js', 29 | 'bower/angular-mocks/angular-mocks.js', 30 | 'src/**/*.module.js', 31 | 'src/**/*.js', 32 | 'test/unit/**/*.js' 33 | ], 34 | 35 | 36 | // list of files to exclude 37 | exclude: [ 38 | ], 39 | 40 | 41 | // preprocess matching files before serving them to the browser 42 | // available preprocessors: https://npmjs.org/browse/keyword/karma-preprocessor 43 | preprocessors: { 44 | }, 45 | 46 | 47 | // test results reporter to use 48 | // possible values: 'dots', 'progress' 49 | // available reporters: https://npmjs.org/browse/keyword/karma-reporter 50 | reporters: ['progress', 'spec'], 51 | 52 | 53 | // web server port 54 | port: 9876, 55 | 56 | 57 | // enable / disable colors in the output (reporters and logs) 58 | colors: true, 59 | 60 | 61 | // level of logging 62 | // possible values: config.LOG_DISABLE || config.LOG_ERROR || config.LOG_WARN || config.LOG_INFO || config.LOG_DEBUG 63 | logLevel: config.LOG_INFO, 64 | 65 | 66 | // enable / disable watching file and executing tests whenever any file changes 67 | autoWatch: true, 68 | 69 | 70 | // start these browsers 71 | // available browser launchers: https://npmjs.org/browse/keyword/karma-launcher 72 | browsers: ['PhantomJS'], 73 | 74 | 75 | // Continuous Integration mode 76 | // if true, Karma captures browsers, runs the tests and exits 77 | singleRun: false 78 | }); 79 | }; 80 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "vg-src", 3 | "version": "0.1.0", 4 | "author": { 5 | "name": "van", 6 | "email": "1321907687@qq.com" 7 | }, 8 | "description": "A simple angular plugin for load images, with placeholder during load ,error or empty.", 9 | "dependencies": {}, 10 | "repository": { 11 | "type": "git", 12 | "url": "git@github.com:VanMess/vgSrc.git" 13 | }, 14 | "devDependencies": { 15 | "chai": "^1.9.1", 16 | "chai-jquery": "^1.2.3", 17 | "gulp": "^3.8.7", 18 | "gulp-concat": "^2.3.4", 19 | "gulp-jshint": "^1.8.4", 20 | "gulp-plumber": "^0.6.6", 21 | "gulp-rename": "^1.2.0", 22 | "gulp-uglify": "^0.3.1", 23 | "jshint-stylish": "^0.4.0", 24 | "karma": "^0.12.22", 25 | "karma-chai": "^0.1.0", 26 | "karma-chai-jquery": "*", 27 | "karma-chrome-launcher": "^0.1.4", 28 | "karma-jasmine": "^0.1.5", 29 | "karma-jquery": "^0.1.0", 30 | "karma-mocha": "^0.1.8", 31 | "karma-phantomjs-launcher": "^0.1.4", 32 | "karma-sinon-chai": "^0.2.0", 33 | "karma-spec-reporter": "^0.0.18", 34 | "mocha": "^1.21.4", 35 | "run-sequence": "^1.0.2", 36 | "sinon": "^1.10.3", 37 | "sinon-chai": "^2.5.0" 38 | }, 39 | "engines": { 40 | "node": ">=0.8.0" 41 | } 42 | } 43 | -------------------------------------------------------------------------------- /sample/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | vgSrc sample 10 | 38 | 39 | 40 | 41 |
42 |
43 |

error images

44 | 45 | 46 | 47 |
48 |
49 |

empty images

50 | 51 | 52 | 53 |
54 |
55 |

apply classes

56 | 57 | 58 |
59 |
60 |

loading status

61 | 62 | 63 | 64 |
65 |
66 |

events

67 | 68 |
69 |
70 |

in ng-repeat

71 |
    72 |
  • 73 | 74 |
  • 75 |
76 |
77 |
78 | 79 | 80 | 81 | 82 | 83 | 84 | -------------------------------------------------------------------------------- /sample/index.js: -------------------------------------------------------------------------------- 1 | !(function(ng, $root) { 2 | ng.module('vgSrc.sample', ['vgSrc']).config([ 3 | 'vgSrcConfigProvider', 4 | function(vgSrcConfigProvider) { 5 | vgSrcConfigProvider.$set({ 6 | debug: false, 7 | error: 'http://ico.ooopic.com/iconset01/status-icons/gif/99589.gif', 8 | onBegin: function($e) { 9 | // console.log('start load:' + $e.src); 10 | }, 11 | onError: function($e) { 12 | // console.log('failure load:' + $e.src); 13 | }, 14 | onLoad: function($e) { 15 | // console.log('complete load:' + $e.src); 16 | } 17 | }); 18 | } 19 | ]).controller('IndexController', [ 20 | '$scope', 21 | function($scope) { 22 | this.errorImg = 'http://ico.ooopic.com/iconset01/status-icons/gif/99589.gif'; 23 | this.emptyImg = 'http://ico.ooopic.com/iconset01/status-icons/gif/99474.gif'; 24 | this.loadingImg = 'http://ico.ooopic.com/iconset01/status-icons/gif/99494.gif'; 25 | 26 | this.currentImg = 'http://attach.bbs.miui.com/forum/201402/21/115847dwxfcspf4c54esin.jpg.thumb.jpg'; 27 | 28 | this.imgList = [ 29 | 'http://attach.bbs.miui.com/forum/201402/21/115847dwxfcspf4c54esin.jpg.thumb.jpg', 30 | 'http://attach.bbs.miui.com/forum/201402/21/115847dwxfcspf4c54esin.jpg.thumb.jpeg', 31 | 'http://pic2.52pk.com/files/150929/1283568_103401945.jpg' 32 | ]; 33 | this.switchImg = function() {}; 34 | this.log = function(content) { 35 | console.log(content); 36 | }; 37 | this.debug = function(content) { 38 | debugger; 39 | console.log(content); 40 | }; 41 | } 42 | ]); 43 | 44 | ng.bootstrap($root, ['vgSrc.sample']); 45 | })(window.angular, window.document); 46 | -------------------------------------------------------------------------------- /src/_end.js: -------------------------------------------------------------------------------- 1 | }); 2 | -------------------------------------------------------------------------------- /src/_start.js: -------------------------------------------------------------------------------- 1 | /* 2 | * angular-image-lazyload - v0.0.1 - 2015-10-14 3 | * https://github.com/VanMess/angular-image-lazyload 4 | * Copyright (c) 2014 Van (http://vanmess.github.io/) 5 | */ 6 | !(function(factory) { 7 | if (typeof define === 'function' && define.amd) { 8 | // AMD 9 | define(['angular'], factory); 10 | } else { 11 | // Global Variables 12 | factory(window.angular); 13 | } 14 | })(function(ng) { 15 | 'use strict'; 16 | -------------------------------------------------------------------------------- /src/vgSrc.config.js: -------------------------------------------------------------------------------- 1 | /* 2 | * The provider of this module, just for setting default config. 3 | */ 4 | var 5 | 6 | // default setting 7 | defaults = { 8 | debug: false, 9 | // place image content when it begin to load real image resource. 10 | loading: '', 11 | 12 | // element would applied this class when it is loading. 13 | loadingCls: '', 14 | 15 | // element would applied this class when it is load success. 16 | loadedCls: '', 17 | 18 | // place image content when image resource fail to load. 19 | error: '', 20 | 21 | // element would applied this class when it faild to load. 22 | errorCls: '', 23 | 24 | // place image content when element's "src" attribute is empty(null,undefine). 25 | empty: '', 26 | 27 | // element would applied this class when src attribute is empty(null,undefine). 28 | emptyCls: '', 29 | 30 | // event occurs when it start to load something. 31 | 'onBegin': ng.noop, 32 | 33 | // event occurs when it fail load something. 34 | 'onError': ng.noop, 35 | 36 | // event occurs when it finish to load something. 37 | 'onLoad': ng.noop 38 | }, 39 | 40 | provider = function() { 41 | var moduleConfig = ng.copy(defaults), 42 | result = { 43 | $set: _setConfig, 44 | $get: function() { 45 | return moduleConfig; 46 | } 47 | }; 48 | 49 | return result; 50 | 51 | function _setConfig(cfg) { 52 | 53 | if (!ng.isDefined(cfg)) return; 54 | 55 | ng.forEach(Object.keys(moduleConfig), function(key) { 56 | if (cfg.hasOwnProperty(key) && key !== 'debug') { 57 | moduleConfig[key] = cfg[key] + ''; 58 | } 59 | }); 60 | 61 | // init debug model 62 | if (cfg.hasOwnProperty('debug')) { 63 | moduleConfig.debug = !!cfg.debug; 64 | } 65 | 66 | // init events 67 | var events = ['onBegin', 'onError', 'onLoad'], 68 | tmp; 69 | ng.forEach(events, function(e) { 70 | tmp = ng.isFunction(cfg[e]) ? cfg[e] : ng.noop; 71 | moduleConfig[e] = (function(_old) { 72 | return function($scope, $e) { 73 | _old($e); 74 | }; 75 | })(tmp); 76 | }); 77 | } 78 | }; 79 | 80 | angular.module('vgSrc').provider('vgSrcConfig', [provider]); 81 | -------------------------------------------------------------------------------- /src/vgSrc.directive.js: -------------------------------------------------------------------------------- 1 | /* 2 | * An angular directive for image loading status present. 3 | */ 4 | var directive = function($parse, defaults) { 5 | var status = ['loading', 'empty', 'error'], 6 | statusCls = ['loadingCls', 'emptyCls', 'errorCls', 'loadedCls'], 7 | events = ['onBegin', 'onError', 'onLoad'], 8 | // angular's directive define object. 9 | defineObj = { 10 | priority: 99, 11 | restrict: 'A', 12 | name: 'vgSrc', 13 | compile: function(element, attrs) { 14 | var attrName = attrs.$normalize(defineObj.name), 15 | srcParser = $parse(attrs[attrName]); 16 | 17 | return function _link($scope, element, attrs) { 18 | var opt = ng.copy(defaults), 19 | $log = opt.debug ? console.log.bind(console) : ng.noop; 20 | // parse everything in status lists. 21 | ng.forEach(status, function(att) { 22 | if (ng.isString(attrs[att])) { 23 | // parse element's setting attribute use ng's '$parse' 24 | // so that users can define the configuration by ng's 'expression'. 25 | opt[att] = $parse(attrs[att])($scope); 26 | } 27 | }); 28 | 29 | // simply copy everything in statusCls lists. 30 | ng.forEach(statusCls, function(att) { 31 | if (ng.isString(attrs[att])) { 32 | opt[att] = attrs[att]; 33 | } 34 | }); 35 | 36 | // parse event handlers 37 | // so that we can occu 38 | ng.forEach(events, function(att) { 39 | if (ng.isString(attrs[att])) { 40 | opt[att] = $parse(attrs[att]); 41 | } 42 | }); 43 | 44 | // watching vgSrc attribute 45 | // so that we could dynamicly fresh element's image when each time user change the value 46 | $scope.$watch(function() { 47 | return srcParser($scope); 48 | }, function _bindImg(newVal, oldVal) { 49 | var $e = { 50 | src: newVal 51 | }; 52 | if (ng.isString(newVal) && newVal.length > 0) { 53 | attrs.$set('src', opt.loading); 54 | _refreshCls(opt['loadingCls']); 55 | opt['onBegin'].call($scope, $scope, $e); 56 | $log('start loading resource:' + $e.src); 57 | 58 | _lazyLoad(newVal, function() { 59 | attrs.$set('src', newVal); 60 | _refreshCls(opt['loadedCls']); 61 | opt['onLoad'].call($scope, $scope, $e); 62 | $log('success load resource:' + $e.src); 63 | }, function() { 64 | attrs.$set('src', opt.error); 65 | _refreshCls(opt['errorCls']); 66 | opt['onError'].call($scope, $scope, $e); 67 | $log('failure load resource:' + $e.src); 68 | }) 69 | } else { 70 | attrs.$set('src', opt.empty); 71 | _refreshCls(opt['emptyCls']); 72 | opt['onError'].call($scope, $scope, $e); 73 | $log('current img is empty'); 74 | } 75 | }); 76 | 77 | // clear element's status class 78 | // and add the new class 79 | function _refreshCls(cls) { 80 | ng.forEach(statusCls, function(cls) { 81 | element.removeClass(opt[cls]); 82 | }); 83 | element.addClass(cls); 84 | } 85 | }; 86 | 87 | } 88 | }; 89 | 90 | return defineObj; 91 | }; 92 | 93 | angular.module('vgSrc').directive('vgSrc', ['$parse', 'vgSrcConfig', directive]); 94 | 95 | /* 96 | * load function to excute a shadow load 97 | */ 98 | function _lazyLoad(src, loadCallback, errorCallback) { 99 | var $imgDom = ng.element(new Image()); 100 | loadCallback = ng.isFunction(loadCallback) ? loadCallback : ng.noop; 101 | errorCallback = ng.isFunction(errorCallback) ? errorCallback : ng.noop; 102 | 103 | $imgDom.bind('error', errorCallback.bind(this)).bind('load', loadCallback.bind(this)).attr('src', src); 104 | } 105 | -------------------------------------------------------------------------------- /src/vgSrc.ie.js: -------------------------------------------------------------------------------- 1 | var DONT_ENUM = "propertyIsEnumerable,isPrototypeOf,hasOwnProperty,toLocaleString,toString,valueOf,constructor".split(","), 2 | hasOwn = ({}).hasOwnProperty; 3 | for (var i in { 4 | toString: 1 5 | }) { 6 | DONT_ENUM = false; 7 | } 8 | Object.keys = Object.keys || (function(obj) { //ecma262v5 15.2.3.14 9 | return function(obj) { 10 | var result = [], 11 | key; 12 | for (key in obj) { 13 | if (hasOwn.call(obj, key)) { 14 | result.push(key); 15 | } 16 | } 17 | if (DONT_ENUM && obj) { 18 | for (var i = 0; i < DONT_ENUM.length; i++) { 19 | key = DONT_ENUM[i++]; 20 | if (hasOwn.call(obj, key)) { 21 | result.push(key); 22 | } 23 | } 24 | } 25 | return result; 26 | }; 27 | })(); 28 | 29 | if (!Function.prototype.bind) { 30 | Function.prototype.bind = function(oThis) { 31 | if (typeof this !== "function") { 32 | // closest thing possible to the ECMAScript 5 internal IsCallable function 33 | throw new TypeError("Function.prototype.bind - what is trying to be bound is not callable"); 34 | } 35 | var aArgs = Array.prototype.slice.call(arguments, 1), 36 | fToBind = this, 37 | fNOP = function() {}, 38 | fBound = function() { 39 | return fToBind.apply(this instanceof fNOP && oThis ? this : oThis, 40 | aArgs.concat(Array.prototype.slice.call(arguments))); 41 | }; 42 | fNOP.prototype = this.prototype; 43 | fBound.prototype = new fNOP(); 44 | return fBound; 45 | }; 46 | } 47 | -------------------------------------------------------------------------------- /src/vgSrc.js: -------------------------------------------------------------------------------- 1 | // Create all modules and define dependencies to make sure they exist 2 | // and are loaded in the correct order to satisfy dependency injection 3 | // before all nested files are concatenated by Gulp 4 | 5 | angular.module('vgSrc', []); 6 | -------------------------------------------------------------------------------- /test/unit/vg-src/vgSrcSpec.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | describe('', function() { 4 | 5 | var module; 6 | var dependencies; 7 | dependencies = []; 8 | 9 | var hasModule = function(module) { 10 | return dependencies.indexOf(module) >= 0; 11 | }; 12 | 13 | beforeEach(function() { 14 | 15 | // Get module 16 | module = angular.module('vgSrc'); 17 | dependencies = module.requires; 18 | }); 19 | 20 | it('should load config module', function() { 21 | expect(hasModule('vgSrc.config')).to.be.ok; 22 | }); 23 | 24 | 25 | 26 | 27 | it('should load directives module', function() { 28 | expect(hasModule('vgSrc.directives')).to.be.ok; 29 | }); 30 | 31 | 32 | 33 | 34 | }); --------------------------------------------------------------------------------