55
55
default_type_icon = "txt.png";
57
57
/* Relative to IVLE root */
58
type_icons_path = "+media/ivle.webapp.core/images/mime";
59
type_icons_path_large = "+media/ivle.webapp.core/images/mime/large";
58
type_icons_path = "media/images/mime";
59
type_icons_path_large = "media/images/mime/large";
61
61
/* Mapping SVN status to icons, just the file's basename */
63
"unversioned": "unversioned.png",
64
"ignored": null, /* Supposed to be innocuous */
65
64
"normal": "normal.png",
66
65
"added": "added.png",
67
66
"missing": "missing.png",
68
67
"deleted": "deleted.png",
69
"replaced": "replaced.png",
70
68
"modified": "modified.png",
71
"conflicted": "conflicted.png",
72
69
"revision": "revision.png"
75
72
/* Mapping SVN status to "nice" strings */
77
74
"unversioned": "Temporary file",
78
"ignored": "Temporary file (ignored)",
79
75
"normal": "Permanent file",
80
76
"added": "Temporary file (scheduled to be added)",
81
77
"missing": "Permanent file (missing)",
137
132
* May be "application/x-www-form-urlencoded" or "multipart/form-data".
138
133
* Defaults to "application/x-www-form-urlencoded".
139
134
* "multipart/form-data" is recommended for large uploads.
140
* \param callback, optional.
141
* A callback function for after the action has been handled.
143
function do_action(action, path, args, content_type, callback)
136
function do_action(action, path, args, content_type, ignore_response)
145
138
args.action = action;
146
139
/* Callback action, when the server returns */
147
var callback_inner = function(response)
140
var callback = function(response)
149
142
/* Check for action errors reported by the server, and report them
151
144
var error = response.getResponseHeader("X-IVLE-Action-Error");
152
if (error != null && error != "")
153
/* Note: This header (in particular) comes URI-encoded, to
154
* allow multi-line error messages. Decode */
155
alert("Error: " + decodeURIComponent(error.toString()) + ".");
146
alert("Error: " + error.toString() + ".");
156
147
/* Now read the response and set up the page accordingly */
157
if (callback != null)
158
callback(path, response);
148
if (ignore_response != true)
149
handle_response(path, response);
160
151
/* Call the server and perform the action. This mutates the server. */
161
ajax_call(callback_inner, service_app, path, args, "POST", content_type);
152
ajax_call(callback, service_app, path, args, "POST", content_type);
164
155
/** Calls the server using Ajax, requesting a directory listing. This should
274
var err = document.createElement("div");
275
var p = dom_make_text_elem("p", "Error: "
276
+ "There was an unexpected server error processing "
277
+ "the selected command.");
279
p = dom_make_text_elem("p", "If the problem persists, please "
280
+ "contact the system administrator.")
282
p = document.createElement("p");
283
var refresh = document.createElement("input");
284
refresh.setAttribute("type", "button");
285
refresh.setAttribute("value", "Back to file view");
286
refresh.setAttribute("onclick", "refresh()");
287
p.appendChild(refresh);
293
var err = document.createElement("div");
294
var p = dom_make_text_elem("p", "Error: "
295
+ "There was an unexpected server error retrieving "
296
+ "the requested file or directory.");
298
p = dom_make_text_elem("p", "If the problem persists, please "
299
+ "contact the system administrator.")
247
handle_error("The server returned an invalid directory listing");
305
250
/* Get "." out, it's special */
306
251
current_file = file_listing["."]; /* Global */
307
252
delete file_listing["."];
309
if ('revision' in listing)
311
current_revision = listing.revision;
314
254
/* Check if this is a directory listing or file contents */
315
255
var isdir = response.getResponseHeader("X-IVLE-Return") == "Dir";
321
/* Top-level dir, with subjects */
322
special_home_listing(listing, subjects, path);
326
/* Not the top-level dir. Do a normal dir listing. */
327
handle_dir_listing(path, listing.listing);
258
handle_dir_listing(path, listing);
447
351
/*** HANDLERS for different types of responses (such as dir listing, file,
451
* message may either be a string, or a DOM node, which will be placed inside
454
354
function handle_error(message)
456
356
var files = document.getElementById("filesbody");
458
if (typeof(message) == "string")
460
txt_elem = dom_make_text_elem("div", "Error: "
461
+ message.toString() + ".")
465
/* Assume message is a DOM node */
466
txt_elem = document.createElement("div");
467
txt_elem.appendChild(message);
357
var txt_elem = dom_make_text_elem("div", "Error: "
358
+ message.toString() + ".")
469
359
txt_elem.setAttribute("class", "padding error");
470
360
files.appendChild(txt_elem);
473
/** Given a path, filename and optional revision, returns a URL to open that
474
* revision of that file.
476
function build_revision_url(path, filename, revision)
478
bits = {'path': app_path(this_app, path, filename)};
479
if (current_revision)
481
bits['query_string'] = 'r=' + revision;
483
return build_url(bits);
486
363
/** Given a mime type, returns the path to the icon.
487
364
* \param type String, Mime type.
488
365
* \param sizelarge Boolean, optional.
641
/* Available if zero or one files are selected,
471
/* Available if exactly one file is selected,
642
472
* and only if this is a file, not a directory */
643
473
var serve = document.getElementById("act_serve");
644
if (numsel <= 1 && !file.isdir && current_file.svnstatus != 'revision')
474
if (numsel == 1 && !file.isdir)
646
476
serve.setAttribute("class", "choice");
647
serve.setAttribute("onclick",
648
"return maybe_save('The last saved version will be served.')");
650
serve.setAttribute("href",
651
app_url(serve_app, current_path));
653
serve.setAttribute("href",
654
app_url(serve_app, current_path, filename));
477
serve.setAttribute("href",
478
app_path(serve_app, current_path, filename));
658
482
serve.setAttribute("class", "disabled");
659
483
serve.removeAttribute("href");
660
serve.removeAttribute("onclick");
664
487
/* Available if exactly one file is selected,
665
488
* and it is a Python file.
667
var run = document.getElementById("act_run");
669
if (numsel <= 1 && !file.isdir && file.type == "text/x-python"
670
&& current_file.svnstatus != 'revision')
674
// In the edit window
675
var localpath = path_join('/home', current_path);
679
// In the browser window
680
var localpath = path_join('/home', current_path, filename);
682
run.setAttribute("class", "choice");
683
run.setAttribute("onclick", "runfile('" + localpath + "')");
687
run.setAttribute("class", "disabled");
688
run.removeAttribute("onclick");
692
/* Always available for current files.
693
494
* If 0 files selected, download the current file or directory as a ZIP.
694
495
* If 1 directory selected, download it as a ZIP.
695
496
* If 1 non-directory selected, download it.
696
497
* If >1 files selected, download them all as a ZIP.
698
499
var download = document.getElementById("act_download");
699
if (current_file.svnstatus == 'revision')
701
download.setAttribute("class", "disabled");
702
download.removeAttribute("onclick");
704
else if (numsel <= 1)
706
download.setAttribute("class", "choice")
709
504
download.setAttribute("href",
710
app_url(download_app, current_path));
505
app_path(download_app, current_path));
712
507
download.setAttribute("title",
713
508
"Download the current directory as a ZIP file");
748
542
var publish = document.getElementById("act_publish");
749
543
var submit = document.getElementById("act_submit");
750
var pubcond = numsel <= 1 && file.isdir;
753
/* If this dir is already published, call it "Unpublish" */
756
publish.setAttribute("value", "unpublish");
757
publish.setAttribute("title" ,"Make it so this directory "
758
+ "can not be seen by anyone on the web");
759
publish.firstChild.nodeValue = "Unpublish";
761
publish.setAttribute("value", "publish");
762
publish.setAttribute("title","Make it so this directory "
763
+ "can be seen by anyone on the web");
764
publish.firstChild.nodeValue = "Publish";
767
set_action_state(["publish", "submit"], pubcond);
544
if (numsel <= 1 && file.isdir)
546
/* TODO: Work out of file is svn'd */
547
/* TODO: If this dir is already published, call it "Unpublish" */
548
publish.setAttribute("class", "choice");
549
publish.removeAttribute("disabled");
550
submit.setAttribute("class", "choice");
551
submit.removeAttribute("disabled");
555
publish.setAttribute("class", "disabled");
556
publish.setAttribute("disabled", "disabled");
557
submit.setAttribute("class", "disabled");
558
submit.setAttribute("disabled", "disabled");
770
/* If exactly 1 non-directory file is selected, and its parent
562
/* If exactly 1 non-directory file is selected/opened, and its parent
771
563
* directory is published.
773
set_action_state("share", numsel == 1 && !file.isdir &&
774
current_file.published);
565
var share = document.getElementById("act_share");
566
if (numsel <= 1 && !file.isdir)
568
/* TODO: Work out if parent dir is published */
569
share.setAttribute("class", "choice");
570
share.removeAttribute("disabled");
574
share.setAttribute("class", "disabled");
575
share.setAttribute("disabled", "disabled");
777
579
/* If exactly 1 file is selected */
778
set_action_state("rename", numsel == 1);
580
var rename = document.getElementById("act_rename");
583
rename.setAttribute("class", "choice");
584
rename.removeAttribute("disabled");
588
rename.setAttribute("class", "disabled");
589
rename.setAttribute("disabled", "disabled");
780
592
/* Delete, cut, copy */
781
593
/* If >= 1 file is selected */
782
set_action_state(["delete", "cut", "copy"], numsel >= 1);
594
var act_delete = document.getElementById("act_delete");
595
var cut = document.getElementById("act_cut");
596
var copy = document.getElementById("act_copy");
599
act_delete.setAttribute("class", "choice");
600
act_delete.removeAttribute("disabled");
601
cut.setAttribute("class", "choice");
602
cut.removeAttribute("disabled");
603
copy.setAttribute("class", "choice");
604
copy.removeAttribute("disabled");
608
act_delete.setAttribute("class", "disabled");
609
act_delete.setAttribute("disabled", "disabled");
610
cut.setAttribute("class", "disabled");
611
cut.setAttribute("disabled", "disabled");
612
copy.setAttribute("class", "disabled");
613
copy.setAttribute("disabled", "disabled");
784
616
/* Paste, new file, new directory, upload */
785
/* Disable if the current file is not a directory */
786
set_action_state(["paste", "newfile", "mkdir", "upload"], current_file.isdir);
617
/* Always enabled (assuming this is a directory) */
788
619
/* Subversion actions */
789
/* These are only useful if we are in a versioned directory and have some
791
set_action_state(["svnadd"], numsel >= 1 && current_file.svnstatus);
792
/* And these are only useful is ALL the selected files are versioned */
793
set_action_state(["svnremove", "svnrevert", "svncopy", "svncut"],
794
numsel >= 1 && current_file.svnstatus && svn_selection);
795
/* Commit is useful if ALL selected files are versioned, or the current
796
* directory is versioned */
797
set_action_state(["svncommit"], current_file.svnstatus &&
798
(numsel >= 1 && svn_selection || numsel == 0));
800
/* Diff, log and update only support one path at the moment, so we must
801
* have 0 or 1 versioned files selected. If 0, the directory must be
803
single_versioned_path = (
805
(numsel == 1 && (svnst = file_listing[selected_files[0]].svnstatus)) ||
806
(numsel == 0 && (svnst = current_file.svnstatus))
807
) && svnstatus_versioned(svnst));
808
set_action_state(["svndiff", "svnupdate"], single_versioned_path);
810
/* We can resolve if we have a file selected and it is conflicted. */
811
set_action_state("svnresolved", single_versioned_path && numsel == 1 && svnst == "conflicted");
813
/* Log should be available for revisions as well. */
814
set_action_state("svnlog", single_versioned_path, true);
816
/* Cleanup should be available for revisions as well. */
817
set_action_state("svncleanup", single_versioned_path, true);
819
single_ivle_versioned_path = (
821
(numsel == 1 && (stat = file_listing[selected_files[0]])) ||
822
(numsel == 0 && (stat = current_file))
823
) && svnstatus_versioned(stat.svnstatus)
825
&& stat.svnurl.substr(0, svn_base.length) == svn_base);
826
set_action_state(["submit"], single_ivle_versioned_path);
828
/* There is currently nothing on the More Actions menu of use
829
* when the current file is not a directory. Hence, just remove
831
* (This makes some of the above decisions somewhat redundant).
832
* We also take this opportunity to show the appropriate actions2
833
* bar for this path. It should either be a save or upload widget.
835
if (current_file.isdir)
837
var actions2_directory = document.getElementById("actions2_directory");
838
actions2_directory.setAttribute("style", "display: inline;");
839
var moreactions = document.getElementById("moreactions_area");
840
moreactions.setAttribute("style", "display: inline;");
844
var actions2_file = document.getElementById("actions2_file");
845
actions2_file.setAttribute("style", "display: inline;");
620
/* TODO: Work out when these are appropriate */
621
var svnadd = document.getElementById("act_svnadd");
622
var svnrevert = document.getElementById("act_svnrevert");
623
var svncommit = document.getElementById("act_svncommit");
626
svnadd.setAttribute("class", "choice");
627
svnadd.removeAttribute("disabled");
628
svnrevert.setAttribute("class", "choice");
629
svnrevert.removeAttribute("disabled");
630
svncommit.setAttribute("class", "choice");
631
svncommit.removeAttribute("disabled");
931
700
action_add(selected_files);
934
action_remove(selected_files);
936
702
case "svnrevert":
937
703
action_revert(selected_files);
940
window.location = path_join(app_url('diff'), current_path, selected_files[0] || '');
943
action_update(selected_files);
946
action_resolved(selected_files);
948
705
case "svncommit":
949
706
action_commit(selected_files);
952
window.location = path_join(app_url('svnlog'), current_path, selected_files[0] || '');
955
action_svncopy(selected_files);
958
action_svncut(selected_files);
961
action_svncleanup(".");
966
/** User clicks "Run" button.
967
* Do an Ajax call and print the test output.
969
function runfile(localpath)
971
if (!maybe_save('The last saved version will be run.')) return false;
973
/* Dump the entire file to the console */
974
var callback = function()
976
console_enter_line("execfile('" + localpath + "')", "block");
978
start_server(callback)
982
711
/** Called when the page loads initially.
984
function browser_init()
713
window.onload = function()
986
715
/* Navigate (internally) to the path in the URL bar.
987
716
* This causes the page to be populated with whatever is at that address,
988
717
* whether it be a directory or a file.
990
var path = get_path();
994
/** Gets the current path of the window */
995
function get_path() {
996
719
var path = parse_url(window.location.href).path;
997
720
/* Strip out root_dir + "/files" from the front of the path */
998
721
var strip = make_path(this_app);