jquery_gantt_rails 0.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (52) hide show
  1. checksums.yaml +15 -0
  2. data/.gitignore +74 -0
  3. data/.idea/encodings.xml +5 -0
  4. data/.idea/jquery_gantt_rails.iml +9 -0
  5. data/.idea/misc.xml +22 -0
  6. data/.idea/modules.xml +9 -0
  7. data/.idea/scopes/scope_settings.xml +5 -0
  8. data/.idea/vcs.xml +7 -0
  9. data/.ruby-version.rb +1 -0
  10. data/Gemfile +3 -0
  11. data/README.md +41 -0
  12. data/Rakefile +1 -0
  13. data/jquery_gantt_rails.gemspec +20 -0
  14. data/lib/jquery_gantt_rails/version.rb +3 -0
  15. data/lib/jquery_gantt_rails/version.rb~ +3 -0
  16. data/lib/jquery_gantt_rails.rb +9 -0
  17. data/vendor/assets/javascripts/jquery_gantt/README.md +28 -0
  18. data/vendor/assets/javascripts/jquery_gantt/add.gif +0 -0
  19. data/vendor/assets/javascripts/jquery_gantt/alert.gif +0 -0
  20. data/vendor/assets/javascripts/jquery_gantt/closeBig.png +0 -0
  21. data/vendor/assets/javascripts/jquery_gantt/del.gif +0 -0
  22. data/vendor/assets/javascripts/jquery_gantt/edit.gif +0 -0
  23. data/vendor/assets/javascripts/jquery_gantt/gantt.css +323 -0
  24. data/vendor/assets/javascripts/jquery_gantt/gantt.html +547 -0
  25. data/vendor/assets/javascripts/jquery_gantt/ganttDrawer.js +752 -0
  26. data/vendor/assets/javascripts/jquery_gantt/ganttGridEditor.js +518 -0
  27. data/vendor/assets/javascripts/jquery_gantt/ganttMaster.js +702 -0
  28. data/vendor/assets/javascripts/jquery_gantt/ganttTask.js +947 -0
  29. data/vendor/assets/javascripts/jquery_gantt/ganttUtilities.js +237 -0
  30. data/vendor/assets/javascripts/jquery_gantt/gantt_compact.css +323 -0
  31. data/vendor/assets/javascripts/jquery_gantt/hasExternalDeps.png +0 -0
  32. data/vendor/assets/javascripts/jquery_gantt/libs/JST/jquery.JST.js +167 -0
  33. data/vendor/assets/javascripts/jquery_gantt/libs/date.js +584 -0
  34. data/vendor/assets/javascripts/jquery_gantt/libs/dateField/images/next.png +0 -0
  35. data/vendor/assets/javascripts/jquery_gantt/libs/dateField/images/prev.png +0 -0
  36. data/vendor/assets/javascripts/jquery_gantt/libs/dateField/jquery.dateField.css +88 -0
  37. data/vendor/assets/javascripts/jquery_gantt/libs/dateField/jquery.dateField.js +212 -0
  38. data/vendor/assets/javascripts/jquery_gantt/libs/i18nJs.js +140 -0
  39. data/vendor/assets/javascripts/jquery_gantt/libs/jquery.livequery.min.js +11 -0
  40. data/vendor/assets/javascripts/jquery_gantt/libs/jquery.timers.js +142 -0
  41. data/vendor/assets/javascripts/jquery_gantt/libs/platform.js +954 -0
  42. data/vendor/assets/javascripts/jquery_gantt/linkArrow.png +0 -0
  43. data/vendor/assets/javascripts/jquery_gantt/milestone.png +0 -0
  44. data/vendor/assets/javascripts/jquery_gantt/platform.css +346 -0
  45. data/vendor/assets/javascripts/jquery_gantt/teamwork-regular-webfont.eot +0 -0
  46. data/vendor/assets/javascripts/jquery_gantt/teamwork-regular-webfont.otf +0 -0
  47. data/vendor/assets/javascripts/jquery_gantt/teamwork-regular-webfont.svg +152 -0
  48. data/vendor/assets/javascripts/jquery_gantt/teamwork-regular-webfont.ttf +0 -0
  49. data/vendor/assets/javascripts/jquery_gantt/teamwork-regular-webfont.woff +0 -0
  50. data/vendor/assets/javascripts/jquery_gantt/teamworkFont.css +16 -0
  51. data/vendor/assets/javascripts/jquery_gantt/twGanttSmall.png +0 -0
  52. metadata +107 -0
@@ -0,0 +1,518 @@
1
+ /*
2
+ Copyright (c) 2012-2013 Open Lab
3
+ Written by Roberto Bicchierai and Silvia Chelazzi http://roberto.open-lab.com
4
+ Permission is hereby granted, free of charge, to any person obtaining
5
+ a copy of this software and associated documentation files (the
6
+ "Software"), to deal in the Software without restriction, including
7
+ without limitation the rights to use, copy, modify, merge, publish,
8
+ distribute, sublicense, and/or sell copies of the Software, and to
9
+ permit persons to whom the Software is furnished to do so, subject to
10
+ the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be
13
+ included in all copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
16
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
17
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
18
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
19
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
20
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
21
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
22
+ */
23
+ function GridEditor(master) {
24
+ this.master = master; // is the a GantEditor instance
25
+ var gridEditor = $.JST.createFromTemplate({}, "TASKSEDITHEAD");
26
+ gridEditor.gridify();
27
+ this.element = gridEditor;
28
+ }
29
+
30
+
31
+ GridEditor.prototype.fillEmptyLines = function() {
32
+ var factory = new TaskFactory();
33
+
34
+ //console.debug("GridEditor.fillEmptyLines");
35
+ var rowsToAdd = 30 - this.element.find(".taskEditRow").size();
36
+
37
+ //fill with empty lines
38
+ for (var i = 0; i < rowsToAdd; i++) {
39
+ var emptyRow = $.JST.createFromTemplate({}, "TASKEMPTYROW");
40
+ //click on empty row create a task and fill above
41
+ var master = this.master;
42
+ emptyRow.click(function(ev) {
43
+ master.beginTransaction();
44
+ var emptyRow = $(this);
45
+ var lastTask;
46
+ var start = new Date().getTime();
47
+ var level = 0;
48
+ if (master.tasks[0]) {
49
+ start = master.tasks[0].start;
50
+ level = master.tasks[0].level + 1;
51
+ }
52
+
53
+ //fill all empty previouses
54
+ emptyRow.prevAll(".emptyRow").andSelf().each(function() {
55
+ var ch = factory.build("tmp_fk" + new Date().getTime(), "", "", level, start, 1);
56
+ var task = master.addTask(ch);
57
+ lastTask = ch;
58
+ });
59
+ master.endTransaction();
60
+ lastTask.rowElement.click();
61
+ lastTask.rowElement.find("[name=name]").focus()//focus to "name" input
62
+ .blur(function() { //if name not inserted -> undo -> remove just added lines
63
+ var imp = $(this);
64
+ if (!imp.isValueChanged())
65
+ master.undo();
66
+ });
67
+ });
68
+ this.element.append(emptyRow);
69
+ }
70
+ };
71
+
72
+
73
+ GridEditor.prototype.addTask = function(task, row) {
74
+ //console.debug("GridEditor.addTask",task,row);
75
+ //var prof = new Profiler("editorAddTaskHtml");
76
+
77
+ //remove extisting row
78
+ this.element.find("[taskId=" + task.id + "]").remove();
79
+
80
+ var taskRow = $.JST.createFromTemplate(task, "TASKROW");
81
+ //save row element on task
82
+ task.rowElement = taskRow;
83
+
84
+ this.bindRowEvents(task, taskRow);
85
+
86
+ if (typeof(row) != "number") {
87
+ var emptyRow = this.element.find(".emptyRow:first"); //tries to fill an empty row
88
+ if (emptyRow.size() > 0)
89
+ emptyRow.replaceWith(taskRow);
90
+ else
91
+ this.element.append(taskRow);
92
+ } else {
93
+ var tr = this.element.find("tr.taskEditRow").eq(row);
94
+ if (tr.size() > 0) {
95
+ tr.before(taskRow);
96
+ } else {
97
+ this.element.append(taskRow);
98
+ }
99
+
100
+ }
101
+ this.element.find(".taskRowIndex").each(function(i, el) {
102
+ $(el).html(i + 1);
103
+ });
104
+ //prof.stop();
105
+
106
+ return taskRow;
107
+ };
108
+
109
+
110
+ GridEditor.prototype.refreshTaskRow = function(task) {
111
+ //console.debug("refreshTaskRow")
112
+ //var profiler = new Profiler("editorRefreshTaskRow");
113
+ var row = task.rowElement;
114
+
115
+ row.find(".taskRowIndex").html(task.getRow() + 1);
116
+ row.find(".indentCell").css("padding-left", task.level * 10);
117
+ row.find("[name=name]").val(task.name);
118
+ row.find("[name=code]").val(task.code);
119
+ row.find("[status]").attr("status", task.status);
120
+
121
+ row.find("[name=duration]").val(task.duration);
122
+ row.find("[name=start]").val(new Date(task.start).format()).updateOldValue(); // called on dates only because for other field is called on focus event
123
+ row.find("[name=end]").val(new Date(task.end).format()).updateOldValue();
124
+ row.find("[name=depends]").val(task.depends);
125
+ row.find(".taskAssigs").html(task.getAssigsString());
126
+
127
+ //profiler.stop();
128
+ };
129
+
130
+ GridEditor.prototype.redraw = function() {
131
+ for (var i = 0; i < this.master.tasks.length; i++) {
132
+ this.refreshTaskRow(this.master.tasks[i]);
133
+ }
134
+ };
135
+
136
+ GridEditor.prototype.reset = function() {
137
+ this.element.find("[taskId]").remove();
138
+ };
139
+
140
+
141
+ GridEditor.prototype.bindRowEvents = function (task, taskRow) {
142
+ var self = this;
143
+ //console.debug("bindRowEvents",this,this.master,this.master.canWrite);
144
+ if (this.master.canWrite) {
145
+ self.bindRowInputEvents(task,taskRow);
146
+
147
+ } else { //cannot write: disable input
148
+ taskRow.find("input").attr("readonly", true);
149
+ }
150
+
151
+ taskRow.find(".edit").click(function() {self.openFullEditor(task,taskRow)});
152
+
153
+ };
154
+
155
+
156
+ GridEditor.prototype.bindRowInputEvents = function (task, taskRow) {
157
+ var self = this;
158
+
159
+ //bind dateField on dates
160
+ taskRow.find(".date").each(function () {
161
+ var el = $(this);
162
+
163
+ el.click(function () {
164
+ var inp = $(this);
165
+ inp.dateField({
166
+ inputField:el
167
+ });
168
+ });
169
+
170
+ el.blur(function (date) {
171
+ var inp = $(this);
172
+ if (inp.isValueChanged()) {
173
+ if (!Date.isValid(inp.val())) {
174
+ alert(GanttMaster.messages["INVALID_DATE_FORMAT"]);
175
+ inp.val(inp.getOldValue());
176
+
177
+ } else {
178
+ var date = Date.parseString(inp.val());
179
+ var row = inp.closest("tr");
180
+ var taskId = row.attr("taskId");
181
+ var task = self.master.getTask(taskId);
182
+ var lstart = task.start;
183
+ var lend = task.end;
184
+
185
+ if (inp.attr("name") == "start") {
186
+ lstart = date.getTime();
187
+ if (lstart >= lend) {
188
+ var end_as_date = new Date(lstart);
189
+ lend = end_as_date.add('d', task.duration).getTime();
190
+ }
191
+
192
+ //update task from editor
193
+ self.master.beginTransaction();
194
+ self.master.moveTask(task, lstart);
195
+ self.master.endTransaction();
196
+
197
+ } else {
198
+ var end_as_date = new Date(date.getTime());
199
+ lend = end_as_date.getTime();
200
+ if (lstart >= lend) {
201
+ end_as_date.add('d', -1 * task.duration);
202
+ lstart = end_as_date.getTime();
203
+ }
204
+
205
+ //update task from editor
206
+ self.master.beginTransaction();
207
+ self.master.changeTaskDates(task, lstart, lend);
208
+ self.master.endTransaction();
209
+ }
210
+
211
+
212
+ inp.updateOldValue(); //in order to avoid multiple call if nothing changed
213
+ }
214
+ }
215
+ });
216
+ });
217
+
218
+
219
+ //binding on blur for task update (date exluded as click on calendar blur and then focus, so will always return false, its called refreshing the task row)
220
+ taskRow.find("input:not(.date)").focus(function () {
221
+ $(this).updateOldValue();
222
+
223
+ }).blur(function () {
224
+ var el = $(this);
225
+ if (el.isValueChanged()) {
226
+ var row = el.closest("tr");
227
+ var taskId = row.attr("taskId");
228
+
229
+ var task = self.master.getTask(taskId);
230
+
231
+ //update task from editor
232
+ var field = el.attr("name");
233
+
234
+ self.master.beginTransaction();
235
+
236
+ if (field == "depends") {
237
+
238
+ var oldDeps = task.depends;
239
+ task.depends = el.val();
240
+ // update links
241
+ var linkOK = self.master.updateLinks(task);
242
+ if (linkOK) {
243
+ //synchronize status fro superiors states
244
+ var sups = task.getSuperiors();
245
+ for (var i = 0; i < sups.length; i++) {
246
+ if (!sups[i].from.synchronizeStatus())
247
+ break;
248
+ }
249
+
250
+ self.master.changeTaskDates(task, task.start, task.end);
251
+ }
252
+
253
+ } else if (field == "duration") {
254
+ var dur = task.duration;
255
+ dur = parseInt(el.val()) || 1;
256
+ el.val(dur);
257
+ var newEnd = computeEndByDuration(task.start, dur);
258
+ self.master.changeTaskDates(task, task.start, newEnd);
259
+
260
+ } else {
261
+ task[field] = el.val();
262
+ }
263
+ self.master.endTransaction();
264
+ }
265
+ });
266
+
267
+
268
+ //change status
269
+ taskRow.find(".taskStatus").click(function () {
270
+ var el = $(this);
271
+ var tr = el.closest("[taskId]");
272
+ var taskId = tr.attr("taskId");
273
+ var task = self.master.getTask(taskId);
274
+
275
+ var changer = $.JST.createFromTemplate({}, "CHANGE_STATUS");
276
+ changer.css("top", tr.position().top + self.element.parent().scrollTop());
277
+ changer.find("[status=" + task.status + "]").addClass("selected");
278
+ changer.find(".taskStatus").click(function () {
279
+ self.master.beginTransaction();
280
+ task.changeStatus($(this).attr("status"));
281
+ self.master.endTransaction();
282
+ el.attr("status", task.status);
283
+ changer.remove();
284
+ el.show();
285
+
286
+ });
287
+ el.hide().oneTime(3000, "hideChanger", function () {
288
+ changer.remove();
289
+ $(this).show();
290
+ });
291
+ el.after(changer);
292
+ });
293
+
294
+
295
+ /*//expand collapse todo to be completed
296
+ taskRow.find(".expcoll").click(function(){
297
+ //expand?
298
+ var el=$(this);
299
+ var taskId=el.closest("[taskId]").attr("taskId");
300
+ var task=self.master.getTask(taskId);
301
+ var descs=task.getDescendant();
302
+ if (el.is(".exp")){
303
+ for (var i=0;i<descs.length;i++)
304
+ descs[i].rowElement.show();
305
+ } else {
306
+ for (var i=0;i<descs.length;i++)
307
+ descs[i].rowElement.hide();
308
+ }
309
+
310
+ });*/
311
+
312
+ //bind row selection
313
+ taskRow.click(function () {
314
+ var row = $(this);
315
+ //var isSel = row.hasClass("rowSelected");
316
+ row.closest("table").find(".rowSelected").removeClass("rowSelected");
317
+ row.addClass("rowSelected");
318
+
319
+ //set current task
320
+ self.master.currentTask = self.master.getTask(row.attr("taskId"));
321
+
322
+ //move highlighter
323
+ if (self.master.currentTask.ganttElement)
324
+ self.master.gantt.highlightBar.css("top", self.master.currentTask.ganttElement.position().top);
325
+
326
+ //if offscreen scroll to element
327
+ var top = row.position().top;
328
+ if (row.position().top > self.element.parent().height()) {
329
+ self.master.gantt.element.parent().scrollTop(row.position().top - self.element.parent().height() + 100);
330
+ }
331
+ });
332
+
333
+ };
334
+
335
+
336
+
337
+ GridEditor.prototype.openFullEditor = function (task, taskRow) {
338
+
339
+ var self=this;
340
+
341
+ //task editor in popup
342
+ var taskId = taskRow.attr("taskId");
343
+ //console.debug(task);
344
+
345
+ //make task editor
346
+ var taskEditor = $.JST.createFromTemplate({}, "TASK_EDITOR");
347
+
348
+ taskEditor.find("#name").val(task.name);
349
+ taskEditor.find("#description").val(task.description);
350
+ taskEditor.find("#code").val(task.code);
351
+ taskEditor.find("#progress").val(task.progress ? parseFloat(task.progress) : 0);
352
+ taskEditor.find("#status").attr("status", task.status);
353
+
354
+ if (task.startIsMilestone)
355
+ taskEditor.find("#startIsMilestone").attr("checked", true);
356
+ if (task.endIsMilestone)
357
+ taskEditor.find("#endIsMilestone").attr("checked", true);
358
+
359
+ taskEditor.find("#duration").val(task.duration);
360
+ taskEditor.find("#start").val(new Date(task.start).format());
361
+ taskEditor.find("#end").val(new Date(task.end).format());
362
+
363
+ //taskEditor.find("[name=depends]").val(task.depends);
364
+
365
+ //make assignments table
366
+ var assigsTable = taskEditor.find("#assigsTable");
367
+ assigsTable.find("[assigId]").remove();
368
+ // loop on already assigned resources
369
+ for (var i = 0; i < task.assigs.length; i++) {
370
+ var assig = task.assigs[i];
371
+ var assigRow = $.JST.createFromTemplate({task:task, assig:assig}, "ASSIGNMENT_ROW");
372
+ assigsTable.append(assigRow);
373
+ }
374
+
375
+ if (!self.master.canWrite) {
376
+ taskEditor.find("input,textarea").attr("readOnly", true);
377
+ taskEditor.find("input:checkbox,select").attr("disabled", true);
378
+ } else {
379
+
380
+ //bind dateField on dates
381
+ taskEditor.find("#start").click(function () {
382
+ $(this).dateField({
383
+ inputField:$(this),
384
+ callback: function (date) {
385
+ var dur = parseInt(taskEditor.find("#duration").val());
386
+ date.clearTime();
387
+ taskEditor.find("#end").val(new Date(computeEndByDuration(date.getTime(), dur)).format());
388
+ }
389
+ });
390
+ });
391
+
392
+ //bind dateField on dates
393
+ taskEditor.find("#end").click(function () {
394
+ $(this).dateField({
395
+ inputField:$(this),
396
+ callback: function (end) {
397
+ var start = Date.parseString(taskEditor.find("#start").val());
398
+ end.setHours(23, 59, 59, 999);
399
+
400
+ if (end.getTime() < start.getTime()) {
401
+ var dur = parseInt(taskEditor.find("#duration").val());
402
+ start = incrementDateByWorkingDays(end.getTime(), -dur);
403
+ taskEditor.find("#start").val(new Date(computeStart(start)).format());
404
+ } else {
405
+ taskEditor.find("#duration").val(recomputeDuration(start.getTime(), end.getTime()));
406
+ }
407
+ }
408
+ });
409
+ });
410
+
411
+ //bind blur on duration
412
+ taskEditor.find("#duration").change(function () {
413
+ var start = Date.parseString(taskEditor.find("#start").val());
414
+ var el = $(this);
415
+ var dur = parseInt(el.val());
416
+ dur = dur <= 0 ? 1 : dur;
417
+ el.val(dur);
418
+ taskEditor.find("#end").val(new Date(computeEndByDuration(start.getTime(), dur)).format());
419
+ });
420
+
421
+ //bind add assignment
422
+ taskEditor.find("#addAssig").click(function () {
423
+ var assigsTable = taskEditor.find("#assigsTable");
424
+ var assigRow = $.JST.createFromTemplate({task:task, assig:{id:"tmp_" + new Date().getTime()}}, "ASSIGNMENT_ROW");
425
+ assigsTable.append(assigRow);
426
+ });
427
+
428
+ taskEditor.find("#status").click(function () {
429
+ var tskStatusChooser = $(this);
430
+ var changer = $.JST.createFromTemplate({}, "CHANGE_STATUS");
431
+ changer.css("top", tskStatusChooser.position().top);
432
+ changer.find("[status=" + task.status + "]").addClass("selected");
433
+ changer.find(".taskStatus").click(function () {
434
+ tskStatusChooser.attr("status", $(this).attr("status"));
435
+ changer.remove();
436
+ tskStatusChooser.show();
437
+ });
438
+ tskStatusChooser.hide().oneTime(3000, "hideChanger", function () {
439
+ changer.remove();
440
+ $(this).show();
441
+ });
442
+ tskStatusChooser.after(changer);
443
+ });
444
+
445
+ //save task
446
+ taskEditor.find("#saveButton").click(function () {
447
+ var task = self.master.getTask(taskId); // get task again because in case of rollback old task is lost
448
+
449
+ self.master.beginTransaction();
450
+ task.name = taskEditor.find("#name").val();
451
+ task.description = taskEditor.find("#description").val();
452
+ task.code = taskEditor.find("#code").val();
453
+ task.progress = parseFloat(taskEditor.find("#progress").val());
454
+ task.duration = parseInt(taskEditor.find("#duration").val());
455
+ task.startIsMilestone = taskEditor.find("#startIsMilestone").is(":checked");
456
+ task.endIsMilestone = taskEditor.find("#endIsMilestone").is(":checked");
457
+
458
+ //set assignments
459
+ taskEditor.find("tr[assigId]").each(function () {
460
+ var trAss = $(this);
461
+ var assId = trAss.attr("assigId");
462
+ var resId = trAss.find("[name=resourceId]").val();
463
+ var roleId = trAss.find("[name=roleId]").val();
464
+ var effort = millisFromString(trAss.find("[name=effort]").val());
465
+
466
+
467
+ //check if an existing assig has been deleted and re-created with the same values
468
+ var found = false;
469
+ for (var i = 0; i < task.assigs.length; i++) {
470
+ var ass = task.assigs[i];
471
+
472
+ if (assId == ass.id) {
473
+ ass.effort = effort;
474
+ ass.roleId = roleId;
475
+ ass.resourceId = resId;
476
+ ass.touched = true;
477
+ found = true;
478
+ break;
479
+
480
+ } else if (roleId == ass.roleId && resId == ass.resourceId) {
481
+ ass.effort = effort;
482
+ ass.touched = true;
483
+ found = true;
484
+ break;
485
+
486
+ }
487
+ }
488
+
489
+ if (!found) { //insert
490
+ var ass = task.createAssignment("tmp_" + new Date().getTime(), resId, roleId, effort);
491
+ ass.touched = true;
492
+ }
493
+
494
+ });
495
+
496
+ //remove untouched assigs
497
+ task.assigs = task.assigs.filter(function (ass) {
498
+ var ret = ass.touched;
499
+ delete ass.touched;
500
+ return ret;
501
+ });
502
+
503
+ //change dates
504
+ task.setPeriod(Date.parseString(taskEditor.find("#start").val()).getTime(), Date.parseString(taskEditor.find("#end").val()).getTime() + (3600000 * 24));
505
+
506
+ //change status
507
+ task.changeStatus(taskEditor.find("#status").attr("status"));
508
+
509
+ if (self.master.endTransaction()) {
510
+ $("#__blackpopup__").trigger("close");
511
+ }
512
+
513
+ });
514
+ }
515
+
516
+ var ndo = createBlackPage(800, 500).append(taskEditor);
517
+
518
+ };