/* This file contains proprietary software owned by Motorola Mobility, Inc.
No rights, expressed or implied, whatsoever to this software are provided by Motorola Mobility, Inc. hereunder.
(c) Copyright 2011 Motorola Mobility, Inc. All Rights Reserved.
*/ //////////////////////////////////////////////////////////////////////// // var Montage = require("montage/core/core").Montage, Component = require("montage/ui/component").Component, FileIo = require("js/io/system/fileio").FileIo, ProjectIo = require("js/io/system/projectio").ProjectIo, TemplateCreator = require("node_modules/tools/template/template-creator").TemplateCreator; //////////////////////////////////////////////////////////////////////// // exports.IoMediator = Montage.create(Component, { //////////////////////////////////////////////////////////////////// // hasTemplate: { enumerable: false, value: false }, //////////////////////////////////////////////////////////////////// // deserializedFromTemplate: { enumerable: false, value: function () { // } }, //////////////////////////////////////////////////////////////////// // fio: { enumerable: false, value: FileIo }, //////////////////////////////////////////////////////////////////// // pio: { enumerable: false, value: ProjectIo }, //////////////////////////////////////////////////////////////////// // getAppTemplatesUrlRegEx: { enumerable: false, value: function () { var regex = new RegExp(chrome.extension.getURL('js/document/templates/montage-html').replace(/\//gi, '\\\/'), 'gi'); return regex; } }, //////////////////////////////////////////////////////////////////// // fileNew: { enumerable: false, value: function (file, url, callback, template) { //Loading template from template URL var xhr = new XMLHttpRequest(), result; xhr.open("GET", url, false); xhr.send(); if (xhr.readyState === 4) { //Making call to create file, checking for return code switch (this.fio.newFile({ uri: file, contents: parseTemplate(xhr.response, template) })) { case 201: result = { status: 201, success: true, uri: file }; break; case 204: result = { status: 204, success: false, uri: file }; break; case 400: result = { status: 400, success: false, uri: file }; break; default: result = { status: 500, success: false, uri: file }; break; } //TODO: Improve template data injection function parseTemplate (content, template) { // if (template.name.toLowerCase() === 'banner' || template.name.toLowerCase() === 'animation') { //Getting dimensions of banner var dimensions = template.id.split('x'); dimensions = {width: String(dimensions[0])+'px', height: String(dimensions[1])+'px'}; // content = content.replace(/Dimensions@@@/gi, "Dimensions@@@"+template.id); content = content.replace(/ninja-banner {}/gi, "ninja-banner {overflow: visible; width: "+dimensions.width+"; height: "+dimensions.height+"}"); content = content.replace(/ninja-content-wrapper {}/gi, "ninja-content-wrapper {overflow: hidden; width: "+dimensions.width+"; height: "+dimensions.height+"}"); } // return content; } } else { result = { status: 500, success: false, uri: file }; } //Sending result to callback if requested for handling if (callback) callback(result); //Codes // 204: File exists | 400: File exists // 201: File succesfully created | 500: Unknown (Probably cloud API not running) } }, //////////////////////////////////////////////////////////////////// // fileOpen: { enumerable: false, value: function (file, callback) { //Reading file (Ninja doesn't really open a file, all in browser memory) var read = this.fio.readFile({ uri: file }), result; //Checking for status switch (read.status) { case 204: //Creating and formatting result object for callbak result = read.file.details; result.root = read.file.details.uri.replace(read.file.details.name, ""); //Checking for type of content to returns if (result.extension !== 'html' && result.extension !== 'htm') { //Simple string result.content = read.file.content; } else { //Object to be used by Ninja Template result.content = this.parseHtmlToNinjaTemplate(read.file.content); } //Status of call result.status = read.status; //Calling back with result if (callback) callback(result); break; case 404: //File does not exists if (callback) callback({ status: read.status }); break; default: //Unknown if (callback) callback({ status: 500 }); break; } /* //////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////// //Return Object Description Object.status (Always presents for handling) 204: File exists (Success) 404: File does not exists (Failure) 500: Unknown (Probably cloud API not running) (Below only present if succesfull 204) Object.content Object.extension Object.name Object.uri Object.creationDate Object.modifiedDate Object.readOnly Object.size //////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////// */ } }, //////////////////////////////////////////////////////////////////// // fileSave: { enumerable: false, value: function (doc, callback) { // var contents, save; // switch (doc.mode) { case 'html': //Getting content from function to properly handle saving assets (as in external if flagged) if (doc.template && (doc.template.type === 'banner' || doc.template.type === 'animation')) { contents = this.parseNinjaTemplateToHtml(doc, true); } else { contents = this.parseNinjaTemplateToHtml(doc); } break; default: contents = doc.content; break; } //Making call to save file save = this.fio.saveFile({ uri: doc.file.uri, contents: contents }); //Checking for callback if (callback) callback(save); } }, //////////////////////////////////////////////////////////////////// // fileSaveAs: { enumerable: false, value: function (copyTo, copyFrom, callback) { //TODO: Implement Save As functionality } }, //////////////////////////////////////////////////////////////////// // fileDelete: { enumerable: false, value: function (file, callback) { //TODO: Implement Delete functionality } }, //////////////////////////////////////////////////////////////////// // getDataDirectory: { value: function (path) { //TODO: Implement user overwrite return this._getUserDirectory(path+'data/'); } }, //////////////////////////////////////////////////////////////////// // getNinjaDirectory: { value: function (path) { //TODO: Implement user overwrite return this._getUserDirectory(this.getDataDirectory(path)+'ninja/'); } }, //////////////////////////////////////////////////////////////////// // getCanvasDirectory: { value: function (path) { //TODO: Implement user overwrite return this._getUserDirectory(this.getNinjaDirectory(path)+'canvas/'); } }, //////////////////////////////////////////////////////////////////// // _getUserDirectory: { value: function (path) { //Checking for data directory var check = this.application.ninja.coreIoApi.fileExists({uri: path}), directory; //Creating directory if doesn't exists switch (check.status) { case 204: //Exists directory = path; break; case 404: //Doesn't exists directory = this.application.ninja.coreIoApi.createDirectory({uri: path}); //Checking for success if (directory.status === 201) { directory = path; } else { //Error directory = null; } break; default: //Error directory = null; break; } //Returning the path to the directory on disk (null for any error) return directory; } }, //////////////////////////////////////////////////////////////////// // parseHtmlToNinjaTemplate: { enumerable: false, value: function (html) { //Creating temp object to mimic HTML var doc = window.document.implementation.createHTMLDocument(), template; //Setting content to temp doc.getElementsByTagName('html')[0].innerHTML = html; //Creating return object return {head: doc.head.innerHTML, body: doc.body.innerHTML, document: doc}; } }, //////////////////////////////////////////////////////////////////// //TODO: Expand to allow more templates, clean up variables parseNinjaTemplateToHtml: { enumerable: false, value: function (template, ninjaWrapper) { //TODO: Improve reference for rootUrl var regexRootUrl, rootUrl = this.application.ninja.coreIoApi.rootUrl + escape((this.application.ninja.documentController.documentHackReference.root.split(this.application.ninja.coreIoApi.cloudData.root)[1])), mjsCreator = template.mjsTemplateCreator.create(), mJsSerialization, montageTemplate; //Creating instance of template creator montageTemplate = mjsCreator.initWithDocument(template.document); //Setting up expression for parsing URLs regexRootUrl = new RegExp(rootUrl.replace(/\//gi, '\\\/'), 'gi'); //Injecting head and body into old document if (montageTemplate._ownerSerialization.length > 0) { template.file.content.document.head.innerHTML = montageTemplate._document.head.innerHTML.replace(regexRootUrl, ''); template.file.content.document.body.innerHTML = montageTemplate._document.body.innerHTML.replace(regexRootUrl, ''); // mJsSerialization = montageTemplate._ownerSerialization; } else { template.file.content.document.head.innerHTML = template.head.innerHTML.replace(regexRootUrl, ''); template.file.content.document.body.innerHTML = template.body.innerHTML.replace(regexRootUrl, ''); } //Copying attributes to maintain same properties as the for (var n in template.body.attributes) { if (template.body.attributes[n].value) { // template.file.content.document.body.setAttribute(template.body.attributes[n].name, template.body.attributes[n].value); } } //TODO: Add attribute copying for and //console.log(template.file.content.document.getElementsByTagName('html')[0].innerHTML); //Getting all CSS (style or link) tags var styletags = template.file.content.document.getElementsByTagName('style'), linktags = template.file.content.document.getElementsByTagName('link'), toremovetags = [], njtemplatetags = template.file.content.document.querySelectorAll('[data-ninja-template]'); ////////////////////////////////////////////////// //TODO: Remove, temp hack, this is to be fixed by Montage var basetags = template.file.content.document.getElementsByTagName('base'); for (var g in basetags) { if (basetags[g].getAttribute) toremovetags.push(basetags[g]); } ////////////////////////////////////////////////// //Adding to tags to be removed form template for (var f in njtemplatetags) { if (njtemplatetags[f].getAttribute) toremovetags.push(njtemplatetags[f]); } //Getting styles tags to be removed from document if (styletags.length) { for (var j = 0; j < styletags.length; j++) { if (styletags[j].getAttribute) { if (styletags[j].getAttribute('data-ninja-uri') !== null && !styletags[j].getAttribute('data-ninja-template')) { toremovetags.push(styletags[j]); } else if (styletags[j].getAttribute('data-ninja-external-url')) { toremovetags.push(styletags[j]); } } } } //Removing styles tags from document for (var h = 0; toremovetags[h]; h++) { try { //Checking head first template.file.content.document.head.removeChild(toremovetags[h]); } catch (e) { } try { //Checking body if not in head template.file.content.document.body.removeChild(toremovetags[h]); } catch (e) { //Error, not found! } } //Removing disabled tags from tags that were not originally disabled by user (Ninja enables all) for (var l in linktags) { if (linktags[l].getAttribute && linktags[l].getAttribute('disabled')) {//TODO: Use querySelectorAll for (var p = 0; toremovetags[p]; p++) { if (toremovetags[p].getAttribute('href') === linktags[l].getAttribute('href')) { if (!toremovetags[p].getAttribute('data-ninja-disabled')) { linktags[l].removeAttribute('disabled'); } } } } } //Checking for type of save: styles =