Dateiverwaltung für die WebBox
ulrich
2021-01-17 f320427aedc5b1bd73da7e08c0e84cef9d7ec66f
commit | author | age
05e9c4 1 function AppVorlage() {
U 2   var self = this;
41ab37 3   this.appMenu;
4e450e 4   //this.vorlagen;
41ab37 5   this.api;
U 6   this.userid;
7   this.pfad = '';
8   this.loc;
9   this.modus = 'kacheln';
ebc1f3 10   this.cm;
4e450e 11   //this.tinymce;
U 12   this.ed;
13   this.tmo;
14   this.tmo2;
41ab37 15   this.PERS_DIR = "Persoenlich";
U 16   this.PUB_DIR = "Oeffentlich";
17   this.DAV_DIR = "Austausch";
18   this.BASE_DIR = "$basis";
19   this.DATA_DIR = "$daten";
20   this.WWW_DIR = "www";
f002d9 21   this.openEditor;
U 22   this.openFileName;
beb124 23   this.cutCopyOperation;
U 24   this.cutCopySrcDir;
25   this.cutCopyFiles;
41ab37 26   this.cache = {}; // mustache template cache
05e9c4 27
94b871 28   this.datei_neuer_text = function () {
ebc1f3 29     //self.meldung_mit_timeout("Neuer Text", 1500);
U 30     self.fm_text_edit('Neue Datei');
31   };
32   
05e9c4 33   /* Funktionen aus App-Vorlage */
94b871 34
4e450e 35   this.init = function () {    
f002d9 36     document.querySelector(".codeeditor-space").style.display = "none";
U 37     document.querySelector("#mce-editor").style.display = "none";
05e9c4 38     self.appMenu = new AppMenu();
U 39     self.appMenu.init(
94b871 40             "data/menu/",
U 41             "hauptmenue.json",
42             "data/tpl/app-menu.tpl",
43             ".west",
44             "8em");
45     document.querySelector('.hamburger').addEventListener('click', function (e) {
05e9c4 46       self.menue_umschalten();
U 47     });
f32042 48     document.querySelector('#top-neuer-text-btn').addEventListener('click', self.datei_neuer_text);
U 49     document.querySelector('#top-neuer-ordner-btn').addEventListener('click', self.datei_neuer_ordner);
50     document.querySelector('#top-close-btn').addEventListener('click', self.fm_menu_datei_schliessen);
58bcbd 51     self.fm_get_login();
41ab37 52     
U 53     var parsedUrl = new URL(window.location.href);
54     var wunschPfad = parsedUrl.searchParams.get("pfad");
55
56     if(wunschPfad !== null && wunschPfad.length > 0) {
57       self.pfad = wunschPfad;
58       self.fm_get_list(wunschPfad);
59     } else {
60       self.fm_get_list('');
61     }
ebc1f3 62     
U 63     self.seitenleiste_umschalten();
41ab37 64     //fm_init_uploader();
58bcbd 65     self.loc = window.location.protocol + '//' + window.location.host;
05e9c4 66   };
U 67
58bcbd 68   this.login_zeigen = function() {
U 69     self.meldung_mit_timeout("Benutzer: " + self.userid, 1500);
70   };
71   
b20b74 72   this.fm_neuer_reiter = function() {
U 73     window.open('/file-cms/ui2/?pfad=' + self.pfad, '_blank');
74   };
75   
94b871 76   this.menue_umschalten = function () {
05e9c4 77     var ham = document.querySelector(".hamburger");
U 78     ham.classList.toggle("is-active"); // hamburger-icon umschalten
79     self.appMenu.toggle(); // menue oeffnen/schliessen
80   };
81
94b871 82   this.info_dialog_zeigen = function () {
05e9c4 83     self.dialog_laden_und_zeigen('data/tpl/dlg-info.tpl', '');
U 84     self.menue_umschalten();
85   };
86
94b871 87   this.seitenleiste_umschalten = function () {
05e9c4 88     var ostDiv = document.querySelector('.ost');
94b871 89     if (ostDiv.classList.contains('ost-open')) {
05e9c4 90       ostDiv.classList.remove('ost-open');
94b871 91       ostDiv.style.flexBasis = '0em';
05e9c4 92     } else {
94b871 93       ostDiv.classList.add('ost-open');
U 94       ostDiv.style.flexBasis = '6em';
05e9c4 95     }
ebc1f3 96     //self.menue_umschalten();
05e9c4 97   };
U 98
94b871 99   this.fusszeile_umschalten = function () {
05e9c4 100     var suedDiv = document.querySelector('.sued');
94b871 101     if (suedDiv.classList.contains('sued-open')) {
05e9c4 102       suedDiv.classList.remove('sued-open');
94b871 103       suedDiv.style.height = '0';
05e9c4 104     } else {
U 105       suedDiv.classList.add('sued-open');
94b871 106       suedDiv.style.height = '1.5em';
05e9c4 107     }
U 108     self.menue_umschalten();
109   };
110
94b871 111   this.meldung_mit_timeout = function (meldung, timeout) {
05e9c4 112     var s = document.querySelector('.sued');
U 113     s.textContent = meldung;
85ba85 114     s.classList.add('sued-open');
U 115     s.style.height = '1.5em';
94b871 116     setTimeout(function () {
05e9c4 117       s.textContent = 'Bereit.';
94b871 118       setTimeout(function () {
05e9c4 119         var suedDiv = document.querySelector('.sued');
94b871 120         if (suedDiv.classList.contains('sued-open')) {
U 121           suedDiv.classList.remove('sued-open');
122           suedDiv.style.height = '0';
05e9c4 123         }
U 124       }, 500);
125     }, timeout);
126   };
ebc1f3 127   
U 128   this.fm_menu_datei_schliessen = function() {
129     if(self.openEditor === 'text') {
130       if(!self.cm.getDoc().isClean()) {
131         self.fm_ask_for_save();
132       } else {
133         self.fm_do_close();
134       }
135     } else {
4e450e 136       if(tinymce.activeEditor.undoManager.hasUndo()) {
ebc1f3 137         self.fm_ask_for_save();
U 138       } else {
139         self.fm_do_close();
140       }
141     }
142   };  
143
144   this.fm_ask_for_save = function() {
145     self.dialog_laden_und_zeigen('data/tpl/dlg-ask-save.tpl', '', function() {
146       // wenn dialog da ist, hier events verknuepfen
147       document.querySelector('#cancel-btn').addEventListener('click', function() {
148         self.fm_do_close();
149         self.dialog_schliessen();
150       });
151       document.querySelector('#speichern-btn').addEventListener('click', function() {
152         self.fm_menu_datei_speichern(function() {
153           self.fm_do_close();
154           self.dialog_schliessen();
155         });
156       });
157     });
158   };
159
160   this.fm_menu_datei_speichern = function(callback) {
161     //var fname = $('.datei-gewaehlt').text();
162     var fname = self.openFileName;
163     if(fname !== undefined && fname !== '') {
164       self.fm_save_file(fname, 'saveTextFile', callback);
165     } else {
166       self.fm_menu_datei_speichern_unter(callback);
167     }
168   };
169   
170   this.fm_menu_datei_speichern_unter = function(callback) {  
171     self.dialog_laden_und_zeigen('data/tpl/dlg-save-as.tpl', '', function() {
5b4d03 172       document.querySelector('#cancel-btn').addEventListener('click', function() {
U 173         //self.fm_do_close();
174         self.dialog_schliessen();
175       });
176       document.querySelector('#speichern-btn').addEventListener('click', function() {
1a9589 177         //self.fm_menu_datei_speichern(function() {
U 178           //console.log(document.querySelector('#datei-name-in').value);
5b4d03 179           self.fm_save_file(document.querySelector('#datei-name-in').value, 'saveTextFileAs', callback);
4e450e 180           if(typeof(callback) !== 'function') {
U 181             self.fm_do_close();
182             self.dialog_schliessen();
183           }
1a9589 184         //});
5b4d03 185       });
ebc1f3 186     });    
4e450e 187   };
U 188   
189   this.fm_menu_neues_dokument = function() {
190     self.fm_dok_edit('');
ebc1f3 191   };
05e9c4 192
U 193   /* Dialog-Funktionen */
194
195   /*
94b871 196    Einen Dialog aus Vorlagen erzeugen
U 197    
198    vurl - URL zur Dialogvorlage
199    msgTpl - URL mit einer Vorlage eines Mitteilungstextes (optional)
200    */
ebc1f3 201   this.dialog_laden_und_zeigen = function (vurl, msgTpl, cb) {
U 202     var vorlage = self.cache[vurl];
203     if(vorlage === undefined) {
204       self.fm_get(vurl, "text", function(antwort) {
205         self.cache[vurl] = antwort;
206         self.dialog_zeigen(vurl, antwort, cb);
207         //self.vorlage_fuellen(vurl, inhalt, cb);
94b871 208       });
05e9c4 209     } else {
ebc1f3 210       self.dialog_zeigen(vurl, vorlage, cb);
05e9c4 211     }
U 212   };
213
ebc1f3 214   this.dialog_zeigen = function (vurl, inhalt, cb) {
05e9c4 215     var dlg = document.querySelector(".dialog");
ebc1f3 216     self.html_erzeugen(
94b871 217             vurl,
U 218             inhalt,
219             function (html) {
220               //dlg.html(html);
1a9589 221               dlg.style.height = '7em';
94b871 222               dlg.innerHTML = html;
U 223               document.querySelector('.close-btn').addEventListener('click', self.dialog_schliessen);
224               //dlg.slideDown(300);
ebc1f3 225               if(typeof(cb) !== 'function') {
U 226                 // ..
227               } else {
228                 cb();
229               }
94b871 230             });
05e9c4 231   };
U 232
58bcbd 233   this.dialog_schliessen = function () {
05e9c4 234     document.querySelector('.close-btn').removeEventListener('click', self.dialog_schliessen);
U 235     //$('.dialog').slideUp(300);
236     var dlg = document.querySelector('.dialog');
237     //dlg.style.display = "none";
238     dlg.style.height = '0';
239     dlg.innerHTML = '';
5b4d03 240     self.removeAllListeners('#cancel-btn');
U 241     self.removeAllListeners('#speichern-btn');
05e9c4 242   };
39e714 243   
U 244   this.fm_auswahl_marke_entfernen = function() {
245     var gew = document.querySelector('.datei-gewaehlt');
246     if(gew !== null) {
247       gew.classList.remove('datei-gewaehlt');
248     }        
249   };
0783c5 250   
U 251   this.fm_view_file = function() {
252     var fname = document.querySelector('.datei-gewaehlt').textContent;
253     var path = self.fm_get_path(self.userid);
254     window.open(path + '/' + fname, '_blank');
255   };
39e714 256
0783c5 257   this.datei_ansehen_oder_bearbeiten = function(dateiname) {
U 258     if(dateiname.endsWith('htmi')) {
259       self.fm_get_file_content('doc', dateiname);
260     } else if(dateiname.endsWith('jpg') || dateiname.endsWith('png') || 
261             dateiname.endsWith('gif') || dateiname.endsWith('pdf')) {
262       
263     } else {
264       self.fm_get_file_content('text', dateiname);      
265     }
39e714 266   };
U 267   
268   /**
85ba85 269    * Hier werden die folgenden Faelle einer Datei- 
U 270    * bzw. Ordnerauswahl ausgefuehrt:
271    * 
272    * geklickt, nicht gewaehlt: 
273    *    auswaehlen, evtl. bestehende andere Auswahl entfernen
274    *    
275    * geklickt, ausgewaehlt: 
276    *    Datei ansehen oder oeffnen bzw. Ordner oeffnen
277    *    
278    * geklickt mit [Strg], nicht gewahlt: 
279    *    auswaehlen, bestehende Auswahl beibehalten (Mehrfachauswahl)
280    *    
281    * geklickt mit [Strg], gewaehlt: 
282    *    Auswahl bei diesem Element entfernen
39e714 283    * 
U 284    * Diese Fälle sind unterteilt in 
85ba85 285    *    1. Kachel-Modus
U 286    *    2. Listenmodus
39e714 287    * 
85ba85 288    * @param {type} ev Klick-Event
U 289    * @returns {undefined} nichts 
39e714 290    */
6648a8 291   this.fm_dateiwahl = function(ev) {
U 292     var elem = ev.target;
293     if(self.modus == 'kacheln') {
294       // Kacheln
0783c5 295       var par = elem.parentElement;
U 296       var dElem = par.querySelector('.dateiname');
6648a8 297       if(elem.classList.contains("icon-folder")) {
39e714 298         // Ordner
0783c5 299         //var par = elem.parentElement;
U 300         var istGewaehlt = dElem.classList.contains('datei-gewaehlt');
39e714 301         if(istGewaehlt) {
U 302           if(/*ev.shiftKey || */ ev.ctrlKey) {
303             // mehrere Ordner sind gewaehlt, diese eine wieder entfernen
0783c5 304             dElem.classList.remove('datei-gewaehlt');
6648a8 305           } else {
39e714 306             // Ordner oeffnen
0783c5 307             var ordner = dElem.textContent.trim();
39e714 308             if(self.pfad.length > 0) {
U 309               self.pfad = self.pfad + '/' + ordner;
310             } else {
311               self.pfad = ordner;
312             }
313             self.fm_get_list(self.pfad);
6648a8 314           }
39e714 315         } else {
U 316           // Ordner auswaehlen
317           if(/*ev.shiftKey || */ ev.ctrlKey) {
318             // Mehrfachauswahl
319           } else {
320             self.fm_auswahl_marke_entfernen();
321           }
0783c5 322           dElem.classList.add('datei-gewaehlt');
6648a8 323         }
U 324       } else if(elem.classList.contains('datei')) {
39e714 325         // Datei
0783c5 326         //var par = elem.parentElement;
U 327         //var dElem = par.querySelector('.dateiname');
328         var istGewaehlt = dElem.classList.contains('datei-gewaehlt');
39e714 329         if(istGewaehlt) {
U 330           if(/*ev.shiftKey || */ ev.ctrlKey) {
331             // mehrere Dateien sind gewaehlt, diese eine wieder entfernen
0783c5 332             dElem.classList.remove('datei-gewaehlt');
39e714 333           } else {
U 334             // Datei zum Bearbeiten oeffnen
0783c5 335             self.datei_ansehen_oder_bearbeiten(dElem.textContent);
39e714 336           }
U 337         } else {
338           if(/*ev.shiftKey || */ ev.ctrlKey) {
339             // mehrere Dateien sollen gewaehlt werden
340           } else {
341             self.fm_auswahl_marke_entfernen();
342           }
0783c5 343           dElem.classList.add('datei-gewaehlt');
39e714 344         }
6648a8 345       } else {
U 346         //console.log('kein folder oder file...');
347       }
348     } else {
4f01b8 349       // Liste
U 350       var pElem = elem.closest('.datei-zeile');
351       var dElem = pElem.querySelector('.datei-elem');
0783c5 352       var dnElem = pElem.querySelector('.dateiname');
U 353       var istGewaehlt = dnElem.classList.contains('datei-gewaehlt');
4f01b8 354       if(dElem.getElementsByTagName("i")[0].classList.contains('icon-doc-text-inv')) {
6648a8 355         // Datei
39e714 356         if(istGewaehlt) {
U 357           if(/*ev.shiftKey || */ ev.ctrlKey) {
358             // mehrere Dateien sind gewaehlt, diese eine wieder entfernen
0783c5 359             dnElem.classList.remove('datei-gewaehlt');
39e714 360           } else {
U 361             // gewaehlte Datei ohne [Strg] geklickt: ansehen oder oeffnen
0783c5 362             self.datei_ansehen_oder_bearbeiten(dnElem.textContent);
39e714 363           }
6648a8 364         } else {
39e714 365           if(/*ev.shiftKey || */ ev.ctrlKey) {
U 366             // mehrere Dateien sollen gewaehlt werden, Auswahl ist weiter unten
367           } else {
368             var ti = document.querySelector('.table-info');
369             if(ti !== null) {
370               ti.classList.remove('table-info');
371             }
372             self.fm_auswahl_marke_entfernen();
4f01b8 373           }
39e714 374           pElem.classList.add('table-info');
0783c5 375           dnElem.classList.add('datei-gewaehlt');
6648a8 376         }
U 377       } else {
378         // Ordner
39e714 379         if(istGewaehlt) {
U 380           if(/*ev.shiftKey || */ ev.ctrlKey) {
381             // ein gewaehlter Ordner ist mit [Strg] geklickt, Auswahl entfernen
0783c5 382             dnElem.classList.remove('datei-gewaehlt');
6648a8 383           } else {
39e714 384             // Ordner oeffnen
0783c5 385             var ordner = dnElem.textContent;
39e714 386             if(self.pfad.length > 0) {
U 387               self.pfad = self.pfad + '/' + ordner;
388             } else {
389               self.pfad = ordner;
390             }
391             self.fm_get_list(self.pfad);
6648a8 392           }
39e714 393         } else {
U 394           if(/*ev.shiftKey || */ ev.ctrlKey) {
395             // nicht gewaehlter Ordner und [Strg]: zu Mehrfachauswahl hinzufuegen
396           } else {
397             // nicht gewaehlter Ordner ohne [Strg]: andere Auswhalen entfernen und weiter unten Ordner auswaehlen
398             self.fm_auswahl_marke_entfernen();
399           }
0783c5 400           dnElem.classList.add('datei-gewaehlt');
6648a8 401         }
U 402       }   
403     }
404   };
405  
41ab37 406   
58bcbd 407   this.fm_render_list = function (fl) {
4f01b8 408     if (self.modus === 'kacheln') {
58bcbd 409       // Kachelansicht
6648a8 410       self.html_erzeugen("data/tpl/kacheln.tpl", fl, function(html) {
U 411         var elem = document.querySelector('#dateien');
412         elem.innerHTML = html;
413         self.addEvtListener('.figure', 'click', self.fm_dateiwahl);
414       });
58bcbd 415     } else {
U 416       // Listenansicht
4f01b8 417       self.html_erzeugen("data/tpl/liste.tpl", fl, function(html) {
U 418         var elem = document.querySelector('#dateien');
419         elem.innerHTML = html;
420         self.addEvtListener('.datei-zeile', 'click', self.fm_dateiwahl);
421       });
58bcbd 422     }
U 423   };
424   
425   this.fm_get_path = function (uid) {
426     var restdir;
427     if (self.pfad.indexOf(self.PUB_DIR) > -1) {
428       restdir = self.pfad.substr(self.PUB_DIR.length);
429     } else if (self.pfad.indexOf(self.PERS_DIR) > -1) {
430       restdir = self.pfad.substr(self.PERS_DIR.length);
431     } else if (self.pfad.indexOf(self.BASE_DIR) > -1) {
432       restdir = self.pfad.substr(self.BASE_DIR.length);
433     } else if (self.pfad.indexOf(self.DATA_DIR) > -1) {
434       restdir = self.pfad.substr(self.DATA_DIR.length);
435     } else if (self.pfad.indexOf(self.DAV_DIR) > -1) {
436       restdir = self.pfad.substr(self.DAV_DIR.length);
437     }
438     if (restdir !== undefined && restdir.startsWith('/')) {
439       restdir = restdir.substr(1);
440       if (restdir.indexOf(self.WWW_DIR) > -1) {
441         restdir = restdir.replace(self.WWW_DIR, 'data');
442       }
443     }
41ab37 444     var pdir = self.fm_get_base(uid);
58bcbd 445     if (restdir.length > 1) {
U 446       return pdir + "/" + restdir;
447     } else {
448       return pdir;
449     }
450   };
451
452   this.fm_get_base = function (uid) {
453     var pdir;
454     if (self.pfad.indexOf(self.PUB_DIR) > -1) {
455       pdir = '/data/' + uid;
456     } else if (self.pfad.indexOf(self.PERS_DIR) > -1) {
457       pdir = '/home/' + uid;
458     } else if (self.pfad.indexOf(self.BASE_DIR) > -1) {
459       pdir = '';
460     } else if (self.pfad.indexOf(self.DATA_DIR) > -1) {
461       pdir = '';
462     }
463     return pdir;
464   };
2864b2 465   
U 466   /**
467    * Aus einem relativen Pfad ein Array aus BcrFile Objekten 
468    * machen
469    * 
470    * @param {String} relPfad  der relative Pfad
471    * @returns {Array}  die BcrFile-Objekte zum Pfad als Array
472    */
473   this.fm_buildBreadcrumb = function(relPfad) {
474     var rp = '';
475     var dirList = new Array();
476     dirList.push(new BcrFile(rp, 'Home'));
477     if(relPfad.length > 1) {
478       var dirs = relPfad.split('/');
479       for(var i = 0; i < dirs.length; i++) {
480         if(rp.length > 0 ) {
481           // weitere Einträge
482           dirList.push(new BcrFile(rp + '/' + dirs[i], dirs[i]));
483           rp = rp + '/' + dirs[i];
484         } else {
485           // erster Eintrag
486           dirList.push(new BcrFile(dirs[i], dirs[i]));
487           rp = dirs[i];
488         }
489       }
490     }
491     return dirList;
492   };
493   
4f01b8 494   /*
U 495    * icon-th-large
496    * icon-th-list
497    * @returns {undefined}
498    */
499   this.fm_ansicht_umschalten = function() {
500     var elem = document.querySelector('#ansicht');
501     var iElem = elem.getElementsByTagName("i")[0];
502     if(iElem.classList.contains('icon-th-list')) {
503       iElem.classList.add('icon-th-large');
504       iElem.classList.remove('icon-th-list');
505       self.modus = 'liste';
506     } else {
507       iElem.classList.add('icon-th-list');
508       iElem.classList.remove('icon-th-large');
509       self.modus = 'kacheln';
510     }
511     self.fm_get_list(self.pfad);    
512   };
513
514   this.fm_set_modus = function() {
515     var elem = document.querySelector('#ansicht');
516     var iElem = elem.getElementsByTagName("i")[0];
517     if(self.modus === 'kacheln') {
518       iElem.classList.add('icon-th-list');    
519       iElem.classList.remove('icon-th-large');
520     } else {
521       iElem.classList.add('icon-th-large');
522       iElem.classList.remove('icon-th-list');
523     }    
524   };
525
526   
2864b2 527   /**
U 528    * Der letzte Eintrag in dirs ist der aktuelle Ordner und 
529    * deshalb ausgegraut.
530    * 
531    * Wenn nur ein Ordner in dirs enthalten ist, dann sind wir 
532    * an der obersten Ebene (Home).
533    * 
534    * @param {Array} dirList ein Array aus BcrFile-Objekten
535    * @returns nichts 
536    */
537   this.fm_renderBreadcrumb = function(dirList) {
538     var elem = document.querySelector('.breadcrumb');
539     if(dirList.length > 1) {
540       var last = dirList.pop();
541       var bcList = new BcrFiles(dirList);
4f01b8 542       
2864b2 543       self.html_erzeugen("data/tpl/bcr.tpl", bcList, function(html) {
4f01b8 544         var htmlGesamt = html;
2864b2 545         self.html_erzeugen("data/tpl/bcr2.tpl", last, function(html) {
4f01b8 546           htmlGesamt += html;          
U 547           self.html_erzeugen("data/tpl/bcr3.tpl", dirList[0], function(html) {
548             htmlGesamt += html;
549             elem.innerHTML = htmlGesamt;
550             self.addEvtListener('.bc-link', 'click', self.fm_bc_click);
551             self.addEvtListener('#ansicht', 'click', self.fm_ansicht_umschalten);
552             self.fm_set_modus();
553           });
554           
2864b2 555         });                  
U 556       });            
557     } else {
558       // oberste Ebene
4f01b8 559       var htmla;
U 560       var htmlb;
2864b2 561       self.html_erzeugen("data/tpl/bcr2.tpl", dirList[0], function(html) {
4f01b8 562         htmla = html;        
U 563         self.html_erzeugen("data/tpl/bcr3.tpl", dirList[0], function(html) {
564           htmlb = html;
565           elem.innerHTML = htmla + htmlb;
566           self.addEvtListener('.bc-link', 'click', self.fm_bc_click);
567           self.addEvtListener('#ansicht', 'click', self.fm_ansicht_umschalten);
568           self.fm_set_modus();
569         });
2864b2 570       });            
U 571     }
572   };
573   
574   /**
575    * Auf den Klick auf ein Breadcrumb-Element reagieren:
576    * Den Ordner-Inhalt des geklickten Elements anzeigen
577    * @returns nichts
578    */
579   this.fm_bc_click = function() {
580     var elem = this;
581     var bcPfad = elem.getAttribute("rpath");
582     if(bcPfad !== undefined) {
583       self.pfad = bcPfad;
584       self.fm_get_list(bcPfad);
585     } else {
586       pfad = '';
587       self.fm_get_list('');
588     }
589   };
590
beb124 591   /* ------- Dateifunktionen Start ----------- */
U 592
593   this.fm_menu_cut = function() {
594     console.log('fm_menu_cut');
595     self.fm_cut_files();
596   };
39e714 597   this.fm_menu_copy = function() {
beb124 598     console.log('fm_menu_kopie');
U 599     self.fm_copy_files();
600   };
601   this.fm_menu_paste = function() {
602     console.log('fm_menu_paste');
603     self.fm_paste_files();
604   };
605   this.fm_menu_delete = function() {
606     self.dialog_laden_und_zeigen('data/tpl/dlg-ask-del.tpl', '', function() {
607       // wenn dialog da ist, hier events verknuepfen
608       document.querySelector('#cancel-btn').addEventListener('click', function() {
609         //self.fm_do_close();
610         self.dialog_schliessen();
611       });
612       document.querySelector('#speichern-btn').addEventListener('click', function() {
39e714 613         //self.fm_menu_datei_speichern(function() {
beb124 614           self.fm_del_files();
U 615           self.dialog_schliessen();
39e714 616         //});
beb124 617       });
U 618     });
619     /*
620     $('#confirmOk').click(function() {
621       $('#confirmOk').attr('onclick','').unbind('click');
622       fm_del_files();
623     });
624     $('#confirmOk').text("Loeschen");
625     $('#confirmOk').removeClass('btn-primary');
626     $('#confirmOk').addClass('btn-danger');
627     $('#confirmClose').hide();
628     $('#confirmModalTitle').text("Löschen");
629     $('#confirmModalBody').text("Wirklich löschen?");
630     $('#confirmModal').modal({
631       keyboard: false,
632       show: true
633     });
634     */
635   };
636   
637   /*
638    * Hier merkt sich die Dateiverwalting die markierten Dateien fuer 
639    * eine Dateioperation. Bei cut wird in der 'paste' Operation auf dem Server 
640    * fuer die 'gemerkten' Dateien ein Move gemacht. Die Dateien werden 
641    * vom gewaehlten Ort zum Zielort verschoben.
642    * 
643    * @returns {undefined}
644    */
645   this.fm_cut_files = function() {
646     console.log('fm_cut_files');
647     self.cutCopySrcDir = self.pfad;
648     self.cutCopyFiles = self.fm_gewaehlte_dateien();
649     self.cutCopyOperation = 'cut';
650   };
651
652   /*
653    * Hier merkt sich die Dateiverwalting die markierten Dateien fuer 
654    * eine Dateioperation. Bei copy wird in der 'paste' Operation auf dem Server 
655    * fuer die 'gemerkten' Dateien eine Kopie gemacht. Die Dateien werden 
656    * am gewaehlten Ort behalten und am Zielort wird eine Kopier der 
657    * gewaehlten DAteien erstellt
658    * 
659    * @returns {undefined}
660    */
661   this.fm_copy_files = function() {
662     console.log('fm_copy_files');
663     self.cutCopySrcDir = self.pfad;
664     self.cutCopyFiles = self.fm_gewaehlte_dateien();
665     self.cutCopyOperation = 'copy';
666   };
667   
668   
669
670   /*
671    * Gewaehlte Dateien feststellen
672    * 
673    * @returns {String} eine per JSON.stringify schon vorbereitete List zum 
674    * Absenden als Parameter an den Server
675    */
676   this.fm_gewaehlte_dateien = function() {
677     //var gewaehlte = $('.datei-gewaehlt');
678     var gewaehlte = document.querySelectorAll('.datei-gewaehlt');
679     /*
680     //console.log('anzahl: ' + gewaehlte.length);
681     var fnames = new Array();
682     var i = 0;
683     while(i < gewaehlte.length) {
684       var str = { "java.lang.String": $(gewaehlte[i]).text()};
685       fnames.push(str);
686       //console.log('loeschen ' + pfad + ' ' + $(gewaehlte[i]).text());
687       //i += 2;
688       i++;
689     }
690     var liste = '{"List":' + JSON.stringify(fnames) + '}';
691     // 'liste' enthaelt:
692     // {"List":[{"java.lang.String":"test3.txt"},{"java.lang.String":"Test1.txt"},{"java.lang.String":"Test2.txt"}]}
693     // und wird auf dem Server zu einer List[ArrayList<String>] bei der jede ArrayList<String> nur ein Element hat 
694     return liste;
695     */
696     return self.fm_dateiliste_bilden(gewaehlte);
697   };
698
699   this.fm_dateiliste_bilden = function(gewaehlte) {
700     var fnames = new Array();
701     var i = 0;
702     while(i < gewaehlte.length) {
703       var str = { "java.lang.String": gewaehlte[i].innerText};
704       fnames.push(str);
705       //console.log('loeschen ' + pfad + ' ' + $(gewaehlte[i]).text());
706       //i += 2;
707       i++;
708     }
709     var liste = '{"List":' + JSON.stringify(fnames) + '}';
710     // 'liste' enthaelt:
711     // {"List":[{"java.lang.String":"test3.txt"},{"java.lang.String":"Test1.txt"},{"java.lang.String":"Test2.txt"}]}
712     // und wird auf dem Server zu einer List[ArrayList<String>] bei der jede ArrayList<String> nur ein Element hat 
713     console.log(liste);
714     return liste;
715   };
716
717   this.fm_menu_html_export = function() {
718     self.fm_export_html();
719   };
720
721   /* --------- Dateifunktionen Ende ---------- */
722
94b871 723   /* API functions */
05e9c4 724
94b871 725   // http://localhost:8079/file-cms/svc?c=de.uhilger.filecms.api.FileMgr&f=JSONNICE&m=list&p=
U 726   this.fm_get_list = function (relPfad) {
727     var m = '?c=de.uhilger.filecms.api.FileMgr&m=list&p=' + relPfad;
728     var u = '../svc' + m;
58bcbd 729     self.fm_get(u, "json", function (respText) {
U 730       var resp = JSON.parse(respText);
94b871 731       if (resp.List[0].FileRef !== undefined) {
U 732         var files = new Array();
733         if (resp.List[0].FileRef instanceof Array) {
734           for (var i = 0; i < resp.List[0].FileRef.length; i++) {
735             files.push(new FileRef(resp.List[0].FileRef[i]));
05e9c4 736           }
U 737         } else {
94b871 738           files.push(new FileRef(resp.List[0].FileRef));
05e9c4 739         }
94b871 740         var fl = new FileList(files);
58bcbd 741         self.fm_render_list(fl);
94b871 742       } else {
58bcbd 743         // #dateien leeren
8ba358 744         var elem = document.querySelector("#dateien");
U 745         elem.innerHTML = '';
94b871 746       }
8ba358 747       
U 748       // Breadcrumb
749       
2864b2 750       var dirList = self.fm_buildBreadcrumb(relPfad);
4f01b8 751       self.fm_renderBreadcrumb(dirList);     
94b871 752     });
U 753   };
f002d9 754   
1a9589 755   this.datei_neuer_ordner = function() {
U 756     self.dialog_laden_und_zeigen('data/tpl/dlg-ask-folder.tpl', '', function() {
757       document.querySelector('#cancel-btn').addEventListener('click', function() {
758         self.dialog_schliessen();
759       });
760       document.querySelector('#speichern-btn').addEventListener('click', function() {
761         var m = '?c=de.uhilger.filecms.api.FileMgr&m=newFolder&p=' + 
762                 self.pfad + 
763                 '&p=' + document.querySelector('#folder-name-in').value;
764         var u = '../svc' + m;
765         self.fm_get(u, "json", function(resp) {
766           self.dialog_schliessen();
767           self.fm_get_list(self.pfad);
768         });
769       });
770     });    
771   };
772
ebc1f3 773   this.fm_edit_as_text = function() {
0783c5 774     self.fm_get_sel_for_file_content('text');
4e450e 775   };
U 776   
777   this.fm_edit_as_doc = function() {
0783c5 778     self.fm_get_sel_for_file_content('doc');
ebc1f3 779   };
U 780   
0783c5 781   this.get_sel_for_file_content = function(typ) {
f002d9 782     var gewaehlte = document.querySelector('.datei-gewaehlt');
U 783     //var fname = $(gewaehlte).find('.dateiname').text();
784
785     var fname = gewaehlte.textContent;
0783c5 786     self.fm_get_file_content(typ, fname);
U 787   };
788   
789   this.fm_get_file_content = function(typ, fname) {
790     //var gewaehlte = document.querySelector('.datei-gewaehlt');
791     //var fname = $(gewaehlte).find('.dateiname').text();
792
793     //var fname = gewaehlte.textContent;
ebc1f3 794     //console.log('fname: ' + fname);
f002d9 795     self.openFileName = fname;
U 796     var m = '?c=de.uhilger.filecms.api.FileMgr&m=getCode&p=' + self.pfad + '&p=' + fname;
797     var u = '../svc' + m;
798     self.fm_get(u, "text", function(resp) {
799       if(typ === 'text') {
800         var mode = "text/x-java";
801         if(fname.endsWith('js')) {
802           mode = 'javascript';
803         } else if(fname.endsWith('xml')) {
804           mode = 'xml';
805         } else if(fname.endsWith('properties')) {
806           mode = 'xml';
807         } else if(fname.endsWith('adoc')) {
808           mode = 'text/x-markdown';
809         }
810         self.fm_text_edit(resp, mode);
811       } else {
812         self.fm_dok_edit(resp);
813       }
814     });
815   };
816
ebc1f3 817   this.fm_save_file = function(saveFileName, method, callback) {
U 818     var content;
819     if(self.openEditor === 'text') {
820       content = self.cm.getValue();
821       self.cm.getDoc().markClean();
822     } else {
4e450e 823       content = self.ed.getContent();
ebc1f3 824       tinymce.activeEditor.undoManager.clear();
U 825     }
826     var m = '?c=de.uhilger.filecms.api.FileMgr&m=' + method;
5b4d03 827     var u = '../svc' + m;
U 828     var data = '&p=' + self.pfad + '&p=' + saveFileName + '&p=' + encodeURIComponent(content);
829     self.fm_post(u, data, "text", function(resp) {
830       // ...
ebc1f3 831     });
U 832     self.openFileName = saveFileName;
833     if(typeof (callback) !== 'function') {
834
835     } else {
836       callback();
837     }
838   };
839
beb124 840   /*
U 841    * Hier wird fuer eine zuvor markierte Liste von Dateien, fuer die 
842    * Cut oder Copy gewaehlt wurde, die Operations ausgefuehrt (move oder copy)
843    * @returns {undefined}
844    */
845   this.fm_paste_files = function() {
846     var m;
847     if(self.cutCopyOperation === 'cut') {
848       //m = '?c=de.uhilger.filecms.api.FileMgr&m=moveFiles'; //&p=' + cutCopySrcDir + '&p=' + pfad + '&p=' + encodeURIComponent(liste);
849       m = '?c=de.uhilger.filecms.api.FileMgr&m=moveFiles&p=' + self.cutCopySrcDir + '&p=' + self.pfad + '&p=' + encodeURIComponent(self.cutCopyFiles);
850     } else {
851       //m = '?c=de.uhilger.filecms.api.FileMgr&m=copyFiles'; //&p=' + cutCopySrcDir + '&p=' + pfad + '&p=' + encodeURIComponent(liste);
852       m = '?c=de.uhilger.filecms.api.FileMgr&m=copyFiles&p=' + self.cutCopySrcDir + '&p=' + self.pfad + '&p=' + encodeURIComponent(self.cutCopyFiles);
853     }
854     var u = '../svc' + m;  
855     self.fm_get(u, "text", function(resp) {
856       // console.log('deleteFiles gab folgendes zurueck: ' + resp);
857       self.fm_get_list(self.pfad);
858     });
859   };
ebc1f3 860
beb124 861   this.fm_del_files = function() {
U 862     var liste = self.fm_gewaehlte_dateien();
863     var m = '?c=de.uhilger.filecms.api.FileMgr&m=deleteFiles&p=' + self.pfad + '&p=' + encodeURIComponent(liste);
864     var u = '../svc' + m;
865     self.fm_get(u, "text", function(resp) {
866       // console.log('deleteFiles gab folgendes zurueck: ' + resp);
867       self.fm_get_list(self.pfad);
868     });
869   };
f002d9 870
U 871 /* ---- codemirror editor handling -------- */
872
873   /*
874   function htmlDecode(value){ 
875     return $('<div/>').html(value).text(); 
876   }
877   */
878
879   this.fm_code_edit = function(content, m) {
ebc1f3 880     //console.log('fm_code_edit content: ' + content.substring(0,30));
U 881     self.cm = CodeMirror.fromTextArea(document.getElementById("editspace"), {
f002d9 882       lineNumbers: true,
U 883       lineWrapping: true,
884       gutters: ["CodeMirror-linenumbers", "breakpoints"],
885       mode: m,
886       viewportMargin : Infinity,
887       tabSize: 2,
888       extraKeys: {
889           "F9": function(cm) {
890           cm.setOption("fullScreen", !cm.getOption("fullScreen"));
891         },
892           "Esc": function(cm) {
893           if (cm.getOption("fullScreen")) cm.setOption("fullScreen", false);
894         },
895           ".": function(cm) {
896           console.log('dot pressed: perhaps look up class or method name');
897           /*
898            * Hier kann man eine Funktion ausloesen, die fuer das 
899            * Wort vor dem Punkt (Name der Klasse) eine Liste mit 
900            * Vorschlaegen fuer Methodennamen einblendet.
901            */
902
903           /*
904            * CodeMirror.Pass laesst das Zeichen zum Editorinhalt durch, 
905            * verhindert aber das Ausloesen von 'keyHandled'
906            */
907           return CodeMirror.Pass; 
908         }
909       }
910     });
911     //cm.setValue(htmlDecode(content));
912     //cm.setValue(content);
ebc1f3 913     self.cm.setValue(self.unescapeHtml(content));
U 914     self.cm.getDoc().markClean();
915     self.cm.on("gutterClick", function(theEditor, lineNumber) {
f002d9 916       var info = theEditor.lineInfo(lineNumber);
U 917       //--lineNumber;
918       //console.log(info.gutterMarkers.breakpoints.message);
919       //var marker = info.gutterMarkers.breakpoints;
920       //$(marker).tooltip('toggle');
921     });
922     /*
923     cm.on("keyHandled", function(theEditor, keyName, event){
924       console.log('cm.keyHandled keyName: ' + keyName + ', event.type: ' + event.type);    
925     });
926     */
927   };
928
929   this.makeMarker = function(msg) {
930     var marker = document.createElement("div");
931     marker.style.color = "#822";
932     marker.innerHTML = "●";
933     //marker.message = msg;
934     $(marker).tooltip({
935       placement: 'right',
936       title: msg,
937       offset: '0 -30'
938     });
939     return marker;
940   };
941
4e450e 942   /* ---- TinyMCE editor handling -------- */
U 943
944   this.fm_dok_editor_init = function(uid) {
945     var base = self.fm_get_path(uid);
946     //var edCount = 0;
947     //console.log("fm_dok_editor_init calling tinymce.init with base: " + base + "/");
948
949     /*
950      * vgl.
951      * http://stackoverflow.com/questions/4651676/how-do-i-remove-tinymce-and-then-re-add-it
952      */
953     //tinymce.EditorManager.execCommand('mceRemoveControl',true, 'textarea.text-editor');
954     //tinymce.EditorManager.execCommand('mceAddControl',true, editor_id);
955     if(self.ed !== undefined) {
956       self.ed.destroy();
957     }
958     if(tinymce !== undefined) {
959       tinymce.remove('textarea.text-editor');
960       tinymce.EditorManager.editors = []; 
961     }
962
963     /*
964      * Konfiguration TinyMCE
965      */
966     tinymce.init({
967       content_css : "/file-cms/ui/mce.css",
968       //content_css: "/jslib/bootstrap/css/bootstrap.min.css",
969       selector: "textarea.text-editor",
970       statusbar: false,
971       menubar: false,
972       plugins: 'advlist charmap code image link lists media table print preview save table textcolor importcss',
973       toolbar: 'undo redo | styleselect | image table | link unlink | bullist numlist | outdent indent | code',
974       /*
975       menu: {
976         file: {title: 'File', items: 'savevers | exit'},
977         edit: {title: 'Edit', items: 'undo redo | cut copy paste pastetext | selectall'},
978         view: {title: 'View', items: 'visualaid | code | link image media | template hr'},
979       },
980       */
981       resize: 'both',
982       importcss_append: true,
983       width: "100%",
984       height: '100%',
985       relative_urls : true, 
986       convert_urls : false, 
987       document_base_url : base + "/",
988       setup: function (editor) {
989         self.ed = editor;
990       }
991     });
992
993     window.addEventListener('resize', self.fm_resize_editor);
994     //$(window).on('resize', self.fm_resize_editor);
995     self.fm_resize_editor();
996   };
997
998   this.fm_resize_editor = function() {
999     window.clearTimeout(self.tmo);
1000     self.tmo = window.setTimeout(function () {
1001       try {
1002         var myHeight = document.querySelector('.inhalt').offsetHeight - 
1003                 document.querySelector('.nord').offsetHeight + 2; // - 4;
1004         //console.log('myHeight: ' + myHeight);
1005         self.ed.theme.resizeTo('100%', myHeight);  // sets the dimensions of the editable area
1006       } catch (err) {
1007       }
1008     }, 200);
1009   };
1010
1011
f002d9 1012   
U 1013   /* -------- Editoren --------- */
1014   
1015   this.fm_text_edit = function(content, mode) {
1016     document.querySelector(".codeeditor-space").style.display = "block";
5b4d03 1017     //document.querySelector(".code-editor-container").style.display = "block";
U 1018     //document.querySelector(".zentrum-behaelter").style.display = "none";
1019     document.querySelector(".breadcrumb").style.display = "none";
1020     document.querySelector(".zentrum").style.display = "none";
f002d9 1021     self.fm_code_edit(content, mode);  
U 1022     self.openEditor = 'text';
1023   };
1024
1025   this.fm_dok_edit = function(content) {
1026     //fm_filectls_hide();
4e450e 1027     self.fm_dok_editor_init(self.userid);
U 1028     document.querySelector("#mce-editor").style.display = "block";
1029     //$("#mce-editor").show();
1030     document.querySelector(".breadcrumb").style.display = "none";
1031     document.querySelector(".zentrum").style.display = "none";
f002d9 1032
4e450e 1033     window.clearTimeout(self.tmo2);
U 1034     self.tmo2 = window.setTimeout(function () {
f002d9 1035       try {
4e450e 1036         self.ed.setContent(content);
U 1037         self.openEditor = 'dok';  
f002d9 1038       } catch (err) {
U 1039       }
1040     }, 200);
1041   };
ebc1f3 1042   
U 1043   this.fm_do_close = function() {
1044     document.querySelector(".codeeditor-space").style.display = "none";
4e450e 1045     document.querySelector("#mce-editor").style.display = "none";
U 1046     // $("#mce-editor").hide();
5b4d03 1047     //document.querySelector(".code-editor-container").style.display = "none";
U 1048     //document.querySelector(".zentrum-behaelter").style.display = "block";
1049     document.querySelector(".breadcrumb").style.display = "block";
1050     document.querySelector(".zentrum").style.display = "block";
ebc1f3 1051     if(self.cm !== undefined) {
U 1052       self.cm.toTextArea();
1053     }
1054     self.openFileName = '';
1055     self.openEditor = '';
1056     self.fm_get_list(self.pfad);
1057   };
05e9c4 1058
94b871 1059   /* -------- An- und Abmelden ------------- */
05e9c4 1060
94b871 1061   this.fm_get_login = function() {
U 1062     var m = '?c=de.uhilger.filecms.pub.SessionManager&m=getSessionUser';
1063     var u = '../pub' + m;
1064     self.fm_get(u, "text", function (resp) {
1065       self.userid = resp;
58bcbd 1066       self.login_zeigen();
U 1067       //document.querySelector("#userMenu").textContent = resp;
94b871 1068     });
U 1069   };
05e9c4 1070
94b871 1071   this.fm_logout = function() {
U 1072     var m = '?c=de.uhilger.filecms.pub.SessionManager&m=expireSession';
1073     var u = '../pub' + m;
1074     self.fm_get(u, "text", function (resp) {
1075       //$('#userMenu').text('nicht angemeldet');
1076       window.location.href = '../logout.html';
1077     });
1078   };
41ab37 1079   
6648a8 1080   /* ----- Hilfsfunktionen ----- */
U 1081
1082   this.serialisieren = function(obj) {
1083     return '{"' + obj.constructor.name + '":' + JSON.stringify(obj) + '}';
1084   };
1085   
1086   this.addEvtListener = function(selector, eventName, func) {
1087     document.querySelectorAll(selector).forEach(elem => { elem.addEventListener(eventName, func); });
1088   };
1089   
1090   this.removeAllListeners = function(id) {
1091     var el = document.getElementById(id);
5b4d03 1092     if(el !== null) {
U 1093       elClone = el.cloneNode(true);
1094       el.parentNode.replaceChild(elClone, el);
1095     }
6648a8 1096   }; // https://stackoverflow.com/questions/19469881/remove-all-event-listeners-of-specific-type
U 1097
ebc1f3 1098   this.escapeHtml = function(text) {
U 1099     text = text.replace(/\u228/g,'&auml;');
1100     text = text.replace(/\u246/g,'&ouml;');
1101     text = text.replace(/\u252/g,'&uuml;');
1102     text = text.replace(/\u196/g,'&Auml;');
1103     text = text.replace(/\u214/g,'&Ouml;');
1104     text = text.replace(/\u220/g,'&Uuml;');
1105     text = text.replace(/\u223/g,'&szlig;');
1106     text = text.replace(/\u26/g,'&amp;');
1107     return text;
1108   };
1109
1110   this.unescapeHtml = function(text) {
1111     text = text.replace(/&auml;/g, String.fromCharCode(228));
1112     text = text.replace(/&ouml;/g, String.fromCharCode(246));
1113     text = text.replace(/&uuml;/g, String.fromCharCode(252));
1114     text = text.replace(/&Auml;/g, String.fromCharCode(196));
1115     text = text.replace(/&Ouml;/g, String.fromCharCode(214));
1116     text = text.replace(/&Uuml;/g, String.fromCharCode(220));
1117     text = text.replace(/&szlig;/g, String.fromCharCode(223));
1118     text = text.replace(/&amp;/g, String.fromCharCode(26));
1119     return text;
1120   };
1121
41ab37 1122   /* ---- Vorlagen ---- */
U 1123
1124   this.html_erzeugen = function(vurl, inhalt, cb) {
1125     var vorlage = self.cache[vurl];
1126     if(vorlage === undefined) {
1127       self.vorlage_laden_und_fuellen(vurl, inhalt, cb);
1128     } else {
1129       self.vorlage_fuellen(vurl, inhalt, cb);
1130     }
1131   };
1132
1133   this.vorlage_fuellen = function(vurl, inhalt, cb) {
1134     cb(Mustache.render(self.cache[vurl], inhalt));
1135   };
1136
1137   this.vorlage_laden_und_fuellen = function(vurl, inhalt, cb) {
1138     self.fm_get(vurl, "text", function(antwort) {
1139       self.cache[vurl] = antwort;
1140       self.vorlage_fuellen(vurl, inhalt, cb);
1141     });
1142   };
94b871 1143
U 1144   /* -------- ajax helper functions ----------- */
1145
58bcbd 1146   this.fm_get = function (u, dtype, scallback) {    
U 1147     var xmlhttp = new XMLHttpRequest();
1148     var url = u;
1149     xmlhttp.onreadystatechange = function() {
5b4d03 1150       if (this.readyState === 4 && this.status === 200) {
58bcbd 1151         scallback(this.responseText);
94b871 1152       }
58bcbd 1153     };
U 1154     xmlhttp.open("GET", url, true);
1155     xmlhttp.send();
94b871 1156   };
ebc1f3 1157   
U 1158   self.fm_post = function(u, d, dtype, scallback) {
1159     var xmlhttp = new XMLHttpRequest();
1160     var url = u;
1161     xmlhttp.onreadystatechange = function() {
5b4d03 1162       if (this.readyState === 4 && this.status === 200) {
ebc1f3 1163         scallback(this.responseText);
U 1164       }
1165     };
1166     xmlhttp.open("POST", url, true);
1167     xmlhttp.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');
5b4d03 1168     xmlhttp.send(d);
ebc1f3 1169   };
U 1170
5b4d03 1171   /*
ebc1f3 1172   this.http_get = function(u, cb)  {
U 1173     self.http_call('GET', u, null, cb);
1174   };
1175   
1176   this.http_post = function(u, data, cb) {
1177     self.http_call('POST', u, data, cb);
1178   };
1179
1180   this.http_call = function (method, u, data, scallback) {    
1181     var xhr = new XMLHttpRequest();
1182     var url = u;
1183     xhr.onreadystatechange = function() {
1184       if (this.readyState === 4 && this.status === 200) {
1185         scallback(this.responseText);
1186       }
1187     };
5b4d03 1188     xhr.open(method, url, false);
ebc1f3 1189     if(method === 'GET')  {
U 1190       xhr.send();
1191     } else if(method === 'POST' || method === 'PUT') {
1192       xhr.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');
1193       xhr.send(data);
1194     }
1195   };
5b4d03 1196   */
ebc1f3 1197
94b871 1198
05e9c4 1199 }