Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Programming exercises: Enhance filtering and sorting for error analysis #9315

Merged
Show file tree
Hide file tree
Changes from 12 commits
Commits
Show all changes
80 commits
Select commit Hold shift + click to select a range
0b1023e
first approach
az108 Sep 13, 2024
e14b18c
first working valid approach
az108 Sep 13, 2024
8e4c046
final approach and server side cleaned with server side tests adapted
az108 Sep 13, 2024
bd0ff93
cleaned client side
az108 Sep 13, 2024
5a4800c
cleaned client side and added new client tests
az108 Sep 13, 2024
0d087aa
changed server side naming
az108 Sep 13, 2024
ec9fe4d
fixed i18n files
az108 Sep 13, 2024
e7d16c7
Merge branch 'develop' into feature/programming-exercises/refactor-an…
az108 Sep 13, 2024
289d2ea
coderabbit and fixed client and server style
az108 Sep 14, 2024
f03def0
Merge branch 'develop' into feature/programming-exercises/refactor-an…
az108 Sep 14, 2024
c4b5503
Patrik feedback
az108 Sep 15, 2024
e6f41e1
Merge remote-tracking branch 'origin/feature/programming-exercises/re…
az108 Sep 15, 2024
b6ad90e
Patrik feedback
az108 Sep 16, 2024
4592cfe
Patrik feedback
az108 Sep 16, 2024
afffba3
Patrik feedback
az108 Sep 16, 2024
e0df0a8
Patrik feedback
az108 Sep 16, 2024
0b6c5fa
Patrik feedback
az108 Sep 16, 2024
4547ae5
Merge branch 'develop' into feature/programming-exercises/refactor-an…
az108 Sep 16, 2024
5a00250
Johannes feedback
az108 Sep 16, 2024
46270e9
Merge remote-tracking branch 'origin/feature/programming-exercises/re…
az108 Sep 16, 2024
fc1f68f
client test fixed
az108 Sep 16, 2024
06b0c5e
sorting removed
az108 Sep 24, 2024
c68a23e
using Page now
az108 Sep 26, 2024
3b0a4f1
working slider for modal
az108 Sep 30, 2024
857f8ab
working slider for modal
az108 Sep 30, 2024
89b2434
single option filter working
az108 Oct 1, 2024
c532903
working modal
az108 Oct 2, 2024
9e00651
added header sorting for occurrence and feedback again
az108 Oct 2, 2024
dcdacc0
Merge remote-tracking branch 'origin/develop' into feature/programmin…
az108 Oct 2, 2024
80df1c6
partially working
az108 Oct 6, 2024
02ce3cf
working but clean up missing
az108 Oct 6, 2024
0f83c13
Merge remote-tracking branch 'origin/develop' into feature/programmin…
az108 Oct 6, 2024
b1e1f23
seperated maxCount call
az108 Oct 6, 2024
b44caca
occurrence slider adjustments
az108 Oct 6, 2024
f13e75a
server tests adjusted and added
az108 Oct 6, 2024
b94ed3b
client side code adjusted
az108 Oct 6, 2024
3c479bd
client side tests added / adjusted
az108 Oct 6, 2024
02f7709
client side tests added / adjusted and coderabbit
az108 Oct 7, 2024
713c060
server style
az108 Oct 7, 2024
f74a15d
Merge branch 'develop' into feature/programming-exercises/refactor-an…
az108 Oct 7, 2024
3531d70
server style
az108 Oct 7, 2024
a4af219
Merge remote-tracking branch 'origin/feature/programming-exercises/re…
az108 Oct 7, 2024
0ca27db
client code coverage
az108 Oct 7, 2024
45f04b1
client test fixed
az108 Oct 7, 2024
38a97f6
client test adjusted
az108 Oct 7, 2024
d91b2a5
range slider adjusted
az108 Oct 7, 2024
6f2f989
range slider adjusted
az108 Oct 7, 2024
6275926
more request changes handled
az108 Oct 8, 2024
5808bf9
more request changes handled
az108 Oct 8, 2024
691fff2
client test adjusted
az108 Oct 8, 2024
2898348
Merge remote-tracking branch 'origin/develop' into feature/programmin…
az108 Oct 8, 2024
c6a55b7
server style adjusted
az108 Oct 8, 2024
cdb6442
server style adjusted
az108 Oct 8, 2024
23f045d
server tests adjusted
az108 Oct 8, 2024
098e8f7
fixed small % _ interaction in searchterm
az108 Oct 9, 2024
34d6fe6
rounded relative count for modal
az108 Oct 9, 2024
e8dd9d3
Merge branch 'develop' into feature/programming-exercises/refactor-an…
az108 Oct 15, 2024
a3de668
ramona feedback
az108 Oct 18, 2024
5ca3241
removed method call from template
az108 Oct 19, 2024
5fb62da
removed form group
az108 Oct 19, 2024
3ca814e
removed form group
az108 Oct 19, 2024
400cc2c
Merge remote-tracking branch 'origin/feature/programming-exercises/re…
az108 Oct 19, 2024
ba080f5
Merge branch 'develop' into feature/programming-exercises/refactor-an…
az108 Oct 19, 2024
4f6d62c
fixed client test
az108 Oct 19, 2024
9660f87
Merge remote-tracking branch 'origin/feature/programming-exercises/re…
az108 Oct 19, 2024
92aa0cd
fixed client test
az108 Oct 19, 2024
2426e0d
fixed client test
az108 Oct 19, 2024
c06a81b
Merge branch 'develop' into feature/programming-exercises/refactor-an…
krusche Oct 19, 2024
b23693d
stephan feedback
az108 Oct 19, 2024
573572a
Merge remote-tracking branch 'origin/feature/programming-exercises/re…
az108 Oct 19, 2024
634c3f3
stephan feedback
az108 Oct 19, 2024
71cd81f
Merge branch 'develop' into feature/programming-exercises/refactor-an…
az108 Oct 19, 2024
5e0076b
comments impproved patrik
az108 Oct 21, 2024
6e20397
Merge remote-tracking branch 'origin/feature/programming-exercises/re…
az108 Oct 21, 2024
9d879c5
Merge branch 'develop' into feature/programming-exercises/refactor-an…
az108 Oct 21, 2024
4bda1ef
Merge remote-tracking branch 'origin/develop' into feature/programmin…
az108 Oct 22, 2024
082343a
merge conflict
az108 Oct 22, 2024
c9182c8
Merge remote-tracking branch 'origin/feature/programming-exercises/re…
az108 Oct 22, 2024
8c97aae
merge conflict
az108 Oct 22, 2024
ef82bff
Merge branch 'develop' into feature/programming-exercises/refactor-an…
az108 Oct 23, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
package de.tum.cit.aet.artemis.assessment.dto;

import com.fasterxml.jackson.annotation.JsonInclude;

import de.tum.cit.aet.artemis.core.dto.SearchResultPageDTO;

@JsonInclude(JsonInclude.Include.NON_EMPTY)
public record FeedbackAnalysisResponseDTO(SearchResultPageDTO<FeedbackDetailDTO> feedbackDetails, long totalItems) {
}
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,9 @@
import java.util.Objects;
import java.util.Optional;
import java.util.Set;
import java.util.function.Predicate;
import java.util.stream.Collectors;
import java.util.stream.IntStream;

import jakarta.annotation.Nullable;
import jakarta.validation.constraints.NotNull;
Expand All @@ -28,6 +30,7 @@
import de.tum.cit.aet.artemis.assessment.domain.FeedbackType;
import de.tum.cit.aet.artemis.assessment.domain.LongFeedbackText;
import de.tum.cit.aet.artemis.assessment.domain.Result;
import de.tum.cit.aet.artemis.assessment.dto.FeedbackAnalysisResponseDTO;
import de.tum.cit.aet.artemis.assessment.dto.FeedbackDetailDTO;
import de.tum.cit.aet.artemis.assessment.repository.ComplaintRepository;
import de.tum.cit.aet.artemis.assessment.repository.ComplaintResponseRepository;
Expand All @@ -40,6 +43,9 @@
import de.tum.cit.aet.artemis.buildagent.dto.ResultBuildJob;
import de.tum.cit.aet.artemis.core.domain.Course;
import de.tum.cit.aet.artemis.core.domain.User;
import de.tum.cit.aet.artemis.core.dto.SearchResultPageDTO;
import de.tum.cit.aet.artemis.core.dto.SortingOrder;
import de.tum.cit.aet.artemis.core.dto.pageablesearch.SearchTermPageableSearchDTO;
import de.tum.cit.aet.artemis.core.exception.BadRequestAlertException;
import de.tum.cit.aet.artemis.core.repository.UserRepository;
import de.tum.cit.aet.artemis.core.security.Role;
Expand Down Expand Up @@ -530,31 +536,66 @@ private Result shouldSaveResult(@NotNull Result result, boolean shouldSave) {
}

/**
* Retrieves aggregated feedback details for a given exercise, calculating relative counts based on the total number of distinct results.
* Retrieves paginated and filtered aggregated feedback details for a given exercise, calculating relative counts based on the total number of distinct results.
* The task numbers are assigned based on the associated test case names, using the set of tasks fetched from the database.
* <br>
* For each feedback detail:
* 1. The relative count is calculated as a percentage of the total number of distinct results for the exercise.
* 2. The task number is determined by matching the test case name with the tasks.
* <br>
* The method supports filtering by a search term across feedback details, test case names, counts, task numbers, and relative counts.
* Sorting is applied based on the specified column and order (ascending or descending).
* The result is paginated based on the provided page number and page size.
*
* @param exerciseId The ID of the exercise for which feedback details should be retrieved.
* @return A list of FeedbackDetailDTO objects, each containing:
* - feedback count,
* - relative count (as a percentage of distinct results),
* - detail text,
* - test case name,
* - determined task number (based on the test case name).
* @param search The pageable search DTO containing page number, page size, sorting options, and a search term for filtering results.
* @return A {@link FeedbackAnalysisResponseDTO} object containing:
* - a {@link SearchResultPageDTO} of paginated feedback details, and
* - the total number of distinct results (distinctResultCount) for the exercise.
*/
public List<FeedbackDetailDTO> findAggregatedFeedbackByExerciseId(long exerciseId) {
public FeedbackAnalysisResponseDTO getFeedbackDetailsOnPage(long exerciseId, SearchTermPageableSearchDTO<String> search) {
long distinctResultCount = studentParticipationRepository.countDistinctResultsByExerciseId(exerciseId);
Set<ProgrammingExerciseTask> tasks = programmingExerciseTaskService.getTasksWithUnassignedTestCases(exerciseId);

List<FeedbackDetailDTO> feedbackDetails = studentParticipationRepository.findAggregatedFeedbackByExerciseId(exerciseId);
String searchTerm = search.getSearchTerm() != null ? search.getSearchTerm().toLowerCase() : "";

return feedbackDetails.stream().map(detail -> {
feedbackDetails = feedbackDetails.stream().map(detail -> {
double relativeCount = (detail.count() * 100.0) / distinctResultCount;
int taskNumber = tasks.stream().filter(task -> task.getTestCases().stream().anyMatch(tc -> tc.getTestName().equals(detail.testCaseName()))).findFirst()
.map(task -> tasks.stream().toList().indexOf(task) + 1).orElse(0);
int taskNumber = determineTaskNumberOfTestCase(detail.testCaseName(), tasks);

return new FeedbackDetailDTO(detail.count(), relativeCount, detail.detailText(), detail.testCaseName(), taskNumber);
}).toList();
}).filter(matchesSearchTerm(searchTerm)).sorted(getComparatorForFeedbackDetails(search)).toList();
az108 marked this conversation as resolved.
Show resolved Hide resolved

return paginateFeedbackDetails(feedbackDetails, search.getPage(), search.getPageSize());
az108 marked this conversation as resolved.
Show resolved Hide resolved
}

private Predicate<FeedbackDetailDTO> matchesSearchTerm(String searchTerm) {
az108 marked this conversation as resolved.
Show resolved Hide resolved
return detail -> searchTerm.isEmpty() || detail.detailText().toLowerCase().contains(searchTerm) || detail.testCaseName().toLowerCase().contains(searchTerm)
|| String.valueOf(detail.count()).contains(searchTerm) || String.valueOf(detail.taskNumber()).contains(searchTerm)
|| String.valueOf(detail.relativeCount()).contains(searchTerm);
}

private Comparator<FeedbackDetailDTO> getComparatorForFeedbackDetails(SearchTermPageableSearchDTO<String> search) {
Map<String, Comparator<FeedbackDetailDTO>> comparators = Map.of("count", Comparator.comparingLong(FeedbackDetailDTO::count), "detailText",
Comparator.comparing(FeedbackDetailDTO::detailText, String.CASE_INSENSITIVE_ORDER), "testCaseName",
Comparator.comparing(FeedbackDetailDTO::testCaseName, String.CASE_INSENSITIVE_ORDER), "taskNumber", Comparator.comparingInt(FeedbackDetailDTO::taskNumber),
"relativeCount", Comparator.comparingDouble(FeedbackDetailDTO::relativeCount));
Comparator<FeedbackDetailDTO> comparator = comparators.getOrDefault(search.getSortedColumn(), (a, b) -> 0);
return search.getSortingOrder() == SortingOrder.ASCENDING ? comparator : comparator.reversed();
}

private int determineTaskNumberOfTestCase(String testCaseName, Set<ProgrammingExerciseTask> tasks) {
return IntStream.range(0, tasks.size()).filter(i -> tasks.stream().toList().get(i).getTestCases().stream().anyMatch(tc -> tc.getTestName().equals(testCaseName)))
az108 marked this conversation as resolved.
Show resolved Hide resolved
.findFirst().orElse(-1) + 1;
}

private FeedbackAnalysisResponseDTO paginateFeedbackDetails(List<FeedbackDetailDTO> feedbackDetails, int page, int pageSize) {
az108 marked this conversation as resolved.
Show resolved Hide resolved
int start = (page - 1) * pageSize;
az108 marked this conversation as resolved.
Show resolved Hide resolved
int end = Math.min(start + pageSize, feedbackDetails.size());
List<FeedbackDetailDTO> paginatedFeedbackDetails = feedbackDetails.subList(start, end);

int totalPages = (feedbackDetails.size() + pageSize - 1) / pageSize;
az108 marked this conversation as resolved.
Show resolved Hide resolved
return new FeedbackAnalysisResponseDTO(new SearchResultPageDTO<>(paginatedFeedbackDetails, totalPages), feedbackDetails.size());
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -27,12 +27,13 @@

import de.tum.cit.aet.artemis.assessment.domain.Feedback;
import de.tum.cit.aet.artemis.assessment.domain.Result;
import de.tum.cit.aet.artemis.assessment.dto.FeedbackDetailDTO;
import de.tum.cit.aet.artemis.assessment.dto.FeedbackAnalysisResponseDTO;
import de.tum.cit.aet.artemis.assessment.dto.ResultWithPointsPerGradingCriterionDTO;
import de.tum.cit.aet.artemis.assessment.repository.ResultRepository;
import de.tum.cit.aet.artemis.assessment.service.ResultService;
import de.tum.cit.aet.artemis.core.domain.Course;
import de.tum.cit.aet.artemis.core.domain.User;
import de.tum.cit.aet.artemis.core.dto.pageablesearch.SearchTermPageableSearchDTO;
import de.tum.cit.aet.artemis.core.exception.BadRequestAlertException;
import de.tum.cit.aet.artemis.core.repository.UserRepository;
import de.tum.cit.aet.artemis.core.security.Role;
Expand Down Expand Up @@ -280,16 +281,21 @@ public ResponseEntity<Result> createResultForExternalSubmission(@PathVariable Lo
}

/**
* GET /exercises/:exerciseId/feedback-details : Retrieves all aggregated feedback details for a given exercise.
* The feedback details include counts and relative counts of feedback occurrences, along with associated test case names and task numbers.
* POST /exercises/{exerciseId}/feedback-details-paged : Retrieves paginated and filtered aggregated feedback details for a given exercise.
* The feedback details include counts and relative counts of feedback occurrences, test case names, and task numbers.
* The method allows filtering by search term and sorting by various fields.
* <br>
* Pagination is applied based on the provided {@link SearchTermPageableSearchDTO}, including page number, page size, sorting order, and search term.
* The response contains both the paginated feedback details and the total count of distinct results for the exercise.
*
* @param exerciseId The ID of the exercise for which feedback details should be retrieved.
* @return A ResponseEntity containing a list of {@link FeedbackDetailDTO}s
* @param search The pageable search DTO containing page number, page size, sorting options, and a search term for filtering results.
* @return A {@link ResponseEntity} containing a {@link FeedbackAnalysisResponseDTO}, which includes the paginated feedback details and the total count of distinct results.
*/
@GetMapping("exercises/{exerciseId}/feedback-details")
@PostMapping("exercises/{exerciseId}/feedback-details-paged")
az108 marked this conversation as resolved.
Show resolved Hide resolved
@EnforceAtLeastEditorInExercise
public ResponseEntity<List<FeedbackDetailDTO>> getAllFeedbackDetailsForExercise(@PathVariable Long exerciseId) {
log.debug("REST request to get all Feedback details for Exercise {}", exerciseId);
return ResponseEntity.ok(resultService.findAggregatedFeedbackByExerciseId(exerciseId));
public ResponseEntity<FeedbackAnalysisResponseDTO> getFeedbackDetailsPaged(@PathVariable long exerciseId, @RequestBody SearchTermPageableSearchDTO<String> search) {
FeedbackAnalysisResponseDTO response = resultService.getFeedbackDetailsOnPage(exerciseId, search);
return ResponseEntity.ok(response);
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
<div class="modal-header">
<h4 jhiTranslate="artemisApp.programmingExercise.configureGrading.feedbackAnalysis.feedbackModal.header"></h4>
</div>
<div class="modal-body">
<div class="d-flex justify-content-between mb-3">
<div class="border">
az108 marked this conversation as resolved.
Show resolved Hide resolved
<span class="modal-label" jhiTranslate="artemisApp.programmingExercise.configureGrading.feedbackAnalysis.task"></span>
<p>{{ feedbackDetail.taskNumber }}</p>
</div>
<div class="border">
<span class="modal-label" jhiTranslate="artemisApp.programmingExercise.configureGrading.feedbackAnalysis.occurrence"></span>
<p>{{ feedbackDetail.relativeCount }}% ({{ feedbackDetail.count }})</p>
</div>
<div class="border">
<span class="modal-label" jhiTranslate="artemisApp.programmingExercise.configureGrading.feedbackAnalysis.testcase"></span>
<p>{{ feedbackDetail.testCaseName }}</p>
</div>
<div class="border">
<span class="modal-label" jhiTranslate="artemisApp.programmingExercise.configureGrading.feedbackAnalysis.errorCategory"></span>
<p>Student Error</p>
az108 marked this conversation as resolved.
Show resolved Hide resolved
</div>
</div>

<h5 class="mb-3" jhiTranslate="artemisApp.programmingExercise.configureGrading.feedbackAnalysis.feedbackModal.feedbackTitle"></h5>
<p>{{ feedbackDetail.detailText }}</p>
</div>
<div class="modal-footer">
<button
type="button"
class="btn btn-outline-primary"
jhiTranslate="artemisApp.programmingExercise.configureGrading.feedbackAnalysis.feedbackModal.ok"
(click)="activeModal.close()"
></button>
</div>
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
.modal-body {
az108 marked this conversation as resolved.
Show resolved Hide resolved
padding: 1rem;
}

.modal-label {
az108 marked this conversation as resolved.
Show resolved Hide resolved
font-weight: bold;
}

.border {
az108 marked this conversation as resolved.
Show resolved Hide resolved
padding: 5px;
az108 marked this conversation as resolved.
Show resolved Hide resolved
border-left: var(--border-color) 1px solid;
border-right: var(--border-color) 1px solid;
border-bottom: var(--border-color) 1px solid;
az108 marked this conversation as resolved.
Show resolved Hide resolved
border-radius: 10px;
}

.modal-header {
border-bottom: none;
padding-left: 1rem;
padding-top: 1.5rem;
az108 marked this conversation as resolved.
Show resolved Hide resolved
}

.modal-body p {
margin-bottom: 0;
}

.modal-footer {
border-top: none;
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
import { Component, Input, inject } from '@angular/core';
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap';
import { FeedbackDetail } from 'app/exercises/programming/manage/grading/feedback-analysis/feedback-analysis.service';
import { ArtemisSharedCommonModule } from 'app/shared/shared-common.module';

@Component({
selector: 'jhi-feedback-modal',
templateUrl: './feedback-modal.component.html',
styleUrls: ['./feedback-modal.component.scss'],
imports: [ArtemisSharedCommonModule],
standalone: true,
})
export class FeedbackModalComponent {
@Input() feedbackDetail!: FeedbackDetail;
az108 marked this conversation as resolved.
Show resolved Hide resolved

activeModal: NgbActiveModal = inject(NgbActiveModal);
}
Original file line number Diff line number Diff line change
@@ -1,27 +1,81 @@
<ng-template #headerTemplate let-column="column" let-label="label">
<th scope="col" class="clickable" (click)="setSortedColumn(column)">
<span [jhiTranslate]="label"></span>
@if (sortedColumn() === column) {
<fa-icon [icon]="sortIcon()"></fa-icon>
}
</th>
</ng-template>

<div class="m-3">
<h2 class="mb-3" jhiTranslate="artemisApp.programmingExercise.configureGrading.feedbackAnalysis.title" [translateValues]="{ exerciseTitle: exerciseTitle }"></h2>
<h2 class="mb-3" jhiTranslate="artemisApp.programmingExercise.configureGrading.feedbackAnalysis.title" [translateValues]="{ exerciseTitle: exerciseTitle() }"></h2>

<div class="d-flex align-items-center mb-3 position-relative">
<input
type="search"
class="form-control"
[placeholder]="'artemisApp.programmingExercise.configureGrading.feedbackAnalysis.search' | artemisTranslate"
aria-label="Search feedback"
[(ngModel)]="searchTerm"
(ngModelChange)="search()"
/>
<fa-icon [icon]="faMagnifyingGlass" class="search-icon"></fa-icon>
</div>

<table class="table table-striped mb-3">
<thead>
<tr>
<th scope="col" jhiTranslate="artemisApp.programmingExercise.configureGrading.feedbackAnalysis.occurrence"></th>
<th scope="col" jhiTranslate="artemisApp.programmingExercise.configureGrading.feedbackAnalysis.feedback"></th>
<th scope="col" jhiTranslate="artemisApp.programmingExercise.configureGrading.feedbackAnalysis.task"></th>
<th scope="col" jhiTranslate="artemisApp.programmingExercise.configureGrading.feedbackAnalysis.testcase"></th>
<th scope="col" jhiTranslate="artemisApp.programmingExercise.configureGrading.feedbackAnalysis.errorCategory"></th>
<ng-container
*ngTemplateOutlet="headerTemplate; context: { column: 'count', label: 'artemisApp.programmingExercise.configureGrading.feedbackAnalysis.occurrence' }"
></ng-container>
<ng-container
*ngTemplateOutlet="headerTemplate; context: { column: 'detailText', label: 'artemisApp.programmingExercise.configureGrading.feedbackAnalysis.feedback' }"
></ng-container>
<ng-container
*ngTemplateOutlet="headerTemplate; context: { column: 'taskNumber', label: 'artemisApp.programmingExercise.configureGrading.feedbackAnalysis.task' }"
></ng-container>
<ng-container
*ngTemplateOutlet="headerTemplate; context: { column: 'testCaseName', label: 'artemisApp.programmingExercise.configureGrading.feedbackAnalysis.testcase' }"
></ng-container>
<ng-container
*ngTemplateOutlet="
headerTemplate;
context: { column: 'relativeCount', label: 'artemisApp.programmingExercise.configureGrading.feedbackAnalysis.errorCategory' }
"
></ng-container>
</tr>
</thead>
<tbody class="table-group-divider">
@for (item of feedbackDetails; track item) {
@for (item of content().resultsOnPage; track item) {
<tr>
<td class="text-center">{{ item.count }} ({{ item.relativeCount | number: '1.0-0' }}%)</td>
<td>{{ item.detailText }}</td>
<td>
{{ item.detailText.length > maxFeedbackDetailTextLength ? (item.detailText | slice: 0 : 100) + '...' : item.detailText }}
</td>
<td class="text-center">{{ item.taskNumber }}</td>
<td>{{ item.testCaseName }}</td>
<td>Student Error</td>
az108 marked this conversation as resolved.
Show resolved Hide resolved
<!-- This is a placeholder, will be covered in follow up PRs -->
<td class="text-center">
<fa-icon [icon]="faMagnifyingGlassPlus" class="ms-2 clickable" (click)="openFeedbackModal(item)"></fa-icon>
</td>
</tr>
}
</tbody>
</table>
<div jhiTranslate="artemisApp.programmingExercise.configureGrading.feedbackAnalysis.totalItems" [translateValues]="{ count: feedbackDetails.length }"></div>

<div class="d-flex flex-column align-items-end mt-2">
<ngb-pagination
[collectionSize]="collectionsSize()"
[pageSize]="pageSize()"
[page]="page()"
(pageChange)="setPage($event)"
size="sm"
class="pagination justify-content-end"
>
</ngb-pagination>

<div class="text-muted text-end">
<span class="small-text" jhiTranslate="artemisApp.programmingExercise.configureGrading.feedbackAnalysis.totalItems" [translateValues]="{ count: totalItems() }"></span>
</div>
</div>
az108 marked this conversation as resolved.
Show resolved Hide resolved
</div>
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
.small-text {
font-size: 0.8rem;
az108 marked this conversation as resolved.
Show resolved Hide resolved
}

.position-relative {
position: relative;
}

.search-icon {
position: absolute;
right: 10px;
pointer-events: none;
}
az108 marked this conversation as resolved.
Show resolved Hide resolved
Loading
Loading