scinote-web/app/assets/javascripts/my_modules/protocols.js

562 lines
16 KiB
JavaScript

//= require protocols/import_export/import
//= require protocols/import_export/export
//= require datatables
// Currently selected row in "load from protocol" modal
var selectedRow = null;
// Initialize edit description modal window
function initEditDescription() {
var editDescriptionModal = $("#manage-module-description-modal");
var editDescriptionModalBody = editDescriptionModal.find(".modal-body");
var editDescriptionModalSubmitBtn = editDescriptionModal.find("[data-action='submit']");
$(".description-link")
.on("ajax:success", function(ev, data, status) {
var descriptionLink = $(".description-refresh");
// Set modal body & title
editDescriptionModalBody.html(data.html);
editDescriptionModal
.find("#manage-module-description-modal-label")
.text(data.title);
editDescriptionModalBody.find("form")
.on("ajax:success", function(ev2, data2, status2) {
// Update module's description in the tab
descriptionLink.html(data2.description_label);
// Close modal
editDescriptionModal.modal("hide");
})
.on("ajax:error", function(ev2, data2, status2) {
// Display errors if needed
$(this).render_form_errors("my_module", data.responseJSON);
});
// Show modal
editDescriptionModal.modal("show");
})
.on("ajax:error", function(ev, data, status) {
// TODO
});
editDescriptionModalSubmitBtn.on("click", function() {
// Submit the form inside the modal
editDescriptionModalBody.find("form").submit();
});
editDescriptionModal.on("hidden.bs.modal", function() {
editDescriptionModalBody.find("form").off("ajax:success ajax:error");
editDescriptionModalBody.html("");
});
}
function bindEditDueDateAjax() {
var editDueDateModal = null;
var editDueDateModalBody = null;
var editDueDateModalTitle = null;
var editDueDateModalSubmitBtn = null;
editDueDateModal = $("#manage-module-due-date-modal");
editDueDateModalBody = editDueDateModal.find(".modal-body");
editDueDateModalTitle = editDueDateModal.find("#manage-module-due-date-modal-label");
editDueDateModalSubmitBtn = editDueDateModal.find("[data-action='submit']");
$(".due-date-link")
.on("ajax:success", function(ev, data, status) {
var dueDateLink = $(".due-date-refresh");
// Load contents
editDueDateModalBody.html(data.html);
editDueDateModalTitle.text(data.title);
// Add listener to form inside modal
editDueDateModalBody.find("form")
.on("ajax:success", function(ev2, data2, status2) {
// Update module's due date
dueDateLink.html(data2.module_header_due_date_label);
// Close modal
editDueDateModal.modal("hide");
})
.on("ajax:error", function(ev2, data2, status2) {
// Display errors if needed
$(this).render_form_errors("my_module", data.responseJSON);
});
// Open modal
editDueDateModal.modal("show");
})
.on("ajax:error", function(ev, data, status) {
// TODO
});
editDueDateModalSubmitBtn.on("click", function() {
// Submit the form inside the modal
editDueDateModalBody.find("form").submit();
});
editDueDateModal.on("hidden.bs.modal", function() {
editDueDateModalBody.find("form").off("ajax:success ajax:error");
editDueDateModalBody.html("");
});
}
function initTutorial() {
var currentStep = Cookies.get('current_tutorial_step');
if (showTutorial() && (currentStep > 8 && currentStep < 12)) {
var resultsTab = $("#results-nav-tab");
var moduleProtocolsTutorial = $("[data-role='tutorial-data']").attr("data-module-protocols-step-text");
var moduleProtocolsClickResultsTutorial = $("[data-role='tutorial-data']").attr("data-module-protocols-click-results-step-text");
introJs()
.setOptions({
steps: [
{
intro: moduleProtocolsTutorial
},
{
intro: moduleProtocolsClickResultsTutorial,
element: resultsTab[0],
tooltipClass: 'custom next-page-link'
}
],
overlayOpacity: '0.1',
doneLabel: 'End tutorial',
skipLabel: 'End tutorial',
nextLabel: 'Next',
showBullets: false,
showStepNumbers: false,
exitOnOverlayClick: false,
exitOnEsc: false,
tooltipClass: 'custom'
})
.onafterchange(function(tarEl) {
Cookies.set('current_tutorial_step', this._currentStep + 10);
if (this._currentStep == 1) {
setTimeout(function() {
$('.next-page-link a.introjs-nextbutton')
.removeClass('introjs-disabled')
.attr('href', resultsTab.find("a").attr('href'));
positionTutorialTooltip();
}, 500);
} else {
positionTutorialTooltip();
}
})
.goToStep(currentStep == 11 ? 2 : 1)
.start();
window.onresize = positionTutorialTooltip;
// Destroy first-time tutorial cookies when skip tutorial
// or end tutorial is clicked
$(".introjs-skipbutton").each(function (){
$(this).click(function (){
Cookies.remove('tutorial_data');
Cookies.remove('current_tutorial_step');
});
});
}
}
function positionTutorialTooltip() {
if (Cookies.get('current_tutorial_step') == 11) {
if ($("#results-nav-tab").offset().left == 0) {
$(".introjs-tooltip").css("left", (window.innerWidth / 2 - 50) + "px");
$(".introjs-tooltip").addClass("repositioned");
} else if ($(".introjs-tooltip").hasClass("repositioned")) {
$(".introjs-tooltip").css("left", "");
$(".introjs-tooltip").removeClass("repositioned");
}
}
};
function showTutorial() {
var tutorialData;
if (Cookies.get('tutorial_data'))
tutorialData = JSON.parse(Cookies.get('tutorial_data'));
else
return false;
var tutorialModuleId = tutorialData[0].qpcr_module;
var currentModuleId = $("[data-role='tutorial-data']").attr("data-module-id");
return tutorialModuleId == currentModuleId;
}
function initCopyToRepository() {
var link = $("[data-action='copy-to-repository']");
var modal = $("#copy-to-repository-modal");
var modalBody = modal.find(".modal-body");
var submitBtn = modal.find(".modal-footer [data-action='submit']");
link
.on("ajax:success", function(e, data) {
modalBody.html(data.html);
modalBody.find("[data-role='copy-to-repository']")
.on("ajax:success", function(e2, data2) {
if (data2.refresh !== null) {
// Reload page
location.reload();
} else {
// Simply hide the modal
modal.modal("hide");
}
})
.on("ajax:error", function(e2, data2) {
// Display errors in form
if (data2.status === 422) {
$(this).render_form_errors("protocol", data2.responseJSON);
} else {
// Simply display global error
alert(data2.responseJSON.message);
}
});
modal.modal("show");
})
.on("ajax:error", function() {} );
submitBtn.on("click", function() {
// Submit the embedded form
modalBody.find("form").submit();
});
modalBody.on("click", "[data-role='link-check']", function() {
var text = $(this).closest(".modal-body").find("[data-role='link-text']");
if ($(this).prop("checked")) {
text.show();
} else {
text.hide();
}
});
modal.on("hidden.bs.modal", function() {
modalBody.find("[data-role='copy-to-repository']")
.off("ajax:success ajax:error");
modalBody.html("");
});
}
function initLinkUpdate() {
var modal = $("#confirm-link-update-modal");
var modalTitle = modal.find(".modal-title");
var modalBody = modal.find(".modal-body");
var updateBtn = modal.find(".modal-footer [data-action='submit']");
$("[data-action='unlink'], [data-action='revert'], [data-action='update-parent'], [data-action='update-self']")
.on("ajax:success", function(e, data) {
modalTitle.html(data.title);
modalBody.html(data.message);
updateBtn.text(data.btn_text);
modal.attr("data-url", data.url);
modal.modal("show");
});
modal.on("hidden.bs.modal", function() {
modalBody.html("");
});
if( !$._data( updateBtn[0], 'events' ) ) {
updateBtn.on("click", function() {
// POST via ajax
$.ajax({
url: modal.attr("data-url"),
type: "POST",
dataType: "json",
success: function(ev, data) {
// Simply reload page
location.reload();
},
error: function(ev, data) {
// Display error message in alert()
alert(ev.responseJSON.message);
// Hide modal
modal.modal("hide");
}
});
});
}
}
function initLoadFromRepository() {
var modal = $("#load-from-repository-modal");
var modalBody = modal.find(".modal-body");
var loadBtn = modal.find(".modal-footer [data-action='submit']");
$("[data-action='load-from-repository']")
.on("ajax:success", function(e, data) {
modalBody.html(data.html);
// Disable load btn
loadBtn.attr("disabled", "disabled");
modal.modal("show");
// Init Datatable on public tab
initLoadFromRepositoryTable(modalBody.find("#public-tab"));
modalBody.find("a[data-toggle='tab']")
.on("hide.bs.tab", function(e) {
// Destroy Handsontable in to-be-hidden tab
var content = $($(e.target).attr("href"));
destroyLoadFromRepositoryTable(content);
})
.on("shown.bs.tab", function(e) {
// Initialize Handsontable in to-be-shown tab
var content = $($(e.target).attr("href"));
initLoadFromRepositoryTable(content);
});
loadBtn.on("click", function() {
loadFromRepository();
});
});
modal.on("hidden.bs.modal", function() {
// Destroy the current Datatable
destroyLoadFromRepositoryTable(modalBody.find(".tab-pane.active"));
modalBody.find("a[data-toggle='tab']")
.off("hide.bs.tab shown.bs.tab");
loadBtn.off("click");
modalBody.html("");
});
}
function initLoadFromRepositoryTable(content) {
var tableEl = content.find("[data-role='datatable']");
var repositoryType = tableEl.data("type");
var datatable = tableEl.DataTable({
order: [[1, "asc"]],
dom: "RBfltpi",
buttons: [],
processing: true,
serverSide: true,
responsive: true,
ajax: {
url: tableEl.data("source"),
type: "POST"
},
colReorder: {
fixedColumnsLeft: 1000000 // Disable reordering
},
columnDefs: [{
targets: 0,
searchable: false,
orderable: false,
sWidth: "1%",
render: function (data, type, full, meta) {
return "<input type='radio'>";
}
}, {
targets: [ 1, 2, 3, 4, 5, 6 ],
searchable: true,
orderable: true
}],
columns: [
{ data: "0" },
{ data: "1" },
{ data: "2" },
{ data: "3" },
{ data: "4" },
{ data: "5" },
{ data: "6" }
],
rowCallback: function(row, data, dataIndex) {
// Get row ID
var rowId = data["DT_RowId"];
$(row).attr("data-row-id", rowId);
// If row ID is in the list of selected row IDs
if(rowId === selectedRow){
$(row).find("input[type='radio']").prop("checked", true);
$(row).addClass("selected");
}
},
fnDrawCallback: function(settings, json) {
animateSpinner(this, false);
},
preDrawCallback: function(settings) {
animateSpinner(this);
}
});
// Handle click on table cells with radio buttons
tableEl.find("tbody").on("click", "td", function(e) {
$(this).parent().find("input[type='radio']").trigger("click");
});
// Handle clicks on radio buttons
tableEl.find("tbody").on("click", "input[type='radio']", function(e) {
// Get row ID
var row = $(this).closest("tr");
var data = datatable.row(row).data();
var rowId = data["DT_RowId"];
// Uncheck all radio buttons
tableEl.find("tbody input[type='radio']")
.prop("checked", false)
.closest("tr")
.removeClass("selected");
// Select the current row
row.find("input[type='radio']").prop("checked", true);
selectedRow = rowId;
row.addClass("selected");
// Enable load btn
content.closest(".modal")
.find(".modal-footer [data-action='submit']")
.removeAttr("disabled");
e.stopPropagation();
});
tableEl.find("tbody").on("click", "a[data-action='filter']", function(e) {
var link = $(this);
var searchInput = $("#protocols-table_wrapper input[type='search']");
var query = link.attr("data-param");
// Re-search data
datatable.search(query).draw();
// Don't propagate this further up
e.stopPropagation();
return false;
});
}
function destroyLoadFromRepositoryTable(content) {
var tableEl = content.find("[data-role='datatable']");
// Unbind event listeners
tableEl.find("tbody").off("click", "a[data-action='filter']");
tableEl.find("tbody").off("click", "input[type='radio']");
tableEl.find("tbody").off("click", "td");
// Destroy datatable
tableEl.DataTable().destroy();
tableEl.find("tbody").html("");
selectedRow = null;
// Disable load btn
content.closest(".modal")
.find(".modal-footer [data-action='submit']")
.attr("disabled", "disabled");
}
function loadFromRepository() {
var modal = $("#load-from-repository-modal");
var check_linked = $("[data-role='protocol-status-bar']")
.text();
var confirm_message= "";
if( check_linked.trim() !== '(unlinked)' ){
confirm_message = I18n.t("protocols.import_export.import_modal.import_to_linked_task");
} else {
confirm_message = I18n.t("my_modules.protocols.load_from_repository_modal.confirm_message");
}
if (selectedRow !== null && confirm(confirm_message) ) {
// POST via ajax
$.ajax({
url: modal.attr("data-url"),
type: "POST",
dataType: "json",
data: { source_id: selectedRow },
success: function(ev, data) {
// Simply reload page
location.reload();
},
error: function(ev, data) {
// Display error message in alert()
alert(ev.responseJSON.message);
// Hide modal
modal.modal("hide");
}
});
}
}
function initRefreshStatusBar() {
$("[data-role='steps-container']")
.on(
"ajax:success",
function(e, data) {
if ($(e.target).is("[data-role='edit-step-form'], [data-role='new-step-form']")) {
// Get the status bar URL
var url = $("[data-role='protocol-status-bar-url']").attr("data-url");
// Fetch new updated at label
$.ajax({
url: url,
type: "GET",
dataType: "json",
success: function (data2) {
$("[data-role='protocol-status-bar']").html(data2.html);
initLinkUpdate();
}
});
}
}
);
}
function initImport() {
var fileInput = $("[data-action='load-from-file']");
// Make sure multiple selections of same file
// always prompt new modal
fileInput.find("input[type='file']").on("click", function() {
this.value = null;
});
// Hack to hide "No file chosen" tooltip
fileInput.attr("title", window.webkitURL ? " " : "");
fileInput.on("change", function(ev) {
var importUrl = fileInput.attr("data-import-url");
importProtocolFromFile(
ev.target.files[0],
importUrl,
null,
true,
function(datas) {
var data = datas[0];
if (data.status === "ok") {
// Simply reload page
location.reload();
} else {
alert(I18n.t("my_modules.protocols.load_from_file_error"));
}
});
// Clear input on self
$(this).val("");
});
}
function initExport() {
var exportBtn = $("[data-action='export']");
exportBtn.on("click", function() {
exportProtocols([ $(this).attr("data-id") ]);
});
}
// On init
bindEditDueDateAjax();
initEditDescription();
initCopyToRepository();
initLinkUpdate();
initLoadFromRepository();
initRefreshStatusBar();
initImport();
initExport();
initTutorial();