diff --git a/Gulpfile.js b/Gulpfile.js index 927067b..3822a84 100644 --- a/Gulpfile.js +++ b/Gulpfile.js @@ -1,7 +1,11 @@ var gulp = require('gulp') , watch = require('gulp-watch') , plumber = require('gulp-plumber') - , concat = require('gulp-concat'); + , concat = require('gulp-concat') + , uglify = require('gulp-uglify') + , rename = require('gulp-rename') + + , pkg = require('./package.json'); gulp.task('default', ['localBuild'], function(){ 'use strict'; @@ -19,6 +23,16 @@ gulp.task('localBuild', function(){ .pipe(gulp.dest('public/javascripts/built/')); }); +gulp.task('publish', function(){ + 'use strict'; + + gulp.src('public/javascripts/action.js') + .pipe(rename('action-v' + pkg.version + '.js')) + .pipe(gulp.dest('packages/' + pkg.version + '/')) + .pipe(uglify()) + .pipe(rename('action-v' + pkg.version + '.min.js')) + .pipe(gulp.dest('packages/' + pkg.version + '/')); +}); // gulp.task('default', function () { // watch({ glob: ['public/javascripts/app.js', 'public/javascripts/components/**/*.js'], emitOnGlob: false }) diff --git a/package.json b/package.json index 232abdb..47c0e90 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { - "name": "application-name", - "version": "0.0.1", + "name": "action", + "version": "0.2.0", "private": true, "scripts": { "start": "node app.js" @@ -14,6 +14,7 @@ "gulp": "^3.6.0", "gulp-uglify": "^0.2.1", "gulp-plumber": "^0.6.0", - "gulp-watch": "^0.5.4" + "gulp-watch": "^0.5.4", + "gulp-rename": "^1.2.0" } } diff --git a/packages/0.2.0/action-v0.2.0.js b/packages/0.2.0/action-v0.2.0.js new file mode 100644 index 0000000..ccd59d0 --- /dev/null +++ b/packages/0.2.0/action-v0.2.0.js @@ -0,0 +1,514 @@ +var action = function(){ + 'use strict'; + + var action = { + eventMe: function(objectIn){ + var returnObject = objectIn + , localEvents = {}; + + //set an emitter id for troubleshooting + returnObject.emitterId = Math.ceil(Math.random() * 10000); + + //create the lcoal event Store + returnObject.eventStore = {}; + + returnObject.emit = function(eventNameIn, eventDataIn, localFlag){ + //add the emitterID to this thing + var that = this + , eventStack + , functionToCall + , i + , isLocal = (typeof localFlag !== 'undefined' && localFlag); + + if(isLocal){ + eventStack = that.eventStore[eventNameIn]; + } else { + eventStack = action.eventStore[eventNameIn]; + } + + // if(typeof eventDataIn !== 'undefined'){ + // //we have some event data + // eventData.payload = eventDataIn; + // } + + //emit the event + if(typeof eventStack !== 'undefined'){ + for(i = 0; i < eventStack.length; i ++){ + if(typeof eventStack[i].scope !== 'undefined'){ + eventStack[i].call.apply(eventStack[i].scope,[eventDataIn, that.emitterId]); + }else{ + eventStack[i].call(eventDataIn, that.emitterId); + } + + + if(eventStack[i].once){ + if(isLocal){ + that.silenceLocal(eventNameIn, eventStack[i].call, true); + } else { + that.silence(eventNameIn, eventStack[i].call, true); + } + } + } + } + }; + + returnObject.emitLocal = function(eventNameIn, eventDataIn){ + var that = this; + + that.emit(eventNameIn, eventDataIn, true); + }; + + returnObject.listen = function(eventNameIn, handlerIn, scopeIn, onceIn, localFlagIn){ + var that = this + , i + , newCheck = true + + //attribute holders and such + , eventName = eventNameIn + , handler = handlerIn + , scope = scopeIn + , local = localFlagIn + , once = onceIn + + //variables for later + , eventStack + , newEvent; + + if(typeof eventNameIn === 'object'){ + //we have an object to split up dude + eventName = eventNameIn.eventName; + handler = eventNameIn.handler; + scope = eventNameIn.scope; + once = eventNameIn.once; + local = eventNameIn.local; + } + + eventStack = (typeof local !== 'undefined' && local) ? that.eventStore[eventNameIn] : action.eventStore[eventNameIn]; + newEvent = (typeof local !== 'undefined' && local) ? that : action; + + if(typeof eventStack !== 'undefined'){ + //already exists check to see if the function is already bound + + for(i = 0; i < eventStack.length; i ++){ + if(eventStack[i].call === handler && eventStack[i].once === false){ + newCheck = false; + break; + } + } + + if(newCheck && typeof scopeIn !== 'undefined'){ + eventStack.push({once: false, call: handler, scope: scope}); + }else if(newCheck){ + eventStack.push({once: false, call:handler}); + } + + } else { + //new event + newEvent.eventStore[eventName] = []; //use an array to store functions + if(typeof scopeIn !== 'undefined'){ + newEvent.eventStore[eventName].push({once: false, call: handler, scope: scope}); + }else{ + newEvent.eventStore[eventName].push({once: false, call: handler}); + } + } + }; + + returnObject.listenLocal = function(eventNameIn, handlerIn, scopeIn, onceIn){ + var that = this; + + //convenience function for local listens + if(typeof eventNameIn === 'object'){ + eventNameIn.local = true; + that.listen(eventNameIn); + } else { + that.listen({ + eventName: eventNameIn + , handler: handlerIn + , scope: scopeIn + , once: onceIn + , local: true + }); + } + }; + + returnObject.listenOnce = function(eventNameIn, handlerIn, scopeIn, localFlagIn){ + //same thing as .listen() but is only triggered once + var that = this + , i + , newCheck = true + , eventStack + , newEvent + + , eventName = eventNameIn + , handler = handlerIn + , scope = scopeIn + , localFlag = localFlagIn; + + if(typeof eventNameIn === 'object'){ + eventName = eventNameIn.eventName; + handler = eventNameIn.handler; + scope = eventNameIn.scope; + localFlag = eventNameIn.local; + } + + if(typeof localFlag !== 'undefined' && localFlag){ + //make it local! + eventStack = that.eventStore[eventName]; + newEvent = that; + }else{ + eventStack = action.eventStore[eventName]; + newEvent = action; + } + + if(typeof eventStack !== 'undefined'){ + //already exists check to see if the function is already bound + + for(i = 0; i < eventStack.length; i ++){ + if(eventStack[i].call === handler && eventStack[i].once === true){ + newCheck = false; + break; + } + } + + if(newCheck){ + eventStack.push({once:true, call: handler, scope: scope}); + } + + } else{ + //new event + newEvent.eventStore[eventNameIn] = []; //use an array to store functions + newEvent.eventStore[eventNameIn].push({once:true, call: handler, scope: scope}); + } + }; + + returnObject.listenOnceLocal = function(eventNameIn, handlerIn, scopeIn){ + var that = this; + + //same thing as .listen() but is only triggered once + if(typeof eventNameIn === 'object'){ + eventNameIn.local = true; + that.listenLocal(eventNameIn); + }else{ + that.listenLocal({ + eventName: eventNameIn + , handler: handlerIn + , scope: scopeIn + , local: true + }); + } + }; + + returnObject.silence = function(eventNameIn, handlerIn, onceIn, scopeIn, localFlagIn){ + //localize variables + var that = this + , i + , truthy = false + , eventName = eventNameIn + , handler = handlerIn + , once = onceIn + , scope = scopeIn + , localFlag = localFlagIn + , store; + + if(typeof eventNameIn === 'object'){ + // passed in a collection of params instead of params + eventName = eventNameIn.eventName; + handler = eventNameIn.handler; + once = eventNameIn.once; + scope = eventNameIn.scope; + localFlag = eventNameIn.local; + } + + //local or global event store? + store = (typeof localFlag === 'undefined' || !localFlag) ? action : that; + + if(typeof store.eventStore[eventName] === 'undefined'){ + //if there is no event with a name... return nothing + return; + } + + //there is an event that matches... proceed + for(i = 0; i < store.eventStore[eventName].length; i ++){ + //reset this variable + truthy = false; + + if(typeof handler !== 'undefined'){ + //function is passed in + if(typeof scope !== 'undefined'){ + //scope is passed in... + if(typeof once === 'boolean'){ + // function + scope + once provides the match + truthy = (handler === store.eventStore[eventName][i].call && scope === store.eventStore[eventName][i].scope && once === store.eventStore[eventName][i].once); + } else { + //function + scope provides the match + truthy = (handler === store.eventStore[eventName][i].call && scope === store.eventStore[eventName][i].scope); + } + } else { + //function + once in for the match + if(typeof once === 'boolean'){ + truthy = (handler === store.eventStore[eventName][i].call && store.eventStore[eventName][i].once === once); + } else { + truthy = (handler === store.eventStore[eventName][i].call); + } + } + } else { + //no function unbind everything by resetting + store.eventStore[eventName] = []; + + //and exit + break; + } + + if(truthy){ + //remove this bad boy + store.eventStore[eventName].splice(i,1); + } + + } + }; + + returnObject.silenceLocal = function(eventNameIn, handlerIn, onceIn, scopeIn){ + var that = this; + + //essentially a convenience function. + if(typeof eventNameIn === 'object'){ + eventNameIn.local = true; + that.silence(eventNameIn); + }else{ + that.silence({ + eventName: eventNameIn + , handler: handlerIn + , once: onceIn + , scope: scopeIn + , local: true + }); + } + }; + + //Event Based state machine + returnObject.requiredEvent = function(name, callback, context, fireMultipleIn){ + var that = this + , stateUpdate; + + that._fireMultiple = (typeof fireMultipleIn !== 'undefined') ? fireMultipleIn : false; + + //init some hidden storage if needed + if(typeof that.stateEvents === 'undefined'){ + that.stateEvents = {}; + } + + if(typeof that._triggeredStateReady === 'undefined'){ + that._triggeredStateReady = false; + } + + that.stateEvents[name] = false; + + stateUpdate = that.stateUpdate(name, that.stateEvents); + + that.listen(name, stateUpdate, that); + that.listen(name, callback, context); + }; + + returnObject.stateUpdate = function(nameIn, stateEventsIn){ + var name = nameIn + , stateEvents = stateEventsIn + , that = this; + + return function(){ + var truthy = true + , key; + + if(typeof stateEvents[name] !== 'undefined'){ + stateEvents[name] = true; + + for(key in stateEvents){ + truthy = truthy && stateEvents[key]; + } + + if(truthy){ + if(!that._triggeredStateReady || that._fireMultiple){ + //feels like a little bit of a hack. + // lets the data finish propogating before triggering the call + setTimeout(that.stateReady, 100); + that._triggeredStateReady = true; + } + } + } + }; + }; + + returnObject.stateReady = function(){ + //this is a default action when all required events have been completed. + // needs to be overridden if you want to do something real + console.log('ready!'); + }; + + returnObject.listen('system:trace', function(emitterIDIn){ + var that = this; + + if(that.emitterId === emitterIDIn){ + that.emit('system:addTraced', that); + } + }, returnObject); + + //execute the init function if it exists + if(typeof returnObject.init === 'function'){ + returnObject.init.apply(returnObject); + } + + return returnObject; + } + + , modelMe: function(objectIn){ + //this is the module for creating a data model object + var that = this + , newModel = that.eventMe({}) + , attributes = {} + , changes = []; + + newModel.get = function(attributeName){ + return attributes[attributeName]; + }; + + newModel.set = function(attributeName, attributeValue){ + var that = this + , key; + + if(typeof attributeName === 'object'){ + //well... this is an object... iterate and rock on + for(key in attributeName){ + if(attributeName.hasOwnProperty(key)){ + //this attribute does not belong to the prototype. Good. + + //TODO: maybe make this do a deep copy to prevent + // pass by reference or switch to clone() + if(key !== 'destroy' && key !== 'fetch' && key !== 'save' && typeof attributeName[key] !== 'function'){ + attributes[key] = attributeName[key]; + that.emitLocal('attribute:changed', key); + } else { + that[key] = attributeName[key]; + } + } + } + } else{ + if(attributeName !== 'destroy' && attributeName !== 'fetch' && attributeName !== 'save'){ + attributes[attributeName] = attributeValue; + that.emitLocal('attribute:changed', attributeName); + } else { + that[attributeName] = attributeValue; + } + } + } + + newModel.flatten = function(){ + return attributes; + } + + newModel.fetch = function(){ + var that = this + , requestUrl = that.get('url'); + + if(typeof requestUrl !== 'undefined'){ + + } else { + //TODO: probably should trigger some sort of error... + throw new action.Error('http', 'No URL defined'); + } + } + + newModel.save = function(){ + //TODO make this talk to a server with the URL + //TODO make it only mark the saved changes clear + var that = this + , requestUrl = that.get('url'); + + if(typeof requestUrl !== 'undefined'){ + + //only do this on success... + changes = []; + } else { + //TODO: probably should trigger some sort of error... + throw new action.Error('http', 'No URL defined'); + } + } + + newModel.getChanges = function(){ + return changes; + } + + newModel.clear = function(){ + attributes = {}; + } + + newModel.destroy = function(){ + //TODO not really working... should get rid of this thing + // and all of its parameters + var that = this + , key; + + setTimeout(function(){ + // delete me; + },0); // not quite working... + + for(key in that){ + // delete this[key]; + } + + //TODO this still doesn't kill the attributes or changes + // private data + } + + newModel.set(objectIn); //set the inital attributes + + newModel.listenLocal('attribute:changed', function(nameIn){ + changes.push(nameIn); + }, this); //maybe eliminate this 'this' + + if(typeof newModel.init === 'function'){ + newModel.init.apply(newModel); + } + + return newModel; + } + + , trace: function(emitterIdIn){ + //log out the function that has the emitterId attached + + //create the traced object/stack + action.traced = action.modelMe({ + stack: [] + , emitterId: emitterIdIn + }); + + action.traced.listen('system:addTraced', function(objectIn){ + var that = this; + + that.get('stack').push(objectIn); + }, action.traced); + + //trigger the event that will cause the trace + action.events.emit('system:trace', emitterIdIn); + } + + , Error: function(typeIn, messageIn){ + return { + type: typeIn + , message: messageIn + } + } + + , eventStore: {} + }; + + //add an events hook for global dealing with events... + action = action.eventMe(action); + + //global error handler y'all + window.onerror = function(error){ + console.log('Error occured'); + console.warn(error); + } + + //return the tweaked function + return action; +}(this); \ No newline at end of file diff --git a/packages/0.2.0/action-v0.2.0.min.js b/packages/0.2.0/action-v0.2.0.min.js new file mode 100644 index 0000000..e7b3ca8 --- /dev/null +++ b/packages/0.2.0/action-v0.2.0.min.js @@ -0,0 +1 @@ +var action=function(){"use strict";var e={eventMe:function(t){var n=t;return n.emitterId=Math.ceil(1e4*Math.random()),n.eventStore={},n.emit=function(t,n,o){var i,c,r=this,a="undefined"!=typeof o&&o;if(i=a?r.eventStore[t]:e.eventStore[t],"undefined"!=typeof i)for(c=0;c