Skip to content
Snippets Groups Projects
Commit d2aae249 authored by Sven Kästle's avatar Sven Kästle
Browse files

feat: Validate textarea input

parent df7bf93c
No related branches found
No related tags found
No related merge requests found
...@@ -33,7 +33,6 @@ body, html { ...@@ -33,7 +33,6 @@ body, html {
float: right; float: right;
} }
.file-upload-area { .file-upload-area {
background-color: blue;
height: 20%; height: 20%;
} }
...@@ -46,6 +45,14 @@ body, html { ...@@ -46,6 +45,14 @@ body, html {
resize: none; resize: none;
flex: 1; flex: 1;
} }
#upload-textarea-form {
height: 100%;
margin-bottom: 15px;
}
label.error {
color: red;
}
.flex .container-fluid{ .flex .container-fluid{
flex: 1; flex: 1;
} }
// initialize userId, userColors and targetId
var userId = randomUserId();
var userColors = new Map();
var userColorsDark = new Map();
var targetId = 200;
// declare document text
var documentText;
/** /**
* This function will fire when the DOM is ready * This function will fire when the DOM is ready
*/ */
$(document).ready(function() { $(document).ready(function() {
/** $('#btnNext').click(function () {
* Context menu handler if ($('#upload-textarea-form').valid()) {
*/ location.href="project-student.jsp?token="+getUserTokenFromUrl();
$.contextMenu({
selector: '.context-menu-one',
callback: function(key, options) {
// close context menu
window.close;
// initialize selected body
var body = getSelectedTextFromTextArea();
// if user selected something
if (body.length > 0) {
// annotationPostRequest
var request = {
userId: userId,
targetId: targetId,
body: body,
startCharacter: window.getSelection().getRangeAt(0).startOffset,
endCharacter: window.getSelection().getRangeAt(0).endOffset
};
console.log(request);
createAnnotation(request, function(response) {
// display the new annotation
displayAnnotation(response);
});
}
},
items: {
"annotation": {name: "Annotation", icon: "edit"}
} }
}); });
$('#btnSave').click(function () {
location.href="project-student.jsp?token="+getUserTokenFromUrl();
});
$('#btnBack').click(function () { $('#btnBack').click(function () {
location.href="project-student.jsp?token="+getUserTokenFromUrl(); location.href="project-student.jsp?token="+getUserTokenFromUrl();
}); });
documentText = $('#documentText').html(); /**
* validation of upload textarea
// fetch annotations from server on page start */
getAnnotations(targetId, function (response) { $('#upload-textarea-form').validate({
// iterate over annotations and display each rules: {
$.each(response, function (i, annotation) { uploadtextarea: {
displayAnnotation(annotation); required: true
}) }
}); },
messages: {
}); uploadtextarea: {
required: "Ein Text wird benötigt"
/** }
* POST: Save an annotation in the database
*
* @param annotationPostRequest The post request
* @param responseHandler The response handler
*/
function createAnnotation(annotationPostRequest, responseHandler) {
var url = "http://localhost:8080/rest/annotations/";
var json = JSON.stringify(annotationPostRequest);
$.ajax({
url: url,
type: "POST",
data: json,
contentType: "application/json",
dataType: "json",
success: function (response) {
responseHandler(response);
}
});
}
/**
* PATCH: Alter an annotation in database
*
* @param id The annotation id
* @param annotationPatchRequest The patch request
* @param responseHandler The response handler
*/
function alterAnnotation(id, annotationPatchRequest, responseHandler) {
var url = "http://localhost:8080/rest/annotations/" + id;
var json = JSON.stringify(annotationPatchRequest);
$.ajax({
url: url,
type: "PATCH",
data: json,
contentType: "application/json",
dataType: "json",
success: function (response) {
responseHandler(response);
}
});
}
/**
* DELETE: Delete an annotation from database
*
* @param id The annotation id
*/
function deleteAnnotation(id) {
var url = "http://localhost:8080/rest/annotations/" + id;
$.ajax({
url: url,
type: "DELETE",
dataType: "json",
success: function (response) {
// Nothing to do
}
});
}
/**
* GET: Get all annotations from database for a specific target
*
*
* @param targetId The target id
* @param responseHandler The response handler
*/
function getAnnotations(targetId, responseHandler) {
var url = "http://localhost:8080/rest/annotations/target/" + targetId;
$.ajax({
url: url,
type: "GET",
dataType: "json",
success: function (response) {
// sort the responding annotations by timestamp (DESC)
response.sort(function (a, b) {
return a.timestamp - b.timestamp;
});
// handle the response
responseHandler(response);
} }
}); });
}
/**
* Delete annotation from list
*
* @param elem The parent li element
* @param id The id of the annotation
*/
function deleteUnstructuredAnnotationHandler(elem, id) {
// remove annotation from list
elem.remove()
// remove annotation from database
deleteAnnotation(id)
}
/** });
* Display annotation in the list
*
* @param annotation The annotation to be displayed
*/
function displayAnnotation(annotation) {
// fetch list of annotations
var list = $('#annotations')
var deleteIcon = "fas fa-trash";
var dateIcon = "fas fa-calendar";
if (isTimestampToday(annotation.timestamp)) {
dateIcon = "fas fa-clock";
}
// insert annotation card
list.prepend(
$('<li>')
.attr('class', 'listelement')
.append(
$('<div>').attr('class', 'annotation-card')
.mouseenter(function () {
$(this).children('.annotation-header').css('background-color', getDarkUserColor(annotation.userId));
})
.mouseleave(function () {
$(this).children('.annotation-header').css('background-color', getUserColor(annotation.userId));
})
.append(
$('<div>').attr('class', 'annotation-header')
.css('background-color', getUserColor(annotation.userId))
.append(
$('<div>').attr('class', 'annotation-header-title')
.append(
$('<div>').attr('class', 'overflow-hidden')
.append(
$('<i>').attr('class', 'fas fa-user')
)
.append(
$('<span>').append(annotation.userId)
)
)
.append(
$('<div>').attr('class', 'overflow-hidden')
.append(
$('<i>').attr('class', 'fas fa-bookmark')
)
.append(
$('<span>').append('title' + annotation.userId)
)
)
)
.append(
$('<div>').attr('class', 'annotation-header-toggle')
.click(function () {
toggleButtonHandler($(this));
})
.append(
$('<i>').attr('class', 'fas fa-chevron-down')
)
)
)
.append(
$('<div>').attr('class', 'annotation-body')
.append(
$('<p>').attr('class', 'overflow-hidden').append(annotation.body)
)
)
.append(
$('<div>').attr('class', 'annotation-footer')
.append(
function () {
if (userId == annotation.userId) {
return $('<div>').attr('class', 'annotation-footer-delete')
.append(
$('<i>').attr('class', deleteIcon)
)
.click(function () {
deleteUnstructuredAnnotationHandler($(this).closest('li'), annotation.id)
})
}
}
)
.append(
$('<div>').attr('class', 'annotation-footer-date overflow-hidden')
.append(
$('<i>').attr('class', dateIcon)
)
.append(
$('<span>').append(timestampToReadableTime(annotation.timestamp))
)
)
)
)
.data('annotation', annotation)
.append(function () {
if ($('#annotations li').filter( ".listelement" ).length > 0) {
return $('<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 userId The user id
*/
function addHighlightedText(startCharacter, endCharacter, userId) {
// create <span> tag with the annotated text
var replacement = $('<span></span>').css('background-color', getUserColor(userId)).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 = documentText.slice(0, startCharacter) + replacementHtml + documentText.slice(endCharacter);
// set new document text
$('#documentText').html(newDocument);
}
/**
* Restore the base text
*/
function deleteHighlightedText() {
$('#documentText').html(documentText);
}
/**
* 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 the text value of the selected text from a textarea
*
* @returns {string} The text
*/
function getSelectedTextFromTextArea() {
// if firefox
if( navigator.userAgent.toLowerCase().indexOf('firefox') > -1 ){
// reference to the textarea
var txtarea = document.getElementById("upload-area");
// get index of first character
var start = txtarea.selectionStart;
// get index of last character
var end = txtarea.selectionEnd;
// return substring from start to end
return txtarea.value.substring(start, end);
}
else 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 userId The id of the user
* @returns {string} The user color
*/
function getUserColor(userId) {
// insert new color if there is no userId key
if (userColors.get(userId) == null) {
generateRandomColor(userId);
}
// return the color
return userColors.get(userId);
}
/**
* Get dark color based on user id
*
* @param userId The id of the user
* @returns {string} The dark user color
*/
function getDarkUserColor(userId) {
// insert new color if there is no userId key
if (userColorsDark.get(userId) == null) {
generateRandomColor(userId);
}
// return the color
return userColorsDark.get(userId);
}
/**
* Generate a random color of the format 'rgb(r, g, b)'
*
* @param userId The given user id
*/
function generateRandomColor(userId) {
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(userId, color);
userColorsDark.set(userId, 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;
// if annotation is from today
if (isTimestampToday(timestamp)) {
// get hours from date
var hours = annotationDate.getHours();
// get minutes from date
var minutes = "0" + annotationDate.getMinutes();
// get seconds from date
// var seconds = "0" + annotationDate.getSeconds();
// build readable timestamp
responseTimestamp = hours + ":" + minutes.substr(-2);
}
// else annotation is not from today
else {
// get date
var date = annotationDate.getDate();
// get month
var month = annotationDate.getMonth();
// get year
var year = annotationDate.getFullYear();
// build readable timestamp
responseTimestamp = date + "." + month + "." + year;
}
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 element The given toggle button
*/
function toggleButtonHandler(element) {
// open and close annotation text
element.parent().siblings(".annotation-body").children("p").toggleClass("overflow-hidden");
// toggle between up and down button
element.children("i").toggleClass("fa-chevron-down fa-chevron-up")
}
function lockButtonHandler() {
var lock = $('#btnLock').children('i')
lock.toggleClass("fa-lock-open fa-lock")
var area = $('#upload-area')
if (area.attr('readonly')) {
area.attr('readonly', false);
}
else {
area.attr('readonly', true);
}
}
/*
MOCKUP FUNCTIONS
*/
function randomUserId() {
return Math.floor((Math.random() * 12) + 1);;
}
...@@ -9,6 +9,11 @@ ...@@ -9,6 +9,11 @@
<head> <head>
<omniDependencies:omniDependencies/> <omniDependencies:omniDependencies/>
<!-- js - jQuery validation plugin -->
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery-validate/1.14.0/jquery.validate.min.js"></script>
<!-- js - jQuery ui position -->
<script src="https://cdnjs.cloudflare.com/ajax/libs/jqueryui/1.12.1/jquery-ui.min.js" type="text/javascript"></script>
<!-- css - upload-unstructured --> <!-- css - upload-unstructured -->
<link rel="stylesheet" type="text/css" href="../assets/css/unstructured-upload.css"> <link rel="stylesheet" type="text/css" href="../assets/css/unstructured-upload.css">
<!-- js - annotationScript --> <!-- js - annotationScript -->
...@@ -26,17 +31,20 @@ ...@@ -26,17 +31,20 @@
<headLine:headLine/> <headLine:headLine/>
</div> </div>
<div class="content-mainpage"> <div class="content-mainpage">
<div class="form-group upload-text" id="documentText"> <form id="upload-textarea-form">
<label for="upload-textarea">Texteingabe</label> <div class="form-group upload-text" id="documentText">
<textarea class="upload-text-textarea form-control" placeholder="Text einfügen..." id="upload-textarea"></textarea> <label for="upload-textarea">Texteingabe</label>
</div> <textarea class="upload-text-textarea form-control" placeholder="Text einfügen..." id="upload-textarea" name="uploadtextarea"></textarea>
</div>
</form>
<div> <div>
<label for="file">Alternativ bitte Datei wählen</label> <label for="file">Alternativ bitte Datei wählen</label>
<input type="file" id="file" name="file"> <input type="file" id="file" name="file">
</div> </div>
<div class="document-text-buttons"> <div class="document-text-buttons">
<button type="button" class="btn btn-secondary document-text-buttons-back" id="btnBack">Zurück</button> <button type="button" class="btn btn-secondary document-text-buttons-back" id="btnBack">Zurück</button>
<button type="button" class="btn btn-primary document-text-buttons-next" id="btnSave">Weiter</button> <button type="button" class="btn btn-primary document-text-buttons-next" id="btnNext">Weiter</button>
</div> </div>
</div> </div>
</div> </div>
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment