feat(exam-correct): general improvement

This commit is contained in:
Gregor Kleen 2020-02-07 20:57:26 +01:00
parent 014036e4e3
commit 23044b28db
5 changed files with 95 additions and 47 deletions

View File

@ -47,6 +47,7 @@ export class ExamCorrect {
_userInputStatus;
_userInputCandidates;
_partInputs;
_partDeleteBoxes;
_dateFormat;
_cIndices;
@ -75,6 +76,7 @@ export class ExamCorrect {
this._userInputStatus = document.getElementById(EXAM_CORRECT_USER_INPUT_STATUS_ID);
this._userInputCandidates = document.getElementById(EXAM_CORRECT_USER_INPUT_CANDIDATES_ID);
this._partInputs = [...this._element.querySelectorAll(`input[${EXAM_CORRECT_PART_INPUT_ATTR}]`)];
this._partDeleteBoxes = [...this._element.querySelectorAll('input.uw-exam-correct--delete-exam-part')];
if (this._sendBtn)
this._sendBtn.addEventListener('click', this._sendCorrectionHandler.bind(this));
@ -84,6 +86,10 @@ export class ExamCorrect {
this._userInput.addEventListener('focusout', this._validateUserInput.bind(this));
else throw new Error('ExamCorrect utility could not detect user input!');
for (let deleteBox of this._partDeleteBoxes) {
deleteBox.addEventListener('change', (() => { this._updatePartDeleteDisabled(deleteBox); }).bind(this));
}
if (!this._userInputStatus) {
throw new Error('ExamCorrect utility could not detect user input status element!');
}
@ -93,7 +99,7 @@ export class ExamCorrect {
}
// TODO get date format by post request
this._dateFormat = 'YYYY-MM-DD HH:mm:ss';
this._dateFormat = 'DD.MM.YYYY HH:mm:ss';
this._cIndices = new Map(
[...this._element.querySelectorAll('[uw-exam-correct-header]')]
@ -116,6 +122,14 @@ export class ExamCorrect {
// TODO destroy handlers on user input candidate elements
}
_updatePartDeleteDisabled(deleteBox) {
const partInput = deleteBox.parentElement.querySelector(`input[${EXAM_CORRECT_PART_INPUT_ATTR}]`);
if (!partInput)
return;
partInput.disabled = deleteBox.checked;
}
_validateUserInput() {
(!this._userInput.value) ? this._userInput.classList.add('no-value') : this._userInput.classList.remove('no-value');
@ -176,11 +190,17 @@ export class ExamCorrect {
const results = {};
for (const input of this._partInputs) {
if (input.reportValidity && !input.reportValidity()) {
if (input.disabled) {
const partKey = input.getAttribute(EXAM_CORRECT_PART_INPUT_ATTR);
if (!partKey) {
console.error('Error while parsing results: Could not detect exam part key attribute');
return;
}
results[partKey] = null;
} else if (input.reportValidity && !input.reportValidity()) {
input.focus();
return;
}
if (input.value) {
} else if (input.value) {
const partKey = input.getAttribute(EXAM_CORRECT_PART_INPUT_ATTR);
if (!partKey) {
console.error('Error while parsing results: Could not detect exam part key attribute');
@ -189,24 +209,25 @@ export class ExamCorrect {
results[partKey] = parseFloat(input.value);
}
}
console.log('results', results);
const result = this._resultSelect.value !== 'none' && this._resultSelect.value;
const result = this._resultSelect && this._resultSelect.value !== 'none' && this._resultSelect.value;
// abort send if there are no results (after validation)
if (Object.keys(results).length <= 0) return;
if (Object.keys(results).length <= 0 && !result) return;
const rowInfo = {
users: [user],
results: results,
result: result === 'delete' ? null : result,
status: STATUS.LOADING,
};
if (results) rowInfo.results = results;
if (result) rowInfo.result = result === 'delete' ? null : result;
this._addRow(rowInfo);
// clear inputs on validation success
this._clearUserInput();
this._partInputs.forEach(clearInput);
this._partDeleteBoxes.forEach(box => { box.checked = false; this._updatePartDeleteDisabled(box); });
const body = {
user: userId || user,
@ -305,39 +326,38 @@ export class ExamCorrect {
case 'success':
status = STATUS.SUCCESS;
if (response.user) {
userElem.setAttribute(EXAM_CORRECT_USER_ATTR, response.user.id);
userElem.innerHTML = userToHTML(response.user);
const timeElem = row.cells.item(0);
timeElem.innerHTML = moment(response.time).format(this._dateFormat);
timeElem.classList.remove('exam-correct--local-time');
newEntry.users = [response.user];
newEntry.results = response.results;
newEntry.result = response.grade;
} else {
console.error('Invalid response');
return;
}
// TODO set edit button visibility
break;
case 'ambiguous':
// TODO set edit button visibility
status = STATUS.AMBIGUOUS;
if (response.users) {
userElem = this._showUserList(row, response.users, results);
newEntry.users = response.users;
newEntry.results = results.partResults;
newEntry.result = results.result;
}
newEntry.users = response.users;
newEntry.results = typeof results === 'undefined' ? {} : results;
newEntry.result = typeof result === 'undefined' ? undefined : result; // eslint-disable-line no-undef
newEntry.message = response.message || null;
break;
case 'failure':
status = STATUS.FAILURE;
newEntry.users = (response.user && [response.user]) || null;
newEntry.results = results;
newEntry.results = typeof results === 'undefined' ? {} : results;
newEntry.message = response.message || null;
newEntry.result = results.result;
newEntry.result = typeof result === 'undefined' ? undefined : result; // eslint-disable-line no-undef
break;
default:
// TODO show tooltip with 'invalid response'
// TODO set edit button visibility
console.error('Invalid response');
return;
}
row.querySelectorAll('.fa-spin').forEach((elem) => {
setStatus(elem, status);
@ -359,19 +379,23 @@ export class ExamCorrect {
statusCell.appendChild(messageElem);
}
const userCell = row.querySelector('.uw-exam-correct--user-cell');
if (userCell && newEntry.users && newEntry.users.length === 1) {
if (userElem && newEntry.users && newEntry.users.length === 1) {
const user = newEntry.users[0];
userCell.innerHTML = userToHTML(user);
userCell.setAttribute(EXAM_CORRECT_USER_ATTR, user);
} else if (userCell && newEntry.users) {
row.replaceChild(userCell, this._showUserList(row, newEntry.users, request.results));
userElem.innerHTML = userToHTML(user);
userElem.setAttribute(EXAM_CORRECT_USER_ATTR, user.id || user);
} else if (userElem && newEntry.users) {
row.replaceChild(userElem, this._showUserList(row, newEntry.users, { partResults: request.results, result: request.grade } ));
}
for (let [k, v] of Object.entries(newEntry.results)) {
const resultCell = row.cells.item(this._cIndices.get(k));
if (v.result)
if (v === null) {
resultCell.innerHTML = '<i class="fas fa-fw fa-trash"></i>';
resultCell.classList.remove('exam-correct--result-unconfirmed');
} else if (v && v.result) {
resultCell.innerHTML = v.result;
resultCell.classList.remove('exam-correct--result-unconfirmed');
}
}
savedEntries.push(newEntry);
@ -429,9 +453,9 @@ export class ExamCorrect {
const body = {
user: listItem.getAttribute(EXAM_CORRECT_USER_ATTR),
results: results.partResults,
grade: results.result,
};
if (results.partResults) body.results = results.partResults;
if (results.result || results.result === null) body.grade = results.result;
this._app.httpClient.post({
url: EXAM_CORRECT_URL_POST,
@ -478,8 +502,13 @@ export class ExamCorrect {
console.error('Could not determine cell index from part key!');
} else {
const partCell = document.createElement('TD');
partCell.innerHTML = partResult;
partCell.classList.add('uw-exam-correct--part-cell');
if (partResult === null) {
partCell.innerHTML = '<i class="fas fa-fw fa-trash"></i>';
} else {
partCell.innerHTML = partResult;
}
partCell.classList.add('uw-exam-correct--part-cell', 'exam-correct--result-unconfirmed');
cells.set(cellIndex, partCell);
}
}

View File

@ -12,7 +12,7 @@ table[uw-exam-correct]
min-width: 200px
th.uw-exam-correct--part-cell, td.uw-exam-correct--part-cell
width: min-content
width: 115px
text-align: center
white-space: nowrap
@ -24,12 +24,16 @@ table[uw-exam-correct]
opacity: .5
cursor: pointer
margin-left: 5px
.uw-exam-correct--delete-exam-part ~ .fa-trash:hover
opacity: 1
&:hover
opacity: 1
.uw-exam-correct--delete-exam-part:checked ~ .fa-trash
opacity: 1
color: var(--color-error)
&:hover
color: var(--color-error-dark)
td#uw-exam-correct__result
width: min-content
select
@ -63,7 +67,6 @@ table[uw-exam-correct]
border: 2px solid var(--color-error)
[uw-exam-correct] tbody ul
list-style: none
li
cursor: pointer
text-decoration: underline
@ -73,9 +76,11 @@ table[uw-exam-correct]
width: calc(100% - 18em/14 - #{$exam-correct--input-status-margin})
i
margin-left: $exam-correct--input-status-margin
ul
margin-top: 7px
.exam-correct--local-time
color: var(--color-fontsec)
.exam-correct--local-time, .exam-correct--result-unconfirmed
opacity: .5
font-style: italic
.exam-correct--success

View File

@ -413,6 +413,7 @@ UnauthorizedAllocationLecturer: Sie sind nicht als Veranstalter für eine Verans
UnauthorizedCorrector: Sie sind nicht als Korrektor für diese Veranstaltung eingetragen.
UnauthorizedSheetCorrector: Sie sind nicht als Korrektor für dieses Übungsblatt eingetragen.
UnauthorizedExamCorrector: Sie sind nicht als Korrektor für diese Prüfung eingetragen.
UnauthorizedExamCorrectorGrade: Sie haben nicht die Berechtigung für diese Prüfung Gesamtergebnisse einzutragen.
UnauthorizedCorrectorAny: Sie sind nicht als Korrektor für eine Veranstaltung eingetragen.
UnauthorizedRegistered: Sie sind nicht als Teilnehmer für diese Veranstaltung registriert.
UnauthorizedAllocationRegistered: Sie sind nicht als Teilnehmer für diese Zentralanmeldung registriert.

View File

@ -411,6 +411,7 @@ UnauthorizedAllocationLecturer: You are no administrator for any of the courses
UnauthorizedCorrector: You are no sheet corrector for this course.
UnauthorizedSheetCorrector: You are no corrector for this sheet.
UnauthorizedExamCorrector: You are no corrector for this exam.
UnauthorizedExamCorrectorGrade: You may not enter overall exam achievements for this exam.
UnauthorizedCorrectorAny: You are no corrector for any course.
UnauthorizedRegistered: You are no participant in this course.
UnauthorizedAllocationRegistered: You are no participant in this central allocation.

View File

@ -10,10 +10,13 @@ import qualified Data.Aeson as JSON
import qualified Database.Esqueleto as E
import qualified Database.Esqueleto.Utils as E
import Database.Persist.Sql (transactionUndo)
import Handler.Utils
import Handler.Utils.Exam (fetchExam)
import qualified Data.HashMap.Strict as HashMap
data CorrectInterfaceUser
= CorrectInterfaceUser
@ -30,7 +33,7 @@ data CorrectInterfaceResponse
= CorrectInterfaceResponseSuccess
{ cirsUser :: CorrectInterfaceUser
, cirsResults :: Map ExamPartNumber (Maybe ExamResultPoints)
, cirsGrade :: Maybe ExamResultPassedGrade
, cirsGrade :: Maybe (Maybe ExamResultPassedGrade)
, cirsTime :: UTCTime
}
| CorrectInterfaceResponseAmbiguous
@ -44,17 +47,18 @@ data CorrectInterfaceResponse
| CorrectInterfaceResponseNoOp
{ cirnUsers :: Set CorrectInterfaceUser
}
deriveJSON defaultOptions
deriveToJSON defaultOptions
{ constructorTagModifier = camelToPathPiece' 3
, fieldLabelModifier = camelToPathPiece' 1
, sumEncoding = TaggedObject "status" "results"
, omitNothingFields = True
} ''CorrectInterfaceResponse
data CorrectInterfaceRequest
= CorrectInterfaceRequest
{ ciqUser :: Either Text (CryptoID UUID (Key User))
, ciqResults :: Maybe (NonNull (Map ExamPartNumber (Maybe Points)))
, ciqGrade :: Maybe ExamResultPassedGrade
, ciqGrade :: Maybe (Maybe ExamResultPassedGrade)
}
instance FromJSON CorrectInterfaceRequest where
@ -62,7 +66,11 @@ instance FromJSON CorrectInterfaceRequest where
ciqUser <- Right <$> o JSON..: "user" <|> Left <$> o JSON..: "user"
results <- o JSON..:? "results"
ciqResults <- for results $ maybe (fail "Results may not be nullable") return . fromNullable
ciqGrade <- o JSON..:? "grade"
ciqGrade <- if
| "grade" `HashMap.member` o
-> Just <$> o JSON..: "grade"
| otherwise
-> pure Nothing
return CorrectInterfaceRequest{..}
@ -104,7 +112,9 @@ postECorrectR tid ssh csh examn = do
CorrectInterfaceRequest{..} <- requireCheckJsonBody
response <- exceptT return return . hoist runDB $ do
let mayEditResults = False
response <- runDB . exceptT (<$ transactionUndo) return $ do
Entity eId Exam{..} <- lift $ fetchExam tid ssh csh examn
euid <- traverse decrypt ciqUser
@ -192,20 +202,22 @@ postECorrectR tid ssh csh examn = do
| otherwise -> return Nothing
| otherwise -> return mempty
newExamResult <- lift $ do
newExamResult <- for ciqGrade $ \ciqGrade' -> lift $ do
unless mayEditResults $
permissionDeniedI MsgUnauthorizedExamCorrectorGrade
mOldResult <- getBy $ UniqueExamResult eId uid
if
| Just (Entity oldId _) <- mOldResult, is _Nothing ciqGrade -> do
delete oldId
audit $ TransactionExamResultDeleted eId uid
return Nothing
| Just result <- ciqGrade -> let
| Just (Entity oldId _) <- mOldResult, is _Nothing ciqGrade' -> do
delete oldId
audit $ TransactionExamResultDeleted eId uid
return Nothing
| Just result <- ciqGrade' -> let
mOld = view passedGrade . examResultResult . entityVal <$> mOldResult
resultGrade = review passedGrade result
passedGrade :: Iso' ExamResultGrade ExamResultPassedGrade
passedGrade = iso (fmap $ bool (Left . view passingGrade) Right examShowGrades) (fmap $ either (review passingGrade) id)
in if
| ciqGrade /= mOld -> do
| ciqGrade' /= mOld -> do
newResult <- upsert ExamResult
{ examResultExam = eId
, examResultUser = uid