Document.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489
  1. SC.loadPackage({ 'Document': {
  2. comment: 'I am the document the user works on.',
  3. properties: {
  4. pageContainer: { comment: 'I store the div element containing the pages\' elements.' },
  5. editingContainer: { comment: 'I am a container holding all user interface components of the editing tool.'},
  6. children: { comment: 'I care for my (array of) children, which are the instances of some Element class.' },
  7. showOutlines: { comment: 'Shall I put outlines around all my child Elements?',
  8. transform: function(aBoolean){
  9. if(aBoolean){
  10. this.get('pageContainer').classList.add('sg-editing-outlines');
  11. this.get('widthMarkers').set({ visible: true });
  12. this.get('grid').set({ visible: true });
  13. }else{
  14. this.get('pageContainer').classList.remove('sg-editing-outlines');
  15. this.get('widthMarkers').set({ visible: false });
  16. this.get('grid').set({ visible: false });
  17. }
  18. return aBoolean;
  19. }
  20. },
  21. interactionInProgress: { comment: 'Some interaction require a modal change of editing state, while the interaction is in progress. This concerns all interactions which listen for mouse moves.',
  22. transform: function(aBoolean){
  23. if(aBoolean){
  24. this.get('pageContainer').classList.add('sg-editing-block-events');
  25. this.get('documentMenu').do('close');
  26. this.get('creationMenu').do('close');
  27. }else{
  28. this.get('pageContainer').classList.remove('sg-editing-block-events');
  29. }
  30. return aBoolean;
  31. }
  32. },
  33. documentMenu: { comment: 'I hold the DocumentMenu.' },
  34. creationMenu: { comment: 'I hold the CreationMenu.' },
  35. widthMarkers: { comment: 'When the document has a defined width, WidthMarkers visualize this and provide interaction.' },
  36. grid: { comment: 'The document always has a Grid to arrange its Elements.' },
  37. layout: { comment: 'The document manages the basic layout: it can either be (centered with a defined width) or can be (of undefined dimensions). To change this property give me a newConfig like { centered: true, width: \'70%\' } or { centered: false }.',
  38. transform: function(newConfig){
  39. var pageContainer = this.get('pageContainer'),
  40. editingContainer = this.get('editingContainer');
  41. if(newConfig.centered){
  42. if(newConfig.width === undefined){
  43. this.do('cropEmptySpaceOnLeftSide');
  44. newConfig.width = this.do('getMinWidth');
  45. }
  46. newConfig.width = newConfig.width > 140 ? newConfig.width : 140;
  47. pageContainer.classList.add('sg-page-centered');
  48. editingContainer.classList.add('sg-editing-container-centered');
  49. pageContainer.style.width = editingContainer.style.width = newConfig.width + 'px';
  50. this.get('widthMarkers').set({ active: true });
  51. }else{
  52. pageContainer.classList.remove('sg-page-centered');
  53. editingContainer.classList.remove('sg-editing-container-centered');
  54. pageContainer.style.width = editingContainer.style.width = null;
  55. this.get('widthMarkers').set({ active: false });
  56. }
  57. var returnConfig = { centered: newConfig.centered, width: newConfig.width };
  58. this.get('grid').do('updateDimensions', returnConfig);
  59. return returnConfig;
  60. }
  61. }
  62. },
  63. methods: {
  64. init: {
  65. comment: 'I set up the Document during system initialization, called from SuperGlue>>init.',
  66. code: function(){
  67. var pageContainer = this.set({ pageContainer: document.body.querySelector('#sg-page') }),
  68. editingContainer = this.set({ editingContainer: (function(){
  69. var editingContainer = document.createElement('div');
  70. editingContainer.setAttribute('id', 'sg-editing-container');
  71. document.body.appendChild(editingContainer);
  72. return editingContainer;
  73. }).call(this)
  74. });
  75. this.set({
  76. documentMenu: SC.init('DocumentMenu'),
  77. creationMenu: SC.init('CreationMenu'),
  78. widthMarkers: SC.init('WidthMarkers', editingContainer),
  79. grid: SC.init('Grid', this)
  80. });
  81. this.set({
  82. layout: {
  83. centered: pageContainer.classList.contains('sg-page-centered'),
  84. width: parseInt(pageContainer.style.width)
  85. }
  86. });
  87. }
  88. },
  89. setUpWorkspace: {
  90. comment: 'I wake up my children from the DOM and prepare the document for work.',
  91. code: function(){
  92. this.set({
  93. children: (function(){
  94. var nodeList = this.get('pageContainer').querySelectorAll('.sg-element'),
  95. children = [];
  96. for (var i = 0, l = nodeList.length; i < l; ++i) {
  97. children.push(SC.do('Element', 'awakeFromDOM', nodeList.item(i)));
  98. }
  99. return children;
  100. }).call(this),
  101. showOutlines: document.querySelector('meta[name=generator]').getAttribute('data-superglue-settings').indexOf('wireframe') > -1
  102. });
  103. this.do('afterLayoutHasChanged');
  104. this.do('registerEventListeners');
  105. }
  106. },
  107. createNewElement: {
  108. comment: 'I create a new Element, with the arguments configuration { classname: aString, top: anInt, left: anInt, width: anInt, height: anInt }.',
  109. code: function(configuration){
  110. var protoHTML = SC.getSharedProperty(configuration.classname, 'protoHTML'),
  111. newNode = (new DOMParser()).parseFromString(protoHTML, 'text/html').body.firstChild,
  112. newChild;
  113. this.get('pageContainer').appendChild(newNode);
  114. newChild = SC.do('Element', 'awakeFromDOM', newNode)
  115. newChild.set({
  116. top: configuration.top,
  117. left: configuration.left,
  118. width: configuration.width,
  119. height: configuration.height
  120. })
  121. this.get('children').push(newChild);
  122. this.do('afterLayoutHasChanged');
  123. return newChild;
  124. }
  125. },
  126. removeElement: {
  127. comment: 'I remove an Element.',
  128. code: function(anElement){
  129. var indexOfElement = this.get('children').indexOf(anElement);
  130. if(indexOfElement >= 0){
  131. this.get('children').splice(indexOfElement, 1);
  132. this.get('pageContainer').removeChild(anElement.get('node'));
  133. }
  134. this.do('afterLayoutHasChanged');
  135. }
  136. },
  137. layerUp: {
  138. comment: 'I move an Element upwards in the layer order.',
  139. code: function(anElement){
  140. var pageContainer = this.get('pageContainer'),
  141. children = this.get('children'),
  142. childIndex = children.indexOf(anElement),
  143. swapTemp;
  144. if(childIndex === (children.length - 1)){
  145. return;
  146. }
  147. swapTemp = children[childIndex + 1];
  148. children[childIndex + 1] = children[childIndex];
  149. children[childIndex] = swapTemp;
  150. pageContainer.insertBefore(
  151. children[childIndex + 1].get('node'),
  152. children[childIndex].get('node').nextElementSibling
  153. );
  154. }
  155. },
  156. layerDown: {
  157. comment: 'I move an Element downwards in the layer order.',
  158. code: function(anElement){
  159. var pageContainer = this.get('pageContainer'),
  160. children = this.get('children'),
  161. childIndex = children.indexOf(anElement),
  162. swapTemp;
  163. if(childIndex === 0){
  164. return;
  165. }
  166. swapTemp = children[childIndex - 1];
  167. children[childIndex - 1] = children[childIndex];
  168. children[childIndex] = swapTemp;
  169. pageContainer.insertBefore(
  170. children[childIndex - 1].get('node'),
  171. children[childIndex].get('node')
  172. );
  173. }
  174. },
  175. getMinWidth: {
  176. comment: 'I calculate the minimal width I need to fit in all my child Elements.',
  177. code: function(){
  178. var minWidth = 0,
  179. children = this.get('children');
  180. for(var i = 0, l = children.length; i < l; i++){
  181. var childWidth = children[i].get('left') + children[i].get('width');
  182. minWidth = (childWidth > minWidth) ? childWidth : minWidth;
  183. }
  184. return minWidth;
  185. }
  186. },
  187. getMinHeight: {
  188. comment: 'I calculate the minimal height to fit in all my child Elements.',
  189. code: function(){
  190. var minHeight = 0,
  191. children = this.get('children');
  192. for(var i = 0, l = children.length; i < l; i++){
  193. var childHeight = children[i].get('top') + children[i].get('height');
  194. minHeight = (childHeight > minHeight) ? childHeight : minHeight;
  195. }
  196. return minHeight;
  197. }
  198. },
  199. getMostLeft: {
  200. comment: 'I get the left value of the most left positioned of all my child Elements.',
  201. code: function(){
  202. var children = this.get('children');
  203. if(children.length === 0){ return 0; }
  204. var mostLeft = children[0].get('left');
  205. for(var i = 0, l = children.length; i < l; i++){
  206. var childLeft = children[i].get('left');
  207. mostLeft = (childLeft < mostLeft) ? childLeft : mostLeft;
  208. }
  209. return mostLeft;
  210. }
  211. },
  212. cropEmptySpaceOnLeftSide: {
  213. comment: 'I crop the empty space on the left side of the page. Should be called when centering the page.',
  214. code: function(){
  215. var children = this.get('children');
  216. if(children.length === 0){ return 0; }
  217. for(var cropWidth = this.do('getMostLeft'), i = 0, l = children.length; i < l; i++){
  218. children[i].set({ left: children[i].get('left') - cropWidth });
  219. }
  220. this.do('afterLayoutHasChanged');
  221. }
  222. },
  223. afterLayoutHasChanged: {
  224. comment: 'After the layout has changed (resizing, moving, or adding anything), I must be called to ensure the layout constraints.',
  225. code: function(){
  226. this.get('widthMarkers').do('updateHeight');
  227. this.get('grid').do('updateDimensions');
  228. }
  229. },
  230. registerEventListeners: {
  231. comment: 'I register eventListeners on the window.document for DocumentMenu and CreationMenu.',
  232. code: function(){
  233. var self = this,
  234. creationMenu = this.get('creationMenu'),
  235. startX,
  236. startY,
  237. dragging,
  238. outOfBounds,
  239. centeredPage,
  240. pseudoMargin,
  241. pageLeft,
  242. pageWidth,
  243. widthMarkersVisible,
  244. gridVisible,
  245. onMouseDown = function(evt){
  246. if(evt.button !== 0) return;
  247. if( evt.clientX > document.documentElement.clientWidth
  248. || evt.clientY > document.documentElement.clientHeight ){
  249. // browser scrollbars
  250. return;
  251. }
  252. startX = evt.pageX;
  253. startY = evt.pageY;
  254. dragging = false;
  255. centeredPage = self.get('layout').centered;
  256. pageWidth = self.get('layout').width;
  257. if(SuperGlue.get('document').get('grid').get('active')){
  258. var gridSize = SuperGlue.get('document').get('grid').get('gridSize');
  259. pageWidth = Math.floor(pageWidth / gridSize) * gridSize;
  260. }
  261. pseudoMargin = centeredPage
  262. ? (parseInt(window.getComputedStyle(document.body).getPropertyValue('width')) - pageWidth) / 2
  263. : 0;
  264. pageLeft = pseudoMargin > 0 ? pseudoMargin : 0;
  265. pageRight = pageLeft + pageWidth;
  266. outOfBounds = startX < pageLeft || (centeredPage && startX > pageRight) || startY < 0;
  267. document.addEventListener('mousemove', onMouseMove, true);
  268. document.addEventListener('mouseup', onMouseUp, true);
  269. evt.preventDefault();
  270. },
  271. onMouseMove = function(evt){
  272. if( !dragging
  273. && ( evt.pageX < startX - 10
  274. || evt.pageX > startX + 10
  275. || evt.pageY < startY - 10
  276. || evt.pageY > startY + 10 )
  277. ){
  278. dragging = true;
  279. self.set({ interactionInProgress: true });
  280. var widthMarkers = SuperGlue.get('document').get('widthMarkers');
  281. grid = SuperGlue.get('document').get('grid');
  282. widthMarkersVisible = widthMarkers.get('visible');
  283. gridVisible = grid.get('visible');
  284. widthMarkers.set({ visible: true });
  285. grid.set({ visible: true });
  286. if(!outOfBounds){
  287. SuperGlue.get('selection').do('clearAll')
  288. creationMenu.do('trigger', {
  289. startX: startX - pageLeft,
  290. startY: startY,
  291. stopX: evt.pageX - pageLeft,
  292. stopY: evt.pageY
  293. });
  294. }
  295. }
  296. if(dragging && !outOfBounds){
  297. if(evt.pageX <= pageLeft){
  298. creationMenu.set({
  299. stopX: 0
  300. });
  301. }else if(evt.pageX >= pageRight){
  302. creationMenu.set({
  303. stopX: pageWidth
  304. });
  305. }else{
  306. creationMenu.set({
  307. stopX: (evt.pageX - pageLeft)
  308. });
  309. }
  310. if(evt.pageY > 0){
  311. creationMenu.set({
  312. stopY: evt.pageY
  313. });
  314. }else{
  315. creationMenu.set({
  316. stopY: 0
  317. });
  318. }
  319. }
  320. },
  321. onMouseUp = function(evt){
  322. if(dragging){
  323. self.set({ interactionInProgress: false });
  324. SuperGlue.get('document').get('widthMarkers').set({ visible: widthMarkersVisible });
  325. SuperGlue.get('document').get('grid').set({ visible: gridVisible });
  326. if( !outOfBounds
  327. &&(evt.pageX < startX - 50
  328. || evt.pageX > startX + 50
  329. || evt.pageY < startY - 50
  330. || evt.pageY > startY + 50)
  331. ){
  332. creationMenu.do('showItems');
  333. }else{
  334. creationMenu.do('close');
  335. }
  336. }else{
  337. if(SuperGlue.get('selection').do('isEmpty')){
  338. if(creationMenu.get('active')){
  339. creationMenu.do('close');
  340. }else{
  341. self.get('documentMenu').do('trigger', { x: evt.pageX, y: evt.pageY });
  342. }
  343. }else{
  344. SuperGlue.get('selection').do('clearAll')
  345. }
  346. }
  347. document.removeEventListener('mousemove', onMouseMove, true);
  348. document.removeEventListener('mouseup', onMouseUp, true);
  349. evt.preventDefault();
  350. };
  351. document.addEventListener('mousedown', onMouseDown, false);
  352. }
  353. }
  354. }
  355. }});