├── .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 | });
--------------------------------------------------------------------------------