diff options
Diffstat (limited to 'js/document/document-html.js')
-rwxr-xr-x | js/document/document-html.js | 304 |
1 files changed, 301 insertions, 3 deletions
diff --git a/js/document/document-html.js b/js/document/document-html.js index b48e004a..841e66ed 100755 --- a/js/document/document-html.js +++ b/js/document/document-html.js | |||
@@ -6,8 +6,9 @@ No rights, expressed or implied, whatsoever to this software are provided by Mot | |||
6 | 6 | ||
7 | //////////////////////////////////////////////////////////////////////// | 7 | //////////////////////////////////////////////////////////////////////// |
8 | // | 8 | // |
9 | var Montage = require("montage/core/core").Montage, | 9 | var Montage = require("montage/core/core").Montage, |
10 | Component = require("montage/ui/component").Component; | 10 | Component = require("montage/ui/component").Component; |
11 | HtmlDocumentModel = require("js/document/models/html").HtmlDocumentModel; | ||
11 | //////////////////////////////////////////////////////////////////////// | 12 | //////////////////////////////////////////////////////////////////////// |
12 | // | 13 | // |
13 | exports.HtmlDocument = Montage.create(Component, { | 14 | exports.HtmlDocument = Montage.create(Component, { |
@@ -16,9 +17,306 @@ exports.HtmlDocument = Montage.create(Component, { | |||
16 | hasTemplate: { | 17 | hasTemplate: { |
17 | enumerable: false, | 18 | enumerable: false, |
18 | value: false | 19 | value: false |
19 | } | 20 | }, |
21 | |||
22 | model: { | ||
23 | value: null | ||
24 | }, | ||
25 | |||
26 | loadDelegate: { | ||
27 | value: null | ||
28 | }, | ||
29 | |||
30 | delegateContext: { | ||
31 | value: null | ||
32 | }, | ||
33 | |||
34 | // Getters for the model. | ||
35 | // TODO: Change how these properties are accessed through Ninja | ||
36 | name: { | ||
37 | get: function() { | ||
38 | return this.model._name; | ||
39 | }, | ||
40 | set: function(value) { | ||
41 | this.model._name = value; | ||
42 | } | ||
43 | }, | ||
44 | |||
45 | // View Properties | ||
46 | // TODO: Move those into a view object - for now dump it here | ||
47 | iframe: { | ||
48 | value: null | ||
49 | }, | ||
50 | |||
51 | uuid: { | ||
52 | get: function() { | ||
53 | return this._uuid; | ||
54 | } | ||
55 | }, | ||
20 | //////////////////////////////////////////////////////////////////// | 56 | //////////////////////////////////////////////////////////////////// |
21 | //////////////////////////////////////////////////////////////////// | 57 | //////////////////////////////////////////////////////////////////// |
58 | init: { | ||
59 | value:function(file, context, callback) { | ||
60 | this.model = Montage.create(HtmlDocumentModel, { | ||
61 | file: { | ||
62 | value: file | ||
63 | } | ||
64 | }); | ||
65 | |||
66 | this.name = file.name; | ||
67 | |||
68 | // this.init(file.name, file.uri, file.extension, iframe, uuid, callback); | ||
69 | |||
70 | this.iframe = this.createView(); | ||
71 | this.iframe.addEventListener("load", this.handleWebTemplateLoad.bind(this), true); | ||
72 | |||
73 | //this.selectionExclude = ["HTML", "BODY", "Viewport", "UserContent", "stageBG"]; | ||
74 | //this.currentView = "design"; | ||
75 | // | ||
76 | |||
77 | this.delegateContext = context; | ||
78 | this.loadDelegate = callback; | ||
79 | } | ||
80 | }, | ||
81 | |||
82 | // Create View | ||
83 | // Move this into a base view object | ||
84 | createView: { | ||
85 | value: function() { | ||
86 | var ifr = document.createElement("iframe"); | ||
87 | ifr.id = "document_" + this._uuid; | ||
88 | |||
89 | |||
90 | ifr.style.border = "none"; | ||
91 | ifr.style.background = "#FFF"; | ||
92 | ifr.style.height = "100%"; | ||
93 | ifr.style.width = "100%"; | ||
94 | |||
95 | // TODO: Reable opacity to display only when done loading | ||
96 | // ifr.style.opacity = 0; | ||
97 | |||
98 | ifr.src = "js/document/templates/montage-web/index.html"; | ||
99 | |||
100 | return document.getElementById("iframeContainer").appendChild(ifr); | ||
101 | } | ||
102 | }, | ||
103 | |||
104 | handleWebTemplateLoad: { | ||
105 | value: function(event) { | ||
106 | //TODO: Remove, also for prototyping | ||
107 | this.application.ninja.documentController._hackRootFlag = true; | ||
108 | |||
109 | |||
110 | //TODO: Clean up, using for prototyping save | ||
111 | // this._templateDocument = {}; | ||
112 | // this._templateDocument.html = this.iframe.contentWindow.document; | ||
113 | // this._templateDocument.body = | ||
114 | |||
115 | this._window = this.iframe.contentWindow; | ||
116 | this._document = this.iframe.contentWindow.document; | ||
117 | this.documentRoot = this.iframe.contentWindow.document.body; | ||
118 | |||
119 | for (var k in this._document.styleSheets) { | ||
120 | if (this._document.styleSheets[k].ownerNode && this._document.styleSheets[k].ownerNode.setAttribute) { | ||
121 | this._document.styleSheets[k].ownerNode.setAttribute('data-ninja-template', 'true'); | ||
122 | } | ||
123 | } | ||
124 | |||
125 | // TODO: We don't need this anymore -> need to setup the main container still | ||
126 | //Adding a handler for the main user document reel to finish loading | ||
127 | //this._document.body.addEventListener("userTemplateDidLoad", this.userTemplateDidLoad.bind(this), false); | ||
128 | this.documentRoot.addEventListener("userTemplateDidLoad", this.userTemplateDidLoad.bind(this), false); | ||
129 | |||
130 | // Live node list of the current loaded document | ||
131 | this._liveNodeList = this.documentRoot.getElementsByTagName('*'); | ||
132 | |||
133 | // TODO Move this to the appropriate location | ||
134 | /* | ||
135 | var len = this._liveNodeList.length; | ||
136 | |||
137 | for(var i = 0; i < len; i++) { | ||
138 | NJUtils.makeModelFromElement(this._liveNodeList[i]); | ||
139 | } | ||
140 | */ | ||
141 | |||
142 | setTimeout(function () { | ||
143 | |||
144 | //////////////////////////////////////////////////////////////////////////////////////////////////////////////////// | ||
145 | //////////////////////////////////////////////////////////////////////////////////////////////////////////////////// | ||
146 | //////////////////////////////////////////////////////////////////////////////////////////////////////////////////// | ||
147 | if(this._document.styleSheets.length) { | ||
148 | //Checking all styleSheets in document | ||
149 | for (var i in this._document.styleSheets) { | ||
150 | //If rules are null, assuming cross-origin issue | ||
151 | if(this._document.styleSheets[i].rules === null) { | ||
152 | //TODO: Revisit URLs and URI creation logic, very hack right now | ||
153 | var fileUri, cssUrl, cssData, query, prefixUrl, fileCouldDirUrl, docRootUrl; | ||
154 | // | ||
155 | docRootUrl = this.application.ninja.coreIoApi.rootUrl+escape((this.application.ninja.documentController.documentHackReference.root.split(this.application.ninja.coreIoApi.cloudData.root)[1]).replace(/\/\//gi, '/')); | ||
156 | //TODO: Parse out relative URLs and map them to absolute | ||
157 | if (this._document.styleSheets[i].href.indexOf(this.application.ninja.coreIoApi.rootUrl) !== -1) { | ||
158 | // | ||
159 | cssUrl = this._document.styleSheets[i].href.split(this.application.ninja.coreIoApi.rootUrl)[1]; | ||
160 | fileUri = this.application.ninja.coreIoApi.cloudData.root+cssUrl; | ||
161 | //TODO: Add error handling for reading file | ||
162 | cssData = this.application.ninja.coreIoApi.readFile({uri: fileUri}); | ||
163 | // | ||
164 | var tag = this.iframe.contentWindow.document.createElement('style'); | ||
165 | tag.setAttribute('type', 'text/css'); | ||
166 | tag.setAttribute('data-ninja-uri', fileUri); | ||
167 | tag.setAttribute('data-ninja-file-url', cssUrl); | ||
168 | tag.setAttribute('data-ninja-file-read-only', JSON.parse(this.application.ninja.coreIoApi.isFileWritable({uri: fileUri}).content).readOnly); | ||
169 | tag.setAttribute('data-ninja-file-name', cssUrl.split('/')[cssUrl.split('/').length-1]); | ||
170 | //Copying attributes to maintain same properties as the <link> | ||
171 | for (var n in this._document.styleSheets[i].ownerNode.attributes) { | ||
172 | if (this._document.styleSheets[i].ownerNode.attributes[n].value && this._document.styleSheets[i].ownerNode.attributes[n].name !== 'disabled' && this._document.styleSheets[i].ownerNode.attributes[n].name !== 'disabled') { | ||
173 | if (this._document.styleSheets[i].ownerNode.attributes[n].value.indexOf(docRootUrl) !== -1) { | ||
174 | tag.setAttribute(this._document.styleSheets[i].ownerNode.attributes[n].name, this._document.styleSheets[i].ownerNode.attributes[n].value.split(docRootUrl)[1]); | ||
175 | } else { | ||
176 | tag.setAttribute(this._document.styleSheets[i].ownerNode.attributes[n].name, this._document.styleSheets[i].ownerNode.attributes[n].value); | ||
177 | } | ||
178 | } | ||
179 | } | ||
180 | // | ||
181 | fileCouldDirUrl = this._document.styleSheets[i].href.split(this._document.styleSheets[i].href.split('/')[this._document.styleSheets[i].href.split('/').length-1])[0]; | ||
182 | |||
183 | //TODO: Make public version of this.application.ninja.ioMediator.getNinjaPropUrlRedirect with dynamic ROOT | ||
184 | tag.innerHTML = cssData.content.replace(/url\(()(.+?)\1\)/g, detectUrl); | ||
185 | |||
186 | function detectUrl (prop) { | ||
187 | return prop.replace(/[^()\\""\\'']+/g, prefixUrl);; | ||
188 | } | ||
189 | |||
190 | function prefixUrl (url) { | ||
191 | if (url !== 'url') { | ||
192 | if (!url.match(/(\b(?:(?:https?|ftp|file|[A-Za-z]+):\/\/|www\.|ftp\.)(?:\([-A-Z0-9+&@#\/%=~_|$?!:,.]*\)|[-A-Z0-9+&@#\/%=~_|$?!:,.])*(?:\([-A-Z0-9+&@#\/%=~_|$?!:,.]*\)|[A-Z0-9+&@#\/%=~_|$]))/gi)) { | ||
193 | url = fileCouldDirUrl+url; | ||
194 | } | ||
195 | } | ||
196 | return url; | ||
197 | } | ||
198 | |||
199 | //Looping through DOM to insert style tag at location of link element | ||
200 | query = this._templateDocument.html.querySelectorAll(['link']); | ||
201 | for (var j in query) { | ||
202 | if (query[j].href === this._document.styleSheets[i].href) { | ||
203 | //Disabling style sheet to reload via inserting in style tag | ||
204 | query[j].setAttribute('disabled', 'true'); | ||
205 | //Inserting tag | ||
206 | this._templateDocument.head.insertBefore(tag, query[j]); | ||
207 | } | ||
208 | } | ||
209 | } else { | ||
210 | console.log('ERROR: Cross-Domain-Stylesheet detected, unable to load in Ninja'); | ||
211 | //None local stylesheet, probably on a CDN (locked) | ||
212 | var tag = this.iframe.contentWindow.document.createElement('style'); | ||
213 | tag.setAttribute('type', 'text/css'); | ||
214 | tag.setAttribute('data-ninja-external-url', this._document.styleSheets[i].href); | ||
215 | tag.setAttribute('data-ninja-file-read-only', "true"); | ||
216 | tag.setAttribute('data-ninja-file-name', this._document.styleSheets[i].href.split('/')[this._document.styleSheets[i].href.split('/').length-1]); | ||
217 | //Copying attributes to maintain same properties as the <link> | ||
218 | for (var n in this._document.styleSheets[i].ownerNode.attributes) { | ||
219 | if (this._document.styleSheets[i].ownerNode.attributes[n].value && this._document.styleSheets[i].ownerNode.attributes[n].name !== 'disabled' && this._document.styleSheets[i].ownerNode.attributes[n].name !== 'disabled') { | ||
220 | if (this._document.styleSheets[i].ownerNode.attributes[n].value.indexOf(docRootUrl) !== -1) { | ||
221 | tag.setAttribute(this._document.styleSheets[i].ownerNode.attributes[n].name, this._document.styleSheets[i].ownerNode.attributes[n].value.split(docRootUrl)[1]); | ||
222 | } else { | ||