From 68ad8f509b63ba599b39df0f28d94a7dea87e5fe Mon Sep 17 00:00:00 2001 From: popomore Date: Wed, 24 Dec 2014 13:05:43 +0800 Subject: [PATCH 01/14] feat(script) generate file with hash when has hash option --- Gruntfile.js | 32 +- package.json | 5 +- tasks/lib/json.js | 4 +- tasks/lib/script.js | 409 +++++++++++------- tasks/lib/style.js | 4 +- tasks/lib/template.js | 8 +- tasks/lib/text.js | 4 +- tasks/transport.js | 3 + test/cases/alias/baz-debug.js.expect | 2 +- test/cases/alias/baz.js.expect | 2 +- test/cases/assets/arale.js | 2 +- test/cases/assets/family/bar/bar.js | 3 + test/cases/cmdid/foo-debug.js.expect | 2 +- test/cases/cmdid/foo.js.expect | 2 +- test/cases/css2js/a-debug.css.js.expect | 2 +- test/cases/css2js/a.css.js.expect | 2 +- test/cases/directory/v.js.expect | 2 +- test/cases/directory/v/a.js.expect | 2 +- test/cases/directory/v/b.js.expect | 2 +- test/cases/duplicate/a.js.expect | 2 +- test/cases/expand/expand-debug.js.expect | 2 +- .../expand/expand-debug.js.expect.expect | 2 +- test/cases/expand/expand.js.expect | 2 +- test/cases/expand/expand.js.expect.expect | 2 +- .../month-debug.handlebars.js.expect | 2 +- .../handlebars/month.handlebars.js.expect | 2 +- .../simple-debug.handlebars.js.expect | 2 +- .../handlebars/simple.handlebars.js.expect | 2 +- test/cases/hash/a-db1b8e6a.js.expect | 4 + test/cases/hash/a.js | 4 + test/cases/hash/a.js.expect | 4 + test/cases/hash/b-4849d846.js.expect | 4 + test/cases/hash/b.js | 4 + test/cases/hash/b.js.expect | 4 + test/cases/hash/c/c.js | 3 + test/cases/id-deps-exist/a.js.expect | 2 +- test/cases/id-deps-exist/b.js.expect | 2 +- test/cases/id-deps-exist/c.js.expect | 2 +- test/cases/json/simple-debug.json.js.expect | 2 +- test/cases/json/simple.json.js.expect | 2 +- test/cases/nested/bar/a.js.expect | 2 +- test/cases/nested/bar/b.js.expect | 2 +- test/cases/nested/bar/c.js.expect | 2 +- test/cases/nested/foo/a.js.expect | 2 +- test/cases/nested/foo/b.js.expect | 2 +- test/cases/nested/foo/c.js.expect | 2 +- test/cases/relative/bar-debug.js.expect | 2 +- test/cases/relative/bar.js.expect | 2 +- test/cases/relative/foo-debug.js.expect | 2 +- test/cases/relative/foo.js.expect | 2 +- test/cases/rely-arale/foo-debug.js.expect | 4 +- test/cases/rely-arale/foo.js.expect | 4 +- test/cases/single/bar/foo-debug.js.expect | 2 +- test/cases/single/bar/foo.js.expect | 2 +- test/cases/single/foo-debug.js.expect | 2 +- test/cases/single/foo.js.expect | 2 +- test/cases/style/a-debug.css.js.expect | 2 +- test/cases/style/a-debug.js.expect | 2 +- test/cases/style/a.css.js.expect | 2 +- test/cases/style/a.js.expect | 2 +- test/cases/text!/a.js.expect | 2 +- test/cases/text/foo-debug.html.js.expect | 2 +- test/cases/text/foo.html.js.expect | 2 +- test/cases/tpl/month-debug.tpl.js.expect | 2 +- test/cases/tpl/month.tpl.js.expect | 2 +- test/cases/tpl/simple-debug.tpl.js.expect | 2 +- test/cases/tpl/simple.tpl.js.expect | 2 +- test/transport.js | 6 +- 68 files changed, 371 insertions(+), 240 deletions(-) create mode 100644 test/cases/assets/family/bar/bar.js create mode 100644 test/cases/hash/a-db1b8e6a.js.expect create mode 100644 test/cases/hash/a.js create mode 100644 test/cases/hash/a.js.expect create mode 100644 test/cases/hash/b-4849d846.js.expect create mode 100644 test/cases/hash/b.js create mode 100644 test/cases/hash/b.js.expect create mode 100644 test/cases/hash/c/c.js diff --git a/Gruntfile.js b/Gruntfile.js index cabd0e2..8939598 100644 --- a/Gruntfile.js +++ b/Gruntfile.js @@ -18,11 +18,11 @@ module.exports = function(grunt) { all: [ 'Gruntfile.js', 'tasks/**/*.js', - '<%= mochaTest.test.src %>', + '<%= mochaTest.test.src %>' ], options: { - jshintrc: '.jshintrc', - }, + jshintrc: '.jshintrc' + } }, clean: { @@ -93,7 +93,10 @@ module.exports = function(grunt) { // rely on other modules rely: { options: { - paths: ['test/cases/assets'] + paths: ['test/cases/assets'], + alias: { + foo: 'arale/class/foo' + } }, files: [{ expand: true, @@ -248,6 +251,25 @@ module.exports = function(grunt) { src: '**/*.js', dest: 'test/expected/directory' }] + }, + + 'hash': { + options: { + paths: ['test/cases/assets'], + alias: { + 'foo': 'arale/class/foo', + 'bar': 'family/bar/bar', + '$': '$' + }, + idleading: 'family/name/', + hash: true + }, + files: [{ + expand: true, + cwd: 'test/cases/hash', + src: '*.js', + dest: 'test/expected/hash' + }] } }, @@ -271,6 +293,6 @@ module.exports = function(grunt) { // By default, lint and run all tests. grunt.registerTask('default', ['jshint']); - grunt.registerTask('test', ['clean', 'transport', 'mochaTest', 'clean']); + grunt.registerTask('test', ['clean', 'transport', 'mochaTest']); }; diff --git a/package.json b/package.json index 28f3ab4..bb88779 100644 --- a/package.json +++ b/package.json @@ -31,7 +31,8 @@ "cmd-util": "~0.3.5", "handlebars": "1.0.11", "uglify-js": "~2.2.5", - "css": "~1.4.0" + "css": "~1.4.0", + "relative": "~1.2.0" }, "devDependencies": { "grunt-contrib-jshint": "~0.1.1", @@ -43,4 +44,4 @@ "keywords": [ "gruntplugin" ] -} \ No newline at end of file +} diff --git a/tasks/lib/json.js b/tasks/lib/json.js index 7c99bbe..510b7b3 100644 --- a/tasks/lib/json.js +++ b/tasks/lib/json.js @@ -16,7 +16,7 @@ exports.init = function(grunt) { var code = format('define("%s", [], %s)', id, data); var astCache = ast.getAst(code); - data = astCache.print_to_string(options.uglify); + data = astCache.print_to_string(options.uglify) + '\n'; grunt.file.write(dest, data); // create debug file @@ -33,7 +33,7 @@ exports.init = function(grunt) { return v + '-debug'; } }); - data = astCache.print_to_string(options.uglify); + data = astCache.print_to_string(options.uglify) + '\n'; grunt.file.write(dest, data); }; return exports; diff --git a/tasks/lib/script.js b/tasks/lib/script.js index 88a7cb7..5650ece 100644 --- a/tasks/lib/script.js +++ b/tasks/lib/script.js @@ -1,225 +1,300 @@ exports.init = function(grunt) { + var crypto = require('crypto'); var path = require('path'); + var join = path.join; var ast = require('cmd-util').ast; var iduri = require('cmd-util').iduri; + var relative = require('relative'); var _ = grunt.util._; + return { + jsParser: jsParser + }; - var exports = {}; - - exports.jsParser = function(fileObj, options) { + function jsParser(fileObj, options) { grunt.log.verbose.writeln('Transport ' + fileObj.src + ' -> ' + fileObj.dest); - var astCache, data = fileObj.srcData || grunt.file.read(fileObj.src); - try { - astCache = ast.getAst(data); - } catch(e) { - grunt.log.error('js parse error ' + fileObj.src.red); - grunt.fail.fatal(e.message + ' [ line:' + e.line + ', col:' + e.col + ', pos:' + e.pos + ' ]'); - } - var meta = ast.parseFirst(astCache); - - if (!meta) { - grunt.log.warn('found non cmd module "' + fileObj.src + '"'); - // do nothing - return; - } + /* + cache every filepath content to generate hash + */ + var fileCache = {}; - if (meta.id) { - grunt.log.verbose.writeln('id exists in "' + fileObj.src + '"'); - } + var file = getFileInfo(path.join(process.cwd(), fileObj.src)); - var deps, depsSpecified = false; - if (meta.dependencyNode) { - deps = meta.dependencies; - depsSpecified = true; - grunt.log.verbose.writeln('dependencies exists in "' + fileObj.src + '"'); - } else { - deps = parseDependencies(fileObj.src, options); - grunt.log.verbose.writeln(deps.length ? - 'found dependencies ' + deps : 'found no dependencies'); - } + if (!file) return; - // create .js file - astCache = ast.modify(astCache, { - id: meta.id ? meta.id : unixy(options.idleading + fileObj.name.replace(/\.js$/, '')), - dependencies: deps, + // create original file, xxx.js + var data = ast.modify(file.contents, { + id: unixy(options.idleading) + getId(file), + dependencies: getDeps(file), require: function(v) { // ignore when deps is specified by developer - return depsSpecified ? v : iduri.parseAlias(options, v); + return file.depsSpecified ? v : iduri.parseAlias(options, v); } - }); - data = astCache.print_to_string(options.uglify); - grunt.file.write(fileObj.dest, addOuterBoxClass(data, options)); + }).print_to_string(options.uglify); + writeFile(data, fileObj.dest); + + // create file with hash, xxx-2cio56s.js + if (options.hash) { + var hash = file.hash; + data = ast.modify(file.contents, { + id: unixy(options.idleading) + getId(file) + '-' + hash, + dependencies: getDeps(file, addHash), + require: function(v) { + // ignore when deps is specified by developer + if (file.depsSpecified) return v; + var depFile = file.depMap[v]; + if (depFile) { + return addHash(depFile); + } + return iduri.parseAlias(options, v); + } + }).print_to_string(options.uglify); + writeFile(data, fileObj.dest.replace(/\.js$/, '-' + hash + '.js')); + } + // create file with debug, xxx-debug.js + if (options.debug) { + data = ast.modify(file.contents, { + id: unixy(options.idleading) + addDebug(getId(file)), + dependencies: getDeps(file, addDebug), + require: function(v) { + // ignore when deps is specified by developer + return file.depsSpecified ? v : addDebug(iduri.parseAlias(options, v)); + } + }).print_to_string(options.uglify); + writeFile(data, addDebug(fileObj.dest)); + } - // create -debug.js file - if (!options.debug) { - return; + function getId(file) { + return file.id || unixy(fileObj.name.replace(/\.js$/, '')); } - var dest = fileObj.dest.replace(/\.js$/, '-debug.js'); - astCache = ast.modify(data, function(v) { - var ext = path.extname(v); + function getDeps(file, fn) { + return file.dependencies.map(function(dep) { + if (typeof dep !== 'string') { + dep = fn ? fn(dep) : dep.id; + } + return dep.replace(/\.js$/, ''); + }); + } + function addDebug(v) { + if (v.id) v = v.id; + var ext = path.extname(v); if (ext && options.parsers[ext]) { return v.replace(new RegExp('\\' + ext + '$'), '-debug' + ext); } else { return v + '-debug'; } - }); - data = astCache.print_to_string(options.uglify); - grunt.file.write(dest, addOuterBoxClass(data, options)); - }; - - - // helpers - // ---------------- - function unixy(uri) { - return uri.replace(/\\/g, '/'); - } - - function getStyleId(options) { - return unixy((options || {}).idleading || '') - .replace(/\/$/, '') - .replace(/\//g, '-') - .replace(/\./g, '_'); - } - - function addOuterBoxClass(data, options) { - // ex. arale/widget/1.0.0/ => arale-widget-1_0_0 - var styleId = getStyleId(options); - if (options.styleBox && styleId) { - data = data.replace(/(\}\)[;\n\r ]*$)/, 'module.exports.outerBoxClass="' + styleId + '";$1'); } - return data; - } - function moduleDependencies(id, options) { - var alias = iduri.parseAlias(options, id); + function addHash(v) { + if (!v.hash) return v.id; - if (iduri.isAlias(options, id) && alias === id) { - // usually this is "$" - return []; + var ext = path.extname(v.id); + if (ext && options.parsers[ext]) { + return v.id.replace(new RegExp('\\' + ext + '$'), '-' + v.hash + ext); + } else { + return v.id + '-' + v.hash; + } } - // don't resolve text!path/to/some.xx, same as seajs-text - if (/^text!/.test(id)) { - return []; + function writeFile(data, dest) { + grunt.file.write(dest, addOuterBoxClass(data + '\n', options)); } - var file = iduri.appendext(alias); - - if (!/\.js$/.test(file)) { - return []; + // helpers + // ---------------- + function unixy(uri) { + return uri.replace(/\\/g, '/'); } - var fpath; - options.paths.some(function(base) { - var filepath = path.join(base, file); - if (grunt.file.exists(filepath)) { - grunt.log.verbose.writeln('find module "' + filepath + '"'); - fpath = filepath; - return true; - } - }); - - if (!fpath) { - grunt.fail.warn("can't find module " + alias); - return []; + function getStyleId() { + return unixy((options || {}).idleading || '') + .replace(/\/$/, '') + .replace(/\//g, '-') + .replace(/\./g, '_'); } - if (!grunt.file.exists(fpath)) { - grunt.fail.warn("can't find " + fpath); - return []; - } - var data = grunt.file.read(fpath); - var parsed = ast.parse(data); - var deps = []; - - var ids = parsed.map(function(meta) { - return meta.id; - }); - - parsed.forEach(function(meta) { - meta.dependencies.forEach(function(dep) { - dep = iduri.absolute(alias, dep); - if (!_.contains(deps, dep) && !_.contains(ids, dep) && !_.contains(ids, dep.replace(/\.js$/, ''))) { - deps.push(dep); - } - }); - }); - return deps; - } - function parseDependencies(fpath, options) { - var rootpath = fpath; - - function relativeDependencies(fpath, options, basefile) { - if (basefile) { - fpath = path.join(path.dirname(basefile), fpath); + function addOuterBoxClass(data) { + // ex. arale/widget/1.0.0/ => arale-widget-1_0_0 + var styleId = getStyleId(options); + if (options.styleBox && styleId) { + data = data.replace(/(\}\)[;\n\r ]*$)/, 'module.exports.outerBoxClass="' + styleId + '";$1'); } - fpath = iduri.appendext(fpath); - - var deps = []; - var moduleDeps = {}; + return data; + } - if (!grunt.file.exists(fpath)) { - if (!/\{\w+\}/.test(fpath)) { - grunt.log.warn("can't find " + fpath); + function parseFileDependencies(filepath, depMap) { + if (!grunt.file.exists(filepath)) { + if (!/\{\w+\}/.test(filepath)) { + grunt.log.warn('can\'t find ' + filepath); } return []; } - var parsed, data = grunt.file.read(fpath); + + var parsed, data = grunt.file.read(filepath); try { parsed = ast.parseFirst(data); } catch(e) { grunt.log.error(e.message + ' [ line:' + e.line + ', col:' + e.col + ', pos:' + e.pos + ' ]'); return []; } - parsed.dependencies.map(function(id) { - return id.replace(/\.js$/, ''); - }).forEach(function(id) { + return parsed.dependencies.map(function(id) { if (id.charAt(0) === '.') { - // fix nested relative dependencies - if (basefile) { - var altId = path.join(path.dirname(fpath), id).replace(/\\/g, '/'); - var dirname = path.dirname(rootpath).replace(/\\/g, '/'); - if ( dirname !== altId ) { - altId = path.relative(dirname, altId); - } else { - // the same name between file and directory - altId = path.relative(dirname, altId + '.js').replace(/\.js$/, ''); - } - altId = altId.replace(/\\/g, '/'); - if (altId.charAt(0) !== '.') { - altId = './' + altId; - } - deps.push(altId); - } else { - deps.push(id); - } - if (/\.js$/.test(iduri.appendext(id))) { - deps = grunt.util._.union(deps, relativeDependencies(id, options, fpath)); - } - } else if (!moduleDeps[id]) { - var alias = iduri.parseAlias(options, id); - deps.push(alias); + var origId = id; + id = iduri.appendext(id); + var depFilepath = path.join(path.dirname(filepath), id); + var depFile = getFileInfo(depFilepath); + if (!depFile) return; + var obj = { + id: origId, + path: depFile.path, + contents: depFile.contents, + hash: depFile.hash, + relative: true + }; + if (depMap) depMap[origId] = obj; + return [obj].concat(parseFileDependencies(depFilepath)); + } else { + return parseModuleDependencies(id); + } + }); + } + + function parseModuleDependencies(id) { + var alias = iduri.parseAlias(options, id); + + if (iduri.isAlias(options, id) && alias === id) { + // usually this is "$" + return [{id: id}]; + } + + // don't resolve text!path/to/some.xx, same as seajs-text + if (/^text!/.test(id)) { + return [{id: id}]; + } - // don't parse no javascript dependencies - var ext = path.extname(alias); - if (ext && ext !== '.js') return; + var fpath = iduri.appendext(alias); - var mdeps = moduleDependencies(id, options); - moduleDeps[id] = mdeps; - deps = grunt.util._.union(deps, mdeps); + var fbase; + options.paths.some(function(base) { + var filepath = path.join(base, fpath); + if (grunt.file.exists(filepath)) { + grunt.log.verbose.writeln('find module "' + filepath + '"'); + fbase = base; + fpath = filepath; + return true; } }); - return deps; + + if (!fpath) { + grunt.fail.warn('can\'t find module ' + alias); + return [{id: id}]; + } + + var file = getFileInfo(fpath); + + // don't parse no javascript dependencies + if (!/\.js$/.test(fpath)) { + return [{ + id: alias, + path: file.path, + hash: file.hash, + contents: file.contents + }]; + } + + var parsed = ast.parse(file.contents); + var deps = parsed.map(function(meta) { + return meta.dependencies.map(function(id) { + id = iduri.absolute(alias, id); + id = iduri.appendext(id); + var file = getFileInfo(path.join(fbase, id)); + if (!file) return; + return { + id: id, + path: file.path, + hash: file.hash, + contents: file.contents + }; + }); + }); + return [{ + id: alias, + path: file.path, + hash: file.hash, + contents: file.contents + }].concat(deps); } - return relativeDependencies(fpath, options); + function getFileInfo(path){ + if (fileCache[path]) { + return fileCache[path]; + } + + if (!grunt.file.exists(path)) return; + var astCache, deps, contents = grunt.file.read(path); + + try { + astCache = ast.getAst(contents); + } catch(e) { + grunt.log.error('js parse error ' + path.red); + grunt.fail.fatal(e.message + ' [ line:' + e.line + ', col:' + e.col + ', pos:' + e.pos + ' ]'); + } + + // get id/deps of origin cmd module + var meta = ast.parseFirst(astCache), depMap = {}; + + if (!meta) { + grunt.log.warn('found non cmd module "' + path + '"'); + // do nothing + return; + } + + if (meta.id) { + grunt.log.verbose.writeln('id exists in "' + path + '"'); + } + + if (meta.dependencyNode) { + deps = meta.dependencies; + grunt.log.verbose.writeln('dependencies exists in "' + path + '"'); + } else { + deps = parseFileDependencies(path, depMap); + deps = grunt.util._.chain(deps) + .flatten() + .filter(function(item) {return typeof item !== 'undefined'}) + .uniq(function(item) {return item.path}) + .each(function(item) { + if (item.relative) item.id = relative(path, item.path); + }) + .value(); + grunt.log.verbose.writeln(deps.length ? + 'found dependencies ' + deps : 'found no dependencies'); + } + + return fileCache[path] = { + id: meta.id, + dependencies: deps, + depMap: depMap, + depsSpecified: typeof meta.dependencyNode !== 'undefined', + contents: contents, + path: path, + hash: typeof meta.dependencyNode !== 'undefined' ? '' : md5(contents, deps) + }; + } } - return exports; + function md5(contents, deps) { + contents = deps.map(function(depFile) { + return depFile.contents || ''; + }) + contents; + return crypto + .createHash('md5') + .update(contents, 'utf8') + .digest('hex') + .slice(0, 8); + } }; diff --git a/tasks/lib/style.js b/tasks/lib/style.js index 7200de7..fc01b10 100644 --- a/tasks/lib/style.js +++ b/tasks/lib/style.js @@ -20,7 +20,7 @@ exports.init = function(grunt) { var id = unixy(options.idleading + fileObj.name.replace(/\.js$/, '')); data = css2js(data, id, options, fileObj); - data = ast.getAst(data).print_to_string(options.uglify); + data = ast.getAst(data).print_to_string(options.uglify) + '\n'; var dest = fileObj.dest + '.js'; grunt.file.write(dest, data); @@ -37,7 +37,7 @@ exports.init = function(grunt) { return v + '-debug'; } }); - data = data.print_to_string(options.uglify); + data = data.print_to_string(options.uglify) + '\n'; grunt.file.write(dest, data); }; diff --git a/tasks/lib/template.js b/tasks/lib/template.js index d688126..74e6d93 100644 --- a/tasks/lib/template.js +++ b/tasks/lib/template.js @@ -16,7 +16,7 @@ exports.init = function(grunt) { var code = format('define("%s", [], "%s")', id, data.replace(/\"/g, '\\\"')); var astCache = ast.getAst(code); - data = astCache.print_to_string(options.uglify); + data = astCache.print_to_string(options.uglify) + '\n'; grunt.file.write(dest, data); // create debug file @@ -33,7 +33,7 @@ exports.init = function(grunt) { return v + '-debug'; } }); - data = astCache.print_to_string(options.uglify); + data = astCache.print_to_string(options.uglify) + '\n'; grunt.file.write(dest, data); }; @@ -67,7 +67,7 @@ exports.init = function(grunt) { var ret = format(template, id, alias, alias, code); var astCache = ast.getAst(ret); - data = astCache.print_to_string(options.uglify); + data = astCache.print_to_string(options.uglify) + '\n'; grunt.file.write(dest, data); // create debug file @@ -84,7 +84,7 @@ exports.init = function(grunt) { return v + '-debug'; } }); - data = astCache.print_to_string(options.uglify); + data = astCache.print_to_string(options.uglify) + '\n'; grunt.file.write(dest, data); }; diff --git a/tasks/lib/text.js b/tasks/lib/text.js index faa94d7..d3f2e45 100644 --- a/tasks/lib/text.js +++ b/tasks/lib/text.js @@ -17,7 +17,7 @@ exports.init = function(grunt) { var id = unixy(options.idleading + fileObj.name.replace(/\.js$/, '')); data = html2js(data, id); - data = ast.getAst(data).print_to_string(options.uglify); + data = ast.getAst(data).print_to_string(options.uglify) + '\n'; var dest = fileObj.dest + '.js'; grunt.file.write(dest, data); @@ -34,7 +34,7 @@ exports.init = function(grunt) { return v + '-debug'; } }); - data = data.print_to_string(options.uglify); + data = data.print_to_string(options.uglify) + '\n'; grunt.file.write(dest, data); }; diff --git a/tasks/transport.js b/tasks/transport.js index a0abf1f..b12bbb7 100644 --- a/tasks/transport.js +++ b/tasks/transport.js @@ -27,6 +27,9 @@ module.exports = function(grunt) { // create a debug file or not debug: true, + // create a file with hash + hash: false, + // process a template or not process: false, diff --git a/test/cases/alias/baz-debug.js.expect b/test/cases/alias/baz-debug.js.expect index e505e55..9373308 100644 --- a/test/cases/alias/baz-debug.js.expect +++ b/test/cases/alias/baz-debug.js.expect @@ -2,4 +2,4 @@ define("baz-debug", [ "arale/class/foo-debug", "$-debug" ], function(require, ex require("arale/class/foo-debug"); require("$-debug"); module.exports = "baz"; -}); \ No newline at end of file +}); diff --git a/test/cases/alias/baz.js.expect b/test/cases/alias/baz.js.expect index bada148..30065eb 100644 --- a/test/cases/alias/baz.js.expect +++ b/test/cases/alias/baz.js.expect @@ -2,4 +2,4 @@ define("baz", [ "arale/class/foo", "$" ], function(require, exports, module) { require("arale/class/foo"); require("$"); module.exports = "baz"; -}); \ No newline at end of file +}); diff --git a/test/cases/assets/arale.js b/test/cases/assets/arale.js index 10b820a..7f79dc1 100644 --- a/test/cases/assets/arale.js +++ b/test/cases/assets/arale.js @@ -1,3 +1,3 @@ -define('arale', ['foo'], function(require, exports, module) { +define('arale', ['arale/class/foo'], function(require, exports, module) { module.exports = 'arale'; }) diff --git a/test/cases/assets/family/bar/bar.js b/test/cases/assets/family/bar/bar.js new file mode 100644 index 0000000..a7eb7e9 --- /dev/null +++ b/test/cases/assets/family/bar/bar.js @@ -0,0 +1,3 @@ +define('family/name/bar', [], function(require, exports, module) { + module.exports = 'family/name/bar'; +}); diff --git a/test/cases/cmdid/foo-debug.js.expect b/test/cases/cmdid/foo-debug.js.expect index 976a58a..cbc9c69 100644 --- a/test/cases/cmdid/foo-debug.js.expect +++ b/test/cases/cmdid/foo-debug.js.expect @@ -1,3 +1,3 @@ define("family/name/1.0.0/foo-debug", [], function(require, exports, module) { module.exports = "foo"; -}); \ No newline at end of file +}); diff --git a/test/cases/cmdid/foo.js.expect b/test/cases/cmdid/foo.js.expect index b6d22f4..42cd3d7 100644 --- a/test/cases/cmdid/foo.js.expect +++ b/test/cases/cmdid/foo.js.expect @@ -1,3 +1,3 @@ define("family/name/1.0.0/foo", [], function(require, exports, module) { module.exports = "foo"; -}); \ No newline at end of file +}); diff --git a/test/cases/css2js/a-debug.css.js.expect b/test/cases/css2js/a-debug.css.js.expect index 2d7f3e1..120c726 100644 --- a/test/cases/css2js/a-debug.css.js.expect +++ b/test/cases/css2js/a-debug.css.js.expect @@ -1,3 +1,3 @@ define("a-debug.css", [], function() { seajs.importStyle("body{color:#fff}"); -}); \ No newline at end of file +}); diff --git a/test/cases/css2js/a.css.js.expect b/test/cases/css2js/a.css.js.expect index 972e288..74984fc 100644 --- a/test/cases/css2js/a.css.js.expect +++ b/test/cases/css2js/a.css.js.expect @@ -1,3 +1,3 @@ define("a.css", [], function() { seajs.importStyle("body{color:#fff}"); -}); \ No newline at end of file +}); diff --git a/test/cases/directory/v.js.expect b/test/cases/directory/v.js.expect index 961b33d..57e91c5 100644 --- a/test/cases/directory/v.js.expect +++ b/test/cases/directory/v.js.expect @@ -1 +1 @@ -define("v", [], function(require, exports, module) {}); \ No newline at end of file +define("v", [], function(require, exports, module) {}); diff --git a/test/cases/directory/v/a.js.expect b/test/cases/directory/v/a.js.expect index 17cc4ea..5d51c13 100644 --- a/test/cases/directory/v/a.js.expect +++ b/test/cases/directory/v/a.js.expect @@ -1,4 +1,4 @@ define("v/a", [ "../v", "./b" ], function(require, exports, module) { require("../v"); require("./b"); -}); \ No newline at end of file +}); diff --git a/test/cases/directory/v/b.js.expect b/test/cases/directory/v/b.js.expect index 6a90f90..1c8c478 100644 --- a/test/cases/directory/v/b.js.expect +++ b/test/cases/directory/v/b.js.expect @@ -1,3 +1,3 @@ define("v/b", [ "../v" ], function(require, exports, module) { require("../v"); -}); \ No newline at end of file +}); diff --git a/test/cases/duplicate/a.js.expect b/test/cases/duplicate/a.js.expect index 82951db..d9f4b30 100644 --- a/test/cases/duplicate/a.js.expect +++ b/test/cases/duplicate/a.js.expect @@ -1,4 +1,4 @@ define("a", [ "./b", "./d", "./c" ], function(require, exports, module) { require("./b"); require("./c"); -}); \ No newline at end of file +}); diff --git a/test/cases/expand/expand-debug.js.expect b/test/cases/expand/expand-debug.js.expect index 9f73da3..d5413d8 100644 --- a/test/cases/expand/expand-debug.js.expect +++ b/test/cases/expand/expand-debug.js.expect @@ -1 +1 @@ -define("expand-debug", [], function() {}); \ No newline at end of file +define("expand-debug", [], function() {}); diff --git a/test/cases/expand/expand-debug.js.expect.expect b/test/cases/expand/expand-debug.js.expect.expect index 9f73da3..d5413d8 100644 --- a/test/cases/expand/expand-debug.js.expect.expect +++ b/test/cases/expand/expand-debug.js.expect.expect @@ -1 +1 @@ -define("expand-debug", [], function() {}); \ No newline at end of file +define("expand-debug", [], function() {}); diff --git a/test/cases/expand/expand.js.expect b/test/cases/expand/expand.js.expect index 78e009b..2f0f5d8 100644 --- a/test/cases/expand/expand.js.expect +++ b/test/cases/expand/expand.js.expect @@ -1 +1 @@ -define("expand", [], function() {}); \ No newline at end of file +define("expand", [], function() {}); diff --git a/test/cases/expand/expand.js.expect.expect b/test/cases/expand/expand.js.expect.expect index 78e009b..2f0f5d8 100644 --- a/test/cases/expand/expand.js.expect.expect +++ b/test/cases/expand/expand.js.expect.expect @@ -1 +1 @@ -define("expand", [], function() {}); \ No newline at end of file +define("expand", [], function() {}); diff --git a/test/cases/handlebars/month-debug.handlebars.js.expect b/test/cases/handlebars/month-debug.handlebars.js.expect index ade6979..beb5c12 100644 --- a/test/cases/handlebars/month-debug.handlebars.js.expect +++ b/test/cases/handlebars/month-debug.handlebars.js.expect @@ -70,4 +70,4 @@ define("month-debug.handlebars", [ "gallery/handlebars/1.0.2/runtime-debug" ], f buffer += "\n\n"; return buffer; }); -}); \ No newline at end of file +}); diff --git a/test/cases/handlebars/month.handlebars.js.expect b/test/cases/handlebars/month.handlebars.js.expect index 95fec75..26f9b5a 100644 --- a/test/cases/handlebars/month.handlebars.js.expect +++ b/test/cases/handlebars/month.handlebars.js.expect @@ -70,4 +70,4 @@ define("month.handlebars", [ "gallery/handlebars/1.0.2/runtime" ], function(requ buffer += "\n\n"; return buffer; }); -}); \ No newline at end of file +}); diff --git a/test/cases/handlebars/simple-debug.handlebars.js.expect b/test/cases/handlebars/simple-debug.handlebars.js.expect index c79a193..c51270f 100644 --- a/test/cases/handlebars/simple-debug.handlebars.js.expect +++ b/test/cases/handlebars/simple-debug.handlebars.js.expect @@ -31,4 +31,4 @@ define("simple-debug.handlebars", [ "gallery/handlebars/1.0.2/runtime-debug" ], buffer += "\n"; return buffer; }); -}); \ No newline at end of file +}); diff --git a/test/cases/handlebars/simple.handlebars.js.expect b/test/cases/handlebars/simple.handlebars.js.expect index 82e8b8f..8885c5b 100644 --- a/test/cases/handlebars/simple.handlebars.js.expect +++ b/test/cases/handlebars/simple.handlebars.js.expect @@ -31,4 +31,4 @@ define("simple.handlebars", [ "gallery/handlebars/1.0.2/runtime" ], function(req buffer += "\n"; return buffer; }); -}); \ No newline at end of file +}); diff --git a/test/cases/hash/a-db1b8e6a.js.expect b/test/cases/hash/a-db1b8e6a.js.expect new file mode 100644 index 0000000..5683962 --- /dev/null +++ b/test/cases/hash/a-db1b8e6a.js.expect @@ -0,0 +1,4 @@ +define("family/name/a-db1b8e6a", [ "./b-4849d846", "./c/c-5bffe50a", "$", "family/bar/bar", "arale/class/foo" ], function(require, exports, module) { + require("./b-4849d846.js"); + require("arale/class/foo"); +}); diff --git a/test/cases/hash/a.js b/test/cases/hash/a.js new file mode 100644 index 0000000..d0353ba --- /dev/null +++ b/test/cases/hash/a.js @@ -0,0 +1,4 @@ +define(function(require, exports, module) { + require('./b'); + require('foo'); +}); diff --git a/test/cases/hash/a.js.expect b/test/cases/hash/a.js.expect new file mode 100644 index 0000000..7880eb8 --- /dev/null +++ b/test/cases/hash/a.js.expect @@ -0,0 +1,4 @@ +define("family/name/a", [ "./b", "./c/c", "$", "family/bar/bar", "arale/class/foo" ], function(require, exports, module) { + require("./b"); + require("arale/class/foo"); +}); diff --git a/test/cases/hash/b-4849d846.js.expect b/test/cases/hash/b-4849d846.js.expect new file mode 100644 index 0000000..d053127 --- /dev/null +++ b/test/cases/hash/b-4849d846.js.expect @@ -0,0 +1,4 @@ +define("family/name/b-4849d846", [ "./c/c-5bffe50a", "$", "family/bar/bar" ], function(require, exports, module) { + require("./c/c-5bffe50a.js"); + require("family/bar/bar"); +}); diff --git a/test/cases/hash/b.js b/test/cases/hash/b.js new file mode 100644 index 0000000..312f769 --- /dev/null +++ b/test/cases/hash/b.js @@ -0,0 +1,4 @@ +define(function(require, exports, module) { + require('./c/c'); + require('bar'); +}); diff --git a/test/cases/hash/b.js.expect b/test/cases/hash/b.js.expect new file mode 100644 index 0000000..548672e --- /dev/null +++ b/test/cases/hash/b.js.expect @@ -0,0 +1,4 @@ +define("family/name/b", [ "./c/c", "$", "family/bar/bar" ], function(require, exports, module) { + require("./c/c"); + require("family/bar/bar"); +}); diff --git a/test/cases/hash/c/c.js b/test/cases/hash/c/c.js new file mode 100644 index 0000000..3d14f0b --- /dev/null +++ b/test/cases/hash/c/c.js @@ -0,0 +1,3 @@ +define(function(require, exports, module) { + require('$'); +}); diff --git a/test/cases/id-deps-exist/a.js.expect b/test/cases/id-deps-exist/a.js.expect index 1fbc12c..6aee1fe 100644 --- a/test/cases/id-deps-exist/a.js.expect +++ b/test/cases/id-deps-exist/a.js.expect @@ -1,3 +1,3 @@ define("a", [ "arale/class/foo" ], function(require, exports, module) { require("arale/class/foo"); -}); \ No newline at end of file +}); diff --git a/test/cases/id-deps-exist/b.js.expect b/test/cases/id-deps-exist/b.js.expect index 0804a86..24bcb45 100644 --- a/test/cases/id-deps-exist/b.js.expect +++ b/test/cases/id-deps-exist/b.js.expect @@ -1,3 +1,3 @@ define("b", [ "./a" ], function(require, exports, module) { require("foo"); -}); \ No newline at end of file +}); diff --git a/test/cases/id-deps-exist/c.js.expect b/test/cases/id-deps-exist/c.js.expect index 434bc06..d0deb6c 100644 --- a/test/cases/id-deps-exist/c.js.expect +++ b/test/cases/id-deps-exist/c.js.expect @@ -1,3 +1,3 @@ define("c", [ "./a" ], function(require, exports, module) { require("foo"); -}); \ No newline at end of file +}); diff --git a/test/cases/json/simple-debug.json.js.expect b/test/cases/json/simple-debug.json.js.expect index fb91672..bc6f541 100644 --- a/test/cases/json/simple-debug.json.js.expect +++ b/test/cases/json/simple-debug.json.js.expect @@ -2,4 +2,4 @@ define("simple-debug.json", [], { foo: "bar", baz: "foobar", test: 255 -}); \ No newline at end of file +}); diff --git a/test/cases/json/simple.json.js.expect b/test/cases/json/simple.json.js.expect index 5645f24..415f088 100644 --- a/test/cases/json/simple.json.js.expect +++ b/test/cases/json/simple.json.js.expect @@ -2,4 +2,4 @@ define("simple.json", [], { foo: "bar", baz: "foobar", test: 255 -}); \ No newline at end of file +}); diff --git a/test/cases/nested/bar/a.js.expect b/test/cases/nested/bar/a.js.expect index 92c23f5..81ef367 100644 --- a/test/cases/nested/bar/a.js.expect +++ b/test/cases/nested/bar/a.js.expect @@ -1,4 +1,4 @@ define("bar/a", [ "./b", "./c", "../foo/a", "../foo/b", "../foo/c" ], function(require, exports, module) { exports.a = require("./b"); exports.c = require("./c"); -}); \ No newline at end of file +}); diff --git a/test/cases/nested/bar/b.js.expect b/test/cases/nested/bar/b.js.expect index 8bfcf77..c538bcd 100644 --- a/test/cases/nested/bar/b.js.expect +++ b/test/cases/nested/bar/b.js.expect @@ -1,3 +1,3 @@ define("bar/b", [ "./c", "../foo/a", "../foo/b", "../foo/c" ], function(require, exports, module) { exports.b = require("./c"); -}); \ No newline at end of file +}); diff --git a/test/cases/nested/bar/c.js.expect b/test/cases/nested/bar/c.js.expect index 7042707..99c92ca 100644 --- a/test/cases/nested/bar/c.js.expect +++ b/test/cases/nested/bar/c.js.expect @@ -1,3 +1,3 @@ define("bar/c", [ "../foo/a", "../foo/b", "../foo/c" ], function(require, exports, module) { exports.c = require("../foo/a"); -}); \ No newline at end of file +}); diff --git a/test/cases/nested/foo/a.js.expect b/test/cases/nested/foo/a.js.expect index 723965e..66cb656 100644 --- a/test/cases/nested/foo/a.js.expect +++ b/test/cases/nested/foo/a.js.expect @@ -1,3 +1,3 @@ define("foo/a", [ "./b", "./c" ], function(require, exports, module) { exports.a = require("./b"); -}); \ No newline at end of file +}); diff --git a/test/cases/nested/foo/b.js.expect b/test/cases/nested/foo/b.js.expect index cdd9c1e..229f243 100644 --- a/test/cases/nested/foo/b.js.expect +++ b/test/cases/nested/foo/b.js.expect @@ -1,3 +1,3 @@ define("foo/b", [ "./c" ], function(require, exports, module) { exports.b = require("./c"); -}); \ No newline at end of file +}); diff --git a/test/cases/nested/foo/c.js.expect b/test/cases/nested/foo/c.js.expect index d3572f5..85f55a8 100644 --- a/test/cases/nested/foo/c.js.expect +++ b/test/cases/nested/foo/c.js.expect @@ -1,3 +1,3 @@ define("foo/c", [], function(require, exports, module) { exports.c = "c"; -}); \ No newline at end of file +}); diff --git a/test/cases/relative/bar-debug.js.expect b/test/cases/relative/bar-debug.js.expect index 9ea8ce2..a854b84 100644 --- a/test/cases/relative/bar-debug.js.expect +++ b/test/cases/relative/bar-debug.js.expect @@ -1,3 +1,3 @@ define("bar-debug", [ "./baz/baz-debug" ], function(require, exports, module) { require("./baz/baz-debug"); -}); \ No newline at end of file +}); diff --git a/test/cases/relative/bar.js.expect b/test/cases/relative/bar.js.expect index 573de5b..8d090f8 100644 --- a/test/cases/relative/bar.js.expect +++ b/test/cases/relative/bar.js.expect @@ -1,3 +1,3 @@ define("bar", [ "./baz/baz" ], function(require, exports, module) { require("./baz/baz"); -}); \ No newline at end of file +}); diff --git a/test/cases/relative/foo-debug.js.expect b/test/cases/relative/foo-debug.js.expect index 5b36c9b..c3d11ea 100644 --- a/test/cases/relative/foo-debug.js.expect +++ b/test/cases/relative/foo-debug.js.expect @@ -1,4 +1,4 @@ define("foo-debug", [ "./bar-debug", "./baz/baz-debug" ], function(require, exports, module) { require("./bar-debug"); require("./baz/baz-debug"); -}); \ No newline at end of file +}); diff --git a/test/cases/relative/foo.js.expect b/test/cases/relative/foo.js.expect index 22cf458..ab8b7ab 100644 --- a/test/cases/relative/foo.js.expect +++ b/test/cases/relative/foo.js.expect @@ -1,4 +1,4 @@ define("foo", [ "./bar", "./baz/baz" ], function(require, exports, module) { require("./bar"); require("./baz/baz"); -}); \ No newline at end of file +}); diff --git a/test/cases/rely-arale/foo-debug.js.expect b/test/cases/rely-arale/foo-debug.js.expect index 6a5a521..adece19 100644 --- a/test/cases/rely-arale/foo-debug.js.expect +++ b/test/cases/rely-arale/foo-debug.js.expect @@ -1,3 +1,3 @@ -define("foo-debug", [ "arale-debug", "foo-debug" ], function(require, exports, module) { +define("foo-debug", [ "arale-debug", "arale/class/foo-debug" ], function(require, exports, module) { require("arale-debug"); -}); \ No newline at end of file +}); diff --git a/test/cases/rely-arale/foo.js.expect b/test/cases/rely-arale/foo.js.expect index 2644f86..9943a53 100644 --- a/test/cases/rely-arale/foo.js.expect +++ b/test/cases/rely-arale/foo.js.expect @@ -1,3 +1,3 @@ -define("foo", [ "arale", "foo" ], function(require, exports, module) { +define("foo", [ "arale", "arale/class/foo" ], function(require, exports, module) { require("arale"); -}); \ No newline at end of file +}); diff --git a/test/cases/single/bar/foo-debug.js.expect b/test/cases/single/bar/foo-debug.js.expect index 64ea609..0a1c7bb 100644 --- a/test/cases/single/bar/foo-debug.js.expect +++ b/test/cases/single/bar/foo-debug.js.expect @@ -1,3 +1,3 @@ define("bar/foo-debug", [], function(require, exports, module) { module.exports = "bar/foo"; -}); \ No newline at end of file +}); diff --git a/test/cases/single/bar/foo.js.expect b/test/cases/single/bar/foo.js.expect index 8f18ebd..2f1cc5d 100644 --- a/test/cases/single/bar/foo.js.expect +++ b/test/cases/single/bar/foo.js.expect @@ -1,3 +1,3 @@ define("bar/foo", [], function(require, exports, module) { module.exports = "bar/foo"; -}); \ No newline at end of file +}); diff --git a/test/cases/single/foo-debug.js.expect b/test/cases/single/foo-debug.js.expect index d48085a..c284d0d 100644 --- a/test/cases/single/foo-debug.js.expect +++ b/test/cases/single/foo-debug.js.expect @@ -1,3 +1,3 @@ define("foo-debug", [], function(require, exports, module) { module.exports = "foo"; -}); \ No newline at end of file +}); diff --git a/test/cases/single/foo.js.expect b/test/cases/single/foo.js.expect index e132403..8fccd5d 100644 --- a/test/cases/single/foo.js.expect +++ b/test/cases/single/foo.js.expect @@ -1,3 +1,3 @@ define("foo", [], function(require, exports, module) { module.exports = "foo"; -}); \ No newline at end of file +}); diff --git a/test/cases/style/a-debug.css.js.expect b/test/cases/style/a-debug.css.js.expect index 3a81d37..6030af6 100644 --- a/test/cases/style/a-debug.css.js.expect +++ b/test/cases/style/a-debug.css.js.expect @@ -1,3 +1,3 @@ define("arale/widget/1.0.0/a-debug.css", [], function() { seajs.importStyle('@charset "utf-8";.arale-widget-1_0_0 body{_color:red}.arale-widget-1_0_0 .content li,.arale-widget-1_0_0 .content a{*display:inline-block}@media screen{.arale-widget-1_0_0 body{font-size:13px}.arale-widget-1_0_0 .content li{background-color:#fff}}@keyframes foo{0%{top:0;left:0}100%{top:100px;left:100%}}:root .arale-widget-1_0_0 .content{font-size:12px}'); -}); \ No newline at end of file +}); diff --git a/test/cases/style/a-debug.js.expect b/test/cases/style/a-debug.js.expect index ce96894..6b3f11d 100644 --- a/test/cases/style/a-debug.js.expect +++ b/test/cases/style/a-debug.js.expect @@ -1,3 +1,3 @@ define("arale/widget/1.0.0/a-debug", [], function(require, exports, module) { exports.foo = function() {}; -module.exports.outerBoxClass="arale-widget-1_0_0";}); \ No newline at end of file +module.exports.outerBoxClass="arale-widget-1_0_0";}); diff --git a/test/cases/style/a.css.js.expect b/test/cases/style/a.css.js.expect index e7f92ba..9067c9e 100644 --- a/test/cases/style/a.css.js.expect +++ b/test/cases/style/a.css.js.expect @@ -1,3 +1,3 @@ define("arale/widget/1.0.0/a.css", [], function() { seajs.importStyle('@charset "utf-8";.arale-widget-1_0_0 body{_color:red}.arale-widget-1_0_0 .content li,.arale-widget-1_0_0 .content a{*display:inline-block}@media screen{.arale-widget-1_0_0 body{font-size:13px}.arale-widget-1_0_0 .content li{background-color:#fff}}@keyframes foo{0%{top:0;left:0}100%{top:100px;left:100%}}:root .arale-widget-1_0_0 .content{font-size:12px}'); -}); \ No newline at end of file +}); diff --git a/test/cases/style/a.js.expect b/test/cases/style/a.js.expect index 9892f3b..18ec5c3 100644 --- a/test/cases/style/a.js.expect +++ b/test/cases/style/a.js.expect @@ -1,3 +1,3 @@ define("arale/widget/1.0.0/a", [], function(require, exports, module) { exports.foo = function() {}; -module.exports.outerBoxClass="arale-widget-1_0_0";}); \ No newline at end of file +module.exports.outerBoxClass="arale-widget-1_0_0";}); diff --git a/test/cases/text!/a.js.expect b/test/cases/text!/a.js.expect index 3c1dbb5..b2c1744 100644 --- a/test/cases/text!/a.js.expect +++ b/test/cases/text!/a.js.expect @@ -1,3 +1,3 @@ define("a", [ "text!/templates/login" ], function() { require("text!/templates/login"); -}); \ No newline at end of file +}); diff --git a/test/cases/text/foo-debug.html.js.expect b/test/cases/text/foo-debug.html.js.expect index bcbcbc6..da47cad 100644 --- a/test/cases/text/foo-debug.html.js.expect +++ b/test/cases/text/foo-debug.html.js.expect @@ -1 +1 @@ -define("foo-debug.html", [], "
foo / bar \\ baz
\n"); \ No newline at end of file +define("foo-debug.html", [], "
foo / bar \\ baz
\n"); diff --git a/test/cases/text/foo.html.js.expect b/test/cases/text/foo.html.js.expect index b21ec3a..c2dfea2 100644 --- a/test/cases/text/foo.html.js.expect +++ b/test/cases/text/foo.html.js.expect @@ -1 +1 @@ -define("foo.html", [], "
foo / bar \\ baz
\n"); \ No newline at end of file +define("foo.html", [], "
foo / bar \\ baz
\n"); diff --git a/test/cases/tpl/month-debug.tpl.js.expect b/test/cases/tpl/month-debug.tpl.js.expect index 2883aad..998c62b 100644 --- a/test/cases/tpl/month-debug.tpl.js.expect +++ b/test/cases/tpl/month-debug.tpl.js.expect @@ -1 +1 @@ -define("month-debug.tpl", [], '
\n{{#each items}}\n\n{{/each}}\n
\n'); \ No newline at end of file +define("month-debug.tpl", [], '
\n{{#each items}}\n\n{{/each}}\n
\n'); diff --git a/test/cases/tpl/month.tpl.js.expect b/test/cases/tpl/month.tpl.js.expect index 74b2981..d86e81c 100644 --- a/test/cases/tpl/month.tpl.js.expect +++ b/test/cases/tpl/month.tpl.js.expect @@ -1 +1 @@ -define("month.tpl", [], '
\n{{#each items}}\n\n{{/each}}\n
\n'); \ No newline at end of file +define("month.tpl", [], '
\n{{#each items}}\n\n{{/each}}\n
\n'); diff --git a/test/cases/tpl/simple-debug.tpl.js.expect b/test/cases/tpl/simple-debug.tpl.js.expect index 1fdc5f1..4d20564 100644 --- a/test/cases/tpl/simple-debug.tpl.js.expect +++ b/test/cases/tpl/simple-debug.tpl.js.expect @@ -1 +1 @@ -define("simple-debug.tpl", [], "{{#each items}}\n

{{_ this}}

\n{{/each}}\n"); \ No newline at end of file +define("simple-debug.tpl", [], "{{#each items}}\n

{{_ this}}

\n{{/each}}\n"); diff --git a/test/cases/tpl/simple.tpl.js.expect b/test/cases/tpl/simple.tpl.js.expect index 78b1efa..64cb8fb 100644 --- a/test/cases/tpl/simple.tpl.js.expect +++ b/test/cases/tpl/simple.tpl.js.expect @@ -1 +1 @@ -define("simple.tpl", [], "{{#each items}}\n

{{_ this}}

\n{{/each}}\n"); \ No newline at end of file +define("simple.tpl", [], "{{#each items}}\n

{{_ this}}

\n{{/each}}\n"); diff --git a/test/transport.js b/test/transport.js index a81d49d..d903b21 100644 --- a/test/transport.js +++ b/test/transport.js @@ -18,9 +18,9 @@ describe('file', function() { if (files.length) { it('should test ' + dir, function() { files.forEach(function(file) { - var expect = fs.readFileSync(path.join(base, dir, file)) + var actual = fs.readFileSync(path.join(base, dir, file)) .toString().replace(/\r|\\r/g, ''); - var actual= fs.readFileSync(path.join(expected, dir, file.replace(/\.expect$/, ''))) + var expect = fs.readFileSync(path.join(expected, dir, file.replace(/\.expect$/, ''))) .toString().replace(/\r|\\r/g, ''); expect.should.eql(actual); }); @@ -43,4 +43,4 @@ function readDirs(dir) { } }); return result; -} \ No newline at end of file +} From e53d68457ec9aa0b144e2bcde42e51be66c15e86 Mon Sep 17 00:00:00 2001 From: popomore Date: Mon, 29 Dec 2014 20:46:46 +0800 Subject: [PATCH 02/14] fix(script) return when it is not javascript --- tasks/lib/script.js | 21 +++++++++++++++++++-- 1 file changed, 19 insertions(+), 2 deletions(-) diff --git a/tasks/lib/script.js b/tasks/lib/script.js index 5650ece..9689a81 100644 --- a/tasks/lib/script.js +++ b/tasks/lib/script.js @@ -2,6 +2,7 @@ exports.init = function(grunt) { var crypto = require('crypto'); var path = require('path'); var join = path.join; + var extname = path.extname; var ast = require('cmd-util').ast; var iduri = require('cmd-util').iduri; var relative = require('relative'); @@ -81,7 +82,7 @@ exports.init = function(grunt) { function addDebug(v) { if (v.id) v = v.id; - var ext = path.extname(v); + var ext = extname(v); if (ext && options.parsers[ext]) { return v.replace(new RegExp('\\' + ext + '$'), '-debug' + ext); } else { @@ -92,7 +93,7 @@ exports.init = function(grunt) { function addHash(v) { if (!v.hash) return v.id; - var ext = path.extname(v.id); + var ext = extname(v.id); if (ext && options.parsers[ext]) { return v.id.replace(new RegExp('\\' + ext + '$'), '-' + v.hash + ext); } else { @@ -134,6 +135,10 @@ exports.init = function(grunt) { return []; } + if (extname(filepath) !== '.js') { + return []; + } + var parsed, data = grunt.file.read(filepath); try { parsed = ast.parseFirst(data); @@ -238,6 +243,18 @@ exports.init = function(grunt) { if (!grunt.file.exists(path)) return; var astCache, deps, contents = grunt.file.read(path); + if (extname(path) !== '.js') { + return fileCache[path] = { + id: undefined, + dependencies: [], + depMap: {}, + depsSpecified: false, + contents: contents, + path: path, + hash: md5(contents, []) + }; + } + try { astCache = ast.getAst(contents); } catch(e) { From 152c8c9be37a37122eb616882ddbf6589992acbc Mon Sep 17 00:00:00 2001 From: popomore Date: Tue, 30 Dec 2014 02:19:32 +0800 Subject: [PATCH 03/14] feat(transport) tpl handlebars json html css2js support hash --- .jshintrc | 4 +- Gruntfile.js | 22 +- index.js | 3 + package.json | 7 +- tasks/lib/common.js | 58 +++ tasks/lib/css2js.js | 42 ++ tasks/lib/handlebars.js | 60 +++ tasks/lib/json.js | 48 +- tasks/lib/script.js | 17 +- tasks/lib/template.js | 135 +---- tasks/lib/text.js | 69 +-- tasks/lib/util.js | 13 + tasks/transport.js | 7 +- test/cases/project/a.css | 5 + test/cases/project/a.handlebars | 1 + test/cases/project/a.html | 5 + test/cases/project/a.js | 8 + test/cases/project/a.json | 0 test/cases/project/a.tpl | 0 test/cases/project/b.css | 5 + test/cases/project/b.js | 3 + .../alice/list/1.0.1/list-debug.css | 70 +++ .../sea-modules/alice/list/1.0.1/list.css | 1 + .../sea-modules/alice/list/1.0.1/package.json | 24 + .../arale/base/1.1.1/base-debug.js | 462 ++++++++++++++++++ .../sea-modules/arale/base/1.1.1/base.js | 1 + .../sea-modules/arale/base/1.1.1/package.json | 27 + .../arale/class/1.1.0/class-debug.js | 148 ++++++ .../sea-modules/arale/class/1.1.0/class.js | 1 + .../arale/class/1.1.0/package.json | 23 + .../arale/events/1.1.0/events-debug.js | 124 +++++ .../sea-modules/arale/events/1.1.0/events.js | 1 + .../arale/events/1.1.0/package.json | 23 + 33 files changed, 1168 insertions(+), 249 deletions(-) create mode 100644 tasks/lib/common.js create mode 100644 tasks/lib/css2js.js create mode 100644 tasks/lib/handlebars.js create mode 100644 tasks/lib/util.js create mode 100644 test/cases/project/a.css create mode 100644 test/cases/project/a.handlebars create mode 100644 test/cases/project/a.html create mode 100644 test/cases/project/a.js create mode 100644 test/cases/project/a.json create mode 100644 test/cases/project/a.tpl create mode 100644 test/cases/project/b.css create mode 100644 test/cases/project/b.js create mode 100644 test/cases/project/sea-modules/alice/list/1.0.1/list-debug.css create mode 100644 test/cases/project/sea-modules/alice/list/1.0.1/list.css create mode 100644 test/cases/project/sea-modules/alice/list/1.0.1/package.json create mode 100644 test/cases/project/sea-modules/arale/base/1.1.1/base-debug.js create mode 100644 test/cases/project/sea-modules/arale/base/1.1.1/base.js create mode 100644 test/cases/project/sea-modules/arale/base/1.1.1/package.json create mode 100644 test/cases/project/sea-modules/arale/class/1.1.0/class-debug.js create mode 100644 test/cases/project/sea-modules/arale/class/1.1.0/class.js create mode 100644 test/cases/project/sea-modules/arale/class/1.1.0/package.json create mode 100644 test/cases/project/sea-modules/arale/events/1.1.0/events-debug.js create mode 100644 test/cases/project/sea-modules/arale/events/1.1.0/events.js create mode 100644 test/cases/project/sea-modules/arale/events/1.1.0/package.json diff --git a/.jshintrc b/.jshintrc index 2fb199d..7d55548 100644 --- a/.jshintrc +++ b/.jshintrc @@ -4,8 +4,8 @@ "latedef": false, "newcap": true, "quotmark": false, - "undef": false, - "unused": false, + "undef": true, + "unused": true, "trailing": true, "lastsemic": true, "asi": false, diff --git a/Gruntfile.js b/Gruntfile.js index 8939598..b2e6d17 100644 --- a/Gruntfile.js +++ b/Gruntfile.js @@ -9,8 +9,8 @@ module.exports = function(grunt) { - var style = require('./').style.init(grunt); - var css2jsParser = style.css2jsParser; + var css2js = require('./').css2js.init(grunt); + var css2jsParser = css2js.cssParser; var jsParser = require('./').script.init(grunt).jsParser; grunt.initConfig({ @@ -270,6 +270,24 @@ module.exports = function(grunt) { src: '*.js', dest: 'test/expected/hash' }] + }, + + project: { + options: { + paths: ['test/cases/project/sea-modules'], + alias: { + 'list': 'alice/list/1.0.1/list.css', + 'base': 'arale/base/1.1.1/base' + }, + idleading: 'family/name/', + hash: true + }, + files: [{ + expand: true, + cwd: 'test/cases/project', + src: '*.*', + dest: 'test/expected/project' + }] } }, diff --git a/index.js b/index.js index 1ee4e89..fefe13d 100644 --- a/index.js +++ b/index.js @@ -1,4 +1,7 @@ exports.style = require('./tasks/lib/style'); exports.script = require('./tasks/lib/script'); +exports.css2js = require('./tasks/lib/css2js'); exports.template = require('./tasks/lib/template'); exports.text = require('./tasks/lib/text'); +exports.handlebars = require('./tasks/lib/handlebars'); +exports.json = require('./tasks/lib/json'); diff --git a/package.json b/package.json index bb88779..4a32b2d 100644 --- a/package.json +++ b/package.json @@ -29,10 +29,11 @@ "dependencies": { "clean-css": "~1.0.1", "cmd-util": "~0.3.5", - "handlebars": "1.0.11", - "uglify-js": "~2.2.5", "css": "~1.4.0", - "relative": "~1.2.0" + "css2str": "~0.1.1", + "handlebars": "1.0.11", + "relative": "~1.2.0", + "uglify-js": "~2.2.5" }, "devDependencies": { "grunt-contrib-jshint": "~0.1.1", diff --git a/tasks/lib/common.js b/tasks/lib/common.js new file mode 100644 index 0000000..ebe183f --- /dev/null +++ b/tasks/lib/common.js @@ -0,0 +1,58 @@ +exports.init = function(grunt, options) { + var format = require('util').format; + var ast = require('cmd-util').ast; + var md5 = require('./util').md5; + var type = options.type; + var factoryParser = options.factoryParser; + var depParser = options.depParser; + var regType = new RegExp('.' + type + '$'); + var retTypeJs = new RegExp('.' + type + '.js$'); + + var exports = {}; + + exports[type + 'Parser'] = function(fileObj, options) { + var dest, id = unixy(options.idleading + fileObj.name.replace(/\.js$/, '')); + var data = fileObj.srcData || grunt.file.read(fileObj.src); + var deps = depParser ? depParser(data, options) : ''; + var factory = factoryParser ? factoryParser(data, options) : '{}'; + var file = { + contents: format('define("%s", [%s], %s)', id, deps, factory), + dest: fileObj.dest + '.js' + }; + + // create .{type}.js + data = ast.modify(file.contents, { + id: id + }).print_to_string(options.uglify); + writeFile(data, file.dest); + + // create debug file xxx-debug.{type}.js + if (options.debug) { + dest = file.dest.replace(retTypeJs, '-debug.' + type + '.js'); + data = ast.modify(file.contents, { + id: id.replace(regType, '-debug.' + type) + }).print_to_string(options.uglify); + writeFile(data, dest); + } + + // create hash file xxx-{hash}.{type}.js + if (options.hash) { + var hash = md5(factory); + dest = file.dest.replace(retTypeJs, '-' + hash + '.' + type + '.js'); + data = ast.modify(file.contents, { + id: id.replace(regType, '-' + hash + '.' + type) + }).print_to_string(options.uglify); + writeFile(data, dest); + } + }; + return exports; + + function writeFile(data, dest) { + grunt.log.writeln('transport ' + dest + ' created'); + grunt.file.write(dest, data + '\n'); + } +}; + +function unixy(uri) { + return uri.replace(/\\/g, '/'); +} diff --git a/tasks/lib/css2js.js b/tasks/lib/css2js.js new file mode 100644 index 0000000..22209d1 --- /dev/null +++ b/tasks/lib/css2js.js @@ -0,0 +1,42 @@ +var format = require('util').format; +var css2str = require('css2str'); +var cleancss = require('clean-css'); +var commonParser = require('./common'); + +exports.init = function(grunt) { + return commonParser.init(grunt, { + type: 'css', + factoryParser: css2js + }); +}; + +function css2js(code, options) { + // if outside css modules, fileObj would be undefined + // then dont add styleBox + var opt = {}; + if (options.styleBox === true) { + // ex. arale/widget/1.0.0/ => arale-widget-1_0_0 + var styleId = unixy((options || {}).idleading || '') + .replace(/\/$/, '') + .replace(/\//g, '-') + .replace(/\./g, '_'); + opt.prefix = ['.', styleId, ' '].join(''); + } + + code = css2str(code, opt); + + // remove comment and format + code = cleancss.process(code, { + keepSpecialComments: 0, + removeEmpty: true + }); + + // transform css to js + // spmjs/spm#581 + var template = 'function() {seajs.importStyle("%s")}'; + return format(template, code); +} + +function unixy(uri) { + return uri.replace(/\\/g, '/'); +} diff --git a/tasks/lib/handlebars.js b/tasks/lib/handlebars.js new file mode 100644 index 0000000..8dd426f --- /dev/null +++ b/tasks/lib/handlebars.js @@ -0,0 +1,60 @@ +var commonParser = require('./common'); + +exports.init = function(grunt) { + var format = require('util').format; + var handlebars = require('handlebars'); + + return commonParser.init(grunt, { + type: 'handlebars', + depParser: function(data, options) { + // handlebars alias + return '"' + options.handlebars.id + '"'; + }, + factoryParser: function(data, options) { + patchHandlebars(handlebars); + var code = handlebars.precompile(data, options.handlebars); + var template = [ + 'function(require, exports, module) {', + 'var Handlebars = require("%s");', + 'var template = Handlebars.template;', + 'module.exports = template(%s);', + '}' + ].join('\n'); + return format(template, options.handlebars.id, code); + } + }); +}; + +// patch for handlebars +function patchHandlebars(Handlebars) { + Handlebars.JavaScriptCompiler.prototype.preamble = function() { + var out = []; + + if (!this.isChild) { + var namespace = this.namespace; + // patch for handlebars + var copies = [ + "helpers = helpers || {};", + "for (var key in " + namespace + ".helpers) {", + " helpers[key] = helpers[key] || " + namespace + ".helpers[key];", + "}" + ].join('\n'); + if (this.environment.usePartial) { copies = copies + " partials = partials || " + namespace + ".partials;"; } + if (this.options.data) { copies = copies + " data = data || {};"; } + out.push(copies); + } else { + out.push(''); + } + + if (!this.environment.isSimple) { + out.push(", buffer = " + this.initializeBuffer()); + } else { + out.push(""); + } + + // track the last context pushed into place to allow skipping the + // getContext opcode when it would be a noop + this.lastContext = 0; + this.source = out; + }; +} diff --git a/tasks/lib/json.js b/tasks/lib/json.js index 510b7b3..0760f00 100644 --- a/tasks/lib/json.js +++ b/tasks/lib/json.js @@ -1,44 +1,10 @@ -exports.init = function(grunt) { - - var path = require('path'); - var format = require('util').format; - var iduri = require('cmd-util').iduri; - var ast = require('cmd-util').ast; - - var exports = {}; - - exports.jsonParser = function(fileObj, options) { - var dest = fileObj.dest + '.js'; - grunt.log.verbose.writeln('Transport ' + fileObj.src + ' -> ' + dest); - - var id = unixy(options.idleading + fileObj.name.replace(/\.js$/, '')); - var data = fileObj.srcData || grunt.file.read(fileObj.src); - var code = format('define("%s", [], %s)', id, data); - var astCache = ast.getAst(code); +var commonParser = require('./common'); - data = astCache.print_to_string(options.uglify) + '\n'; - grunt.file.write(dest, data); - - // create debug file - if (!options.debug) { - return; +exports.init = function(grunt) { + return commonParser.init(grunt, { + type: 'json', + factoryParser: function(data) { + return data || '{}'; } - dest = dest.replace(/\.json\.js$/, '-debug.json.js'); - - astCache = ast.modify(astCache, function(v) { - var ext = path.extname(v); - if (ext && options.parsers[ext]) { - return v.replace(new RegExp('\\' + ext + '$'), '-debug' + ext); - } else { - return v + '-debug'; - } - }); - data = astCache.print_to_string(options.uglify) + '\n'; - grunt.file.write(dest, data); - }; - return exports; + }); }; - -function unixy(uri) { - return uri.replace(/\\/g, '/'); -} diff --git a/tasks/lib/script.js b/tasks/lib/script.js index 9689a81..2151407 100644 --- a/tasks/lib/script.js +++ b/tasks/lib/script.js @@ -1,12 +1,11 @@ exports.init = function(grunt) { - var crypto = require('crypto'); + var path = require('path'); - var join = path.join; var extname = path.extname; var ast = require('cmd-util').ast; var iduri = require('cmd-util').iduri; var relative = require('relative'); - var _ = grunt.util._; + var md5 = require('./util').md5; return { jsParser: jsParser @@ -102,6 +101,7 @@ exports.init = function(grunt) { } function writeFile(data, dest) { + grunt.log.writeln('transport ' + dest + ' created'); grunt.file.write(dest, addOuterBoxClass(data + '\n', options)); } @@ -304,14 +304,5 @@ exports.init = function(grunt) { } } - function md5(contents, deps) { - contents = deps.map(function(depFile) { - return depFile.contents || ''; - }) + contents; - return crypto - .createHash('md5') - .update(contents, 'utf8') - .digest('hex') - .slice(0, 8); - } + }; diff --git a/tasks/lib/template.js b/tasks/lib/template.js index 74e6d93..b1cc038 100644 --- a/tasks/lib/template.js +++ b/tasks/lib/template.js @@ -1,131 +1,10 @@ -exports.init = function(grunt) { - - var path = require('path'); - var format = require('util').format; - var iduri = require('cmd-util').iduri; - var ast = require('cmd-util').ast; - - var exports = {}; - - exports.tplParser = function(fileObj, options) { - var dest = fileObj.dest + '.js'; - grunt.log.verbose.writeln('Transport ' + fileObj.src + ' -> ' + dest); - - var id = unixy(options.idleading + fileObj.name.replace(/\.js$/, '')); - var data = fileObj.srcData || grunt.file.read(fileObj.src); - var code = format('define("%s", [], "%s")', id, data.replace(/\"/g, '\\\"')); - var astCache = ast.getAst(code); - - data = astCache.print_to_string(options.uglify) + '\n'; - grunt.file.write(dest, data); - - // create debug file - if (!options.debug) { - return; - } - dest = dest.replace(/\.tpl\.js$/, '-debug.tpl.js'); - - astCache = ast.modify(astCache, function(v) { - var ext = path.extname(v); - if (ext && options.parsers[ext]) { - return v.replace(new RegExp('\\' + ext + '$'), '-debug' + ext); - } else { - return v + '-debug'; - } - }); - data = astCache.print_to_string(options.uglify) + '\n'; - grunt.file.write(dest, data); - }; - - exports.handlebarsParser = function(fileObj, options) { - var dest = fileObj.dest + '.js'; - grunt.log.verbose.writeln('Transport ' + fileObj.src + ' -> ' + dest); - - var handlebars = require('handlebars'); - - // id for template - var id = unixy(options.idleading + fileObj.name.replace(/\.js$/, '')); - - // handlebars alias - var alias = options.handlebars.id; - - var template = [ - 'define("%s", ["%s"], function(require, exports, module) {', - 'var Handlebars = require("%s");', - 'var template = Handlebars.template;', - 'module.exports = template(', - '%s', - ');', - '})' - ].join('\n'); - - var data = fileObj.srcData || grunt.file.read(fileObj.src); +var commonParser = require('./common'); - patchHandlebars(handlebars); - var code = handlebars.precompile(data, options.handlebars); - - var ret = format(template, id, alias, alias, code); - var astCache = ast.getAst(ret); - - data = astCache.print_to_string(options.uglify) + '\n'; - grunt.file.write(dest, data); - - // create debug file - if (!options.debug) { - return; +exports.init = function(grunt) { + return commonParser.init(grunt, { + type: 'tpl', + factoryParser: function(data) { + return '"' + data.replace(/\"/g, '\\\"') + '"'; } - dest = dest.replace(/\.handlebars\.js$/, '-debug.handlebars.js'); - - astCache = ast.modify(astCache, function(v) { - var ext = path.extname(v); - if (ext && options.parsers[ext]) { - return v.replace(new RegExp('\\' + ext + '$'), '-debug' + ext); - } else { - return v + '-debug'; - } - }); - data = astCache.print_to_string(options.uglify) + '\n'; - grunt.file.write(dest, data); - }; - - return exports; + }); }; - - -// patch for handlebars -function patchHandlebars(Handlebars) { - Handlebars.JavaScriptCompiler.prototype.preamble = function() { - var out = []; - - if (!this.isChild) { - var namespace = this.namespace; - // patch for handlebars - var copies = [ - "helpers = helpers || {};", - "for (var key in " + namespace + ".helpers) {", - " helpers[key] = helpers[key] || " + namespace + ".helpers[key];", - "}" - ].join('\n'); - if (this.environment.usePartial) { copies = copies + " partials = partials || " + namespace + ".partials;"; } - if (this.options.data) { copies = copies + " data = data || {};"; } - out.push(copies); - } else { - out.push(''); - } - - if (!this.environment.isSimple) { - out.push(", buffer = " + this.initializeBuffer()); - } else { - out.push(""); - } - - // track the last context pushed into place to allow skipping the - // getContext opcode when it would be a noop - this.lastContext = 0; - this.source = out; - }; -} - -function unixy(uri) { - return uri.replace(/\\/g, '/'); -} diff --git a/tasks/lib/text.js b/tasks/lib/text.js index d3f2e45..ad68e88 100644 --- a/tasks/lib/text.js +++ b/tasks/lib/text.js @@ -1,61 +1,18 @@ -var path = require('path'); -var format = require('util').format; +var commonParser = require('./common'); exports.init = function(grunt) { - var ast = require('cmd-util').ast; - var iduri = require('cmd-util').iduri; - - var exports = {}; - - exports.html2jsParser = function(fileObj, options) { - // don't transport debug html files - if (/\-debug\.html/.test(fileObj.src)) return; - - grunt.log.verbose.writeln('Transport ' + fileObj.src + ' -> ' + fileObj.dest); - // transport html to js - var data = fileObj.srcData || grunt.file.read(fileObj.src); - var id = unixy(options.idleading + fileObj.name.replace(/\.js$/, '')); - - data = html2js(data, id); - data = ast.getAst(data).print_to_string(options.uglify) + '\n'; - var dest = fileObj.dest + '.js'; - grunt.file.write(dest, data); - - if (!options.debug) { - return; - } - dest = dest.replace(/\.html\.js$/, '-debug.html.js'); - - data = ast.modify(data, function(v) { - var ext = path.extname(v); - if (ext && options.parsers[ext]) { - return v.replace(new RegExp('\\' + ext + '$'), '-debug' + ext); - } else { - return v + '-debug'; - } - }); - data = data.print_to_string(options.uglify) + '\n'; - grunt.file.write(dest, data); - }; - - return exports; + return commonParser.init(grunt, { + type: 'html', + factoryParser: getCode + }); }; - -// helpers -function html2js(code, id) { - var tpl = 'define("%s", [], "%s");'; - - code = code.split(/\r\n|\r|\n/).map(function(line) { - return line.replace(/\\/g, '\\\\'); - }).join('\n'); - - code = format(tpl, id, code.replace(/\"/g, '\\\"')); - return code; +function getCode(data) { + data = data.split(/\r\n|\r|\n/) + .map(function(line) { + return line.replace(/\\/g, '\\\\'); + }) + .join('\n') + .replace(/\"/g, '\\\"'); + return '"' + data + '"'; } - -function unixy(uri) { - return uri.replace(/\\/g, '/'); -} - -exports.html2js = html2js; diff --git a/tasks/lib/util.js b/tasks/lib/util.js new file mode 100644 index 0000000..d1d930e --- /dev/null +++ b/tasks/lib/util.js @@ -0,0 +1,13 @@ +var crypto = require('crypto'); + +exports.md5 = function md5(contents, deps) { + if (!deps) deps = []; + contents = deps.map(function(depFile) { + return depFile.contents || ''; + }) + contents; + return crypto + .createHash('md5') + .update(contents, 'utf8') + .digest('hex') + .slice(0, 8); +}; diff --git a/tasks/transport.js b/tasks/transport.js index b12bbb7..d3543f9 100644 --- a/tasks/transport.js +++ b/tasks/transport.js @@ -8,12 +8,12 @@ module.exports = function(grunt) { var path = require('path'); - var cmd = require('cmd-util'); var text = require('./lib/text').init(grunt); var script = require('./lib/script').init(grunt); var style = require('./lib/style').init(grunt); var template = require('./lib/template').init(grunt); + var handlebars = require('./lib/handlebars').init(grunt); var json = require('./lib/json').init(grunt); grunt.registerMultiTask('transport', 'Transport everything into cmd.', function() { @@ -37,10 +37,10 @@ module.exports = function(grunt) { parsers: { '.js': [script.jsParser], '.css': [style.cssParser], - '.html': [text.html2jsParser], + '.html': [text.htmlParser], '.json': [json.jsonParser], '.tpl': [template.tplParser], - '.handlebars': [template.handlebarsParser] + '.handlebars': [handlebars.handlebarsParser] }, // for handlebars @@ -96,6 +96,5 @@ module.exports = function(grunt) { count++; }); - grunt.log.writeln('transport ' + count.toString().cyan + ' files'); }); }; diff --git a/test/cases/project/a.css b/test/cases/project/a.css new file mode 100644 index 0000000..5e5e336 --- /dev/null +++ b/test/cases/project/a.css @@ -0,0 +1,5 @@ +@import "./b.css"; + +ul { + margin: 0; +} diff --git a/test/cases/project/a.handlebars b/test/cases/project/a.handlebars new file mode 100644 index 0000000..52e672a --- /dev/null +++ b/test/cases/project/a.handlebars @@ -0,0 +1 @@ +
{{{data}}}
diff --git a/test/cases/project/a.html b/test/cases/project/a.html new file mode 100644 index 0000000..e69de94 --- /dev/null +++ b/test/cases/project/a.html @@ -0,0 +1,5 @@ +
+
    +
  • a
  • +
      +
diff --git a/test/cases/project/a.js b/test/cases/project/a.js new file mode 100644 index 0000000..7c10843 --- /dev/null +++ b/test/cases/project/a.js @@ -0,0 +1,8 @@ +define(function(require) { + require('./a.handlebars'); + require('./a.json'); + require('./a.html'); + require('./a.tpl'); + require('./a.css'); + require('./b'); +}); diff --git a/test/cases/project/a.json b/test/cases/project/a.json new file mode 100644 index 0000000..e69de29 diff --git a/test/cases/project/a.tpl b/test/cases/project/a.tpl new file mode 100644 index 0000000..e69de29 diff --git a/test/cases/project/b.css b/test/cases/project/b.css new file mode 100644 index 0000000..1c544ed --- /dev/null +++ b/test/cases/project/b.css @@ -0,0 +1,5 @@ +@import "list" + +html { + margin: 0; +} diff --git a/test/cases/project/b.js b/test/cases/project/b.js new file mode 100644 index 0000000..1a6e688 --- /dev/null +++ b/test/cases/project/b.js @@ -0,0 +1,3 @@ +define(function(require) { + require('base'); +}); diff --git a/test/cases/project/sea-modules/alice/list/1.0.1/list-debug.css b/test/cases/project/sea-modules/alice/list/1.0.1/list-debug.css new file mode 100644 index 0000000..b76f2c0 --- /dev/null +++ b/test/cases/project/sea-modules/alice/list/1.0.1/list-debug.css @@ -0,0 +1,70 @@ +/*! define alice/list/1.0.1/list-debug.css */ +/* alice.list 样式模块 */ + +.ui-list { + margin: 0; + padding: 10px; + list-style: square inside; +} +/* 默认有方角 */ +.ui-list-item { + font-size: 9px; + line-height: 20px; +} +.ui-list-item a { + line-height: 20px; + text-decoration: none; + color: #08c; +} + +.ui-list-item a, +.ui-list-item span.ui-list-item-text { + font-size: 12px; + vertical-align: middle; +} + +.ui-list-item a:hover { + text-decoration: underline; +} + +/* 灰色小方角 */ +.ui-list-gray .ui-list-item { + color: #808080; +} + +/* 灰色小方角+灰色链接 */ +.ui-list-graylink .ui-list-item { + color: #808080; +} + +.ui-list-graylink .ui-list-item a { + color: #666; +} + +/* 没有小图标的 */ +.ui-list-nosquare { + list-style: none; +} + +/* ui-dlist */ + +.ui-dlist { + display: inline-block; + color: #808080; + font-size: 12px; + line-height: 2.2; +} + +.ui-dlist-tit { + float: left; + width: 20%;/* 默认值, 具体根据视觉可改 */ + text-align: right; + margin: 0; +} + +.ui-dlist-det { + float: left; + width: 80%;/* 默认值,具体根据视觉可改 */ + text-align: left; + margin: 0; +} diff --git a/test/cases/project/sea-modules/alice/list/1.0.1/list.css b/test/cases/project/sea-modules/alice/list/1.0.1/list.css new file mode 100644 index 0000000..cad8f96 --- /dev/null +++ b/test/cases/project/sea-modules/alice/list/1.0.1/list.css @@ -0,0 +1 @@ +.ui-list{margin:0;padding:10px;list-style:square inside}.ui-list-item{font-size:9px;line-height:20px}.ui-list-item a{line-height:20px;text-decoration:none;color:#08c}.ui-list-item a,.ui-list-item span.ui-list-item-text{font-size:12px;vertical-align:middle}.ui-list-item a:hover{text-decoration:underline}.ui-list-gray .ui-list-item{color:gray}.ui-list-graylink .ui-list-item{color:gray}.ui-list-graylink .ui-list-item a{color:#666}.ui-list-nosquare{list-style:none}.ui-dlist{display:inline-block;color:gray;font-size:12px;line-height:2.2}.ui-dlist-tit{float:left;width:20%;text-align:right;margin:0}.ui-dlist-det{float:left;width:80%;text-align:left;margin:0} diff --git a/test/cases/project/sea-modules/alice/list/1.0.1/package.json b/test/cases/project/sea-modules/alice/list/1.0.1/package.json new file mode 100644 index 0000000..8924252 --- /dev/null +++ b/test/cases/project/sea-modules/alice/list/1.0.1/package.json @@ -0,0 +1,24 @@ +{ + "name": "list", + "version": "1.0.1", + "family": "alice", + "description": "通用列表样式。", + "keywords": [ + "列表" + ], + "homepage": "http://aliceui.org/list", + "author": "", + "repository": { + "type": "git", + "url": "https://github.com/aliceui/list" + }, + "bugs": { + "url": "https://github.com/aliceui/list/issues" + }, + "license": "MIT", + "spm": { + "alias": { + }, + "output": ["list.css"] + } +} diff --git a/test/cases/project/sea-modules/arale/base/1.1.1/base-debug.js b/test/cases/project/sea-modules/arale/base/1.1.1/base-debug.js new file mode 100644 index 0000000..04ef152 --- /dev/null +++ b/test/cases/project/sea-modules/arale/base/1.1.1/base-debug.js @@ -0,0 +1,462 @@ +define("arale/base/1.1.1/base-debug", [ "arale/class/1.1.0/class-debug", "arale/events/1.1.0/events-debug", "./aspect-debug", "./attribute-debug" ], function(require, exports, module) { + // Base + // --------- + // Base 是一个基础类,提供 Class、Events、Attrs 和 Aspect 支持。 + var Class = require("arale/class/1.1.0/class-debug"); + var Events = require("arale/events/1.1.0/events-debug"); + var Aspect = require("./aspect-debug"); + var Attribute = require("./attribute-debug"); + module.exports = Class.create({ + Implements: [ Events, Aspect, Attribute ], + initialize: function(config) { + this.initAttrs(config); + // Automatically register `this._onChangeAttr` method as + // a `change:attr` event handler. + parseEventsFromInstance(this, this.attrs); + }, + destroy: function() { + this.off(); + for (var p in this) { + if (this.hasOwnProperty(p)) { + delete this[p]; + } + } + // Destroy should be called only once, generate a fake destroy after called + // https://github.com/aralejs/widget/issues/50 + this.destroy = function() {}; + } + }); + function parseEventsFromInstance(host, attrs) { + for (var attr in attrs) { + if (attrs.hasOwnProperty(attr)) { + var m = "_onChange" + ucfirst(attr); + if (host[m]) { + host.on("change:" + attr, host[m]); + } + } + } + } + function ucfirst(str) { + return str.charAt(0).toUpperCase() + str.substring(1); + } +}); + +define("arale/base/1.1.1/aspect-debug", [], function(require, exports) { + // Aspect + // --------------------- + // Thanks to: + // - http://yuilibrary.com/yui/docs/api/classes/Do.html + // - http://code.google.com/p/jquery-aop/ + // - http://lazutkin.com/blog/2008/may/18/aop-aspect-javascript-dojo/ + // 在指定方法执行前,先执行 callback + exports.before = function(methodName, callback, context) { + return weave.call(this, "before", methodName, callback, context); + }; + // 在指定方法执行后,再执行 callback + exports.after = function(methodName, callback, context) { + return weave.call(this, "after", methodName, callback, context); + }; + // Helpers + // ------- + var eventSplitter = /\s+/; + function weave(when, methodName, callback, context) { + var names = methodName.split(eventSplitter); + var name, method; + while (name = names.shift()) { + method = getMethod(this, name); + if (!method.__isAspected) { + wrap.call(this, name); + } + this.on(when + ":" + name, callback, context); + } + return this; + } + function getMethod(host, methodName) { + var method = host[methodName]; + if (!method) { + throw new Error("Invalid method name: " + methodName); + } + return method; + } + function wrap(methodName) { + var old = this[methodName]; + this[methodName] = function() { + var args = Array.prototype.slice.call(arguments); + var beforeArgs = [ "before:" + methodName ].concat(args); + // prevent if trigger return false + if (this.trigger.apply(this, beforeArgs) === false) return; + var ret = old.apply(this, arguments); + var afterArgs = [ "after:" + methodName, ret ].concat(args); + this.trigger.apply(this, afterArgs); + return ret; + }; + this[methodName].__isAspected = true; + } +}); + +define("arale/base/1.1.1/attribute-debug", [], function(require, exports) { + // Attribute + // ----------------- + // Thanks to: + // - http://documentcloud.github.com/backbone/#Model + // - http://yuilibrary.com/yui/docs/api/classes/AttributeCore.html + // - https://github.com/berzniz/backbone.getters.setters + // 负责 attributes 的初始化 + // attributes 是与实例相关的状态信息,可读可写,发生变化时,会自动触发相关事件 + exports.initAttrs = function(config) { + // initAttrs 是在初始化时调用的,默认情况下实例上肯定没有 attrs,不存在覆盖问题 + var attrs = this.attrs = {}; + // Get all inherited attributes. + var specialProps = this.propsInAttrs || []; + mergeInheritedAttrs(attrs, this, specialProps); + // Merge user-specific attributes from config. + if (config) { + mergeUserValue(attrs, config); + } + // 对于有 setter 的属性,要用初始值 set 一下,以保证关联属性也一同初始化 + setSetterAttrs(this, attrs, config); + // Convert `on/before/afterXxx` config to event handler. + parseEventsFromAttrs(this, attrs); + // 将 this.attrs 上的 special properties 放回 this 上 + copySpecialProps(specialProps, this, attrs, true); + }; + // Get the value of an attribute. + exports.get = function(key) { + var attr = this.attrs[key] || {}; + var val = attr.value; + return attr.getter ? attr.getter.call(this, val, key) : val; + }; + // Set a hash of model attributes on the object, firing `"change"` unless + // you choose to silence it. + exports.set = function(key, val, options) { + var attrs = {}; + // set("key", val, options) + if (isString(key)) { + attrs[key] = val; + } else { + attrs = key; + options = val; + } + options || (options = {}); + var silent = options.silent; + var override = options.override; + var now = this.attrs; + var changed = this.__changedAttrs || (this.__changedAttrs = {}); + for (key in attrs) { + if (!attrs.hasOwnProperty(key)) continue; + var attr = now[key] || (now[key] = {}); + val = attrs[key]; + if (attr.readOnly) { + throw new Error("This attribute is readOnly: " + key); + } + // invoke setter + if (attr.setter) { + val = attr.setter.call(this, val, key); + } + // 获取设置前的 prev 值 + var prev = this.get(key); + // 获取需要设置的 val 值 + // 如果设置了 override 为 true,表示要强制覆盖,就不去 merge 了 + // 都为对象时,做 merge 操作,以保留 prev 上没有覆盖的值 + if (!override && isPlainObject(prev) && isPlainObject(val)) { + val = merge(merge({}, prev), val); + } + // set finally + now[key].value = val; + // invoke change event + // 初始化时对 set 的调用,不触发任何事件 + if (!this.__initializingAttrs && !isEqual(prev, val)) { + if (silent) { + changed[key] = [ val, prev ]; + } else { + this.trigger("change:" + key, val, prev, key); + } + } + } + return this; + }; + // Call this method to manually fire a `"change"` event for triggering + // a `"change:attribute"` event for each changed attribute. + exports.change = function() { + var changed = this.__changedAttrs; + if (changed) { + for (var key in changed) { + if (changed.hasOwnProperty(key)) { + var args = changed[key]; + this.trigger("change:" + key, args[0], args[1], key); + } + } + delete this.__changedAttrs; + } + return this; + }; + // for test + exports._isPlainObject = isPlainObject; + // Helpers + // ------- + var toString = Object.prototype.toString; + var hasOwn = Object.prototype.hasOwnProperty; + /** + * Detect the JScript [[DontEnum]] bug: + * In IE < 9 an objects own properties, shadowing non-enumerable ones, are + * made non-enumerable as well. + * https://github.com/bestiejs/lodash/blob/7520066fc916e205ef84cb97fbfe630d7c154158/lodash.js#L134-L144 + */ + /** Detect if own properties are iterated after inherited properties (IE < 9) */ + var iteratesOwnLast; + (function() { + var props = []; + function Ctor() { + this.x = 1; + } + Ctor.prototype = { + valueOf: 1, + y: 1 + }; + for (var prop in new Ctor()) { + props.push(prop); + } + iteratesOwnLast = props[0] !== "x"; + })(); + var isArray = Array.isArray || function(val) { + return toString.call(val) === "[object Array]"; + }; + function isString(val) { + return toString.call(val) === "[object String]"; + } + function isFunction(val) { + return toString.call(val) === "[object Function]"; + } + function isWindow(o) { + return o != null && o == o.window; + } + function isPlainObject(o) { + // Must be an Object. + // Because of IE, we also have to check the presence of the constructor + // property. Make sure that DOM nodes and window objects don't + // pass through, as well + if (!o || toString.call(o) !== "[object Object]" || o.nodeType || isWindow(o)) { + return false; + } + try { + // Not own constructor property must be Object + if (o.constructor && !hasOwn.call(o, "constructor") && !hasOwn.call(o.constructor.prototype, "isPrototypeOf")) { + return false; + } + } catch (e) { + // IE8,9 Will throw exceptions on certain host objects #9897 + return false; + } + var key; + // Support: IE<9 + // Handle iteration over inherited properties before own properties. + // http://bugs.jquery.com/ticket/12199 + if (iteratesOwnLast) { + for (key in o) { + return hasOwn.call(o, key); + } + } + // Own properties are enumerated firstly, so to speed up, + // if last one is own, then all properties are own. + for (key in o) {} + return key === undefined || hasOwn.call(o, key); + } + function isEmptyObject(o) { + if (!o || toString.call(o) !== "[object Object]" || o.nodeType || isWindow(o) || !o.hasOwnProperty) { + return false; + } + for (var p in o) { + if (o.hasOwnProperty(p)) return false; + } + return true; + } + function merge(receiver, supplier) { + var key, value; + for (key in supplier) { + if (supplier.hasOwnProperty(key)) { + value = supplier[key]; + // 只 clone 数组和 plain object,其他的保持不变 + if (isArray(value)) { + value = value.slice(); + } else if (isPlainObject(value)) { + var prev = receiver[key]; + isPlainObject(prev) || (prev = {}); + value = merge(prev, value); + } + receiver[key] = value; + } + } + return receiver; + } + var keys = Object.keys; + if (!keys) { + keys = function(o) { + var result = []; + for (var name in o) { + if (o.hasOwnProperty(name)) { + result.push(name); + } + } + return result; + }; + } + function mergeInheritedAttrs(attrs, instance, specialProps) { + var inherited = []; + var proto = instance.constructor.prototype; + while (proto) { + // 不要拿到 prototype 上的 + if (!proto.hasOwnProperty("attrs")) { + proto.attrs = {}; + } + // 将 proto 上的特殊 properties 放到 proto.attrs 上,以便合并 + copySpecialProps(specialProps, proto.attrs, proto); + // 为空时不添加 + if (!isEmptyObject(proto.attrs)) { + inherited.unshift(proto.attrs); + } + // 向上回溯一级 + proto = proto.constructor.superclass; + } + // Merge and clone default values to instance. + for (var i = 0, len = inherited.length; i < len; i++) { + merge(attrs, normalize(inherited[i])); + } + } + function mergeUserValue(attrs, config) { + merge(attrs, normalize(config, true)); + } + function copySpecialProps(specialProps, receiver, supplier, isAttr2Prop) { + for (var i = 0, len = specialProps.length; i < len; i++) { + var key = specialProps[i]; + if (supplier.hasOwnProperty(key)) { + receiver[key] = isAttr2Prop ? receiver.get(key) : supplier[key]; + } + } + } + var EVENT_PATTERN = /^(on|before|after)([A-Z].*)$/; + var EVENT_NAME_PATTERN = /^(Change)?([A-Z])(.*)/; + function parseEventsFromAttrs(host, attrs) { + for (var key in attrs) { + if (attrs.hasOwnProperty(key)) { + var value = attrs[key].value, m; + if (isFunction(value) && (m = key.match(EVENT_PATTERN))) { + host[m[1]](getEventName(m[2]), value); + delete attrs[key]; + } + } + } + } + // Converts `Show` to `show` and `ChangeTitle` to `change:title` + function getEventName(name) { + var m = name.match(EVENT_NAME_PATTERN); + var ret = m[1] ? "change:" : ""; + ret += m[2].toLowerCase() + m[3]; + return ret; + } + function setSetterAttrs(host, attrs, config) { + var options = { + silent: true + }; + host.__initializingAttrs = true; + for (var key in config) { + if (config.hasOwnProperty(key)) { + if (attrs[key].setter) { + host.set(key, config[key], options); + } + } + } + delete host.__initializingAttrs; + } + var ATTR_SPECIAL_KEYS = [ "value", "getter", "setter", "readOnly" ]; + // normalize `attrs` to + // + // { + // value: 'xx', + // getter: fn, + // setter: fn, + // readOnly: boolean + // } + // + function normalize(attrs, isUserValue) { + var newAttrs = {}; + for (var key in attrs) { + var attr = attrs[key]; + if (!isUserValue && isPlainObject(attr) && hasOwnProperties(attr, ATTR_SPECIAL_KEYS)) { + newAttrs[key] = attr; + continue; + } + newAttrs[key] = { + value: attr + }; + } + return newAttrs; + } + function hasOwnProperties(object, properties) { + for (var i = 0, len = properties.length; i < len; i++) { + if (object.hasOwnProperty(properties[i])) { + return true; + } + } + return false; + } + // 对于 attrs 的 value 来说,以下值都认为是空值: null, undefined, '', [], {} + function isEmptyAttrValue(o) { + return o == null || // null, undefined + (isString(o) || isArray(o)) && o.length === 0 || // '', [] + isEmptyObject(o); + } + // 判断属性值 a 和 b 是否相等,注意仅适用于属性值的判断,非普适的 === 或 == 判断。 + function isEqual(a, b) { + if (a === b) return true; + if (isEmptyAttrValue(a) && isEmptyAttrValue(b)) return true; + // Compare `[[Class]]` names. + var className = toString.call(a); + if (className != toString.call(b)) return false; + switch (className) { + // Strings, numbers, dates, and booleans are compared by value. + case "[object String]": + // Primitives and their corresponding object wrappers are + // equivalent; thus, `"5"` is equivalent to `new String("5")`. + return a == String(b); + + case "[object Number]": + // `NaN`s are equivalent, but non-reflexive. An `equal` + // comparison is performed for other numeric values. + return a != +a ? b != +b : a == 0 ? 1 / a == 1 / b : a == +b; + + case "[object Date]": + case "[object Boolean]": + // Coerce dates and booleans to numeric primitive values. + // Dates are compared by their millisecond representations. + // Note that invalid dates with millisecond representations + // of `NaN` are not equivalent. + return +a == +b; + + // RegExps are compared by their source patterns and flags. + case "[object RegExp]": + return a.source == b.source && a.global == b.global && a.multiline == b.multiline && a.ignoreCase == b.ignoreCase; + + // 简单判断数组包含的 primitive 值是否相等 + case "[object Array]": + var aString = a.toString(); + var bString = b.toString(); + // 只要包含非 primitive 值,为了稳妥起见,都返回 false + return aString.indexOf("[object") === -1 && bString.indexOf("[object") === -1 && aString === bString; + } + if (typeof a != "object" || typeof b != "object") return false; + // 简单判断两个对象是否相等,只判断第一层 + if (isPlainObject(a) && isPlainObject(b)) { + // 键值不相等,立刻返回 false + if (!isEqual(keys(a), keys(b))) { + return false; + } + // 键相同,但有值不等,立刻返回 false + for (var p in a) { + if (a[p] !== b[p]) return false; + } + return true; + } + // 其他情况返回 false, 以避免误判导致 change 事件没发生 + return false; + } +}); diff --git a/test/cases/project/sea-modules/arale/base/1.1.1/base.js b/test/cases/project/sea-modules/arale/base/1.1.1/base.js new file mode 100644 index 0000000..1e2ef17 --- /dev/null +++ b/test/cases/project/sea-modules/arale/base/1.1.1/base.js @@ -0,0 +1 @@ +define("arale/base/1.1.1/base",["arale/class/1.1.0/class","arale/events/1.1.0/events","./aspect","./attribute"],function(a,b,c){function d(a,b){for(var c in b)if(b.hasOwnProperty(c)){var d="_onChange"+e(c);a[d]&&a.on("change:"+c,a[d])}}function e(a){return a.charAt(0).toUpperCase()+a.substring(1)}var f=a("arale/class/1.1.0/class"),g=a("arale/events/1.1.0/events"),h=a("./aspect"),i=a("./attribute");c.exports=f.create({Implements:[g,h,i],initialize:function(a){this.initAttrs(a),d(this,this.attrs)},destroy:function(){this.off();for(var a in this)this.hasOwnProperty(a)&&delete this[a];this.destroy=function(){}}})}),define("arale/base/1.1.1/aspect",[],function(a,b){function c(a,b,c,g){for(var h,i,j=b.split(f);h=j.shift();)i=d(this,h),i.__isAspected||e.call(this,h),this.on(a+":"+h,c,g);return this}function d(a,b){var c=a[b];if(!c)throw new Error("Invalid method name: "+b);return c}function e(a){var b=this[a];this[a]=function(){var c=Array.prototype.slice.call(arguments),d=["before:"+a].concat(c);if(this.trigger.apply(this,d)!==!1){var e=b.apply(this,arguments),f=["after:"+a,e].concat(c);return this.trigger.apply(this,f),e}},this[a].__isAspected=!0}b.before=function(a,b,d){return c.call(this,"before",a,b,d)},b.after=function(a,b,d){return c.call(this,"after",a,b,d)};var f=/\s+/}),define("arale/base/1.1.1/attribute",[],function(a,b){function c(a){return"[object String]"===t.call(a)}function d(a){return"[object Function]"===t.call(a)}function e(a){return null!=a&&a==a.window}function f(a){if(!a||"[object Object]"!==t.call(a)||a.nodeType||e(a))return!1;try{if(a.constructor&&!u.call(a,"constructor")&&!u.call(a.constructor.prototype,"isPrototypeOf"))return!1}catch(b){return!1}var c;if(s)for(c in a)return u.call(a,c);for(c in a);return void 0===c||u.call(a,c)}function g(a){if(!a||"[object Object]"!==t.call(a)||a.nodeType||e(a)||!a.hasOwnProperty)return!1;for(var b in a)if(a.hasOwnProperty(b))return!1;return!0}function h(a,b){var c,d;for(c in b)if(b.hasOwnProperty(c)){if(d=b[c],v(d))d=d.slice();else if(f(d)){var e=a[c];f(e)||(e={}),d=h(e,d)}a[c]=d}return a}function i(a,b,c){for(var d=[],e=b.constructor.prototype;e;)e.hasOwnProperty("attrs")||(e.attrs={}),k(c,e.attrs,e),g(e.attrs)||d.unshift(e.attrs),e=e.constructor.superclass;for(var f=0,i=d.length;i>f;f++)h(a,o(d[f]))}function j(a,b){h(a,o(b,!0))}function k(a,b,c,d){for(var e=0,f=a.length;f>e;e++){var g=a[e];c.hasOwnProperty(g)&&(b[g]=d?b.get(g):c[g])}}function l(a,b){for(var c in b)if(b.hasOwnProperty(c)){var e,f=b[c].value;d(f)&&(e=c.match(x))&&(a[e[1]](m(e[2]),f),delete b[c])}}function m(a){var b=a.match(y),c=b[1]?"change:":"";return c+=b[2].toLowerCase()+b[3]}function n(a,b,c){var d={silent:!0};a.__initializingAttrs=!0;for(var e in c)c.hasOwnProperty(e)&&b[e].setter&&a.set(e,c[e],d);delete a.__initializingAttrs}function o(a,b){var c={};for(var d in a){var e=a[d];c[d]=!b&&f(e)&&p(e,z)?e:{value:e}}return c}function p(a,b){for(var c=0,d=b.length;d>c;c++)if(a.hasOwnProperty(b[c]))return!0;return!1}function q(a){return null==a||(c(a)||v(a))&&0===a.length||g(a)}function r(a,b){if(a===b)return!0;if(q(a)&&q(b))return!0;var c=t.call(a);if(c!=t.call(b))return!1;switch(c){case"[object String]":return a==String(b);case"[object Number]":return a!=+a?b!=+b:0==a?1/a==1/b:a==+b;case"[object Date]":case"[object Boolean]":return+a==+b;case"[object RegExp]":return a.source==b.source&&a.global==b.global&&a.multiline==b.multiline&&a.ignoreCase==b.ignoreCase;case"[object Array]":var d=a.toString(),e=b.toString();return-1===d.indexOf("[object")&&-1===e.indexOf("[object")&&d===e}if("object"!=typeof a||"object"!=typeof b)return!1;if(f(a)&&f(b)){if(!r(w(a),w(b)))return!1;for(var g in a)if(a[g]!==b[g])return!1;return!0}return!1}b.initAttrs=function(a){var b=this.attrs={},c=this.propsInAttrs||[];i(b,this,c),a&&j(b,a),n(this,b,a),l(this,b),k(c,this,b,!0)},b.get=function(a){var b=this.attrs[a]||{},c=b.value;return b.getter?b.getter.call(this,c,a):c},b.set=function(a,b,d){var e={};c(a)?e[a]=b:(e=a,d=b),d||(d={});var g=d.silent,i=d.override,j=this.attrs,k=this.__changedAttrs||(this.__changedAttrs={});for(a in e)if(e.hasOwnProperty(a)){var l=j[a]||(j[a]={});if(b=e[a],l.readOnly)throw new Error("This attribute is readOnly: "+a);l.setter&&(b=l.setter.call(this,b,a));var m=this.get(a);!i&&f(m)&&f(b)&&(b=h(h({},m),b)),j[a].value=b,this.__initializingAttrs||r(m,b)||(g?k[a]=[b,m]:this.trigger("change:"+a,b,m,a))}return this},b.change=function(){var a=this.__changedAttrs;if(a){for(var b in a)if(a.hasOwnProperty(b)){var c=a[b];this.trigger("change:"+b,c[0],c[1],b)}delete this.__changedAttrs}return this},b._isPlainObject=f;var s,t=Object.prototype.toString,u=Object.prototype.hasOwnProperty;!function(){function a(){this.x=1}var b=[];a.prototype={valueOf:1,y:1};for(var c in new a)b.push(c);s="x"!==b[0]}();var v=Array.isArray||function(a){return"[object Array]"===t.call(a)},w=Object.keys;w||(w=function(a){var b=[];for(var c in a)a.hasOwnProperty(c)&&b.push(c);return b});var x=/^(on|before|after)([A-Z].*)$/,y=/^(Change)?([A-Z])(.*)/,z=["value","getter","setter","readOnly"]}); diff --git a/test/cases/project/sea-modules/arale/base/1.1.1/package.json b/test/cases/project/sea-modules/arale/base/1.1.1/package.json new file mode 100644 index 0000000..7c7bfe8 --- /dev/null +++ b/test/cases/project/sea-modules/arale/base/1.1.1/package.json @@ -0,0 +1,27 @@ +{ + "name": "base", + "family": "arale", + "version": "1.1.1", + "description": "Base 是一个基础类,提供 Class、Events、Attribute 和 Aspect 支持。", + "keywords": ["infrastructure"], + "homepage": "http://aralejs.org/base", + "author": "贯高 ", + "maintainers": [ + "玉伯 ", + "贯高 " + ], + "repository": { + "type": "git", + "url": "https://github.com/aralejs/base.git" + }, + "bugs": { + "url": "https://github.com/aralejs/base/issues" + }, + "spm": { + "alias": { + "class": "arale/class/1.1.0/class", + "events": "arale/events/1.1.0/events" + }, + "output": ["base.js"] + } +} diff --git a/test/cases/project/sea-modules/arale/class/1.1.0/class-debug.js b/test/cases/project/sea-modules/arale/class/1.1.0/class-debug.js new file mode 100644 index 0000000..df60888 --- /dev/null +++ b/test/cases/project/sea-modules/arale/class/1.1.0/class-debug.js @@ -0,0 +1,148 @@ +define("arale/class/1.1.0/class-debug", [], function(require, exports, module) { + // Class + // ----------------- + // Thanks to: + // - http://mootools.net/docs/core/Class/Class + // - http://ejohn.org/blog/simple-javascript-inheritance/ + // - https://github.com/ded/klass + // - http://documentcloud.github.com/backbone/#Model-extend + // - https://github.com/joyent/node/blob/master/lib/util.js + // - https://github.com/kissyteam/kissy/blob/master/src/seed/src/kissy.js + // The base Class implementation. + function Class(o) { + // Convert existed function to Class. + if (!(this instanceof Class) && isFunction(o)) { + return classify(o); + } + } + module.exports = Class; + // Create a new Class. + // + // var SuperPig = Class.create({ + // Extends: Animal, + // Implements: Flyable, + // initialize: function() { + // SuperPig.superclass.initialize.apply(this, arguments) + // }, + // Statics: { + // COLOR: 'red' + // } + // }) + // + Class.create = function(parent, properties) { + if (!isFunction(parent)) { + properties = parent; + parent = null; + } + properties || (properties = {}); + parent || (parent = properties.Extends || Class); + properties.Extends = parent; + // The created class constructor + function SubClass() { + // Call the parent constructor. + parent.apply(this, arguments); + // Only call initialize in self constructor. + if (this.constructor === SubClass && this.initialize) { + this.initialize.apply(this, arguments); + } + } + // Inherit class (static) properties from parent. + if (parent !== Class) { + mix(SubClass, parent, parent.StaticsWhiteList); + } + // Add instance properties to the subclass. + implement.call(SubClass, properties); + // Make subclass extendable. + return classify(SubClass); + }; + function implement(properties) { + var key, value; + for (key in properties) { + value = properties[key]; + if (Class.Mutators.hasOwnProperty(key)) { + Class.Mutators[key].call(this, value); + } else { + this.prototype[key] = value; + } + } + } + // Create a sub Class based on `Class`. + Class.extend = function(properties) { + properties || (properties = {}); + properties.Extends = this; + return Class.create(properties); + }; + function classify(cls) { + cls.extend = Class.extend; + cls.implement = implement; + return cls; + } + // Mutators define special properties. + Class.Mutators = { + Extends: function(parent) { + var existed = this.prototype; + var proto = createProto(parent.prototype); + // Keep existed properties. + mix(proto, existed); + // Enforce the constructor to be what we expect. + proto.constructor = this; + // Set the prototype chain to inherit from `parent`. + this.prototype = proto; + // Set a convenience property in case the parent's prototype is + // needed later. + this.superclass = parent.prototype; + }, + Implements: function(items) { + isArray(items) || (items = [ items ]); + var proto = this.prototype, item; + while (item = items.shift()) { + mix(proto, item.prototype || item); + } + }, + Statics: function(staticProperties) { + mix(this, staticProperties); + } + }; + // Shared empty constructor function to aid in prototype-chain creation. + function Ctor() {} + // See: http://jsperf.com/object-create-vs-new-ctor + var createProto = Object.__proto__ ? function(proto) { + return { + __proto__: proto + }; + } : function(proto) { + Ctor.prototype = proto; + return new Ctor(); + }; + // Helpers + // ------------ + function mix(r, s, wl) { + // Copy "all" properties including inherited ones. + for (var p in s) { + if (s.hasOwnProperty(p)) { + if (wl && indexOf(wl, p) === -1) continue; + // 在 iPhone 1 代等设备的 Safari 中,prototype 也会被枚举出来,需排除 + if (p !== "prototype") { + r[p] = s[p]; + } + } + } + } + var toString = Object.prototype.toString; + var isArray = Array.isArray || function(val) { + return toString.call(val) === "[object Array]"; + }; + var isFunction = function(val) { + return toString.call(val) === "[object Function]"; + }; + var indexOf = Array.prototype.indexOf ? function(arr, item) { + return arr.indexOf(item); + } : function(arr, item) { + for (var i = 0, len = arr.length; i < len; i++) { + if (arr[i] === item) { + return i; + } + } + return -1; + }; +}); diff --git a/test/cases/project/sea-modules/arale/class/1.1.0/class.js b/test/cases/project/sea-modules/arale/class/1.1.0/class.js new file mode 100644 index 0000000..9f6062e --- /dev/null +++ b/test/cases/project/sea-modules/arale/class/1.1.0/class.js @@ -0,0 +1 @@ +define("arale/class/1.1.0/class",[],function(a,b,c){function d(a){return this instanceof d||!l(a)?void 0:f(a)}function e(a){var b,c;for(b in a)c=a[b],d.Mutators.hasOwnProperty(b)?d.Mutators[b].call(this,c):this.prototype[b]=c}function f(a){return a.extend=d.extend,a.implement=e,a}function g(){}function h(a,b,c){for(var d in b)if(b.hasOwnProperty(d)){if(c&&-1===m(c,d))continue;"prototype"!==d&&(a[d]=b[d])}}c.exports=d,d.create=function(a,b){function c(){a.apply(this,arguments),this.constructor===c&&this.initialize&&this.initialize.apply(this,arguments)}return l(a)||(b=a,a=null),b||(b={}),a||(a=b.Extends||d),b.Extends=a,a!==d&&h(c,a,a.StaticsWhiteList),e.call(c,b),f(c)},d.extend=function(a){return a||(a={}),a.Extends=this,d.create(a)},d.Mutators={Extends:function(a){var b=this.prototype,c=i(a.prototype);h(c,b),c.constructor=this,this.prototype=c,this.superclass=a.prototype},Implements:function(a){k(a)||(a=[a]);for(var b,c=this.prototype;b=a.shift();)h(c,b.prototype||b)},Statics:function(a){h(this,a)}};var i=Object.__proto__?function(a){return{__proto__:a}}:function(a){return g.prototype=a,new g},j=Object.prototype.toString,k=Array.isArray||function(a){return"[object Array]"===j.call(a)},l=function(a){return"[object Function]"===j.call(a)},m=Array.prototype.indexOf?function(a,b){return a.indexOf(b)}:function(a,b){for(var c=0,d=a.length;d>c;c++)if(a[c]===b)return c;return-1}}); diff --git a/test/cases/project/sea-modules/arale/class/1.1.0/package.json b/test/cases/project/sea-modules/arale/class/1.1.0/package.json new file mode 100644 index 0000000..10d4a97 --- /dev/null +++ b/test/cases/project/sea-modules/arale/class/1.1.0/package.json @@ -0,0 +1,23 @@ +{ + "name": "class", + "family": "arale", + "version": "1.1.0", + "keywords": ["infrastructure"], + "description": "提供简洁的 OO 实现。", + "homepage": "http://aralejs.org/class/", + "author": "玉伯 ", + "maintainers": [ + "玉伯 ", + "贯高 " + ], + "repository": { + "type": "git", + "url": "https://github.com/aralejs/class.git" + }, + "bugs": { + "url": "https://github.com/aralejs/class/issues" + }, + "spm": { + "output": ["class.js"] + } +} diff --git a/test/cases/project/sea-modules/arale/events/1.1.0/events-debug.js b/test/cases/project/sea-modules/arale/events/1.1.0/events-debug.js new file mode 100644 index 0000000..c2434c4 --- /dev/null +++ b/test/cases/project/sea-modules/arale/events/1.1.0/events-debug.js @@ -0,0 +1,124 @@ +define("arale/events/1.1.0/events-debug", [], function() { + // Events + // ----------------- + // Thanks to: + // - https://github.com/documentcloud/backbone/blob/master/backbone.js + // - https://github.com/joyent/node/blob/master/lib/events.js + // Regular expression used to split event strings + var eventSplitter = /\s+/; + // A module that can be mixed in to *any object* in order to provide it + // with custom events. You may bind with `on` or remove with `off` callback + // functions to an event; `trigger`-ing an event fires all callbacks in + // succession. + // + // var object = new Events(); + // object.on('expand', function(){ alert('expanded'); }); + // object.trigger('expand'); + // + function Events() {} + // Bind one or more space separated events, `events`, to a `callback` + // function. Passing `"all"` will bind the callback to all events fired. + Events.prototype.on = function(events, callback, context) { + var cache, event, list; + if (!callback) return this; + cache = this.__events || (this.__events = {}); + events = events.split(eventSplitter); + while (event = events.shift()) { + list = cache[event] || (cache[event] = []); + list.push(callback, context); + } + return this; + }; + // Remove one or many callbacks. If `context` is null, removes all callbacks + // with that function. If `callback` is null, removes all callbacks for the + // event. If `events` is null, removes all bound callbacks for all events. + Events.prototype.off = function(events, callback, context) { + var cache, event, list, i; + // No events, or removing *all* events. + if (!(cache = this.__events)) return this; + if (!(events || callback || context)) { + delete this.__events; + return this; + } + events = events ? events.split(eventSplitter) : keys(cache); + // Loop through the callback list, splicing where appropriate. + while (event = events.shift()) { + list = cache[event]; + if (!list) continue; + if (!(callback || context)) { + delete cache[event]; + continue; + } + for (i = list.length - 2; i >= 0; i -= 2) { + if (!(callback && list[i] !== callback || context && list[i + 1] !== context)) { + list.splice(i, 2); + } + } + } + return this; + }; + // Trigger one or many events, firing all bound callbacks. Callbacks are + // passed the same arguments as `trigger` is, apart from the event name + // (unless you're listening on `"all"`, which will cause your callback to + // receive the true name of the event as the first argument). + Events.prototype.trigger = function(events) { + var cache, event, all, list, i, len, rest = [], args, returned = { + status: true + }; + if (!(cache = this.__events)) return this; + events = events.split(eventSplitter); + // Fill up `rest` with the callback arguments. Since we're only copying + // the tail of `arguments`, a loop is much faster than Array#slice. + for (i = 1, len = arguments.length; i < len; i++) { + rest[i - 1] = arguments[i]; + } + // For each event, walk through the list of callbacks twice, first to + // trigger the event, then to trigger any `"all"` callbacks. + while (event = events.shift()) { + // Copy callback lists to prevent modification. + if (all = cache.all) all = all.slice(); + if (list = cache[event]) list = list.slice(); + // Execute event callbacks. + callEach(list, rest, this, returned); + // Execute "all" callbacks. + callEach(all, [ event ].concat(rest), this, returned); + } + return returned.status; + }; + // Mix `Events` to object instance or Class function. + Events.mixTo = function(receiver) { + receiver = receiver.prototype || receiver; + var proto = Events.prototype; + for (var p in proto) { + if (proto.hasOwnProperty(p)) { + receiver[p] = proto[p]; + } + } + }; + // Helpers + // ------- + var keys = Object.keys; + if (!keys) { + keys = function(o) { + var result = []; + for (var name in o) { + if (o.hasOwnProperty(name)) { + result.push(name); + } + } + return result; + }; + } + // Execute callbacks + function callEach(list, args, context, returned) { + var r; + if (list) { + for (var i = 0, len = list.length; i < len; i += 2) { + r = list[i].apply(list[i + 1] || context, args); + // trigger will return false if one of the callbacks return false + r === false && returned.status && (returned.status = false); + } + } + } + return Events; +}); diff --git a/test/cases/project/sea-modules/arale/events/1.1.0/events.js b/test/cases/project/sea-modules/arale/events/1.1.0/events.js new file mode 100644 index 0000000..e375d8a --- /dev/null +++ b/test/cases/project/sea-modules/arale/events/1.1.0/events.js @@ -0,0 +1 @@ +define("arale/events/1.1.0/events",[],function(){function a(){}function b(a,b,c,d){var e;if(a)for(var f=0,g=a.length;g>f;f+=2)e=a[f].apply(a[f+1]||c,b),e===!1&&d.status&&(d.status=!1)}var c=/\s+/;a.prototype.on=function(a,b,d){var e,f,g;if(!b)return this;for(e=this.__events||(this.__events={}),a=a.split(c);f=a.shift();)g=e[f]||(e[f]=[]),g.push(b,d);return this},a.prototype.off=function(a,b,e){var f,g,h,i;if(!(f=this.__events))return this;if(!(a||b||e))return delete this.__events,this;for(a=a?a.split(c):d(f);g=a.shift();)if(h=f[g])if(b||e)for(i=h.length-2;i>=0;i-=2)b&&h[i]!==b||e&&h[i+1]!==e||h.splice(i,2);else delete f[g];return this},a.prototype.trigger=function(a){var d,e,f,g,h,i,j=[],k={status:!0};if(!(d=this.__events))return this;for(a=a.split(c),h=1,i=arguments.length;i>h;h++)j[h-1]=arguments[h];for(;e=a.shift();)(f=d.all)&&(f=f.slice()),(g=d[e])&&(g=g.slice()),b(g,j,this,k),b(f,[e].concat(j),this,k);return k.status},a.mixTo=function(b){b=b.prototype||b;var c=a.prototype;for(var d in c)c.hasOwnProperty(d)&&(b[d]=c[d])};var d=Object.keys;return d||(d=function(a){var b=[];for(var c in a)a.hasOwnProperty(c)&&b.push(c);return b}),a}); diff --git a/test/cases/project/sea-modules/arale/events/1.1.0/package.json b/test/cases/project/sea-modules/arale/events/1.1.0/package.json new file mode 100644 index 0000000..fff9283 --- /dev/null +++ b/test/cases/project/sea-modules/arale/events/1.1.0/package.json @@ -0,0 +1,23 @@ +{ + "name": "events", + "family": "arale", + "version": "1.1.0", + "description": "提供基本的事件添加、移除和触发功能。", + "keywords": ["infrastructure"], + "homepage": "http://aralejs.org/events/", + "author": "玉伯 ", + "maintainers": [ + "玉伯 ", + "贯高 " + ], + "repository": { + "type": "git", + "url": "https://github.com/aralejs/events.git" + }, + "bugs": { + "url": "https://github.com/aralejs/events/issues" + }, + "spm": { + "output": ["events.js"] + } +} From 8b2b7414c3ff9a280d12a00aeae79c0c8a4367f5 Mon Sep 17 00:00:00 2001 From: popomore Date: Tue, 30 Dec 2014 16:49:29 +0800 Subject: [PATCH 04/14] feat(css) css support hash --- tasks/lib/style.js | 232 +++++++++++++++------------------------------ 1 file changed, 77 insertions(+), 155 deletions(-) diff --git a/tasks/lib/style.js b/tasks/lib/style.js index fc01b10..244a9ca 100644 --- a/tasks/lib/style.js +++ b/tasks/lib/style.js @@ -1,182 +1,104 @@ -var path = require('path'); -var format = require('util').format; -var css = require('cmd-util').css; -var cssParse = require('css').parse; -var cssStringify = require('css').stringify; exports.init = function(grunt) { - var ast = require('cmd-util').ast; var iduri = require('cmd-util').iduri; + var format = require('util').format; + var css = require('cmd-util').css; + var md5 = require('./util').md5; + var join = require('path').join; + var dirname = require('path').dirname; var exports = {}; - exports.css2jsParser = function(fileObj, options) { - // don't transport debug css files - if (/\-debug\.css$/.test(fileObj.src)) return; - grunt.log.verbose.writeln('Transport ' + fileObj.src + ' -> ' + fileObj.dest); - - // transport css to js - var data = fileObj.srcData || grunt.file.read(fileObj.src); - var id = unixy(options.idleading + fileObj.name.replace(/\.js$/, '')); - - data = css2js(data, id, options, fileObj); - data = ast.getAst(data).print_to_string(options.uglify) + '\n'; - var dest = fileObj.dest + '.js'; - grunt.file.write(dest, data); - - if (!options.debug) { - return; - } - dest = dest.replace(/\.css\.js$/, '-debug.css.js'); - - data = ast.modify(data, function(v) { - var ext = path.extname(v); - if (ext && options.parsers[ext]) { - return v.replace(new RegExp('\\' + ext + '$'), '-debug' + ext); - } else { - return v + '-debug'; - } - }); - data = data.print_to_string(options.uglify) + '\n'; - grunt.file.write(dest, data); - }; - // the real css parser exports.cssParser = function(fileObj, options) { var data = fileObj.srcData || grunt.file.read(fileObj.src); - data = css.parse(data); - grunt.log.verbose.writeln('Transport ' + fileObj.src + ' -> ' + fileObj.dest); - var ret = css.stringify(data[0].code, function(node) { - if (node.type === 'import' && node.id) { - if (node.id.charAt(0) === '.') { - return node; - } - if (/^https?:\/\//.test(node.id)) { - return node; - } - if (!iduri.isAlias(options, node.id)) { - grunt.log.warn('alias ' + node.id + ' not defined.'); - } else { - node.id = iduri.parseAlias(options, node.id); - if (!/\.css$/.test(node.id)) { - node.id += '.css'; - } - return node; - } - } - }); + var dest = fileObj.dest; + var ret = parseCss(data); + var id = unixy(options.idleading + fileObj.name); + var code = format('/*! define %s */\n%s', id, ret); + writeFile(code, dest); + + if (options.debug) { + dest = fileObj.dest.replace(/\.css$/, '-debug.css'); + ret = parseCss(data, addDebug); + id = unixy(options.idleading + fileObj.name.replace(/\.css$/, '-debug.css')); + code = format('/*! define %s */\n%s', id, ret); + writeFile(code, dest); + } - var id = unixy(options.idleading + fileObj.name.replace(/\.js$/, '')); - var banner = format('/*! define %s */', id); - grunt.file.write(fileObj.dest, [banner, ret].join('\n')); + if (options.hash) { + var hash = md5(data); + dest = fileObj.dest.replace(/\.css$/, '-' + hash + '.css'); + ret = parseCss(data, addHash); + id = unixy(options.idleading + fileObj.name.replace(/\.css$/, '-' + hash + '.css')); + code = format('/*! define %s */\n%s', id, ret); + writeFile(code, dest); + } - // create -debug.css file - if (options.debug === false) { - return; + function addDebug(node) { + node.id = node.id.replace(/\.css$/, '-debug.css'); } - var dest = fileObj.dest.replace(/\.css$/, '-debug.css'); - ret = css.stringify(data[0].code, function(node) { - if (node.type === 'import' && node.id) { - var alias = node.id; - if (alias.charAt(0) === '.') { - node.id = alias.replace(/(\.css)?$/, '-debug.css'); - return node; - } - if (/^https?:\/\//.test(node.id)) { - return node; + function addHash(node) { + var code; + if (node.id.charAt(0) === '.') { + code = grunt.file.read(join(dirname(fileObj.src), node.id)); + } else { + var i = 0, path; + while(path = options.paths[i++]) { + var file = join(path, node.id); + if (grunt.file.exists(file)) { + code = grunt.file.read(file); + break; + } } - alias = iduri.parseAlias(options, alias); - if (/\.css$/.test(alias)) { - node.id = alias.replace(/\.css$/, '-debug.css'); - } else { - node.id = alias + '-debug.css'; + if (!code) { + grunt.log.warn('fail find file ' + node.id); } - return node; } - }); - id = id.replace(/(\.css)?$/, '-debug.css'); - banner = format('/*! define %s */', id); - grunt.file.write(dest, [banner, ret].join('\n')); - }; - - return exports; -}; - + var hash = md5(code); + node.id = node.id.replace(/\.css$/, '-' + hash + '.css'); + } -// helpers -function unixy(uri) { - return uri.replace(/\\/g, '/'); -} + function parseCss(data, editId) { + if (!editId) { + editId = function() {}; + } -function parseRules(rules, prefix) { - return rules.map(function(o) { - if (o.selectors) { - o.selectors = o.selectors.map(function(selector) { - // handle :root selector {} - if (selector.indexOf(':root') === 0) { - return ':root ' + prefix + selector.replace(':root', ' '); + return css.stringify(css.parse(data)[0].code, function(node) { + if (node.type === 'import' && node.id) { + if (node.id.charAt(0) === '.') { + editId(node); + return node; + } + if (/^https?:\/\//.test(node.id)) { + return node; + } + if (!iduri.isAlias(options, node.id)) { + grunt.log.warn('alias ' + node.id + ' not defined.'); + } else { + node.id = iduri.parseAlias(options, node.id); + if (!/\.css$/.test(node.id)) { + node.id += '.css'; + } + editId(node); + return node; + } } - return prefix + selector; }); } - if (o.rules) { - o.rules = parseRules(o.rules, prefix); - } - return o; - }); -} - -function css2js(code, id, options, fileObj) { - // ex. arale/widget/1.0.0/ => arale-widget-1_0_0 - var styleId = unixy((options || {}).idleading || '') - .replace(/\/$/, '') - .replace(/\//g, '-') - .replace(/\./g, '_'); - var prefix = ['.', styleId, ' '].join(''); + }; - var addStyleBox = false; - if (options.styleBox === true) { - addStyleBox = true; - } else if (options.styleBox && options.styleBox.length) { - options.styleBox.forEach(function(file) { - if (file === fileObj.name) { - addStyleBox = true; - } - }); - } + return exports; - // if outside css modules, fileObj would be undefined - // then dont add styleBox - if (addStyleBox && styleId && fileObj) { - var data = cssParse(code); - data.stylesheet.rules = parseRules(data.stylesheet.rules, prefix); - code = cssStringify(data); + function writeFile(data, dest) { + grunt.log.writeln('transport ' + dest + ' created'); + grunt.file.write(dest, data + '\n'); } +}; - // remove comment and format - var cleancss = require('clean-css'); - code = cleancss.process(code, { - keepSpecialComments: 0, - removeEmpty: true - }); - - // transform css to js - // spmjs/spm#581 - var tpl = [ - 'define("%s", [], function() {', - "seajs.importStyle('%s')", - '});' - ].join('\n'); - - // spmjs/spm#651 - code = code.split(/\r\n|\r|\n/).map(function(line) { - return line.replace(/\\/g, '\\\\'); - }).join('\n'); - - code = format(tpl, id, code.replace(/\'/g, '\\\'')); - return code; +// helpers +function unixy(uri) { + return uri.replace(/\\/g, '/'); } - -exports.css2js = css2js; From 7624c5c2437ba20f7efe2e10f965eabc87374f89 Mon Sep 17 00:00:00 2001 From: popomore Date: Wed, 31 Dec 2014 01:06:35 +0800 Subject: [PATCH 05/14] fix(hash) fix some problem with hash --- Gruntfile.js | 7 +++-- tasks/lib/common.js | 12 ++++++-- tasks/lib/css2js.js | 17 +++++++++-- tasks/lib/style.js | 18 ++---------- tasks/lib/util.js | 2 +- test/cases/css/alias-debug.css.expect | 2 +- test/cases/css/alias.css.expect | 2 +- test/cases/css/ie8-debug.css.expect | 2 +- test/cases/css/ie8.css.expect | 2 +- test/cases/css/simple-debug.css.expect | 2 +- test/cases/css/simple.css.expect | 2 +- ...b1b8e6a.js.expect => a-da24e8b4.js.expect} | 4 +-- ...849d846.js.expect => b-5821d03c.js.expect} | 2 +- test/cases/project/a-2cee5097.html.js.expect | 1 + .../project/a-670b8177.handlebars.js.expect | 28 +++++++++++++++++++ test/cases/project/a-d41d8cd9.json.js.expect | 1 + test/cases/project/a-d41d8cd9.tpl.js.expect | 1 + test/cases/project/a-e1f4111c.css.expect | 5 ++++ test/cases/project/a-f5f5c4f5.js.expect | 8 ++++++ test/cases/project/b-aa896723.css.expect | 5 ++++ test/cases/project/b-e5f793e3.js.expect | 3 ++ test/cases/style/a-debug.css.js.expect | 2 +- test/cases/style/a.css | 3 +- test/cases/style/a.css.js.expect | 2 +- 24 files changed, 95 insertions(+), 38 deletions(-) rename test/cases/hash/{a-db1b8e6a.js.expect => a-da24e8b4.js.expect} (55%) rename test/cases/hash/{b-4849d846.js.expect => b-5821d03c.js.expect} (65%) create mode 100644 test/cases/project/a-2cee5097.html.js.expect create mode 100644 test/cases/project/a-670b8177.handlebars.js.expect create mode 100644 test/cases/project/a-d41d8cd9.json.js.expect create mode 100644 test/cases/project/a-d41d8cd9.tpl.js.expect create mode 100644 test/cases/project/a-e1f4111c.css.expect create mode 100644 test/cases/project/a-f5f5c4f5.js.expect create mode 100644 test/cases/project/b-aa896723.css.expect create mode 100644 test/cases/project/b-e5f793e3.js.expect diff --git a/Gruntfile.js b/Gruntfile.js index b2e6d17..22ec519 100644 --- a/Gruntfile.js +++ b/Gruntfile.js @@ -188,7 +188,7 @@ module.exports = function(grunt) { '.css': [css2jsParser], '.js': [jsParser] }, - styleBox: ["a.css"], + styleBox: ['a.css'], idleading: 'arale/widget/1.0.0/' }, files: [{ @@ -280,12 +280,13 @@ module.exports = function(grunt) { 'base': 'arale/base/1.1.1/base' }, idleading: 'family/name/', - hash: true + hash: true, + debug: false }, files: [{ expand: true, cwd: 'test/cases/project', - src: '*.*', + src: ['*.*', '!*.expect'], dest: 'test/expected/project' }] } diff --git a/tasks/lib/common.js b/tasks/lib/common.js index ebe183f..6b2914d 100644 --- a/tasks/lib/common.js +++ b/tasks/lib/common.js @@ -13,8 +13,9 @@ exports.init = function(grunt, options) { exports[type + 'Parser'] = function(fileObj, options) { var dest, id = unixy(options.idleading + fileObj.name.replace(/\.js$/, '')); var data = fileObj.srcData || grunt.file.read(fileObj.src); + var hash = md5(data); var deps = depParser ? depParser(data, options) : ''; - var factory = factoryParser ? factoryParser(data, options) : '{}'; + var factory = factoryParser ? factoryParser(data, options, fileObj) : '{}'; var file = { contents: format('define("%s", [%s], %s)', id, deps, factory), dest: fileObj.dest + '.js' @@ -30,14 +31,19 @@ exports.init = function(grunt, options) { if (options.debug) { dest = file.dest.replace(retTypeJs, '-debug.' + type + '.js'); data = ast.modify(file.contents, { - id: id.replace(regType, '-debug.' + type) + id: id.replace(regType, '-debug.' + type), + dependencies: function(id) { + return id + '-debug'; + }, + require: function(id) { + return id + '-debug'; + } }).print_to_string(options.uglify); writeFile(data, dest); } // create hash file xxx-{hash}.{type}.js if (options.hash) { - var hash = md5(factory); dest = file.dest.replace(retTypeJs, '-' + hash + '.' + type + '.js'); data = ast.modify(file.contents, { id: id.replace(regType, '-' + hash + '.' + type) diff --git a/tasks/lib/css2js.js b/tasks/lib/css2js.js index 22209d1..253b7af 100644 --- a/tasks/lib/css2js.js +++ b/tasks/lib/css2js.js @@ -10,11 +10,22 @@ exports.init = function(grunt) { }); }; -function css2js(code, options) { +function css2js(code, options, fileObj) { + var addStyleBox = false; + if (options.styleBox === true) { + addStyleBox = true; + } else if (options.styleBox && options.styleBox.length) { + options.styleBox.forEach(function(file) { + if (file === fileObj.name) { + addStyleBox = true; + } + }); + } + // if outside css modules, fileObj would be undefined // then dont add styleBox var opt = {}; - if (options.styleBox === true) { + if (addStyleBox && fileObj) { // ex. arale/widget/1.0.0/ => arale-widget-1_0_0 var styleId = unixy((options || {}).idleading || '') .replace(/\/$/, '') @@ -33,7 +44,7 @@ function css2js(code, options) { // transform css to js // spmjs/spm#581 - var template = 'function() {seajs.importStyle("%s")}'; + var template = 'function() {seajs.importStyle(\'%s\')}'; return format(template, code); } diff --git a/tasks/lib/style.js b/tasks/lib/style.js index 244a9ca..56e189a 100644 --- a/tasks/lib/style.js +++ b/tasks/lib/style.js @@ -41,24 +41,10 @@ exports.init = function(grunt) { } function addHash(node) { - var code; if (node.id.charAt(0) === '.') { - code = grunt.file.read(join(dirname(fileObj.src), node.id)); - } else { - var i = 0, path; - while(path = options.paths[i++]) { - var file = join(path, node.id); - if (grunt.file.exists(file)) { - code = grunt.file.read(file); - break; - } - } - if (!code) { - grunt.log.warn('fail find file ' + node.id); - } + var code = grunt.file.read(join(dirname(fileObj.src), node.id)); + node.id = node.id.replace(/\.css$/, '-' + md5(code) + '.css'); } - var hash = md5(code); - node.id = node.id.replace(/\.css$/, '-' + hash + '.css'); } function parseCss(data, editId) { diff --git a/tasks/lib/util.js b/tasks/lib/util.js index d1d930e..e4420ec 100644 --- a/tasks/lib/util.js +++ b/tasks/lib/util.js @@ -4,7 +4,7 @@ exports.md5 = function md5(contents, deps) { if (!deps) deps = []; contents = deps.map(function(depFile) { return depFile.contents || ''; - }) + contents; + }).join('') + contents; return crypto .createHash('md5') .update(contents, 'utf8') diff --git a/test/cases/css/alias-debug.css.expect b/test/cases/css/alias-debug.css.expect index 81811c0..8651ad5 100644 --- a/test/cases/css/alias-debug.css.expect +++ b/test/cases/css/alias-debug.css.expect @@ -3,4 +3,4 @@ /*! import ./ie8-debug.css */ .alias { color: red; -} \ No newline at end of file +} diff --git a/test/cases/css/alias.css.expect b/test/cases/css/alias.css.expect index 1f34227..669dd09 100644 --- a/test/cases/css/alias.css.expect +++ b/test/cases/css/alias.css.expect @@ -3,4 +3,4 @@ /*! import ./ie8.css */ .alias { color: red; -} \ No newline at end of file +} diff --git a/test/cases/css/ie8-debug.css.expect b/test/cases/css/ie8-debug.css.expect index 851b9a1..ddb647c 100644 --- a/test/cases/css/ie8-debug.css.expect +++ b/test/cases/css/ie8-debug.css.expect @@ -2,4 +2,4 @@ body { font-size: 14px\0; font-weight: bold; -} \ No newline at end of file +} diff --git a/test/cases/css/ie8.css.expect b/test/cases/css/ie8.css.expect index f9c9605..79372d8 100644 --- a/test/cases/css/ie8.css.expect +++ b/test/cases/css/ie8.css.expect @@ -2,4 +2,4 @@ body { font-size: 14px\0; font-weight: bold; -} \ No newline at end of file +} diff --git a/test/cases/css/simple-debug.css.expect b/test/cases/css/simple-debug.css.expect index 5dce05d..0e00e05 100644 --- a/test/cases/css/simple-debug.css.expect +++ b/test/cases/css/simple-debug.css.expect @@ -1,4 +1,4 @@ /*! define simple-debug.css */ body { color: '#fff'; -} \ No newline at end of file +} diff --git a/test/cases/css/simple.css.expect b/test/cases/css/simple.css.expect index 807ed3d..cb0ee6d 100644 --- a/test/cases/css/simple.css.expect +++ b/test/cases/css/simple.css.expect @@ -1,4 +1,4 @@ /*! define simple.css */ body { color: '#fff'; -} \ No newline at end of file +} diff --git a/test/cases/hash/a-db1b8e6a.js.expect b/test/cases/hash/a-da24e8b4.js.expect similarity index 55% rename from test/cases/hash/a-db1b8e6a.js.expect rename to test/cases/hash/a-da24e8b4.js.expect index 5683962..b37066b 100644 --- a/test/cases/hash/a-db1b8e6a.js.expect +++ b/test/cases/hash/a-da24e8b4.js.expect @@ -1,4 +1,4 @@ -define("family/name/a-db1b8e6a", [ "./b-4849d846", "./c/c-5bffe50a", "$", "family/bar/bar", "arale/class/foo" ], function(require, exports, module) { - require("./b-4849d846.js"); +define("family/name/a-da24e8b4", [ "./b-5821d03c", "./c/c-5bffe50a", "$", "family/bar/bar", "arale/class/foo" ], function(require, exports, module) { + require("./b-5821d03c.js"); require("arale/class/foo"); }); diff --git a/test/cases/hash/b-4849d846.js.expect b/test/cases/hash/b-5821d03c.js.expect similarity index 65% rename from test/cases/hash/b-4849d846.js.expect rename to test/cases/hash/b-5821d03c.js.expect index d053127..d4eaf73 100644 --- a/test/cases/hash/b-4849d846.js.expect +++ b/test/cases/hash/b-5821d03c.js.expect @@ -1,4 +1,4 @@ -define("family/name/b-4849d846", [ "./c/c-5bffe50a", "$", "family/bar/bar" ], function(require, exports, module) { +define("family/name/b-5821d03c", [ "./c/c-5bffe50a", "$", "family/bar/bar" ], function(require, exports, module) { require("./c/c-5bffe50a.js"); require("family/bar/bar"); }); diff --git a/test/cases/project/a-2cee5097.html.js.expect b/test/cases/project/a-2cee5097.html.js.expect new file mode 100644 index 0000000..dd95ba5 --- /dev/null +++ b/test/cases/project/a-2cee5097.html.js.expect @@ -0,0 +1 @@ +define("family/name/a-2cee5097.html", [], "
\n
    \n
  • a
  • \n
      \n
\n"); diff --git a/test/cases/project/a-670b8177.handlebars.js.expect b/test/cases/project/a-670b8177.handlebars.js.expect new file mode 100644 index 0000000..747d3ee --- /dev/null +++ b/test/cases/project/a-670b8177.handlebars.js.expect @@ -0,0 +1,28 @@ +define("family/name/a-670b8177.handlebars", [ "gallery/handlebars/1.0.2/runtime" ], function(require, exports, module) { + var Handlebars = require("gallery/handlebars/1.0.2/runtime"); + var template = Handlebars.template; + module.exports = template(function(Handlebars, depth0, helpers, partials, data) { + this.compilerInfo = [ 3, ">= 1.0.0-rc.4" ]; + helpers = helpers || {}; + for (var key in Handlebars.helpers) { + helpers[key] = helpers[key] || Handlebars.helpers[key]; + } + data = data || {}; + var buffer = "", stack1, functionType = "function"; + buffer += "
"; + if (stack1 = helpers.data) { + stack1 = stack1.call(depth0, { + hash: {}, + data: data + }); + } else { + stack1 = depth0.data; + stack1 = typeof stack1 === functionType ? stack1.apply(depth0) : stack1; + } + if (stack1 || stack1 === 0) { + buffer += stack1; + } + buffer += "
\n"; + return buffer; + }); +}); diff --git a/test/cases/project/a-d41d8cd9.json.js.expect b/test/cases/project/a-d41d8cd9.json.js.expect new file mode 100644 index 0000000..96c1ec4 --- /dev/null +++ b/test/cases/project/a-d41d8cd9.json.js.expect @@ -0,0 +1 @@ +define("family/name/a-d41d8cd9.json", [], {}); diff --git a/test/cases/project/a-d41d8cd9.tpl.js.expect b/test/cases/project/a-d41d8cd9.tpl.js.expect new file mode 100644 index 0000000..ee43aac --- /dev/null +++ b/test/cases/project/a-d41d8cd9.tpl.js.expect @@ -0,0 +1 @@ +define("family/name/a-d41d8cd9.tpl", [], ""); diff --git a/test/cases/project/a-e1f4111c.css.expect b/test/cases/project/a-e1f4111c.css.expect new file mode 100644 index 0000000..fd14e9d --- /dev/null +++ b/test/cases/project/a-e1f4111c.css.expect @@ -0,0 +1,5 @@ +/*! define family/name/a-e1f4111c.css */ +/*! import ./b-aa896723.css */ +ul { + margin: 0; +} diff --git a/test/cases/project/a-f5f5c4f5.js.expect b/test/cases/project/a-f5f5c4f5.js.expect new file mode 100644 index 0000000..23f1123 --- /dev/null +++ b/test/cases/project/a-f5f5c4f5.js.expect @@ -0,0 +1,8 @@ +define("family/name/a-f5f5c4f5", [ "./a-670b8177.handlebars", "./a-d41d8cd9.json", "./a-2cee5097.html", "./a-d41d8cd9.tpl", "./a-e1f4111c.css", "./b-e5f793e3", "arale/base/1.1.1/base", "arale/class/1.1.0/class", "arale/events/1.1.0/events" ], function(require) { + require("./a-670b8177.handlebars"); + require("./a-d41d8cd9.json"); + require("./a-2cee5097.html"); + require("./a-d41d8cd9.tpl"); + require("./a-e1f4111c.css"); + require("./b-e5f793e3.js"); +}); diff --git a/test/cases/project/b-aa896723.css.expect b/test/cases/project/b-aa896723.css.expect new file mode 100644 index 0000000..bfeaa53 --- /dev/null +++ b/test/cases/project/b-aa896723.css.expect @@ -0,0 +1,5 @@ +/*! define family/name/b-aa896723.css */ +/*! import alice/list/1.0.1/list.css */ +html { + margin: 0; +} diff --git a/test/cases/project/b-e5f793e3.js.expect b/test/cases/project/b-e5f793e3.js.expect new file mode 100644 index 0000000..f190278 --- /dev/null +++ b/test/cases/project/b-e5f793e3.js.expect @@ -0,0 +1,3 @@ +define("family/name/b-e5f793e3", [ "arale/base/1.1.1/base", "arale/class/1.1.0/class", "arale/events/1.1.0/events" ], function(require) { + require("arale/base/1.1.1/base"); +}); diff --git a/test/cases/style/a-debug.css.js.expect b/test/cases/style/a-debug.css.js.expect index 6030af6..17191e5 100644 --- a/test/cases/style/a-debug.css.js.expect +++ b/test/cases/style/a-debug.css.js.expect @@ -1,3 +1,3 @@ define("arale/widget/1.0.0/a-debug.css", [], function() { - seajs.importStyle('@charset "utf-8";.arale-widget-1_0_0 body{_color:red}.arale-widget-1_0_0 .content li,.arale-widget-1_0_0 .content a{*display:inline-block}@media screen{.arale-widget-1_0_0 body{font-size:13px}.arale-widget-1_0_0 .content li{background-color:#fff}}@keyframes foo{0%{top:0;left:0}100%{top:100px;left:100%}}:root .arale-widget-1_0_0 .content{font-size:12px}'); + seajs.importStyle('@charset "utf-8";.arale-widget-1_0_0 body{_color:red}.arale-widget-1_0_0 .content li,.arale-widget-1_0_0 .content a{*display:inline-block;background:url(a.jpg)}@media screen{.arale-widget-1_0_0 body{font-size:13px}.arale-widget-1_0_0 .content li{background-color:#fff}}@keyframes foo{0%{top:0;left:0}100%{top:100px;left:100%}}:root .arale-widget-1_0_0 .content{font-size:12px}'); }); diff --git a/test/cases/style/a.css b/test/cases/style/a.css index c62d865..781a186 100644 --- a/test/cases/style/a.css +++ b/test/cases/style/a.css @@ -3,13 +3,14 @@ @import url("fineprint.css"); body { - + _color: red; } /* { } */ .content li, .content a { *display: inline-block; + background: url("a.jpg") } @media screen { diff --git a/test/cases/style/a.css.js.expect b/test/cases/style/a.css.js.expect index 9067c9e..44f4032 100644 --- a/test/cases/style/a.css.js.expect +++ b/test/cases/style/a.css.js.expect @@ -1,3 +1,3 @@ define("arale/widget/1.0.0/a.css", [], function() { - seajs.importStyle('@charset "utf-8";.arale-widget-1_0_0 body{_color:red}.arale-widget-1_0_0 .content li,.arale-widget-1_0_0 .content a{*display:inline-block}@media screen{.arale-widget-1_0_0 body{font-size:13px}.arale-widget-1_0_0 .content li{background-color:#fff}}@keyframes foo{0%{top:0;left:0}100%{top:100px;left:100%}}:root .arale-widget-1_0_0 .content{font-size:12px}'); + seajs.importStyle('@charset "utf-8";.arale-widget-1_0_0 body{_color:red}.arale-widget-1_0_0 .content li,.arale-widget-1_0_0 .content a{*display:inline-block;background:url(a.jpg)}@media screen{.arale-widget-1_0_0 body{font-size:13px}.arale-widget-1_0_0 .content li{background-color:#fff}}@keyframes foo{0%{top:0;left:0}100%{top:100px;left:100%}}:root .arale-widget-1_0_0 .content{font-size:12px}'); }); From b1281da9c4efd8c6a96d7272cc139d8baae3a7f5 Mon Sep 17 00:00:00 2001 From: popomore Date: Wed, 31 Dec 2014 18:31:05 +0800 Subject: [PATCH 06/14] fix(css2js) export css2js function as same as before --- tasks/lib/css2js.js | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/tasks/lib/css2js.js b/tasks/lib/css2js.js index 253b7af..60ff376 100644 --- a/tasks/lib/css2js.js +++ b/tasks/lib/css2js.js @@ -9,6 +9,14 @@ exports.init = function(grunt) { factoryParser: css2js }); }; +exports.css2js = function(code, id, options, fileObj) { + var tpl = [ + 'define("%s", [], function() {', + "seajs.importStyle('%s')", + '});' + ].join('\n'); + return format(tpl, id, css2js(code, options, fileObj)); +}; function css2js(code, options, fileObj) { var addStyleBox = false; From f9acb0bbe64ec78b33a40f08021dee8b7245ffcf Mon Sep 17 00:00:00 2001 From: popomore Date: Wed, 31 Dec 2014 18:32:23 +0800 Subject: [PATCH 07/14] bump 0.5.0-rc --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index 4a32b2d..bbd956e 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "grunt-cmd-transport", "description": "Transport javascript into cmd.", - "version": "0.4.1", + "version": "0.5.0-rc", "homepage": "https://github.com/spmjs/grunt-cmd-transport", "author": { "name": "Hsiaoming Yang", From ee036ae5af06fa1da5183503ae5a047a7d82debc Mon Sep 17 00:00:00 2001 From: popomore Date: Sun, 4 Jan 2015 13:31:32 +0800 Subject: [PATCH 08/14] fix(script) will not generate hash when require css package --- Gruntfile.js | 3 ++- tasks/lib/script.js | 7 ++++-- test/cases/project/a-05fba13d.js.expect | 8 +++++++ test/cases/project/a-f5f5c4f5.js.expect | 8 ------- test/cases/project/b-0ec4c8ca.js.expect | 4 ++++ test/cases/project/b-e5f793e3.js.expect | 3 --- test/cases/project/b.js | 1 + .../alice/loading/1.0.0/loading-debug.css | 14 +++++++++++ .../alice/loading/1.0.0/loading.css | 1 + .../alice/loading/1.0.0/package.json | 23 +++++++++++++++++++ 10 files changed, 58 insertions(+), 14 deletions(-) create mode 100644 test/cases/project/a-05fba13d.js.expect delete mode 100644 test/cases/project/a-f5f5c4f5.js.expect create mode 100644 test/cases/project/b-0ec4c8ca.js.expect delete mode 100644 test/cases/project/b-e5f793e3.js.expect create mode 100644 test/cases/project/sea-modules/alice/loading/1.0.0/loading-debug.css create mode 100644 test/cases/project/sea-modules/alice/loading/1.0.0/loading.css create mode 100644 test/cases/project/sea-modules/alice/loading/1.0.0/package.json diff --git a/Gruntfile.js b/Gruntfile.js index 22ec519..110c2fc 100644 --- a/Gruntfile.js +++ b/Gruntfile.js @@ -277,7 +277,8 @@ module.exports = function(grunt) { paths: ['test/cases/project/sea-modules'], alias: { 'list': 'alice/list/1.0.1/list.css', - 'base': 'arale/base/1.1.1/base' + 'base': 'arale/base/1.1.1/base', + 'loading': 'alice/loading/1.0.0/loading.css' }, idleading: 'family/name/', hash: true, diff --git a/tasks/lib/script.js b/tasks/lib/script.js index 2151407..617e4dd 100644 --- a/tasks/lib/script.js +++ b/tasks/lib/script.js @@ -90,6 +90,7 @@ exports.init = function(grunt) { } function addHash(v) { + if (v.id.charAt(0) !== '.') return v.id; if (!v.hash) return v.id; var ext = extname(v.id); @@ -202,6 +203,10 @@ exports.init = function(grunt) { var file = getFileInfo(fpath); + if (!file) { + return [{id: id}]; + } + // don't parse no javascript dependencies if (!/\.js$/.test(fpath)) { return [{ @@ -303,6 +308,4 @@ exports.init = function(grunt) { }; } } - - }; diff --git a/test/cases/project/a-05fba13d.js.expect b/test/cases/project/a-05fba13d.js.expect new file mode 100644 index 0000000..e98a203 --- /dev/null +++ b/test/cases/project/a-05fba13d.js.expect @@ -0,0 +1,8 @@ +define("family/name/a-05fba13d", [ "./a-670b8177.handlebars", "./a-d41d8cd9.json", "./a-2cee5097.html", "./a-d41d8cd9.tpl", "./a-e1f4111c.css", "./b-0ec4c8ca", "arale/base/1.1.1/base", "arale/class/1.1.0/class", "arale/events/1.1.0/events", "alice/loading/1.0.0/loading.css" ], function(require) { + require("./a-670b8177.handlebars"); + require("./a-d41d8cd9.json"); + require("./a-2cee5097.html"); + require("./a-d41d8cd9.tpl"); + require("./a-e1f4111c.css"); + require("./b-0ec4c8ca.js"); +}); diff --git a/test/cases/project/a-f5f5c4f5.js.expect b/test/cases/project/a-f5f5c4f5.js.expect deleted file mode 100644 index 23f1123..0000000 --- a/test/cases/project/a-f5f5c4f5.js.expect +++ /dev/null @@ -1,8 +0,0 @@ -define("family/name/a-f5f5c4f5", [ "./a-670b8177.handlebars", "./a-d41d8cd9.json", "./a-2cee5097.html", "./a-d41d8cd9.tpl", "./a-e1f4111c.css", "./b-e5f793e3", "arale/base/1.1.1/base", "arale/class/1.1.0/class", "arale/events/1.1.0/events" ], function(require) { - require("./a-670b8177.handlebars"); - require("./a-d41d8cd9.json"); - require("./a-2cee5097.html"); - require("./a-d41d8cd9.tpl"); - require("./a-e1f4111c.css"); - require("./b-e5f793e3.js"); -}); diff --git a/test/cases/project/b-0ec4c8ca.js.expect b/test/cases/project/b-0ec4c8ca.js.expect new file mode 100644 index 0000000..bc952c6 --- /dev/null +++ b/test/cases/project/b-0ec4c8ca.js.expect @@ -0,0 +1,4 @@ +define("family/name/b-0ec4c8ca", [ "arale/base/1.1.1/base", "arale/class/1.1.0/class", "arale/events/1.1.0/events", "alice/loading/1.0.0/loading.css" ], function(require) { + require("arale/base/1.1.1/base"); + require("alice/loading/1.0.0/loading.css"); +}); diff --git a/test/cases/project/b-e5f793e3.js.expect b/test/cases/project/b-e5f793e3.js.expect deleted file mode 100644 index f190278..0000000 --- a/test/cases/project/b-e5f793e3.js.expect +++ /dev/null @@ -1,3 +0,0 @@ -define("family/name/b-e5f793e3", [ "arale/base/1.1.1/base", "arale/class/1.1.0/class", "arale/events/1.1.0/events" ], function(require) { - require("arale/base/1.1.1/base"); -}); diff --git a/test/cases/project/b.js b/test/cases/project/b.js index 1a6e688..61869ca 100644 --- a/test/cases/project/b.js +++ b/test/cases/project/b.js @@ -1,3 +1,4 @@ define(function(require) { require('base'); + require('loading'); }); diff --git a/test/cases/project/sea-modules/alice/loading/1.0.0/loading-debug.css b/test/cases/project/sea-modules/alice/loading/1.0.0/loading-debug.css new file mode 100644 index 0000000..240291d --- /dev/null +++ b/test/cases/project/sea-modules/alice/loading/1.0.0/loading-debug.css @@ -0,0 +1,14 @@ +/*! define alice/loading/1.0.0/loading-debug.css */ +/* alice.loading 样式模块 */ + +.ui-loading { + width: 50px; + height: 50px; + background-repeat: no-repeat; + background-image: url(''); + *background-image: url(https://i.alipayobjects.com/e/201305/M9UQl3TuH.gif); + text-align:center; + line-height: 50px; + font-size: 11px; + color: #777; +} diff --git a/test/cases/project/sea-modules/alice/loading/1.0.0/loading.css b/test/cases/project/sea-modules/alice/loading/1.0.0/loading.css new file mode 100644 index 0000000..8ca19e7 --- /dev/null +++ b/test/cases/project/sea-modules/alice/loading/1.0.0/loading.css @@ -0,0 +1 @@ +.ui-loading{width:50px;height:50px;background-repeat:no-repeat;background-image:url('');*background-image:url(https://i.alipayobjects.com/e/201305/M9UQl3TuH.gif);text-align:center;line-height:50px;font-size:11px;color:#777} diff --git a/test/cases/project/sea-modules/alice/loading/1.0.0/package.json b/test/cases/project/sea-modules/alice/loading/1.0.0/package.json new file mode 100644 index 0000000..a3499ea --- /dev/null +++ b/test/cases/project/sea-modules/alice/loading/1.0.0/package.json @@ -0,0 +1,23 @@ +{ + "family": "alice", + "name": "loading", + "description": "alice loading style.", + "version": "1.0.0", + "homepage": "http://aliceui.org/loading/", + "keywords": [ + "加载中" + ], + "repository": { + "type": "git", + "url": "https://github.com/aliceui/loading/" + }, + "bugs": { + "url": "https://github.com/aliceui/loading/issues" + }, + "spm": { + "alias": {}, + "output": [ + "loading.css" + ] + } +} From 5c9681c766c17a090033be80779605b887e29b9f Mon Sep 17 00:00:00 2001 From: popomore Date: Sun, 4 Jan 2015 13:34:03 +0800 Subject: [PATCH 09/14] bump 0.5.0-rc.1 --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index bbd956e..f5e3246 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "grunt-cmd-transport", "description": "Transport javascript into cmd.", - "version": "0.5.0-rc", + "version": "0.5.0-rc.1", "homepage": "https://github.com/spmjs/grunt-cmd-transport", "author": { "name": "Hsiaoming Yang", From 7887395c2fafaf7a8aca1f35e63f0d7360fa270a Mon Sep 17 00:00:00 2001 From: popomore Date: Sun, 4 Jan 2015 15:25:29 +0800 Subject: [PATCH 10/14] feat(text) clean space in html using htmlclean #78 --- package.json | 1 + tasks/lib/text.js | 8 ++------ test/cases/project/a-2cee5097.html.js.expect | 2 +- test/cases/text/foo-debug.html.js.expect | 2 +- test/cases/text/foo.html | 6 +++++- test/cases/text/foo.html.js.expect | 2 +- 6 files changed, 11 insertions(+), 10 deletions(-) diff --git a/package.json b/package.json index f5e3246..19ab4da 100644 --- a/package.json +++ b/package.json @@ -32,6 +32,7 @@ "css": "~1.4.0", "css2str": "~0.1.1", "handlebars": "1.0.11", + "htmlclean": "~2.2.3", "relative": "~1.2.0", "uglify-js": "~2.2.5" }, diff --git a/tasks/lib/text.js b/tasks/lib/text.js index ad68e88..ab6b9ac 100644 --- a/tasks/lib/text.js +++ b/tasks/lib/text.js @@ -1,3 +1,4 @@ +var htmlclean = require('htmlclean'); var commonParser = require('./common'); exports.init = function(grunt) { @@ -8,11 +9,6 @@ exports.init = function(grunt) { }; function getCode(data) { - data = data.split(/\r\n|\r|\n/) - .map(function(line) { - return line.replace(/\\/g, '\\\\'); - }) - .join('\n') - .replace(/\"/g, '\\\"'); + data = htmlclean(data).replace(/(\"|\'|\\)/g, '\\$1'); return '"' + data + '"'; } diff --git a/test/cases/project/a-2cee5097.html.js.expect b/test/cases/project/a-2cee5097.html.js.expect index dd95ba5..d8f7029 100644 --- a/test/cases/project/a-2cee5097.html.js.expect +++ b/test/cases/project/a-2cee5097.html.js.expect @@ -1 +1 @@ -define("family/name/a-2cee5097.html", [], "
\n
    \n
  • a
  • \n
      \n
\n"); +define("family/name/a-2cee5097.html", [], "
  • a
    "); diff --git a/test/cases/text/foo-debug.html.js.expect b/test/cases/text/foo-debug.html.js.expect index da47cad..b422b80 100644 --- a/test/cases/text/foo-debug.html.js.expect +++ b/test/cases/text/foo-debug.html.js.expect @@ -1 +1 @@ -define("foo-debug.html", [], "
    foo / bar \\ baz
    \n"); +define("foo-debug.html", [], "

    \\n The 'clean' \\ \"HTML is here.\" /

    "); diff --git a/test/cases/text/foo.html b/test/cases/text/foo.html index 4ca63db..8e3d413 100644 --- a/test/cases/text/foo.html +++ b/test/cases/text/foo.html @@ -1 +1,5 @@ -
    foo / bar \ baz
    +
    +

    \n + The 'clean' \ "HTML is here." /

    + +
    diff --git a/test/cases/text/foo.html.js.expect b/test/cases/text/foo.html.js.expect index c2dfea2..4e164f7 100644 --- a/test/cases/text/foo.html.js.expect +++ b/test/cases/text/foo.html.js.expect @@ -1 +1 @@ -define("foo.html", [], "
    foo / bar \\ baz
    \n"); +define("foo.html", [], "

    \\n The 'clean' \\ \"HTML is here.\" /

    "); From e9e5f15afc43d320159fac31961257c73443e61c Mon Sep 17 00:00:00 2001 From: popomore Date: Tue, 13 Jan 2015 16:07:38 +0800 Subject: [PATCH 11/14] update --- Gruntfile.js | 1 + test/cases/project/a.js | 1 + .../arale/dialog/1.3.1/confirmbox-debug.js | 713 +++++ .../arale/dialog/1.3.1/confirmbox.js | 1 + .../arale/dialog/1.3.1/dialog-debug.css | 121 + .../arale/dialog/1.3.1/dialog-debug.js | 418 +++ .../sea-modules/arale/dialog/1.3.1/dialog.css | 1 + .../sea-modules/arale/dialog/1.3.1/dialog.js | 1 + .../arale/dialog/1.3.1/package.json | 39 + .../arale/easing/1.0.0/easing-debug.js | 190 ++ .../sea-modules/arale/easing/1.0.0/easing.js | 1 + .../arale/easing/1.0.0/package.json | 28 + .../iframe-shim/1.0.2/iframe-shim-debug.js | 76 + .../arale/iframe-shim/1.0.2/iframe-shim.js | 1 + .../arale/iframe-shim/1.0.2/package.json | 32 + .../arale/overlay/1.1.4/mask-debug.js | 244 ++ .../sea-modules/arale/overlay/1.1.4/mask.js | 1 + .../arale/overlay/1.1.4/overlay-debug.js | 204 ++ .../arale/overlay/1.1.4/overlay.js | 1 + .../arale/overlay/1.1.4/package.json | 29 + .../arale/position/1.0.1/package.json | 29 + .../arale/position/1.0.1/position-debug.js | 199 ++ .../arale/position/1.0.1/position.js | 1 + .../arale/templatable/0.9.2/package.json | 32 + .../templatable/0.9.2/templatable-debug.js | 149 + .../arale/templatable/0.9.2/templatable.js | 1 + .../arale/widget/1.1.1/package.json | 27 + .../arale/widget/1.1.1/widget-debug.js | 468 +++ .../sea-modules/arale/widget/1.1.1/widget.js | 1 + .../handlebars/1.0.2/handlebars-debug.js | 2760 +++++++++++++++++ .../gallery/handlebars/1.0.2/handlebars.js | 2 + .../gallery/handlebars/1.0.2/package.json | 20 + .../gallery/handlebars/1.0.2/runtime-debug.js | 324 ++ .../gallery/handlebars/1.0.2/runtime.js | 1 + 34 files changed, 6117 insertions(+) create mode 100644 test/cases/project/sea-modules/arale/dialog/1.3.1/confirmbox-debug.js create mode 100644 test/cases/project/sea-modules/arale/dialog/1.3.1/confirmbox.js create mode 100644 test/cases/project/sea-modules/arale/dialog/1.3.1/dialog-debug.css create mode 100644 test/cases/project/sea-modules/arale/dialog/1.3.1/dialog-debug.js create mode 100644 test/cases/project/sea-modules/arale/dialog/1.3.1/dialog.css create mode 100644 test/cases/project/sea-modules/arale/dialog/1.3.1/dialog.js create mode 100644 test/cases/project/sea-modules/arale/dialog/1.3.1/package.json create mode 100644 test/cases/project/sea-modules/arale/easing/1.0.0/easing-debug.js create mode 100644 test/cases/project/sea-modules/arale/easing/1.0.0/easing.js create mode 100644 test/cases/project/sea-modules/arale/easing/1.0.0/package.json create mode 100644 test/cases/project/sea-modules/arale/iframe-shim/1.0.2/iframe-shim-debug.js create mode 100644 test/cases/project/sea-modules/arale/iframe-shim/1.0.2/iframe-shim.js create mode 100644 test/cases/project/sea-modules/arale/iframe-shim/1.0.2/package.json create mode 100644 test/cases/project/sea-modules/arale/overlay/1.1.4/mask-debug.js create mode 100644 test/cases/project/sea-modules/arale/overlay/1.1.4/mask.js create mode 100644 test/cases/project/sea-modules/arale/overlay/1.1.4/overlay-debug.js create mode 100644 test/cases/project/sea-modules/arale/overlay/1.1.4/overlay.js create mode 100644 test/cases/project/sea-modules/arale/overlay/1.1.4/package.json create mode 100644 test/cases/project/sea-modules/arale/position/1.0.1/package.json create mode 100644 test/cases/project/sea-modules/arale/position/1.0.1/position-debug.js create mode 100644 test/cases/project/sea-modules/arale/position/1.0.1/position.js create mode 100644 test/cases/project/sea-modules/arale/templatable/0.9.2/package.json create mode 100644 test/cases/project/sea-modules/arale/templatable/0.9.2/templatable-debug.js create mode 100644 test/cases/project/sea-modules/arale/templatable/0.9.2/templatable.js create mode 100644 test/cases/project/sea-modules/arale/widget/1.1.1/package.json create mode 100644 test/cases/project/sea-modules/arale/widget/1.1.1/widget-debug.js create mode 100644 test/cases/project/sea-modules/arale/widget/1.1.1/widget.js create mode 100644 test/cases/project/sea-modules/gallery/handlebars/1.0.2/handlebars-debug.js create mode 100644 test/cases/project/sea-modules/gallery/handlebars/1.0.2/handlebars.js create mode 100644 test/cases/project/sea-modules/gallery/handlebars/1.0.2/package.json create mode 100644 test/cases/project/sea-modules/gallery/handlebars/1.0.2/runtime-debug.js create mode 100644 test/cases/project/sea-modules/gallery/handlebars/1.0.2/runtime.js diff --git a/Gruntfile.js b/Gruntfile.js index 110c2fc..7c3a9cc 100644 --- a/Gruntfile.js +++ b/Gruntfile.js @@ -278,6 +278,7 @@ module.exports = function(grunt) { alias: { 'list': 'alice/list/1.0.1/list.css', 'base': 'arale/base/1.1.1/base', + 'confirmbox': 'arale/dialog/1.3.1/confirmbox', 'loading': 'alice/loading/1.0.0/loading.css' }, idleading: 'family/name/', diff --git a/test/cases/project/a.js b/test/cases/project/a.js index 7c10843..05d5952 100644 --- a/test/cases/project/a.js +++ b/test/cases/project/a.js @@ -5,4 +5,5 @@ define(function(require) { require('./a.tpl'); require('./a.css'); require('./b'); + require('confirmbox'); }); diff --git a/test/cases/project/sea-modules/arale/dialog/1.3.1/confirmbox-debug.js b/test/cases/project/sea-modules/arale/dialog/1.3.1/confirmbox-debug.js new file mode 100644 index 0000000..336e6cb --- /dev/null +++ b/test/cases/project/sea-modules/arale/dialog/1.3.1/confirmbox-debug.js @@ -0,0 +1,713 @@ +define("arale/dialog/1.3.1/confirmbox-debug", [ "$-debug", "./dialog-debug", "arale/overlay/1.1.4/overlay-debug", "arale/position/1.0.1/position-debug", "arale/iframe-shim/1.0.2/iframe-shim-debug", "arale/widget/1.1.1/widget-debug", "arale/base/1.1.1/base-debug", "arale/class/1.1.0/class-debug", "arale/events/1.1.0/events-debug", "arale/overlay/1.1.4/mask-debug", "arale/templatable/0.9.2/templatable-debug", "gallery/handlebars/1.0.2/handlebars-debug", "./dialog-debug.handlebars", "./confirmbox-debug.handlebars", "./dialog-debug.css" ], function(require, exports, module) { + var $ = require("$-debug"), Dialog = require("./dialog-debug"); + var template = require("./confirmbox-debug.handlebars"); + require("./dialog-debug.css"); + // ConfirmBox + // ------- + // ConfirmBox 是一个有基础模板和样式的对话框组件。 + var ConfirmBox = Dialog.extend({ + attrs: { + title: "默认标题", + confirmTpl: '确定', + cancelTpl: '取消', + message: "默认内容" + }, + setup: function() { + ConfirmBox.superclass.setup.call(this); + var model = { + classPrefix: this.get("classPrefix"), + message: this.get("message"), + title: this.get("title"), + confirmTpl: this.get("confirmTpl"), + cancelTpl: this.get("cancelTpl"), + hasFoot: this.get("confirmTpl") || this.get("cancelTpl") + }; + this.set("content", template(model)); + }, + events: { + "click [data-role=confirm]": function(e) { + e.preventDefault(); + this.trigger("confirm"); + }, + "click [data-role=cancel]": function(e) { + e.preventDefault(); + this.trigger("cancel"); + this.hide(); + } + }, + _onChangeMessage: function(val) { + this.$("[data-role=message]").html(val); + }, + _onChangeTitle: function(val) { + this.$("[data-role=title]").html(val); + }, + _onChangeConfirmTpl: function(val) { + this.$("[data-role=confirm]").html(val); + }, + _onChangeCancelTpl: function(val) { + this.$("[data-role=cancel]").html(val); + } + }); + ConfirmBox.alert = function(message, callback, options) { + var defaults = { + message: message, + title: "", + cancelTpl: "", + closeTpl: "", + onConfirm: function() { + callback && callback(); + this.hide(); + } + }; + new ConfirmBox($.extend(null, defaults, options)).show().after("hide", function() { + this.destroy(); + }); + }; + ConfirmBox.confirm = function(message, title, onConfirm, onCancel, options) { + // support confirm(message, title, onConfirm, options) + if (typeof onCancel === "object" && !options) { + options = onCancel; + } + var defaults = { + message: message, + title: title || "确认框", + closeTpl: "", + onConfirm: function() { + onConfirm && onConfirm(); + this.hide(); + }, + onCancel: function() { + onCancel && onCancel(); + this.hide(); + } + }; + new ConfirmBox($.extend(null, defaults, options)).show().after("hide", function() { + this.destroy(); + }); + }; + ConfirmBox.show = function(message, callback, options) { + var defaults = { + message: message, + title: "", + confirmTpl: false, + cancelTpl: false + }; + new ConfirmBox($.extend(null, defaults, options)).show().before("hide", function() { + callback && callback(); + }).after("hide", function() { + this.destroy(); + }); + }; + module.exports = ConfirmBox; + module.exports.outerBoxClass = "arale-dialog-1_3_1"; +}); + +define("arale/dialog/1.3.1/dialog-debug", [ "$-debug", "arale/overlay/1.1.4/overlay-debug", "arale/position/1.0.1/position-debug", "arale/iframe-shim/1.0.2/iframe-shim-debug", "arale/widget/1.1.1/widget-debug", "arale/base/1.1.1/base-debug", "arale/class/1.1.0/class-debug", "arale/events/1.1.0/events-debug", "arale/overlay/1.1.4/mask-debug", "arale/templatable/0.9.2/templatable-debug", "gallery/handlebars/1.0.2/handlebars-debug" ], function(require, exports, module) { + var $ = require("$-debug"), Overlay = require("arale/overlay/1.1.4/overlay-debug"), mask = require("arale/overlay/1.1.4/mask-debug"), Events = require("arale/events/1.1.0/events-debug"), Templatable = require("arale/templatable/0.9.2/templatable-debug"); + // Dialog + // --- + // Dialog 是通用对话框组件,提供显隐关闭、遮罩层、内嵌iframe、内容区域自定义功能。 + // 是所有对话框类型组件的基类。 + var Dialog = Overlay.extend({ + Implements: Templatable, + attrs: { + // 模板 + template: require("arale/dialog/1.3.1/dialog-debug.handlebars"), + // 对话框触发点 + trigger: { + value: null, + getter: function(val) { + return $(val); + } + }, + // 统一样式前缀 + classPrefix: "ui-dialog", + // 指定内容元素,可以是 url 地址 + content: { + value: null, + setter: function(val) { + // 判断是否是 url 地址 + if (/^(https?:\/\/|\/|\.\/|\.\.\/)/.test(val)) { + this._type = "iframe"; + // 用 ajax 的方式而不是 iframe 进行载入 + if (val.indexOf("?ajax") > 0 || val.indexOf("&ajax") > 0) { + this._ajax = true; + } + } + return val; + } + }, + // 是否有背景遮罩层 + hasMask: true, + // 关闭按钮可以自定义 + closeTpl: "×", + // 默认宽度 + width: 500, + // 默认高度 + height: null, + // iframe 类型时,dialog 的最初高度 + initialHeight: 300, + // 简单的动画效果 none | fade + effect: "none", + // 不用解释了吧 + zIndex: 999, + // 是否自适应高度 + autoFit: true, + // 默认定位居中 + align: { + value: { + selfXY: [ "50%", "50%" ], + baseXY: [ "50%", "42%" ] + }, + getter: function(val) { + // 高度超过窗口的 42/50 浮层头部顶住窗口 + // https://github.com/aralejs/dialog/issues/41 + if (this.element.height() > $(window).height() * .84) { + return { + selfXY: [ "50%", "0" ], + baseXY: [ "50%", "0" ] + }; + } + return val; + } + } + }, + parseElement: function() { + this.set("model", { + classPrefix: this.get("classPrefix") + }); + Dialog.superclass.parseElement.call(this); + this.contentElement = this.$("[data-role=content]"); + // 必要的样式 + this.contentElement.css({ + height: "100%", + zoom: 1 + }); + // 关闭按钮先隐藏 + // 后面当 onRenderCloseTpl 时,如果 closeTpl 不为空,会显示出来 + // 这样写是为了回避 arale.base 的一个问题: + // 当属性初始值为''时,不会进入 onRender 方法 + // https://github.com/aralejs/base/issues/7 + this.$("[data-role=close]").hide(); + }, + events: { + "click [data-role=close]": function(e) { + e.preventDefault(); + this.hide(); + } + }, + show: function() { + // iframe 要在载入完成才显示 + if (this._type === "iframe") { + // ajax 读入内容并 append 到容器中 + if (this._ajax) { + this._ajaxHtml(); + } else { + // iframe 还未请求完,先设置一个固定高度 + !this.get("height") && this.contentElement.css("height", this.get("initialHeight")); + this._showIframe(); + } + } + Dialog.superclass.show.call(this); + return this; + }, + hide: function() { + // 把 iframe 状态复原 + if (this._type === "iframe" && this.iframe) { + this.iframe.attr({ + src: "javascript:'';" + }); + // 原来只是将 iframe 的状态复原 + // 但是发现在 IE6 下,将 src 置为 javascript:''; 会出现 404 页面 + this.iframe.remove(); + this.iframe = null; + } + Dialog.superclass.hide.call(this); + clearInterval(this._interval); + delete this._interval; + return this; + }, + destroy: function() { + this.element.remove(); + this._hideMask(); + clearInterval(this._interval); + return Dialog.superclass.destroy.call(this); + }, + setup: function() { + Dialog.superclass.setup.call(this); + this._setupTrigger(); + this._setupMask(); + this._setupKeyEvents(); + this._setupFocus(); + toTabed(this.element); + toTabed(this.get("trigger")); + // 默认当前触发器 + this.activeTrigger = this.get("trigger").eq(0); + }, + // onRender + // --- + _onRenderContent: function(val) { + if (this._type !== "iframe") { + var value; + // 有些情况会报错 + try { + value = $(val); + } catch (e) { + value = []; + } + if (value[0]) { + this.contentElement.empty().append(value); + } else { + this.contentElement.empty().html(val); + } + // #38 #44 + this._setPosition(); + } + }, + _onRenderCloseTpl: function(val) { + if (val === "") { + this.$("[data-role=close]").html(val).hide(); + } else { + this.$("[data-role=close]").html(val).show(); + } + }, + // 覆盖 overlay,提供动画 + _onRenderVisible: function(val) { + if (val) { + if (this.get("effect") === "fade") { + // 固定 300 的动画时长,暂不可定制 + this.element.fadeIn(300); + } else { + this.element.show(); + } + } else { + this.element.hide(); + } + }, + // 私有方法 + // --- + // 绑定触发对话框出现的事件 + _setupTrigger: function() { + this.delegateEvents(this.get("trigger"), "click", function(e) { + e.preventDefault(); + // 标识当前点击的元素 + this.activeTrigger = $(e.currentTarget); + this.show(); + }); + }, + // 绑定遮罩层事件 + _setupMask: function() { + var that = this; + // 存放 mask 对应的对话框 + mask._dialogs = mask._dialogs || []; + this.after("show", function() { + if (!this.get("hasMask")) { + return; + } + // not using the z-index + // because multiable dialogs may share same mask + mask.set("zIndex", that.get("zIndex")).show(); + mask.element.insertBefore(that.element); + // 避免重复存放 + var existed = false; + for (var i = 0; i < mask._dialogs.length; i++) { + if (mask._dialogs[i] === that) { + existed = true; + } + } + // 依次存放对应的对话框 + if (!existed) { + mask._dialogs.push(that); + } + }); + this.after("hide", this._hideMask); + }, + // 隐藏 mask + _hideMask: function() { + if (!this.get("hasMask")) { + return; + } + mask._dialogs && mask._dialogs.pop(); + if (mask._dialogs && mask._dialogs.length > 0) { + var last = mask._dialogs[mask._dialogs.length - 1]; + mask.set("zIndex", last.get("zIndex")); + mask.element.insertBefore(last.element); + } else { + mask.hide(); + } + }, + // 绑定元素聚焦状态 + _setupFocus: function() { + this.after("show", function() { + this.element.focus(); + }); + this.after("hide", function() { + // 关于网页中浮层消失后的焦点处理 + // http://www.qt06.com/post/280/ + this.activeTrigger && this.activeTrigger.focus(); + }); + }, + // 绑定键盘事件,ESC关闭窗口 + _setupKeyEvents: function() { + this.delegateEvents($(document), "keyup.esc", function(e) { + if (e.keyCode === 27) { + this.get("visible") && this.hide(); + } + }); + }, + _showIframe: function() { + var that = this; + // 若未创建则新建一个 + if (!this.iframe) { + this._createIframe(); + } + // 开始请求 iframe + this.iframe.attr({ + src: this._fixUrl(), + name: "dialog-iframe" + new Date().getTime() + }); + // 因为在 IE 下 onload 无法触发 + // http://my.oschina.net/liangrockman/blog/24015 + // 所以使用 jquery 的 one 函数来代替 onload + // one 比 on 好,因为它只执行一次,并在执行后自动销毁 + this.iframe.one("load", function() { + // 如果 dialog 已经隐藏了,就不需要触发 onload + if (!that.get("visible")) { + return; + } + // 绑定自动处理高度的事件 + if (that.get("autoFit")) { + clearInterval(that._interval); + that._interval = setInterval(function() { + that._syncHeight(); + }, 300); + } + that._syncHeight(); + that._setPosition(); + that.trigger("complete:show"); + }); + }, + _fixUrl: function() { + var s = this.get("content").match(/([^?#]*)(\?[^#]*)?(#.*)?/); + s.shift(); + s[1] = (s[1] && s[1] !== "?" ? s[1] + "&" : "?") + "t=" + new Date().getTime(); + return s.join(""); + }, + _createIframe: function() { + var that = this; + this.iframe = $("