// initialize userToken, userColors var userColors = new Map(); var userColorsDark = new Map(); // declare document text, start and end character var startCharacter, endCharacter; /** * This function will fire when the DOM is ready */ $(document).ready(function() { let fullSubmissionId = getQueryVariable("fullSubmissionId"); let category = getQueryVariable("category"); // fetch full submission from database getFullSubmission(getQueryVariable("fullSubmissionId"), function (response) { // set text $('#documentText').html(response.text); // fetch submission parts getSubmissionPart(fullSubmissionId, category, function (response) { let body = response.body; // save body $('#documentText').data("body", body); let offset = 0; for (let i = 0; i < body.length; i++) { addHighlightedSubmissionPart(body[i].startCharacter, body[i].endCharacter, offset); // add char count of '<span class="categoryText"></span>' offset += 34; } // scroll document text to first span element let documentText = $('#documentText'); let span = $('#documentText span').first(); documentText.scrollTo(span); }, function () { // error }) }, function () { // error }); // connect to websocket on page ready connect(fullSubmissionId, category); /** * Context menu handler */ $.contextMenu({ selector: '.context-menu-one', callback: function(key, options) { // handle annotation context click handleAnnotationClick() }, items: { "annotation": {name: "Annotation", icon: "edit"} } }); /** * continue button */ $('#btnContinue').click(function () { location.href="givefeedback.jsp?token=" + getUserTokenFromUrl(); }); /** * back button */ $('#btnBack').click(function () { location.href="project-student.jsp?token=" + getUserTokenFromUrl() + "&projectId=" + getQueryVariable("projectId"); }); /** * validation of annotation create form inside the modal */ $('#annotation-create-form').validate({ rules: { title: { required: true, maxlength: 120 }, comment: { required: true, maxlength: 400 } }, messages: { title: { required: "Ein Titel wird benötigt", maxlength: "Maximal 120 Zeichen erlaubt" }, comment: { required: "Ein Kommentar wird benötigt", maxlength: "Maximal 400 Zeichen erlaubt" } } }); /** * validation of annotation edit form inside the modal */ $('#annotation-edit-form').validate({ rules: { title: { required: true, maxlength: 120 }, comment: { required: true, maxlength: 400 } }, messages: { title: { required: "Ein Titel wird benötigt", maxlength: "Maximal 120 Zeichen erlaubt" }, comment: { required: "Ein Kommentar wird benötigt", maxlength: "Maximal 400 Zeichen erlaubt" } } }); /** * Save button of the annotation create modal * hide modal and build new annotation */ $('#btnSave').click(function () { if ($('#annotation-create-form').valid()) { // get title and comment from form var title = $('#annotation-form-title').val(); var comment = $('#annotation-form-comment').val(); // hide and clear the modal $('#annotation-create-modal').modal('hide'); // save the new annotation in db and display it saveNewAnnotation(title, comment, startCharacter, endCharacter); } }); /** * Edit button of the annotation edit modal * hide modal and alter the annotation */ $('#btnEdit').click(function () { if ($('#annotation-edit-form').valid()) { // get title and comment from clicked annotation card var id = $('#annotation-edit-modal').data('id'); var card = $('#' + id); var title = card.find('.annotation-header-data-title').text(); var comment = card.find('.annotation-body-text').text(); // get title and comment from form var newTitle = $('#annotation-edit-form-title').val(); var newComment = $('#annotation-edit-form-comment').val(); // compare new and old card content if (title !== newTitle || comment !== newComment) { // build patch request var annotationPatchRequest = { title: newTitle, comment: newComment }; // send alter request to server alterAnnotation(id, annotationPatchRequest, function (response) { // send altered annotation to websocket send("EDIT", id); // alter the annotation card card.find('.annotation-header-data-title').text(newTitle); card.find('.annotation-body-text').text(newComment); // handle drop down button showAndHideToggleButton(); // hide and clear the modal $('#annotation-edit-modal').modal('hide'); }) } } }); /** * Delete an annotation from list and server */ $('#btnDelete').click(function () { // get id from edit modal var id = $('#annotation-edit-modal').data('id'); // delte annotation from server and from list deleteAnnotation(id, function () { // send delete request to websocket send("DELETE", id); // remove annotation from list $('#' + id).closest('.listelement').remove() // remove highlighted text deleteHighlightedText(); // hide and clear the modal $('#annotation-edit-modal').modal('hide'); }) }); /** * Clear the title and comment input field of the create modal */ $('#annotation-create-modal').on('hidden.bs.modal', function(){ // clear title $('#annotation-form-title').val(''); // clear comment $('#annotation-form-comment').val('') }); /** * Clear the title and comment input field of the edit modal */ $('#annotation-edit-modal').on('hidden.bs.modal', function(e){ // clear title $('#annotation-edit-form-title').val(''); // clear comment $('#annotation-edit-form-comment').val('') }); // fetch annotations from server on page start getAnnotations(fullSubmissionId, category, function (response) { // iterate over annotations and display each $.each(response, function (i, annotation) { displayAnnotation(annotation); }) // handle drop down button showAndHideToggleButton(); }); }); /** * This will be called on page resize */ $( window ).resize(function() { // handle drop down button for every annotation showAndHideToggleButton(); }); /** * Display annotation in the list * * @param annotation The annotation to be displayed */ function displayAnnotation(annotation) { // fetch list of annotations var list = $('#annotations') var editIcon = "fas fa-edit"; var dateIcon = "fas fa-calendar"; if (isTimestampToday(annotation.timestamp)) { dateIcon = "fas fa-clock"; } // declare variables let display, input, filter, user, title, comment; input = $('#annotation-search'); filter = input.val().toLowerCase(); user = annotation.userToken.toLowerCase(); title = annotation.body.title.toLowerCase(); comment = annotation.body.comment.toLowerCase(); // hiding based on user, title and comment if (user.indexOf(filter) > -1 || title.indexOf(filter) > -1 || comment.indexOf(filter) > -1) { display = '' } else { display = 'none' } // insert annotation card list.prepend( // list element $('<li>') .css('display', display) .attr('class', 'listelement') .append( // annotation card $('<div>').attr('class', 'annotation-card') .attr('id', annotation.id) .mouseenter(function () { $(this).children('.annotation-header').css('background-color', getDarkUserColor(annotation.userToken)); }) .mouseleave(function () { $(this).children('.annotation-header').css('background-color', getUserColor(annotation.userToken)); }) .append( // annotation header $('<div>').attr('class', 'annotation-header') .css('background-color', getUserColor(annotation.userToken)) .append( // header data $('<div>').attr('class', 'annotation-header-data') .append( // user $('<div>').attr('class', 'overflow-hidden') .append( $('<i>').attr('class', 'fas fa-user') ) .append( $('<span>').attr('class', 'annotation-header-data-user').append(annotation.userToken) ) ) .append( // title $('<div>').attr('class', 'overflow-hidden') .append( $('<i>').attr('class', 'fas fa-bookmark') ) .append( $('<span>').attr('class', 'annotation-header-data-title').append(annotation.body.title) ) ) ) .append( // unfold button $('<div>').attr('class', 'annotation-header-toggle') .click(function () { toggleButtonHandler(annotation.id); }) .append( $('<i>').attr('class', 'fas fa-chevron-down') ) ) ) .append( // annotation body $('<div>').attr('class', 'annotation-body') .append( $('<p>').attr('class', 'overflow-hidden annotation-body-text').append(annotation.body.comment) ) ) .append( // annotation footer $('<div>').attr('class', 'annotation-footer') .append( // edit function () { if (getUserTokenFromUrl() === annotation.userToken) { return $('<div>').attr('class', 'annotation-footer-edit') .append( $('<i>').attr('class', editIcon) ) .click(function () { editAnnotationHandler(annotation.id) }) } } ) .append( // timestamp $('<div>').attr('class', 'flex-one overflow-hidden') .append( $('<i>').attr('class', dateIcon) ) .append( $('<span>').append(timestampToReadableTime(annotation.timestamp)) ) ) ) ) .data('annotation', annotation) .mouseenter(function () { addHighlightedAnnotation(annotation.body.startCharacter, annotation.body.endCharacter, annotation.userToken); // scroll document text to anchor element let documentText = $('#documentText'); let anchor = $('#anchor'); documentText.scrollTo(anchor); }) .mouseleave(function () { deleteHighlightedText(); }) .append( $('<div>').attr('class', 'spacing') ) ); } /** * Add a highlighted text at specific position * * @param startCharacter The offset of the start character * @param endCharacter The offset of the end character * @param userToken The user token */ function addHighlightedAnnotation(startCharacter, endCharacter, userToken) { let offset = calculateExtraOffset(startCharacter); // initialize variables let documentText = $('#documentText').text(); let documentHtml = $('#documentText').html(); // create <span> tag with the annotated text var replacement = $('<span></span>').attr('id', 'anchor').css('background-color', getUserColor(userToken)).html(documentText.slice(startCharacter, endCharacter)); // wrap an <p> tag around the replacement, get its parent (the <p>) and ask for the html var replacementHtml = replacement.wrap('<p/>').parent().html(); // insert the replacementHtml var newDocument = documentHtml.slice(0, startCharacter + offset) + replacementHtml + documentHtml.slice(endCharacter + offset); // set new document text $('#documentText').html(newDocument); } /** * Add a highlighted text at specific position * * @param startCharacter The offset of the start character * @param endCharacter The offset of the end character * @param offset The calculated extra offset depending on already highlighted text */ function addHighlightedSubmissionPart(startCharacter, endCharacter, offset) { var documentText = $('#documentText').text(); var documentHtml = $('#documentText').html(); // create <span> tag with the annotated text var replacement = $('<span></span>').attr('class', 'categoryText').html(documentText.slice(startCharacter, endCharacter)); // wrap an <p> tag around the replacement, get its parent (the <p>) and ask for the html var replacementHtml = replacement.wrap('<p/>').parent().html(); // insert the replacementHtml var newDocument = documentHtml.slice(0, startCharacter + offset) + replacementHtml + documentHtml.slice(endCharacter + offset); // set new document text $('#documentText').html(newDocument); } /** * Iterate over all data arrays and calculate the offset for a given start character * * @param startCharacter The given start character * @returns {number} The offset */ function calculateExtraOffset(startCharacter) { // get submission part body let body = $('#documentText').data("body"); let extraOffset = 0; for (let i = 0; i < body.length; i++) { if (body[i].startCharacter <= startCharacter) { extraOffset += 27; } if (body[i].endCharacter <= startCharacter) { extraOffset += 7; } } return extraOffset; } /** * Restore the base text */ function deleteHighlightedText() { let documentText = $('#documentText'); let highlight = documentText.find('#anchor'); let text = highlight.text(); highlight.replaceWith(text); } /** * Get the text value of the selected text * * @returns {string} The text */ function getSelectedText() { if(window.getSelection){ return window.getSelection().toString(); } else if(document.getSelection){ return document.getSelection(); } else if(document.selection){ return document.selection.createRange().text; } } /** * Get color based on user id * * @param userToken The id of the user * @returns {string} The user color */ function getUserColor(userToken) { // insert new color if there is no userToken key if (userColors.get(userToken) == null) { generateRandomColor(userToken); } // return the color return userColors.get(userToken); } /** * Get dark color based on user id * * @param userToken The token of the user * @returns {string} The dark user color */ function getDarkUserColor(userToken) { // insert new color if there is no userToken key if (userColorsDark.get(userToken) == null) { generateRandomColor(userToken); } // return the color return userColorsDark.get(userToken); } /** * Generate a random color of the format 'rgb(r, g, b)' * * @param userToken The given user token */ function generateRandomColor(userToken) { var r = Math.floor(Math.random()*56)+170; var g = Math.floor(Math.random()*56)+170; var b = Math.floor(Math.random()*56)+170; var r_d = r - 50; var g_d = g - 50; var b_d = b - 50; var color = 'rgb(' + r + ',' + g + ',' + b + ')'; var colorDark = 'rgb(' + r_d + ',' + g_d + ',' + b_d + ')'; userColors.set(userToken, color); userColorsDark.set(userToken, colorDark); } /** * Calculate and build a readable timestamp from an unix timestamp * * @param timestamp A unix timestamp * @returns {string} A readable timestamp */ function timestampToReadableTime(timestamp) { // build Date object from timestamp var annotationDate = new Date(timestamp); // declare response var responseTimestamp; // get hours from date var hours = "0" + annotationDate.getHours(); // get minutes from date var minutes = "0" + annotationDate.getMinutes(); // if annotation is from today if (isTimestampToday(timestamp)) { // build readable timestamp in format HH:mm responseTimestamp = hours.substr(-2) + ":" + minutes.substr(-2); } // else annotation is not from today else { // get date var date = "0" + annotationDate.getDate(); // get month var month = "0" + annotationDate.getMonth(); // get year var year = "" + annotationDate.getFullYear(); // build readable timestamp dd.MM.yy HH:mm responseTimestamp = date.substr(-2) + "." + month.substr(-2) + "." + year.substr(-2) + " " + hours.substr(-2) + ":" + minutes.substr(-2); } return responseTimestamp; } /** * Check if given timestamp is from today * * @param timestamp The given timestamp in milliseconds * @returns {boolean} Returns true if the timestamp is from today */ function isTimestampToday(timestamp) { // now var now = new Date(); // build Date object from timestamp var date = new Date(timestamp); // return true if timestamp is today if (now.getDate() == date.getDate() && now.getMonth() == date.getMonth() && now.getFullYear() == date.getFullYear()) { return true; } else { return false; } } /** * Toggle between the toggle button status * * @param id The id of the clicked annotation */ function toggleButtonHandler(id) { // the clicked annotation card var card = $('#' + id); // open and close annotation text card.find(".annotation-body").children("p").toggleClass("overflow-hidden"); // toggle between up and down button card.find('.annotation-header-toggle').children("i").toggleClass("fa-chevron-down fa-chevron-up") } /** * Save a new annotation in database and list * * @param title The title of the new annotation * @param comment The comment of the new annotation * @param startCharacter The startCharacter based on the annotated text * @param endCharacter The endCharacter based on the annotated text */ function saveNewAnnotation(title, comment, startCharacter, endCharacter) { // initialize target let targetId = getQueryVariable("fullSubmissionId"); let targetCategory = getQueryVariable("category"); let userToken = getUserTokenFromUrl(); // build annotationPostRequest var annotationPostRequest = { userToken: userToken, targetId: targetId, targetCategory: targetCategory, body: { title: title, comment: comment, startCharacter: startCharacter, endCharacter: endCharacter } }; // send new annotation to back-end and display it in list createAnnotation(annotationPostRequest, function(response) { // send new annotation to websocket send("CREATE", response.id); // display the new annotation displayAnnotation(response); }); } /** * Open edit modal with title and comment from given card * * @param id The id of the clicked annotation */ function editAnnotationHandler(id) { // the clicked annotation card var card = $('#' + id); // get title and comment var title = card.find('.annotation-header-data-title').text(); var comment = card.find('.annotation-body-text').text(); // set title and comment $('#annotation-edit-form-title').val(title); $('#annotation-edit-form-comment').val(comment); // display annotation edit modal and pass id $('#annotation-edit-modal').data('id', id).modal("show"); } /** * Change title and comment from annotation by given annotation * * @param annotation The given altered annotation */ function editAnnotationValues(annotation) { // find annotation var annotationElement = $('#' + annotation.id); // set title and comment annotationElement.find('.annotation-header-data-title').text(annotation.body.title); annotationElement.find('.annotation-body-text').text(annotation.body.comment); // handle drop down button showAndHideToggleButtonById(annotation.id); } /** * Show or hide the drop down button for every annotation card. * Call this on page resize and after annotations GET */ function showAndHideToggleButton() { // iterate over each annotation card $('#annotations').find('li').each(function () { // find the comment element, clone and hide it var comment = $(this).find('.annotation-body').children('p'); var clone = comment.clone() .css({display: 'inline', width: 'auto', visibility: 'hidden'}) .appendTo('body'); var cloneWidth = clone.width(); // remove the element from the page clone.remove(); // show drop down button only if text was truncated if(cloneWidth > comment.width()) { $(this).find('.annotation-header-toggle').show(); $(this).find('.annotation-header-data').css('width', 'calc(100% - 40px)'); } else { $(this).find('.annotation-header-toggle').hide(); $(this).find('.annotation-header-data').css('width', '100%'); } }) } /** * Show or hide the drop down button for a given annotation card. * * @param id The id of the annotation */ function showAndHideToggleButtonById(id) { // find annotation var annotationElement = $('#' + id); // find the comment element, clone and hide it var comment = annotationElement.find('.annotation-body').children('p'); var clone = comment.clone() .css({display: 'inline', width: 'auto', visibility: 'hidden'}) .appendTo('body'); var cloneWidth = clone.width(); // remove the element from the page clone.remove(); // show drop down button only if text was truncated if(cloneWidth > comment.width()) { annotationElement.find('.annotation-header-toggle').show(); annotationElement.find('.annotation-header-data').css('width', 'calc(100% - 40px)'); } else { annotationElement.find('.annotation-header-toggle').hide(); annotationElement.find('.annotation-header-data').css('width', '100%'); } } /** * Handle the annotation click and show the modal * */ function handleAnnotationClick() { // if saved selection's range count is > 0 let sel = rangy.getSelection(); if (sel.rangeCount > 0) { // calculate character range offset from range let range = sel.getRangeAt(0); let offsets = range.toCharacterRange($('#documentText')[0]); // if selected text's length is > 0 let selectedText = getSelectedText(); if (selectedText.length > 0) { // save start and end character and handle the selection startCharacter = offsets.start; endCharacter = offsets.end; if (isAnnotationInRange(startCharacter, endCharacter)) { // display annotation create modal $('#annotation-create-modal').modal("show"); } else { window.alert("Annotationen sind nur in vorgehobenen Bereichen möglich") } } } } /** * Checks if user selected area is inside submission part range * * @param start The start character of the selection * @param end The end character of the selection * @returns {boolean} Returns true if the selection is in range */ function isAnnotationInRange(start, end) { let body = $('#documentText').data("body"); for (let i = 0; i < body.length; i++) { if (body[i].startCharacter <= start && end <= body[i].endCharacter) { return true; } } return false; } /** * Display or hide annotations based on filter string */ function searchAnnotation() { // declare variables let input, filter; input = $('#annotation-search'); filter = input.val().toLowerCase(); // iterate over annotation card and hide those who don't match the search query $('#annotations').find('li').each(function () { let user, title, comment; user = $(this).find('.annotation-header-data-user').text().toLowerCase(); title = $(this).find('.annotation-header-data-title').text().toLowerCase(); comment = $(this).find('.annotation-body-text').text().toLowerCase(); // hiding based on user, title and comment if (user.indexOf(filter) > -1 || title.indexOf(filter) > -1 || comment.indexOf(filter) > -1) { $(this).css('display', '') } else { $(this).css('display', 'none') } }); }