/** @class Application is used to create new rio application classes. It provides functionality for dependency management, routing, history management and page management. @extends rio.Attr */ rio.Application = { /** Creates an instance of rio.Application. @param {String} name (optional) The name of this Application. Used primarily for testing reports. @param {Object} extends (optional) An Attr class to use as a superclass. @param {Object} args (optional) The definition of the class. @returns a new instance of type Application @type rio.Application @example rio.apps.example = rio.Application.create({ require: ["pages/example_page"], requireCss: ["css_reset", "example"], routes: { "": "examplePage" }, attrAccessors: [], attrReaders: [], methods: { initialize: function(options) { }, examplePage: function() { return new rio.pages.ExamplePage(); } } }); */ create: function() { var args = $A(arguments); if (args.length > 0 && args.last() != undefined && !args.last().ATTR) { args[args.size() - 1] = Object.extend({ noExtend: true }, args.last()); } var app = rio.Attr.create.apply(this, args); app.addMethods( /** @scope rio.Application.prototype */ { /** @private */ initHistory: function() { dhtmlHistory.initialize(); dhtmlHistory.addListener(this.applyHistoryEntry.bind(this)); }, /** @private */ applyHistoryEntry: function(location, historyData) { if (this.__revertingTransientHistoryEntry) { this.addHistoryEntry(this.__revertingTransientHistoryEntry[0], this.__revertingTransientHistoryEntry[1]); this.__revertingTransientHistoryEntry = false; } else { this.navigateTo(location, true); } }, /** @private */ addHistoryEntry: function(location, transient) { if (historyStorage.hasKey(this.getCurrentLocation()) && historyStorage.get(this.getCurrentLocation()).transient) { this.__revertingTransientHistoryEntry = [location, transient]; history.back(1); } else { dhtmlHistory.add(location, { transient: transient }); } }, /** @private */ resize: function() { this.getCurrentPage().resize(); }, /** @private */ keyPress: function(e) { var currentPage = this.getCurrentPage(); if (currentPage) { this.getCurrentPage().keyPress(e); } }, /** @private */ keyDown: function(e) { var keyMap = this.getKeyMap(); if (keyMap) { keyMap.handle(e); } var currentPage = this.getCurrentPage(); if (currentPage) { this.getCurrentPage()._keyDown(e); } }, /** This method is called just before the page is unloaded. This can be triggered by following a link, closing the window, using the back button, etc. This method is meant to be overriden */ unload: function() { // meant to be overriden }, /** @private */ getKeyMap: function() { if (this._keyMap) { return this._keyMap; } if (!this.keyMap) { return; } this._keyMap = rio.KeyMap.build(this.keyMap()); return this._keyMap; }, /** @private */ launch: function() { document.observe("keypress", this.keyPress.bind(this)); document.observe("keydown", this.keyDown.bind(this)); Event.observe(window, "beforeunload", this.unload.bind(this)); if (this.noRoutes()) { return; } this.initHistory(); this.navigateTo(this.getCurrentLocation()); Event.observe(window, "resize", this.resize.bind(this)); rio.Application._afterLaunchFunctions.each(function(fcn) { fcn(this); }); rio.Application._afterLaunchFunctions.clear(); this._launched = true; }, /** @private */ launched: function() { return this._launched || false; }, /** @private */ noRoutes: function() { return (app.__routes == undefined) || ($H(app.__routes).keys().size() == 0); }, /** @private */ avoidAnimation: function() { return Prototype.Browser.IE; }, /** @private */ matchRoutePath: function(path) { return Object.keys(app.__routes).detect(function(routePath) { if (routePath == "") { return true; } var routeParts = routePath.split("/"); var pathParts = path.split("/"); var match = true; for(var i=0; iYou are better off specifying routes when creating an application with a 'routes' parameter. */ route: function(path, target){ if (!this.__routes) { this.__routes = {}; } this.__routes[path] = target; var parts = path.split("/"); if (parts.length > 1) { for (var i=0; i 0 && args.last() != undefined && !args.last().ATTR) { var initializers = args.last(); if (initializers.requireCss) { rio.Application.includeCss(initializers.requireCss); } Object.keys(initializers.routes || {}).each(function(name) { app.route(name, initializers.routes[name]); }); rio.Application.extend(app, initializers.methods || {}); } return app; }, /** @private */ extend: function(app, extension) { rio.Attr.extend(app, extension); }, /** Alias of rio.Application.require @param {String} fileName The path to the javascript file that will be loaded. */ include: function(fileName) { this.require(fileName); // rio.boot.loadFile(fileName); }, /** Alias of rio.Application.require @param {String} fileName The path to the javascript file that will be loaded. */ require: function(fileName) { rio.require(fileName); }, /** @private */ injectCss: function() { var toLoad = []; rio.boot.loadedStylesheets.each(function(s) { if (!rio.preloadedStylesheets.include(s)) { toLoad.push(s); } }); if (toLoad.empty()) { return; } var query = toLoad.map(function(f) { return "files[]=" + f; }).join("&"); var linkHtml = rio.Tag.link("", { href: rio.url("/rio/stylesheets?" + query + "&" + rio.cacheKey), media: "screen", rel: "stylesheet", type: "text/css" }); Element.head().insert({ bottom: linkHtml }); if (rio.ContainerLayout) { rio.ContainerLayout.resize(); } }, /* Requires a css file @param {String} toInclude The path to the stylesheet that will be loaded. */ includeCss: function(toInclude) { // Because of a bug in IE, we need to remove and readd all link tags every time a new one is added. var include = function(fileName) { if (rio.boot.loadedStylesheets.include(fileName)) { return; } rio.boot.loadedStylesheets.push(fileName); if (rio.preloadedStylesheets.include(fileName)) { return; } if (rio.environment.autoConcatCss && !(rio.app && rio.app.launched())) { // Do nothing } else { var linkHtml = rio.Tag.link("", { href: rio.url("/stylesheets/" + fileName + ".css"), media: "screen", rel: "stylesheet", type: "text/css" }); Element.head().insert({ bottom: linkHtml }); } }.bind(this); if (Object.isString(toInclude)) { include(toInclude); } if (Object.isArray(toInclude)) { toInclude.reverse().each(function(css) { include(css); }); } }, /** @private */ getToken: function() { return this._token || rio.environment.railsToken; }, /** @private */ setToken: function(token) { this._token = token; }, /** @private */ _afterLaunchFunctions: [], /** @private */ afterLaunch: function(afterLoadFunction) { if (rio.app && rio.app.launched()) { afterLoadFunction(rio.app); } else { this._afterLaunchFunctions.push(afterLoadFunction); } }, /** This causes the application to fail and log a 'fail' error message. If the application class has a fail method, that method will be called with the message passed in here. @param {String} msg The application failure message @param {String} msg A more in depth description of the application failure */ fail: function(msg, description) { try { if (rio.app && rio.app.fail) { rio.app.fail(msg); rio.error("FAIL: " + msg, description || "", true); } else { alert("OOPS: " + msg + "\n\nTry refreshing the page or come back later."); } } catch(e) { // Ignore errors during fail } }, toString: function() { return "Application"; } }; if (!window.dhtmlHistoryCreated) { window.dhtmlHistory.create({ toJSON: function(o) { return Object.toJSON(o); }, fromJSON: function(s) { return s.evalJSON(); } }); window.dhtmlHistoryCreated = true; }