From fb98ec486d4f7834280ad47de1b14583e7ccdfe6 Mon Sep 17 00:00:00 2001 From: Tarlan Ismayilsoy Date: Mon, 7 Aug 2023 11:16:32 +0400 Subject: [PATCH 01/21] Fix incorrect filters --- Themis/Extensions/ExamExtension.swift | 2 +- Themis/Extensions/ExerciseExtension.swift | 10 +++++++--- Themis/ViewModels/Course/CourseViewModel.swift | 6 ++++-- 3 files changed, 12 insertions(+), 6 deletions(-) diff --git a/Themis/Extensions/ExamExtension.swift b/Themis/Extensions/ExamExtension.swift index d9dbe0d0..c4e8d36d 100644 --- a/Themis/Extensions/ExamExtension.swift +++ b/Themis/Extensions/ExamExtension.swift @@ -20,7 +20,7 @@ extension Exam { if let publishResultsDate = self.publishResultsDate { return publishResultsDate <= .now } - return false + return true } public var exercises: [Exercise] { diff --git a/Themis/Extensions/ExerciseExtension.swift b/Themis/Extensions/ExerciseExtension.swift index 16bc4e45..462082d0 100644 --- a/Themis/Extensions/ExerciseExtension.swift +++ b/Themis/Extensions/ExerciseExtension.swift @@ -41,11 +41,15 @@ extension Exercise { } var isCurrentlyInAssessment: Bool { - supportsAssessment && hasSomethingToAssess + supportsAssessment && isDueDatePastButAssessmentDueDateNot } - var isViewOnly: Bool { - supportsAssessment && !hasSomethingToAssess + private var isDueDatePastButAssessmentDueDateNot: Bool { + guard let dueDate = self.baseExercise.dueDate, + let assessmentDueDate = self.baseExercise.assessmentDueDate else { + return false + } + return dueDate < .now && .now < assessmentDueDate } private var hasSomethingToAssess: Bool { diff --git a/Themis/ViewModels/Course/CourseViewModel.swift b/Themis/ViewModels/Course/CourseViewModel.swift index 6c78d8bd..ccf17e81 100644 --- a/Themis/ViewModels/Course/CourseViewModel.swift +++ b/Themis/ViewModels/Course/CourseViewModel.swift @@ -106,8 +106,10 @@ class CourseViewModel: ObservableObject { log.error(String(describing: error)) } - assessableExercises = (courseForAssessment.value?.exercises ?? []).filter({ $0.supportsAssessment }) - viewOnlyExercises = Array(Set(shownCourse?.exercises ?? []).subtracting(Set(assessableExercises))).filter({ $0.supportsAssessment }) + let exercisesOfShownCourse = courseForAssessment.value?.exercises ?? [] + + assessableExercises = exercisesOfShownCourse.filter({ $0.isCurrentlyInAssessment }) + viewOnlyExercises = exercisesOfShownCourse.filter({ !$0.isCurrentlyInAssessment }) } } From fe26424dab07f25c42a3214561266105dfdc4d36 Mon Sep 17 00:00:00 2001 From: Tarlan Ismayilsoy Date: Mon, 7 Aug 2023 11:42:08 +0400 Subject: [PATCH 02/21] Enable view-only exercises --- Themis/ViewModels/Exercise/ExerciseViewModel.swift | 2 +- Themis/Views/Courses/CourseView.swift | 4 +--- 2 files changed, 2 insertions(+), 4 deletions(-) diff --git a/Themis/ViewModels/Exercise/ExerciseViewModel.swift b/Themis/ViewModels/Exercise/ExerciseViewModel.swift index 423d3d2f..c689fb3b 100644 --- a/Themis/ViewModels/Exercise/ExerciseViewModel.swift +++ b/Themis/ViewModels/Exercise/ExerciseViewModel.swift @@ -81,7 +81,7 @@ class ExerciseViewModel: ObservableObject { var isAssessmentPossible: Bool { (exercise.value?.isCurrentlyInAssessment ?? false) - || exam?.isOver ?? false + || ((exam?.isOver ?? false) && !(exam?.isAssessmentDue ?? true)) } @MainActor diff --git a/Themis/Views/Courses/CourseView.swift b/Themis/Views/Courses/CourseView.swift index 5ac4744c..d9dfda4c 100644 --- a/Themis/Views/Courses/CourseView.swift +++ b/Themis/Views/Courses/CourseView.swift @@ -33,10 +33,8 @@ struct CourseView: View { ExerciseGroups(courseVM: courseVM, type: .inAssessment) .padding(.bottom) ExerciseGroups(courseVM: courseVM, type: .viewOnly) - .disabled(true) // TODO: remove once view-only mode is fully implemented } - .padding(.horizontal, 20) - .padding(.vertical, 20) + .padding(20) } .environmentObject(courseVM) .refreshable { From b35da641da29e5a3c308c60f96de7dc34548344b Mon Sep 17 00:00:00 2001 From: Tarlan Ismayilsoy Date: Mon, 7 Aug 2023 11:50:32 +0400 Subject: [PATCH 03/21] Fix: exams are not updated when switching courses --- .../ViewModels/Course/CourseViewModel.swift | 23 +++++++++++++------ 1 file changed, 16 insertions(+), 7 deletions(-) diff --git a/Themis/ViewModels/Course/CourseViewModel.swift b/Themis/ViewModels/Course/CourseViewModel.swift index ccf17e81..e34580bb 100644 --- a/Themis/ViewModels/Course/CourseViewModel.swift +++ b/Themis/ViewModels/Course/CourseViewModel.swift @@ -80,8 +80,7 @@ class CourseViewModel: ObservableObject { shownCourseID = pickerCourseIDs.isEmpty ? nil : pickerCourseIDs[0] } - assessableExams = shownCourse?.exams?.filter({ $0.isOver && !$0.isAssessmentDue }) ?? [] - viewOnlyExams = shownCourse?.exams?.filter({ !$0.isOver || $0.isAssessmentDue }) ?? [] + setExamsForShownCourse() } } @@ -104,16 +103,26 @@ class CourseViewModel: ObservableObject { if case .failure(let error) = courseForAssessment { log.error(String(describing: error)) + } else if let courseValueForAssessment = courseForAssessment.value { + setExercises(for: courseValueForAssessment) + setExamsForShownCourse() } - - let exercisesOfShownCourse = courseForAssessment.value?.exercises ?? [] - - assessableExercises = exercisesOfShownCourse.filter({ $0.isCurrentlyInAssessment }) - viewOnlyExercises = exercisesOfShownCourse.filter({ !$0.isCurrentlyInAssessment }) } } func courseForID(id: Int) -> Course? { courses.first { $0.id == id } } + + private func setExercises(for shownCourse: Course) { + let exercisesOfShownCourse = shownCourse.exercises ?? [] + + assessableExercises = exercisesOfShownCourse.filter({ $0.isCurrentlyInAssessment }) + viewOnlyExercises = exercisesOfShownCourse.filter({ !$0.isCurrentlyInAssessment }) + } + + private func setExamsForShownCourse() { + assessableExams = shownCourse?.exams?.filter({ $0.isOver && !$0.isAssessmentDue }) ?? [] + viewOnlyExams = shownCourse?.exams?.filter({ !$0.isOver || $0.isAssessmentDue }) ?? [] + } } From 312c3b30e58bc0b4f0ef211e6c562aec7995fda3 Mon Sep 17 00:00:00 2001 From: Tarlan Ismayilsoy Date: Mon, 14 Aug 2023 14:28:30 +0400 Subject: [PATCH 04/21] Create AthenaService; Fetch suggestions; Display special highlights for suggestions --- .../RoundedCornerLayoutManager.swift | 14 +++- .../Sources/CodeEditor/UXCodeTextView.swift | 23 +++--- Themis.xcodeproj/project.pbxproj | 32 +++++++- Themis/Extensions/ColorExtension.swift | 2 +- .../Models/{ => Programming}/Repository.swift | 0 .../SourceCodeLanguage.swift | 0 Themis/Models/Text/TextBlockRef.swift | 14 ++++ .../Services/Assessment/AthenaService.swift | 37 ++++++++++ .../TextExerciseRendererViewModel.swift | 74 ++++++++++++++++++- .../Text Exercise/TextExerciseRenderer.swift | 5 +- 10 files changed, 177 insertions(+), 24 deletions(-) rename Themis/Models/{ => Programming}/Repository.swift (100%) rename Themis/Models/{ => Programming}/SourceCodeLanguage.swift (100%) create mode 100644 Themis/Models/Text/TextBlockRef.swift create mode 100644 Themis/Services/Assessment/AthenaService.swift diff --git a/CodeEditor/Sources/CodeEditor/RoundedCornerLayoutManager.swift b/CodeEditor/Sources/CodeEditor/RoundedCornerLayoutManager.swift index d7c0a9ab..a2e96e21 100644 --- a/CodeEditor/Sources/CodeEditor/RoundedCornerLayoutManager.swift +++ b/CodeEditor/Sources/CodeEditor/RoundedCornerLayoutManager.swift @@ -61,9 +61,17 @@ class RoundedCornerLayoutManager: NSLayoutManager { lineRect.origin.y += containerOrigin.y lineRect = lineRect.integral.insetBy(dx: 0.4, dy: 0.4) - /// The roundedReect is responsible for the round Corners - let path = UIBezierPath(roundedRect: lineRect, cornerRadius: height * 0.2) - path.fill() + if underlineVal == .thick { + lineRect.origin.y += height + lineRect.size.height = 1 + + let path = UIBezierPath(roundedRect: lineRect, cornerRadius: height * 0.2) + path.lineWidth = height * 0.1 + path.stroke() + } else { + let path = UIBezierPath(roundedRect: lineRect, cornerRadius: height * 0.2) + path.fill() + } } private func numViewWidth() -> CGFloat { diff --git a/CodeEditor/Sources/CodeEditor/UXCodeTextView.swift b/CodeEditor/Sources/CodeEditor/UXCodeTextView.swift index f2c97eef..d01c0724 100644 --- a/CodeEditor/Sources/CodeEditor/UXCodeTextView.swift +++ b/CodeEditor/Sources/CodeEditor/UXCodeTextView.swift @@ -444,12 +444,7 @@ final class UXCodeTextView: UXTextView, HighlightDelegate, UIScrollViewDelegate guard text.hasRange(hRange.range) else { continue } - self.textStorage.addAttributes( - [ - .underlineStyle: NSUnderlineStyle.single.rawValue, - .underlineColor: hRange.color, - .link: hRange.id.uuidString // equals feedback id - ], range: hRange.range) + addUnderlineAttribute(for: hRange) } } } @@ -460,16 +455,20 @@ final class UXCodeTextView: UXTextView, HighlightDelegate, UIScrollViewDelegate guard text.hasRange(hRange.range) else { continue } - self.textStorage.addAttributes( - [ - .underlineStyle: NSUnderlineStyle.single.rawValue, - .underlineColor: hRange.color, - .link: hRange.id.uuidString // equals feedback id - ], range: hRange.range) + addUnderlineAttribute(for: hRange) } } } + private func addUnderlineAttribute(for highlightedRange: HighlightedRange) { + self.textStorage.addAttributes( + [ + .underlineStyle: highlightedRange.isSuggested ? NSUnderlineStyle.thick.rawValue : NSUnderlineStyle.single.rawValue, + .underlineColor: highlightedRange.color, + .link: highlightedRange.id.uuidString // equals feedback id + ], range: highlightedRange.range) + } + private func getGlyphIndex(textView: UXCodeTextView, point: CGPoint) -> Int { let point = CGPoint(x: point.x, y: point.y - (self.font?.pointSize ?? 0.0) / 2.0) diff --git a/Themis.xcodeproj/project.pbxproj b/Themis.xcodeproj/project.pbxproj index a7c64bae..d743e579 100644 --- a/Themis.xcodeproj/project.pbxproj +++ b/Themis.xcodeproj/project.pbxproj @@ -83,6 +83,8 @@ 65F019012A1CCC0300BB1C98 /* UnknownSubmissionServiceImpl.swift in Sources */ = {isa = PBXBuildFile; fileRef = 65F019002A1CCC0300BB1C98 /* UnknownSubmissionServiceImpl.swift */; }; 65F019052A1CDCE400BB1C98 /* AssessmentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 65F019042A1CDCE400BB1C98 /* AssessmentView.swift */; }; 65F019072A1CDE1200BB1C98 /* TextAssessmentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 65F019062A1CDE1200BB1C98 /* TextAssessmentView.swift */; }; + 65FAB2522A89FEC000AC533E /* AthenaService.swift in Sources */ = {isa = PBXBuildFile; fileRef = 65FAB2512A89FEC000AC533E /* AthenaService.swift */; }; + 65FAB2562A8A048F00AC533E /* TextBlockRef.swift in Sources */ = {isa = PBXBuildFile; fileRef = 65FAB2552A8A048F00AC533E /* TextBlockRef.swift */; }; 83396D2E29155935003EF727 /* ThemisApp.swift in Sources */ = {isa = PBXBuildFile; fileRef = 83396D2D29155935003EF727 /* ThemisApp.swift */; }; 83396D3029155935003EF727 /* ContentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 83396D2F29155935003EF727 /* ContentView.swift */; }; 83396D3229155935003EF727 /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 83396D3129155935003EF727 /* Assets.xcassets */; }; @@ -227,6 +229,9 @@ 65F019002A1CCC0300BB1C98 /* UnknownSubmissionServiceImpl.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UnknownSubmissionServiceImpl.swift; sourceTree = ""; }; 65F019042A1CDCE400BB1C98 /* AssessmentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AssessmentView.swift; sourceTree = ""; }; 65F019062A1CDE1200BB1C98 /* TextAssessmentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TextAssessmentView.swift; sourceTree = ""; }; + 65FAB2512A89FEC000AC533E /* AthenaService.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AthenaService.swift; sourceTree = ""; }; + 65FAB2552A8A048F00AC533E /* TextBlockRef.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TextBlockRef.swift; sourceTree = ""; }; + 65FAB2572A8A0FD800AC533E /* artemis-ios-core-modules */ = {isa = PBXFileReference; lastKnownFileType = wrapper; name = "artemis-ios-core-modules"; path = "../artemis-ios-core-modules"; sourceTree = ""; }; 83396D2A29155935003EF727 /* Themis.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = Themis.app; sourceTree = BUILT_PRODUCTS_DIR; }; 83396D2D29155935003EF727 /* ThemisApp.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ThemisApp.swift; sourceTree = ""; }; 83396D2F29155935003EF727 /* ContentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = ContentView.swift; path = Themis/Views/ContentView.swift; sourceTree = SOURCE_ROOT; }; @@ -259,7 +264,7 @@ DAFFA3FA297F066200EA72B8 /* ArtemisDateHelpers.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ArtemisDateHelpers.swift; sourceTree = ""; }; DAFFA3FC297F12A800EA72B8 /* ExerciseListItem.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ExerciseListItem.swift; sourceTree = ""; }; E119ED9A2922DE4700626DEA /* AssessmentResult.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = AssessmentResult.swift; path = Themis/Models/Result/AssessmentResult.swift; sourceTree = SOURCE_ROOT; }; - E119ED9E2922F48A00626DEA /* Repository.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = Repository.swift; path = Themis/Models/Repository.swift; sourceTree = SOURCE_ROOT; }; + E119ED9E2922F48A00626DEA /* Repository.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = Repository.swift; path = Themis/Models/Programming/Repository.swift; sourceTree = SOURCE_ROOT; }; E168C722296DCCE100A1D1D4 /* ThemisButtonStyle.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ThemisButtonStyle.swift; sourceTree = ""; }; E168C726296DDB3D00A1D1D4 /* CustomProgressView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CustomProgressView.swift; sourceTree = ""; }; E171F00D2933B5DD00236BC4 /* CodeEditorViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CodeEditorViewModel.swift; sourceTree = ""; }; @@ -343,11 +348,11 @@ 4065F854292E8729005DAEE8 /* Models */ = { isa = PBXGroup; children = ( + 65FAB2542A8A048100AC533E /* Text */, + 65FAB2532A8A047800AC533E /* Programming */, 65101DD12A33360A007B598A /* Result */, 6581C67B2A305F1D00E49E24 /* Feedback */, 4065F857292E8A08005DAEE8 /* Appearance */, - E119ED9E2922F48A00626DEA /* Repository.swift */, - E200F786297F10B300DE269D /* SourceCodeLanguage.swift */, ); name = Models; path = Themis/Models; @@ -563,6 +568,7 @@ 65E6A0FE2A13D1630058E5D6 /* ProgrammingAssessmentServiceImpl.swift */, 6595D2282A2744410033C1B5 /* TextAssessmentServiceImpl.swift */, 6595D22A2A275A740033C1B5 /* UnknownAssessmentServiceImpl.swift */, + 65FAB2512A89FEC000AC533E /* AthenaService.swift */, ); path = Assessment; sourceTree = ""; @@ -595,9 +601,27 @@ path = Modifiers; sourceTree = ""; }; + 65FAB2532A8A047800AC533E /* Programming */ = { + isa = PBXGroup; + children = ( + E119ED9E2922F48A00626DEA /* Repository.swift */, + E200F786297F10B300DE269D /* SourceCodeLanguage.swift */, + ); + path = Programming; + sourceTree = ""; + }; + 65FAB2542A8A048100AC533E /* Text */ = { + isa = PBXGroup; + children = ( + 65FAB2552A8A048F00AC533E /* TextBlockRef.swift */, + ); + path = Text; + sourceTree = ""; + }; 83396D2129155934003EF727 = { isa = PBXGroup; children = ( + 65FAB2572A8A0FD800AC533E /* artemis-ios-core-modules */, E2E3852B2943800100F5854D /* CodeEditor */, 83396D2C29155935003EF727 /* Themis */, 83396D3D29155936003EF727 /* ThemisTests */, @@ -1067,6 +1091,7 @@ F922443A297BFD1A008E4374 /* CircularProgressView.swift in Sources */, 652A98022A07D51C00EB1F32 /* ToolbarCancelButton.swift in Sources */, 6567A90329EFF85200A5EFDA /* ScorePicker.swift in Sources */, + 65FAB2522A89FEC000AC533E /* AthenaService.swift in Sources */, 65E6A0FA2A13C5C90058E5D6 /* ExamServiceImpl.swift in Sources */, 65EDED372A33103A003BF14B /* UserFacingErrorExtension.swift in Sources */, 650538BC2A1E226A00C45E18 /* MockAssessmentViewModel.swift in Sources */, @@ -1138,6 +1163,7 @@ E2E460DD29292ECF00ECC0A5 /* Stack.swift in Sources */, A9752D7B2992AAEB004441D1 /* ExamSectionDetailView.swift in Sources */, DA3F0F03294A308E00A7B807 /* EditFeedbackView.swift in Sources */, + 65FAB2562A8A048F00AC533E /* TextBlockRef.swift in Sources */, 65F018FF2A1CBD8D00BB1C98 /* TextSubmissionServiceImpl.swift in Sources */, E119ED9F2922F48A00626DEA /* Repository.swift in Sources */, 6581C6812A305F6E00E49E24 /* TextFeedbackDetail.swift in Sources */, diff --git a/Themis/Extensions/ColorExtension.swift b/Themis/Extensions/ColorExtension.swift index 81aeb65f..655e607b 100644 --- a/Themis/Extensions/ColorExtension.swift +++ b/Themis/Extensions/ColorExtension.swift @@ -25,7 +25,7 @@ extension Color { static let positiveFeedbackBackground = Color("positiveFeedbackBackground") static let positiveFeedbackPointBackground = Color("positiveFeedbackPointBackground") static let positiveFeedbackText = Color("positiveFeedbackText") - static let positiveTextHighlight = UIColor(.init(netHex: 0xB0FFB4), darkModeColor: .init(netHex: 0x48A34B)).suColor + static let positiveTextHighlight = UIColor(.init(netHex: 0x4ED955), darkModeColor: .init(netHex: 0x48A34B)).suColor // Negative Feedback static let negativeFeedbackBackground = Color("negativeFeedbackBackground") diff --git a/Themis/Models/Repository.swift b/Themis/Models/Programming/Repository.swift similarity index 100% rename from Themis/Models/Repository.swift rename to Themis/Models/Programming/Repository.swift diff --git a/Themis/Models/SourceCodeLanguage.swift b/Themis/Models/Programming/SourceCodeLanguage.swift similarity index 100% rename from Themis/Models/SourceCodeLanguage.swift rename to Themis/Models/Programming/SourceCodeLanguage.swift diff --git a/Themis/Models/Text/TextBlockRef.swift b/Themis/Models/Text/TextBlockRef.swift new file mode 100644 index 00000000..7f05b73f --- /dev/null +++ b/Themis/Models/Text/TextBlockRef.swift @@ -0,0 +1,14 @@ +// +// TextBlockRef.swift +// Themis +// +// Created by Tarlan Ismayilsoy on 14.08.23. +// + +import Foundation +import SharedModels + +struct TextBlockRef: Codable { + var block: TextBlock + var feedback: Feedback +} diff --git a/Themis/Services/Assessment/AthenaService.swift b/Themis/Services/Assessment/AthenaService.swift new file mode 100644 index 00000000..7dcb36ca --- /dev/null +++ b/Themis/Services/Assessment/AthenaService.swift @@ -0,0 +1,37 @@ +// +// AthenaService.swift +// Themis +// +// Created by Tarlan Ismayilsoy on 14.08.23. +// + +import Foundation +import SharedModels +import APIClient + +/// A service that communicates with the Athena-related endpoints of Artemis to handle automatic feedback suggestions +/// For more info about Athena: https://github.com/ls1intum/Athena +struct AthenaService { + + let client = APIClient() + + // MARK: - Get Feedback Suggestions + private struct GetFeedbackSuggestionsRequest: APIRequest { + typealias Response = [TextBlockRef] + + let exerciseId: Int + let submissionId: Int + + var method: HTTPMethod { + .get + } + + var resourceName: String { + "api/athena/exercises/\(exerciseId)/submissions/\(submissionId)/feedback-suggestions" + } + } + + func getFeedbackSuggestions(exerciseId: Int, submissionId: Int) async throws -> [TextBlockRef] { + try await client.sendRequest(GetFeedbackSuggestionsRequest(exerciseId: exerciseId, submissionId: submissionId)).get().0 + } +} diff --git a/Themis/ViewModels/Assessment/TextExerciseRendererViewModel.swift b/Themis/ViewModels/Assessment/TextExerciseRendererViewModel.swift index f2a12466..d31c31fe 100644 --- a/Themis/ViewModels/Assessment/TextExerciseRendererViewModel.swift +++ b/Themis/ViewModels/Assessment/TextExerciseRendererViewModel.swift @@ -51,7 +51,7 @@ class TextExerciseRendererViewModel: ObservableObject { /// Needed for creating new text blocks private var submissionId: Int? - + private var suggestedRefs = [TextBlockRef]() /// Sets this VM up based on the given participation and optional submission /// - Parameters: @@ -77,6 +77,59 @@ class TextExerciseRendererViewModel: ObservableObject { submissionId = textSubmission.id content = textSubmission.text ?? content setupHighlights(basedOn: blocks, and: feedbacks) + + if let participation { + fetchSuggestions(for: textSubmission, participation) + } + } + + // TODO: make sure this is not called for read-only and finished submissions + private func fetchSuggestions(for textSubmission: TextSubmission, _ participation: BaseParticipation) { + guard let exerciseId = participation.exercise?.id, + let submissionId = textSubmission.id else { + log.error("Could not fetch suggestions for submission: #\(submissionId ?? -1)") + return + } + + Task { [weak self] in + do { + var blockRefs = try await AthenaService().getFeedbackSuggestions(exerciseId: exerciseId, submissionId: submissionId) + log.verbose("Fetched \(blockRefs.count) suggestions") + + blockRefs = self?.removeOverlappingRefs(blockRefs) ?? [] + await self?.setupHighlights(basedOn: blockRefs) + self?.suggestedRefs = blockRefs + } catch { + log.error(String(describing: error)) + } + } + } + + private func removeOverlappingRefs(_ blockRefs: [TextBlockRef]) -> [TextBlockRef] { + var result = [TextBlockRef]() + + for blockRef in blockRefs { + guard let startIndex = blockRef.block.startIndex, + let endIndex = blockRef.block.endIndex else { + continue + } + let blockRefRange = startIndex.. Date: Mon, 14 Aug 2023 14:35:51 +0400 Subject: [PATCH 05/21] Make overlap calculation more efficient --- .../TextExerciseRendererViewModel.swift | 20 +++++-------------- 1 file changed, 5 insertions(+), 15 deletions(-) diff --git a/Themis/ViewModels/Assessment/TextExerciseRendererViewModel.swift b/Themis/ViewModels/Assessment/TextExerciseRendererViewModel.swift index d31c31fe..ac0a623c 100644 --- a/Themis/ViewModels/Assessment/TextExerciseRendererViewModel.swift +++ b/Themis/ViewModels/Assessment/TextExerciseRendererViewModel.swift @@ -106,7 +106,7 @@ class TextExerciseRendererViewModel: ObservableObject { } private func removeOverlappingRefs(_ blockRefs: [TextBlockRef]) -> [TextBlockRef] { - var result = [TextBlockRef]() + var rangeToBlockRef = [Range: TextBlockRef]() for blockRef in blockRefs { guard let startIndex = blockRef.block.startIndex, @@ -114,21 +114,11 @@ class TextExerciseRendererViewModel: ObservableObject { continue } let blockRefRange = startIndex.. Date: Sun, 27 Aug 2023 12:53:07 +0400 Subject: [PATCH 06/21] Fix SwiftLint error --- Themis.xcodeproj/project.pbxproj | 2 +- Themis/Views/Courses/CourseView.swift | 41 ++++++++++++++------------- 2 files changed, 23 insertions(+), 20 deletions(-) diff --git a/Themis.xcodeproj/project.pbxproj b/Themis.xcodeproj/project.pbxproj index a7c64bae..b1bd49f9 100644 --- a/Themis.xcodeproj/project.pbxproj +++ b/Themis.xcodeproj/project.pbxproj @@ -1032,7 +1032,7 @@ ); runOnlyForDeploymentPostprocessing = 0; shellPath = /bin/sh; - shellScript = "if [ \"${CONFIGURATION}\" == \"Debug\" ]; then\n export PATH=\"$PATH:/opt/homebrew/bin\"\n if which swiftlint > /dev/null; then\n swiftlint --config swiftlint.yml && swiftlint --config swiftlint.yml\n else\n echo \"warning: SwiftLint not installed, download from https://github.com/realm/SwiftLint\"\n fi\nfi\n"; + shellScript = "if [ \"${CONFIGURATION}\" == \"Debug\" ]; then\n export PATH=\"$PATH:/opt/homebrew/bin\"\n if which swiftlint > /dev/null; then\n swiftlint --config swiftlint.yml\n else\n echo \"warning: SwiftLint not installed, download from https://github.com/realm/SwiftLint\"\n fi\nfi\n"; }; /* End PBXShellScriptBuildPhase section */ diff --git a/Themis/Views/Courses/CourseView.swift b/Themis/Views/Courses/CourseView.swift index d9dfda4c..a8099539 100644 --- a/Themis/Views/Courses/CourseView.swift +++ b/Themis/Views/Courses/CourseView.swift @@ -44,25 +44,7 @@ struct CourseView: View { } } .navigationTitle(navTitle) - .toolbar { - ToolbarItemGroup(placement: .cancellationAction) { - logoutButton - userFirstName - } - ToolbarItem(placement: .primaryAction) { - Picker("", selection: $courseVM.shownCourseID) { - ForEach(courseVM.pickerCourseIDs, id: \.self) { courseID in - if let courseID { - Text(courseVM.courseForID(id: courseID)?.title ?? "Invalid") - .padding(.leading, 40) - } - } - } - .onChange(of: courseVM.shownCourseID, perform: { _ in courseVM.fetchShownCourseAndSetExercises() }) - .isHidden(courseVM.showEmptyMessage) - .padding(-10) // compensates for Picker's default padding - } - } + .toolbar(content: buildToolbar) } .task { courseVM.fetchAllCourses() @@ -70,6 +52,27 @@ struct CourseView: View { } .errorAlert(error: $courseVM.error) } + + @ToolbarContentBuilder + private func buildToolbar() -> some ToolbarContent { + ToolbarItemGroup(placement: .cancellationAction) { + logoutButton + userFirstName + } + ToolbarItem(placement: .primaryAction) { + Picker("", selection: $courseVM.shownCourseID) { + ForEach(courseVM.pickerCourseIDs, id: \.self) { courseID in + if let courseID { + Text(courseVM.courseForID(id: courseID)?.title ?? "Invalid") + .padding(.leading, 40) + } + } + } + .onChange(of: courseVM.shownCourseID, perform: { _ in courseVM.fetchShownCourseAndSetExercises() }) + .isHidden(courseVM.showEmptyMessage) + .padding(-10) // compensates for Picker's default padding + } + } private var logoutButton: some View { Button { From b282470e1828ffee4a8bde36c540ecd8cffdf4e2 Mon Sep 17 00:00:00 2001 From: Tarlan Ismayilsoy Date: Tue, 29 Aug 2023 13:12:20 +0400 Subject: [PATCH 07/21] Create FeedbackSuggestion subtypes; Open sheet for adding suggested feedback --- CodeEditor/Package.swift | 4 +- .../Suggestion/FeedbackSuggestion.swift | 20 ++++++++ .../ProgrammingFeedbackSuggestion.swift} | 5 +- .../Suggestion/TextFeedbackSuggestion.swift | 33 ++++++++++++ .../CodeEditor/Models/TextBlockRef.swift | 20 ++++++++ .../RoundedCornerLayoutManager.swift | 10 ++-- .../Sources/CodeEditor/UXCodeTextView.swift | 2 +- .../CodeEditor/Utils/EditorBindings.swift | 4 +- Themis.xcodeproj/project.pbxproj | 12 ----- Themis/API/ThemisAPI.swift | 4 +- .../Models/Feedback/AssessmentFeedback.swift | 20 ++++++++ Themis/Models/Text/TextBlockRef.swift | 14 ----- .../Services/Assessment/AthenaService.swift | 1 + .../Assessment/CodeEditorViewModel.swift | 11 ++-- .../TextExerciseRendererViewModel.swift | 51 +++++++++++++++---- .../EditFeedback/AddFeedbackView.swift | 2 +- .../EditFeedback/EditFeedbackViewBase.swift | 18 ++----- .../Views/Assessment/FeedbackDelegate.swift | 6 +-- .../Views/Assessment/TextAssessmentView.swift | 14 +++++ 19 files changed, 179 insertions(+), 72 deletions(-) create mode 100644 CodeEditor/Sources/CodeEditor/Models/Suggestion/FeedbackSuggestion.swift rename CodeEditor/Sources/CodeEditor/{Utils/FeedbackSuggestion.swift => Models/Suggestion/ProgrammingFeedbackSuggestion.swift} (90%) create mode 100644 CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift create mode 100644 CodeEditor/Sources/CodeEditor/Models/TextBlockRef.swift delete mode 100644 Themis/Models/Text/TextBlockRef.swift diff --git a/CodeEditor/Package.swift b/CodeEditor/Package.swift index aeb8d765..d057c287 100644 --- a/CodeEditor/Package.swift +++ b/CodeEditor/Package.swift @@ -1,4 +1,4 @@ -// swift-tools-version:5.3 +// swift-tools-version:5.7 import PackageDescription @@ -7,7 +7,7 @@ let package = Package( name: "CodeEditor", platforms: [ - .macOS(.v10_15), .iOS(.v13) + .macOS(.v10_15), .iOS(.v16) ], products: [ diff --git a/CodeEditor/Sources/CodeEditor/Models/Suggestion/FeedbackSuggestion.swift b/CodeEditor/Sources/CodeEditor/Models/Suggestion/FeedbackSuggestion.swift new file mode 100644 index 00000000..e16bed58 --- /dev/null +++ b/CodeEditor/Sources/CodeEditor/Models/Suggestion/FeedbackSuggestion.swift @@ -0,0 +1,20 @@ +// +// FeedbackSuggestion.swift +// +// +// Created by Tarlan Ismayilsoy on 29.08.23. +// + +import Foundation +import SharedModels + +public protocol FeedbackSuggestion: Equatable { + var id: UUID { get } + var text: String { get } + var credits: Double { get } +} + +public extension FeedbackSuggestion { + var text: String { "" } + var credits: Double { 0.0 } +} diff --git a/CodeEditor/Sources/CodeEditor/Utils/FeedbackSuggestion.swift b/CodeEditor/Sources/CodeEditor/Models/Suggestion/ProgrammingFeedbackSuggestion.swift similarity index 90% rename from CodeEditor/Sources/CodeEditor/Utils/FeedbackSuggestion.swift rename to CodeEditor/Sources/CodeEditor/Models/Suggestion/ProgrammingFeedbackSuggestion.swift index 1fe1f7c2..aa53006a 100644 --- a/CodeEditor/Sources/CodeEditor/Utils/FeedbackSuggestion.swift +++ b/CodeEditor/Sources/CodeEditor/Models/Suggestion/ProgrammingFeedbackSuggestion.swift @@ -1,13 +1,14 @@ // -// FeedbackSuggestion.swift +// ProgrammingFeedbackSuggestion.swift // Themis // // Created by Andreas Cselovszky on 25.01.23. // import Foundation +import SharedModels -public struct FeedbackSuggestion: Decodable, Equatable { +public struct ProgrammingFeedbackSuggestion: FeedbackSuggestion, Decodable { public let id = UUID() public let exerciseId: Int public let participationId: Int diff --git a/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift b/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift new file mode 100644 index 00000000..4e4344fb --- /dev/null +++ b/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift @@ -0,0 +1,33 @@ +// +// TextFeedbackSuggestion.swift +// +// +// Created by Tarlan Ismayilsoy on 29.08.23. +// + +import Foundation +import SharedModels + +public struct TextFeedbackSuggestion: FeedbackSuggestion { + public let blockRef: TextBlockRef + + public var id: UUID { + blockRef.id + } + + public var text: String { + blockRef.feedback.detailText ?? "" + } + + public var credits: Double { + blockRef.feedback.credits ?? 0.0 + } + + public init(blockRef: TextBlockRef) { + self.blockRef = blockRef + } + + public static func == (lhs: TextFeedbackSuggestion, rhs: TextFeedbackSuggestion) -> Bool { + lhs.id == rhs.id && lhs.blockRef.id == rhs.blockRef.id + } +} diff --git a/CodeEditor/Sources/CodeEditor/Models/TextBlockRef.swift b/CodeEditor/Sources/CodeEditor/Models/TextBlockRef.swift new file mode 100644 index 00000000..3dc10ed2 --- /dev/null +++ b/CodeEditor/Sources/CodeEditor/Models/TextBlockRef.swift @@ -0,0 +1,20 @@ +// +// TextBlockRef.swift +// Themis +// +// Created by Tarlan Ismayilsoy on 14.08.23. +// + +import Foundation +import SharedModels + +public struct TextBlockRef: Codable { + public var block: TextBlock + public var feedback: Feedback + + private enum CodingKeys: String, CodingKey { + case block, feedback + } + + public let id = UUID() // not decoded +} diff --git a/CodeEditor/Sources/CodeEditor/RoundedCornerLayoutManager.swift b/CodeEditor/Sources/CodeEditor/RoundedCornerLayoutManager.swift index a2e96e21..70eed2bf 100644 --- a/CodeEditor/Sources/CodeEditor/RoundedCornerLayoutManager.swift +++ b/CodeEditor/Sources/CodeEditor/RoundedCornerLayoutManager.swift @@ -16,7 +16,7 @@ class RoundedCornerLayoutManager: NSLayoutManager { let paraStyle = NSMutableParagraphStyle() var diffLines = [Int]() var isNewFile = false - var feedbackSuggestions = [FeedbackSuggestion]() + var feedbackSuggestions = [any FeedbackSuggestion]() override init() { super.init() @@ -156,7 +156,7 @@ class RoundedCornerLayoutManager: NSLayoutManager { } } self.drawDiffLines(paraNumber, rect, origin) - self.drawFeedbackSuggestions(paraNumber, rect, origin) + self.drawProgrammingFeedbackSuggestions(paraNumber, rect, origin) } UIGraphicsPopContext() @@ -198,13 +198,15 @@ class RoundedCornerLayoutManager: NSLayoutManager { UIGraphicsPopContext() } - private func drawFeedbackSuggestions(_ paraNumber: Int, _ rect: CGRect, _ origin: CGPoint) { + private func drawProgrammingFeedbackSuggestions(_ paraNumber: Int, _ rect: CGRect, _ origin: CGPoint) { let ctx = UIGraphicsGetCurrentContext() guard let ctx else { return } UIGraphicsPushContext(ctx) ctx.setFillColor(CGColor(red: 0, green: 0.2, blue: 0.8, alpha: 0.8)) ctx.setStrokeColor(CGColor(red: 0, green: 0.2, blue: 0.8, alpha: 0.8)) - if self.feedbackSuggestions.contains(where: { paraNumber + 1 >= $0.fromLine && paraNumber + 1 <= $0.toLine }) { + + let programmingSuggestions = feedbackSuggestions.compactMap({ $0 as? ProgrammingFeedbackSuggestion }) + if programmingSuggestions.contains(where: { paraNumber + 1 >= $0.fromLine && paraNumber + 1 <= $0.toLine }) { let path = CGPath( rect: CGRect( x: rect.origin.x, diff --git a/CodeEditor/Sources/CodeEditor/UXCodeTextView.swift b/CodeEditor/Sources/CodeEditor/UXCodeTextView.swift index d01c0724..c65546f9 100644 --- a/CodeEditor/Sources/CodeEditor/UXCodeTextView.swift +++ b/CodeEditor/Sources/CodeEditor/UXCodeTextView.swift @@ -62,7 +62,7 @@ final class UXCodeTextView: UXTextView, HighlightDelegate, UIScrollViewDelegate var highlightedRanges: [HighlightedRange] = [] var dragSelection: Range? - var feedbackSuggestions: [FeedbackSuggestion] = [] + var feedbackSuggestions: [ProgrammingFeedbackSuggestion] = [] var showsLineNumbers: Bool private var firstPoint: CGPoint? diff --git a/CodeEditor/Sources/CodeEditor/Utils/EditorBindings.swift b/CodeEditor/Sources/CodeEditor/Utils/EditorBindings.swift index 590c7792..07621384 100644 --- a/CodeEditor/Sources/CodeEditor/Utils/EditorBindings.swift +++ b/CodeEditor/Sources/CodeEditor/Utils/EditorBindings.swift @@ -24,7 +24,7 @@ public struct EditorBindings { public var diffLines: [Int] public var isNewFile: Bool public var showsLineNumbers: Bool - public var feedbackSuggestions: [FeedbackSuggestion] + public var feedbackSuggestions: [ProgrammingFeedbackSuggestion] public var selectedFeedbackSuggestionId: Binding public init(source: Binding, @@ -48,7 +48,7 @@ public struct EditorBindings { diffLines: [Int] = [], isNewFile: Bool = false, showsLineNumbers: Bool = true, - feedbackSuggestions: [FeedbackSuggestion], + feedbackSuggestions: [ProgrammingFeedbackSuggestion], selectedFeedbackSuggestionId: Binding ) { self.source = source diff --git a/Themis.xcodeproj/project.pbxproj b/Themis.xcodeproj/project.pbxproj index d743e579..ceccd5c0 100644 --- a/Themis.xcodeproj/project.pbxproj +++ b/Themis.xcodeproj/project.pbxproj @@ -84,7 +84,6 @@ 65F019052A1CDCE400BB1C98 /* AssessmentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 65F019042A1CDCE400BB1C98 /* AssessmentView.swift */; }; 65F019072A1CDE1200BB1C98 /* TextAssessmentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 65F019062A1CDE1200BB1C98 /* TextAssessmentView.swift */; }; 65FAB2522A89FEC000AC533E /* AthenaService.swift in Sources */ = {isa = PBXBuildFile; fileRef = 65FAB2512A89FEC000AC533E /* AthenaService.swift */; }; - 65FAB2562A8A048F00AC533E /* TextBlockRef.swift in Sources */ = {isa = PBXBuildFile; fileRef = 65FAB2552A8A048F00AC533E /* TextBlockRef.swift */; }; 83396D2E29155935003EF727 /* ThemisApp.swift in Sources */ = {isa = PBXBuildFile; fileRef = 83396D2D29155935003EF727 /* ThemisApp.swift */; }; 83396D3029155935003EF727 /* ContentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 83396D2F29155935003EF727 /* ContentView.swift */; }; 83396D3229155935003EF727 /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 83396D3129155935003EF727 /* Assets.xcassets */; }; @@ -230,7 +229,6 @@ 65F019042A1CDCE400BB1C98 /* AssessmentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AssessmentView.swift; sourceTree = ""; }; 65F019062A1CDE1200BB1C98 /* TextAssessmentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TextAssessmentView.swift; sourceTree = ""; }; 65FAB2512A89FEC000AC533E /* AthenaService.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AthenaService.swift; sourceTree = ""; }; - 65FAB2552A8A048F00AC533E /* TextBlockRef.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TextBlockRef.swift; sourceTree = ""; }; 65FAB2572A8A0FD800AC533E /* artemis-ios-core-modules */ = {isa = PBXFileReference; lastKnownFileType = wrapper; name = "artemis-ios-core-modules"; path = "../artemis-ios-core-modules"; sourceTree = ""; }; 83396D2A29155935003EF727 /* Themis.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = Themis.app; sourceTree = BUILT_PRODUCTS_DIR; }; 83396D2D29155935003EF727 /* ThemisApp.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ThemisApp.swift; sourceTree = ""; }; @@ -348,7 +346,6 @@ 4065F854292E8729005DAEE8 /* Models */ = { isa = PBXGroup; children = ( - 65FAB2542A8A048100AC533E /* Text */, 65FAB2532A8A047800AC533E /* Programming */, 65101DD12A33360A007B598A /* Result */, 6581C67B2A305F1D00E49E24 /* Feedback */, @@ -610,14 +607,6 @@ path = Programming; sourceTree = ""; }; - 65FAB2542A8A048100AC533E /* Text */ = { - isa = PBXGroup; - children = ( - 65FAB2552A8A048F00AC533E /* TextBlockRef.swift */, - ); - path = Text; - sourceTree = ""; - }; 83396D2129155934003EF727 = { isa = PBXGroup; children = ( @@ -1163,7 +1152,6 @@ E2E460DD29292ECF00ECC0A5 /* Stack.swift in Sources */, A9752D7B2992AAEB004441D1 /* ExamSectionDetailView.swift in Sources */, DA3F0F03294A308E00A7B807 /* EditFeedbackView.swift in Sources */, - 65FAB2562A8A048F00AC533E /* TextBlockRef.swift in Sources */, 65F018FF2A1CBD8D00BB1C98 /* TextSubmissionServiceImpl.swift in Sources */, E119ED9F2922F48A00626DEA /* Repository.swift in Sources */, 6581C6812A305F6E00E49E24 /* TextFeedbackDetail.swift in Sources */, diff --git a/Themis/API/ThemisAPI.swift b/Themis/API/ThemisAPI.swift index f525ed02..2d75a83b 100644 --- a/Themis/API/ThemisAPI.swift +++ b/Themis/API/ThemisAPI.swift @@ -69,7 +69,7 @@ extension ThemisAPI { } /// gets a feedback suggestion for a submission from Themis-ML - static func getFeedbackSuggestions(exerciseId: Int, participationId: Int) async throws -> [FeedbackSuggestion] { + static func getFeedbackSuggestions(exerciseId: Int, participationId: Int) async throws -> [ProgrammingFeedbackSuggestion] { let request = Request( method: .post, path: "/feedback_suggestions", @@ -79,7 +79,7 @@ extension ThemisAPI { participation_id: participationId ) ) - return try await sendRequest([FeedbackSuggestion].self, request: request) + return try await sendRequest([ProgrammingFeedbackSuggestion].self, request: request) } private static func removeTrailingSlash(from string: String) -> String { diff --git a/Themis/Models/Feedback/AssessmentFeedback.swift b/Themis/Models/Feedback/AssessmentFeedback.swift index 1d6335de..52dbddfd 100644 --- a/Themis/Models/Feedback/AssessmentFeedback.swift +++ b/Themis/Models/Feedback/AssessmentFeedback.swift @@ -7,6 +7,7 @@ import Foundation import SharedModels +import CodeEditor enum ThemisFeedbackScope { case inline @@ -62,3 +63,22 @@ extension AssessmentFeedback: Equatable, Hashable { hasher.combine(scope) } } + +extension AssessmentFeedback { + init(basedOn suggestion: any FeedbackSuggestion, _ incompleteFeedbackDetail: FeedbackDetail?) { + var newIncompleteFeedbackDetail = incompleteFeedbackDetail + + if var incompleteFeedbackDetail = incompleteFeedbackDetail as? ProgrammingFeedbackDetail, + let codeSuggestion = suggestion as? ProgrammingFeedbackSuggestion { + let lines = NSRange(location: codeSuggestion.fromLine, length: codeSuggestion.toLine - codeSuggestion.fromLine) + incompleteFeedbackDetail.lines = lines + newIncompleteFeedbackDetail = incompleteFeedbackDetail + } + + self.init(baseFeedback: Feedback(detailText: suggestion.text, + credits: suggestion.credits, + type: .MANUAL_UNREFERENCED), + scope: .inline, + detail: newIncompleteFeedbackDetail) + } +} diff --git a/Themis/Models/Text/TextBlockRef.swift b/Themis/Models/Text/TextBlockRef.swift deleted file mode 100644 index 7f05b73f..00000000 --- a/Themis/Models/Text/TextBlockRef.swift +++ /dev/null @@ -1,14 +0,0 @@ -// -// TextBlockRef.swift -// Themis -// -// Created by Tarlan Ismayilsoy on 14.08.23. -// - -import Foundation -import SharedModels - -struct TextBlockRef: Codable { - var block: TextBlock - var feedback: Feedback -} diff --git a/Themis/Services/Assessment/AthenaService.swift b/Themis/Services/Assessment/AthenaService.swift index 7dcb36ca..099c910d 100644 --- a/Themis/Services/Assessment/AthenaService.swift +++ b/Themis/Services/Assessment/AthenaService.swift @@ -8,6 +8,7 @@ import Foundation import SharedModels import APIClient +import CodeEditor /// A service that communicates with the Athena-related endpoints of Artemis to handle automatic feedback suggestions /// For more info about Athena: https://github.com/ls1intum/Athena diff --git a/Themis/ViewModels/Assessment/CodeEditorViewModel.swift b/Themis/ViewModels/Assessment/CodeEditorViewModel.swift index 5b8fcf59..6df265a5 100644 --- a/Themis/ViewModels/Assessment/CodeEditorViewModel.swift +++ b/Themis/ViewModels/Assessment/CodeEditorViewModel.swift @@ -32,7 +32,7 @@ class CodeEditorViewModel: ObservableObject { @Published var allowsInlineFeedbackOperations = true @Published var selectedFeedbackForEditingId = UUID() @Published var error: Error? - @Published var feedbackSuggestions = [FeedbackSuggestion]() + @Published var feedbackSuggestions = [ProgrammingFeedbackSuggestion]() @Published var selectedFeedbackSuggestionId = "" var scrollUtils = ScrollUtils(range: nil, offsets: [:]) @@ -54,7 +54,7 @@ class CodeEditorViewModel: ObservableObject { return nil } - var selectedFeedbackSuggestion: FeedbackSuggestion? { + var selectedFeedbackSuggestion: ProgrammingFeedbackSuggestion? { feedbackSuggestions.first { "\($0.id)" == selectedFeedbackSuggestionId } } @@ -122,7 +122,7 @@ class CodeEditorViewModel: ObservableObject { } @MainActor - func addFeedbackSuggestionInlineHighlight(feedbackSuggestion: FeedbackSuggestion, feedbackId: UUID) { + func addFeedbackSuggestionInlineHighlight(feedbackSuggestion: ProgrammingFeedbackSuggestion, feedbackId: UUID) { if let file = selectedFile, let code = file.code { guard let range = getLineRange(text: code, fromLine: feedbackSuggestion.fromLine, toLine: feedbackSuggestion.toLine) else { return @@ -311,7 +311,10 @@ extension CodeEditorViewModel: FeedbackDelegate { } @MainActor - func onFeedbackSuggestionSelection(_ suggestion: FeedbackSuggestion, _ feedback: AssessmentFeedback) { + func onFeedbackSuggestionSelection(_ suggestion: any FeedbackSuggestion, _ feedback: AssessmentFeedback) { + guard let suggestion = suggestion as? ProgrammingFeedbackSuggestion else { + return + } addFeedbackSuggestionInlineHighlight(feedbackSuggestion: suggestion, feedbackId: feedback.id) } diff --git a/Themis/ViewModels/Assessment/TextExerciseRendererViewModel.swift b/Themis/ViewModels/Assessment/TextExerciseRendererViewModel.swift index ac0a623c..d33c4c42 100644 --- a/Themis/ViewModels/Assessment/TextExerciseRendererViewModel.swift +++ b/Themis/ViewModels/Assessment/TextExerciseRendererViewModel.swift @@ -108,6 +108,7 @@ class TextExerciseRendererViewModel: ObservableObject { private func removeOverlappingRefs(_ blockRefs: [TextBlockRef]) -> [TextBlockRef] { var rangeToBlockRef = [Range: TextBlockRef]() + // Remove block references overlapping among themselves for blockRef in blockRefs { guard let startIndex = blockRef.block.startIndex, let endIndex = blockRef.block.endIndex else { @@ -117,11 +118,38 @@ class TextExerciseRendererViewModel: ObservableObject { rangeToBlockRef[blockRefRange] = blockRef } + // Remove block references overlapping with existing inline highlights added by the user + for highlight in inlineHighlights { + for blockRefRange in rangeToBlockRef.keys { + if let existingManualHighlightRange = Range(highlight.range), + blockRefRange.overlaps(existingManualHighlightRange) { + rangeToBlockRef.removeValue(forKey: blockRefRange) + } + } + } + let result = Array(rangeToBlockRef.values) log.verbose("\(result.count) suggestions are remaining after removing overlaps") return result } + func getSuggestion(byId id: UUID) -> TextFeedbackSuggestion? { + guard let blockRef = suggestedRefs.first(where: { $0.id == id }) else { + return nil + } + return TextFeedbackSuggestion(blockRef: blockRef) + } + + /// Generates a `TextFeedbackDetail` instance based on the available data. Some fields might be missing + func generateIncompleteFeedbackDetail() -> TextFeedbackDetail { + let block = TextBlock(submissionId: submissionId, + text: textAtSelectedSection, + startIndex: selectedSection?.lowerBound, + endIndex: selectedSection?.upperBound) + return TextFeedbackDetail(block: block) + } + + // MARK: - Highlight-related code private func setupHighlights(basedOn blocks: [TextBlock], and feedbacks: [AssessmentFeedback], shouldWipeUndo: Bool = true) { blocks.forEach { block in guard let startIndex = block.startIndex, @@ -153,15 +181,15 @@ class TextExerciseRendererViewModel: ObservableObject { let endIndex = block.endIndex else { continue } - // TODO: create an AssessmentFeedback + let range = NSRange(startIndex.. TextFeedbackDetail { - let block = TextBlock(submissionId: submissionId, - text: textAtSelectedSection, - startIndex: selectedSection?.lowerBound, - endIndex: selectedSection?.upperBound) - return TextFeedbackDetail(block: block) + private func deleteHighlight(for suggestion: TextFeedbackSuggestion) { + inlineHighlights.removeAll(where: { $0.id == suggestion.id }) } } @@ -211,4 +234,12 @@ extension TextExerciseRendererViewModel: FeedbackDelegate { func onFeedbackDeletion(_ feedback: AssessmentFeedback) { deleteHighlight(for: feedback) } + + func onFeedbackSuggestionSelection(_ suggestion: any FeedbackSuggestion, _ feedback: AssessmentFeedback) { + guard let suggestion = suggestion as? TextFeedbackSuggestion else { + return + } + deleteHighlight(for: suggestion) + createHighlight(for: feedback) + } } diff --git a/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/AddFeedbackView.swift b/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/AddFeedbackView.swift index c898d21f..862e183a 100644 --- a/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/AddFeedbackView.swift +++ b/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/AddFeedbackView.swift @@ -13,7 +13,7 @@ struct AddFeedbackView: View { var assessmentResult: AssessmentResult weak var feedbackDelegate: (any FeedbackDelegate)? var incompleteFeedback: AssessmentFeedback? - var feedbackSuggestion: FeedbackSuggestion? + var feedbackSuggestion: (any FeedbackSuggestion)? let scope: ThemisFeedbackScope let gradingCriteria: [GradingCriterion] diff --git a/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/EditFeedbackViewBase.swift b/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/EditFeedbackViewBase.swift index a9b0f045..73c8870c 100644 --- a/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/EditFeedbackViewBase.swift +++ b/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/EditFeedbackViewBase.swift @@ -15,7 +15,7 @@ struct EditFeedbackViewBase: View { weak var feedbackDelegate: (any FeedbackDelegate)? var idForUpdate: UUID? var incompleteFeedback: AssessmentFeedback? - var feedbackSuggestion: FeedbackSuggestion? + var feedbackSuggestion: (any FeedbackSuggestion)? let title: String? let isEditing: Bool @@ -139,20 +139,8 @@ struct EditFeedbackViewBase: View { feedbackDelegate?.onFeedbackDeletion(feedback) } - private func addFeedbackSuggestionToFeedbacks(feedbackSuggestion: FeedbackSuggestion) { - guard var incompleteFeedbackDetail = incompleteFeedback?.detail as? ProgrammingFeedbackDetail else { - return - } - - let lines = NSRange(location: feedbackSuggestion.fromLine, length: feedbackSuggestion.toLine - feedbackSuggestion.fromLine) - incompleteFeedbackDetail.lines = lines - - let feedback = AssessmentFeedback( - baseFeedback: Feedback(detailText: feedbackSuggestion.text, - credits: feedbackSuggestion.credits, - type: .MANUAL_UNREFERENCED), - scope: .inline, - detail: incompleteFeedbackDetail) + private func addFeedbackSuggestionToFeedbacks(feedbackSuggestion: any FeedbackSuggestion) { + let feedback = AssessmentFeedback(basedOn: feedbackSuggestion, incompleteFeedback?.detail) assessmentResult.addFeedback(feedback: feedback) feedbackDelegate?.onFeedbackSuggestionSelection(feedbackSuggestion, feedback) diff --git a/Themis/Views/Assessment/FeedbackDelegate.swift b/Themis/Views/Assessment/FeedbackDelegate.swift index 418f4868..6a9cc3f7 100644 --- a/Themis/Views/Assessment/FeedbackDelegate.swift +++ b/Themis/Views/Assessment/FeedbackDelegate.swift @@ -13,7 +13,7 @@ protocol FeedbackDelegate: ObservableObject, AnyObject { func onFeedbackCreation(_ feedback: AssessmentFeedback) func onFeedbackUpdate(_ feedback: AssessmentFeedback) func onFeedbackDeletion(_ feedback: AssessmentFeedback) - func onFeedbackSuggestionSelection(_ suggestion: FeedbackSuggestion, _ feedback: AssessmentFeedback) + func onFeedbackSuggestionSelection(_ suggestion: any FeedbackSuggestion, _ feedback: AssessmentFeedback) func onFeedbackCellTap(_ feedback: AssessmentFeedback, participationId: Int?, templateParticipationId: Int?) } @@ -24,8 +24,8 @@ extension FeedbackDelegate { func onFeedbackUpdate(_ feedback: AssessmentFeedback) {} func onFeedbackDeletion(_ feedback: AssessmentFeedback) {} - - func onFeedbackSuggestionSelection(_ suggestion: FeedbackSuggestion, _ feedback: AssessmentFeedback) {} + + func onFeedbackSuggestionSelection(_ suggestion: any FeedbackSuggestion, _ feedback: AssessmentFeedback) {} func onFeedbackCellTap(_ feedback: AssessmentFeedback, participationId: Int?, templateParticipationId: Int?) {} } diff --git a/Themis/Views/Assessment/TextAssessmentView.swift b/Themis/Views/Assessment/TextAssessmentView.swift index 156488b9..e65643d1 100644 --- a/Themis/Views/Assessment/TextAssessmentView.swift +++ b/Themis/Views/Assessment/TextAssessmentView.swift @@ -7,6 +7,7 @@ import SwiftUI import SharedModels +import CodeEditor struct TextAssessmentView: View { @ObservedObject var assessmentVM: AssessmentViewModel @@ -64,6 +65,7 @@ struct TextAssessmentView: View { }) .sheet(isPresented: $textExerciseRendererVM.showEditFeedback) { if let feedback = assessmentVM.getFeedback(byId: textExerciseRendererVM.selectedFeedbackForEditingId) { + // The user tapped on a manual feedback to edit EditFeedbackView( assessmentResult: assessmentVM.assessmentResult, feedbackDelegate: textExerciseRendererVM, @@ -72,6 +74,18 @@ struct TextAssessmentView: View { gradingCriteria: assessmentVM.gradingCriteria, showSheet: $textExerciseRendererVM.showEditFeedback ) + } else if let suggestion = textExerciseRendererVM.getSuggestion(byId: textExerciseRendererVM.selectedFeedbackForEditingId) { + // The user tapped on a feedback suggestion. This is not actually an edit action, but is triggered as one + AddFeedbackView( + assessmentResult: assessmentVM.assessmentResult, + feedbackDelegate: textExerciseRendererVM, + incompleteFeedback: AssessmentFeedback(scope: .inline, + detail: TextFeedbackDetail(block: suggestion.blockRef.block)), + feedbackSuggestion: suggestion, + scope: .inline, + gradingCriteria: assessmentVM.gradingCriteria, + showSheet: $textExerciseRendererVM.showEditFeedback + ) } } .onChange(of: assessmentVM.fontSize, perform: { textExerciseRendererVM.fontSize = $0 }) From aad55cd2c4422c7b18416b3d819785bb9a26d24f Mon Sep 17 00:00:00 2001 From: Tarlan Ismayilsoy Date: Tue, 29 Aug 2023 14:03:13 +0400 Subject: [PATCH 08/21] Make suggested feedbacks deletable; Add robot icon to suggested feedback review sheet --- .../Robot.imageset/Contents.json | 12 +++++++++ .../Robot.imageset/robot-solid.png | Bin 0 -> 5840 bytes Themis/Assets.xcassets/colors/Contents.json | 6 +++++ .../Contents.json | 0 .../Contents.json | 0 .../Contents.json | 0 .../Contents.json | 0 .../Contents.json | 0 .../Contents.json | 0 .../Contents.json | 0 .../Contents.json | 0 .../Contents.json | 0 .../Contents.json | 12 ++++----- .../sidebarBackground.colorset/Contents.json | 12 ++++----- .../themisBackground.colorset/Contents.json | 12 ++++----- .../themisDarkRed.colorset/Contents.json | 0 .../themisGreen.colorset/Contents.json | 6 ++--- .../themisPrimary.colorset/Contents.json | 6 ++--- .../themisRed.colorset/Contents.json | 0 .../themisSecondary.colorset/Contents.json | 0 .../TextExerciseRendererViewModel.swift | 7 +++++ .../EditFeedback/AddFeedbackView.swift | 3 ++- .../EditFeedback/EditFeedbackViewBase.swift | 24 ++++++++++++++---- .../Views/Assessment/FeedbackDelegate.swift | 3 +++ 24 files changed, 73 insertions(+), 30 deletions(-) create mode 100644 Themis/Assets.xcassets/Robot.imageset/Contents.json create mode 100644 Themis/Assets.xcassets/Robot.imageset/robot-solid.png create mode 100644 Themis/Assets.xcassets/colors/Contents.json rename Themis/Assets.xcassets/{ => colors}/negativeFeedbackBackground.colorset/Contents.json (100%) rename Themis/Assets.xcassets/{ => colors}/negativeFeedbackPointBackground.colorset/Contents.json (100%) rename Themis/Assets.xcassets/{ => colors}/negativeFeedbackText.colorset/Contents.json (100%) rename Themis/Assets.xcassets/{ => colors}/neutralFeedbackBackground.colorset/Contents.json (100%) rename Themis/Assets.xcassets/{ => colors}/neutralFeedbackPointBackground.colorset/Contents.json (100%) rename Themis/Assets.xcassets/{ => colors}/neutralFeedbackText.colorset/Contents.json (100%) rename Themis/Assets.xcassets/{ => colors}/positiveFeedbackBackground.colorset/Contents.json (100%) rename Themis/Assets.xcassets/{ => colors}/positiveFeedbackPointBackground.colorset/Contents.json (100%) rename Themis/Assets.xcassets/{ => colors}/positiveFeedbackText.colorset/Contents.json (100%) rename Themis/Assets.xcassets/{ => colors}/selectedFileBackground.colorset/Contents.json (76%) rename Themis/Assets.xcassets/{ => colors}/sidebarBackground.colorset/Contents.json (76%) rename Themis/Assets.xcassets/{ => colors}/themisBackground.colorset/Contents.json (76%) rename Themis/Assets.xcassets/{ => colors}/themisDarkRed.colorset/Contents.json (100%) rename Themis/Assets.xcassets/{ => colors}/themisGreen.colorset/Contents.json (88%) rename Themis/Assets.xcassets/{ => colors}/themisPrimary.colorset/Contents.json (78%) rename Themis/Assets.xcassets/{ => colors}/themisRed.colorset/Contents.json (100%) rename Themis/Assets.xcassets/{ => colors}/themisSecondary.colorset/Contents.json (100%) diff --git a/Themis/Assets.xcassets/Robot.imageset/Contents.json b/Themis/Assets.xcassets/Robot.imageset/Contents.json new file mode 100644 index 00000000..4af2c747 --- /dev/null +++ b/Themis/Assets.xcassets/Robot.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "robot-solid.png", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/Themis/Assets.xcassets/Robot.imageset/robot-solid.png b/Themis/Assets.xcassets/Robot.imageset/robot-solid.png new file mode 100644 index 0000000000000000000000000000000000000000..af5227233145149ec4e54e98a5f06a234aaacd6a GIT binary patch literal 5840 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 kT0La_mb Date: Wed, 30 Aug 2023 21:14:23 +0400 Subject: [PATCH 09/21] Move suggestion fetching logic into TextAssessmentViewModel --- .../Suggestion/TextFeedbackSuggestion.swift | 4 +- .../CodeEditor/Models/TextBlockRef.swift | 7 +- Themis.xcodeproj/project.pbxproj | 6 +- .../Extensions/Core/TextBlockExtension.swift | 18 +++ Themis/Extensions/SubmissionExtension.swift | 4 + .../Models/Feedback/AssessmentFeedback.swift | 6 +- Themis/Models/Result/AssessmentResult.swift | 12 ++ .../Assessment/AssessmentViewModel.swift | 5 + .../Text/TextAssessmentViewModel.swift | 89 ++++++++++++++- .../Text/TextExerciseRendererViewModel.swift | 107 ++++-------------- .../SubmissionListViewModel.swift | 4 +- .../EditFeedback/EditFeedbackViewBase.swift | 12 +- .../Text Exercise/TextAssessmentView.swift | 5 +- 13 files changed, 178 insertions(+), 101 deletions(-) create mode 100644 Themis/Extensions/Core/TextBlockExtension.swift diff --git a/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift b/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift index 4e4344fb..20b100b5 100644 --- a/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift +++ b/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift @@ -12,7 +12,7 @@ public struct TextFeedbackSuggestion: FeedbackSuggestion { public let blockRef: TextBlockRef public var id: UUID { - blockRef.id + blockRef.associatedAssessmentFeedbackId ?? UUID() } public var text: String { @@ -28,6 +28,6 @@ public struct TextFeedbackSuggestion: FeedbackSuggestion { } public static func == (lhs: TextFeedbackSuggestion, rhs: TextFeedbackSuggestion) -> Bool { - lhs.id == rhs.id && lhs.blockRef.id == rhs.blockRef.id + lhs.id == rhs.id && lhs.text == rhs.text && lhs.credits == rhs.credits } } diff --git a/CodeEditor/Sources/CodeEditor/Models/TextBlockRef.swift b/CodeEditor/Sources/CodeEditor/Models/TextBlockRef.swift index 3dc10ed2..5da30207 100644 --- a/CodeEditor/Sources/CodeEditor/Models/TextBlockRef.swift +++ b/CodeEditor/Sources/CodeEditor/Models/TextBlockRef.swift @@ -12,9 +12,14 @@ public struct TextBlockRef: Codable { public var block: TextBlock public var feedback: Feedback + public init(block: TextBlock, feedback: Feedback) { + self.block = block + self.feedback = feedback + } + private enum CodingKeys: String, CodingKey { case block, feedback } - public let id = UUID() // not decoded + public var associatedAssessmentFeedbackId: UUID? // not decoded } diff --git a/Themis.xcodeproj/project.pbxproj b/Themis.xcodeproj/project.pbxproj index ed35ab5d..0585bb84 100644 --- a/Themis.xcodeproj/project.pbxproj +++ b/Themis.xcodeproj/project.pbxproj @@ -25,6 +25,7 @@ 6530B0F72A6BE17E00CBCA71 /* ModelingAssessmentResult.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6530B0F62A6BE17E00CBCA71 /* ModelingAssessmentResult.swift */; }; 6530B0F92A6BF86700CBCA71 /* ModelingFeedbackDetail.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6530B0F82A6BF86700CBCA71 /* ModelingFeedbackDetail.swift */; }; 653452B62A068E4D008D9598 /* ParticipationExtension.swift in Sources */ = {isa = PBXBuildFile; fileRef = 653452B52A068E4D008D9598 /* ParticipationExtension.swift */; }; + 6541DA832A9FA8EA0061AFAF /* TextBlockExtension.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6541DA822A9FA8E90061AFAF /* TextBlockExtension.swift */; }; 654BC9972A6FF7630081CF22 /* UMLRenderer.swift in Sources */ = {isa = PBXBuildFile; fileRef = 654BC9932A6FF7630081CF22 /* UMLRenderer.swift */; }; 654BC9982A6FF7630081CF22 /* ModelingAssessmentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 654BC9942A6FF7630081CF22 /* ModelingAssessmentView.swift */; }; 654BC9992A6FF7630081CF22 /* UMLClassDiagramRenderer.swift in Sources */ = {isa = PBXBuildFile; fileRef = 654BC9962A6FF7630081CF22 /* UMLClassDiagramRenderer.swift */; }; @@ -208,6 +209,7 @@ 6530B0F62A6BE17E00CBCA71 /* ModelingAssessmentResult.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ModelingAssessmentResult.swift; sourceTree = ""; }; 6530B0F82A6BF86700CBCA71 /* ModelingFeedbackDetail.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ModelingFeedbackDetail.swift; sourceTree = ""; }; 653452B52A068E4D008D9598 /* ParticipationExtension.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ParticipationExtension.swift; sourceTree = ""; }; + 6541DA822A9FA8E90061AFAF /* TextBlockExtension.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TextBlockExtension.swift; sourceTree = ""; }; 654BC9932A6FF7630081CF22 /* UMLRenderer.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = UMLRenderer.swift; sourceTree = ""; }; 654BC9942A6FF7630081CF22 /* ModelingAssessmentView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ModelingAssessmentView.swift; sourceTree = ""; }; 654BC9962A6FF7630081CF22 /* UMLClassDiagramRenderer.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = UMLClassDiagramRenderer.swift; sourceTree = ""; }; @@ -279,7 +281,6 @@ 65F019042A1CDCE400BB1C98 /* AssessmentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AssessmentView.swift; sourceTree = ""; }; 65F019062A1CDE1200BB1C98 /* TextAssessmentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TextAssessmentView.swift; sourceTree = ""; }; 65FAB2512A89FEC000AC533E /* AthenaService.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AthenaService.swift; sourceTree = ""; }; - 65FAB2572A8A0FD800AC533E /* artemis-ios-core-modules */ = {isa = PBXFileReference; lastKnownFileType = wrapper; name = "artemis-ios-core-modules"; path = "../artemis-ios-core-modules"; sourceTree = ""; }; 83396D2A29155935003EF727 /* Themis.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = Themis.app; sourceTree = BUILT_PRODUCTS_DIR; }; 83396D2D29155935003EF727 /* ThemisApp.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ThemisApp.swift; sourceTree = ""; }; 83396D2F29155935003EF727 /* ContentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = ContentView.swift; path = Themis/Views/ContentView.swift; sourceTree = SOURCE_ROOT; }; @@ -505,6 +506,7 @@ children = ( 65153F9E2A0F9B260020B6FE /* LoginVMExtension.swift */, 65EDED362A33103A003BF14B /* UserFacingErrorExtension.swift */, + 6541DA822A9FA8E90061AFAF /* TextBlockExtension.swift */, ); path = Core; sourceTree = ""; @@ -752,7 +754,6 @@ 83396D2129155934003EF727 = { isa = PBXGroup; children = ( - 65FAB2572A8A0FD800AC533E /* artemis-ios-core-modules */, E2E3852B2943800100F5854D /* CodeEditor */, 83396D2C29155935003EF727 /* Themis */, 83396D3D29155936003EF727 /* ThemisTests */, @@ -1287,6 +1288,7 @@ 65B49F402A07F29400C9A45F /* PaneViewModel.swift in Sources */, 65F019072A1CDE1200BB1C98 /* TextAssessmentView.swift in Sources */, 65B49F3E2A07ED6100C9A45F /* ToolbarFileTreeToggleButton.swift in Sources */, + 6541DA832A9FA8EA0061AFAF /* TextBlockExtension.swift in Sources */, 6530B0F72A6BE17E00CBCA71 /* ModelingAssessmentResult.swift in Sources */, E2192EE5291EBFFE0092CE58 /* AuthenticationView.swift in Sources */, 650538B92A1E1EF200C45E18 /* ExerciseMockExtension.swift in Sources */, diff --git a/Themis/Extensions/Core/TextBlockExtension.swift b/Themis/Extensions/Core/TextBlockExtension.swift new file mode 100644 index 00000000..5be8de5d --- /dev/null +++ b/Themis/Extensions/Core/TextBlockExtension.swift @@ -0,0 +1,18 @@ +// +// TextBlockExtension.swift +// Themis +// +// Created by Tarlan Ismayilsoy on 30.08.23. +// + +import Foundation +import SharedModels + +extension TextBlock { + var range: Range? { + guard let startIndex, let endIndex else { + return nil + } + return startIndex ..< endIndex + } +} diff --git a/Themis/Extensions/SubmissionExtension.swift b/Themis/Extensions/SubmissionExtension.swift index ee5cbdb6..eee459e5 100644 --- a/Themis/Extensions/SubmissionExtension.swift +++ b/Themis/Extensions/SubmissionExtension.swift @@ -18,6 +18,10 @@ extension BaseSubmission { func getExercise(as: T.Type = (any BaseExercise).self) -> T? { self.getParticipation()?.getExercise(as: T.self) } + + var isAssessed: Bool { + results?.last?.completionDate != nil + } } extension Submission { diff --git a/Themis/Models/Feedback/AssessmentFeedback.swift b/Themis/Models/Feedback/AssessmentFeedback.swift index 52dbddfd..5347adb0 100644 --- a/Themis/Models/Feedback/AssessmentFeedback.swift +++ b/Themis/Models/Feedback/AssessmentFeedback.swift @@ -65,7 +65,7 @@ extension AssessmentFeedback: Equatable, Hashable { } extension AssessmentFeedback { - init(basedOn suggestion: any FeedbackSuggestion, _ incompleteFeedbackDetail: FeedbackDetail?) { + init(basedOn suggestion: any FeedbackSuggestion, _ incompleteFeedbackDetail: FeedbackDetail?, _ detailText: String, _ credits: Double) { var newIncompleteFeedbackDetail = incompleteFeedbackDetail if var incompleteFeedbackDetail = incompleteFeedbackDetail as? ProgrammingFeedbackDetail, @@ -75,8 +75,8 @@ extension AssessmentFeedback { newIncompleteFeedbackDetail = incompleteFeedbackDetail } - self.init(baseFeedback: Feedback(detailText: suggestion.text, - credits: suggestion.credits, + self.init(baseFeedback: Feedback(detailText: detailText, + credits: credits, type: .MANUAL_UNREFERENCED), scope: .inline, detail: newIncompleteFeedbackDetail) diff --git a/Themis/Models/Result/AssessmentResult.swift b/Themis/Models/Result/AssessmentResult.swift index 60e7f0ef..feaf03b0 100644 --- a/Themis/Models/Result/AssessmentResult.swift +++ b/Themis/Models/Result/AssessmentResult.swift @@ -105,6 +105,18 @@ class AssessmentResult: Encodable, ObservableObject { return computedFeedbacks[index] } + @discardableResult + func replace(feedbackWithId id: UUID, with newFeedback: AssessmentFeedback) -> AssessmentFeedback? { + guard let index = (feedbacks.firstIndex { $0.id == id }) else { + return nil + } + + undoManager.beginUndoGrouping() + + computedFeedbacks[index] = newFeedback + return computedFeedbacks[index] + } + @discardableResult func updateFeedback(feedback: AssessmentFeedback) -> AssessmentFeedback? { guard let index = (feedbacks.firstIndex { $0.id == feedback.id }) else { diff --git a/Themis/ViewModels/Assessment/AssessmentViewModel.swift b/Themis/ViewModels/Assessment/AssessmentViewModel.swift index 0e4826f3..41b7c451 100644 --- a/Themis/ViewModels/Assessment/AssessmentViewModel.swift +++ b/Themis/ViewModels/Assessment/AssessmentViewModel.swift @@ -209,6 +209,11 @@ class AssessmentViewModel: ObservableObject { func getFeedback(byId id: UUID) -> AssessmentFeedback? { assessmentResult.feedbacks.first(where: { $0.id == id }) } + + func getManualFeedback(byId id: UUID) -> AssessmentFeedback? { + let manualFeedbacks = assessmentResult.inlineFeedback + assessmentResult.generalFeedback + return manualFeedbacks.first(where: { $0.id == id }) + } } enum AssessmentViewModelFactory { diff --git a/Themis/ViewModels/Assessment/Text/TextAssessmentViewModel.swift b/Themis/ViewModels/Assessment/Text/TextAssessmentViewModel.swift index e70a59b2..c474457c 100644 --- a/Themis/ViewModels/Assessment/Text/TextAssessmentViewModel.swift +++ b/Themis/ViewModels/Assessment/Text/TextAssessmentViewModel.swift @@ -8,8 +8,15 @@ import Foundation import Common import SharedModels +import CodeEditor class TextAssessmentViewModel: AssessmentViewModel { + private var suggestedBlockRefs = [TextBlockRef]() + + private var shouldFetchSuggestions: Bool { + !readOnly && submission?.isAssessed == false && assessmentResult.automaticFeedback.isEmpty + } + @MainActor override func initSubmission() async { guard submission == nil else { @@ -26,9 +33,13 @@ class TextAssessmentViewModel: AssessmentViewModel { } } + if shouldFetchSuggestions { + await fetchSuggestions() + } + ThemisUndoManager.shared.removeAllActions() } - + @MainActor private func getParticipationForSubmission(participationId: Int?, submissionId: Int?) async { guard let submissionId else { @@ -53,4 +64,80 @@ class TextAssessmentViewModel: AssessmentViewModel { log.info(String(describing: error)) } } + + private func fetchSuggestions() async { + guard let exerciseId = participation?.exercise?.id, + let submissionId = submission?.id else { + log.error("Could not fetch suggestions for submission: #\(submissionId ?? -1)") + return + } + + do { + var blockRefs = try await AthenaService().getFeedbackSuggestions(exerciseId: exerciseId, submissionId: submissionId) + log.verbose("Fetched \(blockRefs.count) suggestions") + + blockRefs = removeOverlappingRefs(blockRefs) + + suggestedBlockRefs = blockRefs + await saveBlockRefsAsAssessmentFeedbacks() + } catch { + log.error(String(describing: error)) + } + } + + @MainActor + private func saveBlockRefsAsAssessmentFeedbacks() { + var suggestedFeedbacks = [AssessmentFeedback]() + + for index in 0 ..< suggestedBlockRefs.count { + let blockRef = suggestedBlockRefs[index] + let assessmentFeedback = AssessmentFeedback(baseFeedback: blockRef.feedback, + scope: .inline, + detail: TextFeedbackDetail(block: blockRef.block)) + suggestedFeedbacks.append(assessmentFeedback) + suggestedBlockRefs[index].associatedAssessmentFeedbackId = assessmentFeedback.id + } + + assessmentResult.computedFeedbacks = assessmentResult.computedFeedbacks + suggestedFeedbacks + } + + private func removeOverlappingRefs(_ blockRefs: [TextBlockRef]) -> [TextBlockRef] { + var rangeToBlockRef = [Range: TextBlockRef]() + + for blockRef in blockRefs { + guard let startIndex = blockRef.block.startIndex, + let endIndex = blockRef.block.endIndex else { + continue + } + let blockRefRange = startIndex.. TextFeedbackSuggestion? { + var suggestionBlockRef: TextBlockRef? + + if let blockRef = suggestedBlockRefs.first(where: { $0.associatedAssessmentFeedbackId == id }) { + suggestionBlockRef = blockRef + } + // this happens when `fetchSuggestions()` is not called and automatic feedbacks embedded into the submission are used + else if let assessmentFeedback = assessmentResult.automaticFeedback.first(where: { $0.id == id }), + let textSubmission = submission as? TextSubmission, + let blocks = textSubmission.blocks, + let block = blocks.first(where: { $0.id == assessmentFeedback.baseFeedback.reference }) { + var blockRef = TextBlockRef(block: block, feedback: assessmentFeedback.baseFeedback) + blockRef.associatedAssessmentFeedbackId = id + suggestionBlockRef = blockRef + } + + if let suggestionBlockRef { + return TextFeedbackSuggestion(blockRef: suggestionBlockRef) + } else { + return nil + } + } } diff --git a/Themis/ViewModels/Assessment/Text/TextExerciseRendererViewModel.swift b/Themis/ViewModels/Assessment/Text/TextExerciseRendererViewModel.swift index 694614e7..84cd3573 100644 --- a/Themis/ViewModels/Assessment/Text/TextExerciseRendererViewModel.swift +++ b/Themis/ViewModels/Assessment/Text/TextExerciseRendererViewModel.swift @@ -42,7 +42,6 @@ class TextExerciseRendererViewModel: ExerciseRendererViewModel { /// Needed for creating new text blocks private var submissionId: Int? - private var suggestedRefs = [TextBlockRef]() /// Sets this VM up based on the given participation and optional submission /// - Parameters: @@ -61,74 +60,13 @@ class TextExerciseRendererViewModel: ExerciseRendererViewModel { return } - let feedbacks = assessmentResult.inlineFeedback + let feedbacks = assessmentResult.inlineFeedback + assessmentResult.automaticFeedback let blocks = textSubmission.blocks ?? [] inlineHighlights.removeAll() submissionId = textSubmission.id content = textSubmission.text ?? content setupHighlights(basedOn: blocks, and: feedbacks) - - if let participation { - fetchSuggestions(for: textSubmission, participation) - } - } - - // TODO: make sure this is not called for read-only and finished submissions - private func fetchSuggestions(for textSubmission: TextSubmission, _ participation: BaseParticipation) { - guard let exerciseId = participation.exercise?.id, - let submissionId = textSubmission.id else { - log.error("Could not fetch suggestions for submission: #\(submissionId ?? -1)") - return - } - - Task { [weak self] in - do { - var blockRefs = try await AthenaService().getFeedbackSuggestions(exerciseId: exerciseId, submissionId: submissionId) - log.verbose("Fetched \(blockRefs.count) suggestions") - - blockRefs = self?.removeOverlappingRefs(blockRefs) ?? [] - await self?.setupHighlights(basedOn: blockRefs) - self?.suggestedRefs = blockRefs - } catch { - log.error(String(describing: error)) - } - } - } - - private func removeOverlappingRefs(_ blockRefs: [TextBlockRef]) -> [TextBlockRef] { - var rangeToBlockRef = [Range: TextBlockRef]() - - // Remove block references overlapping among themselves - for blockRef in blockRefs { - guard let startIndex = blockRef.block.startIndex, - let endIndex = blockRef.block.endIndex else { - continue - } - let blockRefRange = startIndex.. TextFeedbackSuggestion? { - guard let blockRef = suggestedRefs.first(where: { $0.id == id }) else { - return nil - } - return TextFeedbackSuggestion(blockRef: blockRef) } /// Generates a `TextFeedbackDetail` instance based on the available data. Some fields might be missing @@ -142,19 +80,20 @@ class TextExerciseRendererViewModel: ExerciseRendererViewModel { // MARK: - Highlight-related code private func setupHighlights(basedOn blocks: [TextBlock], and feedbacks: [AssessmentFeedback], shouldWipeUndo: Bool = true) { - blocks.forEach { block in - guard let startIndex = block.startIndex, + feedbacks.forEach { assessmentFeedback in + guard let block = findBlock(for: assessmentFeedback, using: blocks), + let startIndex = block.startIndex, let endIndex = block.endIndex, - let feedback = feedbacks.first(where: { $0.baseFeedback.reference == block.id }), startIndex < endIndex else { return } let range = NSRange(startIndex.. TextBlock? { + if let block = blocks.first(where: { $0.id == assessmentFeedback.baseFeedback.reference }) { + return block + } else { + return (assessmentFeedback.detail as? TextFeedbackDetail)?.block } - - undoManager.removeAllActions() } private func updateHighlightColor(for feedback: AssessmentFeedback) { @@ -210,6 +140,12 @@ class TextExerciseRendererViewModel: ExerciseRendererViewModel { private func deleteHighlight(for suggestion: TextFeedbackSuggestion) { inlineHighlights.removeAll(where: { $0.id == suggestion.id }) + undoManager.endUndoGrouping() + } + + private func replaceHighlight(for suggestion: TextFeedbackSuggestion, withHighlightFor feedback: AssessmentFeedback) { + inlineHighlights.removeAll(where: { $0.id == suggestion.id }) + createHighlight(for: feedback) } } @@ -230,8 +166,7 @@ extension TextExerciseRendererViewModel: FeedbackDelegate { guard let suggestion = suggestion as? TextFeedbackSuggestion else { return } - deleteHighlight(for: suggestion) - createHighlight(for: feedback) + replaceHighlight(for: suggestion, withHighlightFor: feedback) } func onFeedbackSuggestionDiscard(_ suggestion: any FeedbackSuggestion) { diff --git a/Themis/ViewModels/SubmissionList/SubmissionListViewModel.swift b/Themis/ViewModels/SubmissionList/SubmissionListViewModel.swift index f943678f..ba441b2e 100644 --- a/Themis/ViewModels/SubmissionList/SubmissionListViewModel.swift +++ b/Themis/ViewModels/SubmissionList/SubmissionListViewModel.swift @@ -14,11 +14,11 @@ class SubmissionListViewModel: ObservableObject { @Published var error: Error? var submittedSubmissions: [Submission] { - submissions.filter { $0.baseSubmission.results?.last?.completionDate != nil } + submissions.filter { $0.baseSubmission.isAssessed } } var openSubmissions: [Submission] { - submissions.filter { $0.baseSubmission.results?.last?.completionDate == nil } + submissions.filter { !$0.baseSubmission.isAssessed } } @MainActor diff --git a/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/EditFeedbackViewBase.swift b/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/EditFeedbackViewBase.swift index 96016f7e..9c4f8330 100644 --- a/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/EditFeedbackViewBase.swift +++ b/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/EditFeedbackViewBase.swift @@ -146,6 +146,7 @@ struct EditFeedbackViewBase: View { private func deleteFeedback() { if let feedbackSuggestion { + assessmentResult.deleteFeedback(id: feedbackSuggestion.id) feedbackDelegate?.onFeedbackSuggestionDiscard(feedbackSuggestion) } else if let feedback = assessmentResult.feedbacks.first(where: { idForUpdate == $0.id }) { assessmentResult.deleteFeedback(id: feedback.id) @@ -154,9 +155,16 @@ struct EditFeedbackViewBase: View { } private func addFeedbackSuggestionToFeedbacks(feedbackSuggestion: any FeedbackSuggestion) { - let feedback = AssessmentFeedback(basedOn: feedbackSuggestion, incompleteFeedback?.detail) + var feedback = AssessmentFeedback(basedOn: feedbackSuggestion, incompleteFeedback?.detail, detailText, score) + + // Try to replace the existing automatic feedback + let newFeedback = assessmentResult.replace(feedbackWithId: feedbackSuggestion.id, with: feedback) + + // No automatic feedback found -> simply add a new feedback for the given suggestion + if newFeedback == nil { + assessmentResult.addFeedback(feedback: feedback) + } - assessmentResult.addFeedback(feedback: feedback) feedbackDelegate?.onFeedbackSuggestionSelection(feedbackSuggestion, feedback) } diff --git a/Themis/Views/Assessment/Text Exercise/TextAssessmentView.swift b/Themis/Views/Assessment/Text Exercise/TextAssessmentView.swift index 3050f62a..d2c4919f 100644 --- a/Themis/Views/Assessment/Text Exercise/TextAssessmentView.swift +++ b/Themis/Views/Assessment/Text Exercise/TextAssessmentView.swift @@ -64,7 +64,7 @@ struct TextAssessmentView: View { ) }) .sheet(isPresented: $textExerciseRendererVM.showEditFeedback) { - if let feedback = assessmentVM.getFeedback(byId: textExerciseRendererVM.selectedFeedbackForEditingId) { + if let feedback = assessmentVM.getManualFeedback(byId: textExerciseRendererVM.selectedFeedbackForEditingId) { // The user tapped on a manual feedback to edit EditFeedbackView( assessmentResult: assessmentVM.assessmentResult, @@ -74,7 +74,8 @@ struct TextAssessmentView: View { gradingCriteria: assessmentVM.gradingCriteria, showSheet: $textExerciseRendererVM.showEditFeedback ) - } else if let suggestion = textExerciseRendererVM.getSuggestion(byId: textExerciseRendererVM.selectedFeedbackForEditingId) { + } else if let textAssessmentVM = assessmentVM as? TextAssessmentViewModel, + let suggestion = textAssessmentVM.getSuggestion(byAssessmentFeedbackId: textExerciseRendererVM.selectedFeedbackForEditingId) { // The user tapped on a feedback suggestion. This is not actually an edit action, but is triggered as one AddFeedbackView( assessmentResult: assessmentVM.assessmentResult, From 36304f5d9ff1d1b72e5dd5ffa49d3a054a7f056a Mon Sep 17 00:00:00 2001 From: Tarlan Ismayilsoy Date: Wed, 30 Aug 2023 21:39:21 +0400 Subject: [PATCH 10/21] Fix a SwiftLint error --- Themis/ViewModels/Assessment/Text/TextAssessmentViewModel.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Themis/ViewModels/Assessment/Text/TextAssessmentViewModel.swift b/Themis/ViewModels/Assessment/Text/TextAssessmentViewModel.swift index c474457c..40064dc2 100644 --- a/Themis/ViewModels/Assessment/Text/TextAssessmentViewModel.swift +++ b/Themis/ViewModels/Assessment/Text/TextAssessmentViewModel.swift @@ -98,7 +98,7 @@ class TextAssessmentViewModel: AssessmentViewModel { suggestedBlockRefs[index].associatedAssessmentFeedbackId = assessmentFeedback.id } - assessmentResult.computedFeedbacks = assessmentResult.computedFeedbacks + suggestedFeedbacks + assessmentResult.computedFeedbacks += suggestedFeedbacks } private func removeOverlappingRefs(_ blockRefs: [TextBlockRef]) -> [TextBlockRef] { From 5cabbc44f59b51afbb5f284e4304e7bde0936eda Mon Sep 17 00:00:00 2001 From: Tarlan Ismayilsoy Date: Wed, 30 Aug 2023 21:51:16 +0400 Subject: [PATCH 11/21] Fix a SwiftLint warning --- .../CorrectionSidebar/EditFeedback/EditFeedbackViewBase.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/EditFeedbackViewBase.swift b/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/EditFeedbackViewBase.swift index 9c4f8330..5002c8b3 100644 --- a/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/EditFeedbackViewBase.swift +++ b/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/EditFeedbackViewBase.swift @@ -155,7 +155,7 @@ struct EditFeedbackViewBase: View { } private func addFeedbackSuggestionToFeedbacks(feedbackSuggestion: any FeedbackSuggestion) { - var feedback = AssessmentFeedback(basedOn: feedbackSuggestion, incompleteFeedback?.detail, detailText, score) + let feedback = AssessmentFeedback(basedOn: feedbackSuggestion, incompleteFeedback?.detail, detailText, score) // Try to replace the existing automatic feedback let newFeedback = assessmentResult.replace(feedbackWithId: feedbackSuggestion.id, with: feedback) From f644ebb5385ce3aa34c401b27229e03c46526e6f Mon Sep 17 00:00:00 2001 From: Tarlan Ismayilsoy Date: Thu, 7 Sep 2023 15:54:49 +0400 Subject: [PATCH 12/21] SwiftLint warning fix --- Themis/Views/Assessment/Text Exercise/TextAssessmentView.swift | 1 + 1 file changed, 1 insertion(+) diff --git a/Themis/Views/Assessment/Text Exercise/TextAssessmentView.swift b/Themis/Views/Assessment/Text Exercise/TextAssessmentView.swift index d2c4919f..a2fc5b8b 100644 --- a/Themis/Views/Assessment/Text Exercise/TextAssessmentView.swift +++ b/Themis/Views/Assessment/Text Exercise/TextAssessmentView.swift @@ -121,5 +121,6 @@ struct TextAssessmentView_Previews: PreviewProvider { assessmentResult: assessmentVM.assessmentResult as! TextAssessmentResult, exercise: Exercise.mockText) .previewInterfaceOrientation(.landscapeRight) + // swiftlint: enable force_cast } } From c3e74d273f8a985808e56cb02c1cd47539e6cc34 Mon Sep 17 00:00:00 2001 From: Tarlan Ismayilsoy Date: Thu, 7 Sep 2023 15:57:16 +0400 Subject: [PATCH 13/21] Fix SwiftLint syntax error --- .../Views/Assessment/Text Exercise/TextAssessmentView.swift | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Themis/Views/Assessment/Text Exercise/TextAssessmentView.swift b/Themis/Views/Assessment/Text Exercise/TextAssessmentView.swift index a2fc5b8b..740f4a03 100644 --- a/Themis/Views/Assessment/Text Exercise/TextAssessmentView.swift +++ b/Themis/Views/Assessment/Text Exercise/TextAssessmentView.swift @@ -116,11 +116,11 @@ struct TextAssessmentView_Previews: PreviewProvider { @StateObject private static var assessmentVM = MockAssessmentViewModel(exercise: Exercise.mockText, readOnly: false) static var previews: some View { - // swiftlint: disable force_cast + // swiftlint:disable force_cast TextAssessmentView(assessmentVM: assessmentVM, assessmentResult: assessmentVM.assessmentResult as! TextAssessmentResult, exercise: Exercise.mockText) .previewInterfaceOrientation(.landscapeRight) - // swiftlint: enable force_cast + // swiftlint:enable force_cast } } From 6cfdd8663fece5da96758fd6ecdf5f32b9568051 Mon Sep 17 00:00:00 2001 From: Tarlan Ismayilsoy Date: Sun, 10 Sep 2023 11:54:17 +0400 Subject: [PATCH 14/21] Attempt to fix the SPM issue --- CodeEditor/Package.swift | 7 ++++--- .../CodeEditor/Models/Suggestion/FeedbackSuggestion.swift | 1 - .../Models/Suggestion/ProgrammingFeedbackSuggestion.swift | 1 - .../Models/Suggestion/TextFeedbackSuggestion.swift | 1 - 4 files changed, 4 insertions(+), 6 deletions(-) diff --git a/CodeEditor/Package.swift b/CodeEditor/Package.swift index d057c287..31396d79 100644 --- a/CodeEditor/Package.swift +++ b/CodeEditor/Package.swift @@ -7,7 +7,7 @@ let package = Package( name: "CodeEditor", platforms: [ - .macOS(.v10_15), .iOS(.v16) + .iOS(.v16) ], products: [ @@ -15,10 +15,11 @@ let package = Package( ], dependencies: [ - .package(url: "https://github.com/raspu/Highlightr", from: "2.1.2") + .package(url: "https://github.com/raspu/Highlightr", from: "2.1.2"), + .package(url: "https://github.com/ls1intum/artemis-ios-core-modules", from: "3.3.2"), ], targets: [ - .target(name: "CodeEditor", dependencies: [ "Highlightr" ]) + .target(name: "CodeEditor", dependencies: [ "Highlightr", .product(name: "SharedModels", package: "artemis-ios-core-modules")]) ] ) diff --git a/CodeEditor/Sources/CodeEditor/Models/Suggestion/FeedbackSuggestion.swift b/CodeEditor/Sources/CodeEditor/Models/Suggestion/FeedbackSuggestion.swift index e16bed58..37e944ef 100644 --- a/CodeEditor/Sources/CodeEditor/Models/Suggestion/FeedbackSuggestion.swift +++ b/CodeEditor/Sources/CodeEditor/Models/Suggestion/FeedbackSuggestion.swift @@ -6,7 +6,6 @@ // import Foundation -import SharedModels public protocol FeedbackSuggestion: Equatable { var id: UUID { get } diff --git a/CodeEditor/Sources/CodeEditor/Models/Suggestion/ProgrammingFeedbackSuggestion.swift b/CodeEditor/Sources/CodeEditor/Models/Suggestion/ProgrammingFeedbackSuggestion.swift index aa53006a..6b605c12 100644 --- a/CodeEditor/Sources/CodeEditor/Models/Suggestion/ProgrammingFeedbackSuggestion.swift +++ b/CodeEditor/Sources/CodeEditor/Models/Suggestion/ProgrammingFeedbackSuggestion.swift @@ -6,7 +6,6 @@ // import Foundation -import SharedModels public struct ProgrammingFeedbackSuggestion: FeedbackSuggestion, Decodable { public let id = UUID() diff --git a/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift b/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift index 20b100b5..11e932df 100644 --- a/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift +++ b/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift @@ -6,7 +6,6 @@ // import Foundation -import SharedModels public struct TextFeedbackSuggestion: FeedbackSuggestion { public let blockRef: TextBlockRef From ea1c5bcf26479fe7894577601f676d85ca339d24 Mon Sep 17 00:00:00 2001 From: Tarlan Ismayilsoy Date: Sun, 10 Sep 2023 12:06:24 +0400 Subject: [PATCH 15/21] Fix warning --- .../Sources/CodeEditor/RoundedCornerLayoutManager.swift | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/CodeEditor/Sources/CodeEditor/RoundedCornerLayoutManager.swift b/CodeEditor/Sources/CodeEditor/RoundedCornerLayoutManager.swift index 70eed2bf..138e5c5e 100644 --- a/CodeEditor/Sources/CodeEditor/RoundedCornerLayoutManager.swift +++ b/CodeEditor/Sources/CodeEditor/RoundedCornerLayoutManager.swift @@ -200,7 +200,9 @@ class RoundedCornerLayoutManager: NSLayoutManager { private func drawProgrammingFeedbackSuggestions(_ paraNumber: Int, _ rect: CGRect, _ origin: CGPoint) { let ctx = UIGraphicsGetCurrentContext() - guard let ctx else { return } + guard let ctx else { + return + } UIGraphicsPushContext(ctx) ctx.setFillColor(CGColor(red: 0, green: 0.2, blue: 0.8, alpha: 0.8)) ctx.setStrokeColor(CGColor(red: 0, green: 0.2, blue: 0.8, alpha: 0.8)) From 3754686df076edf66f2374b7f0fe48407cbd510d Mon Sep 17 00:00:00 2001 From: Tarlan Ismayilsoy Date: Thu, 26 Oct 2023 19:48:34 +0400 Subject: [PATCH 16/21] Adapt everything fo ther new server-side changes --- .../Suggestion/FeedbackSuggestion.swift | 18 +-- .../ProgrammingFeedbackSuggestion.swift | 29 ++++- .../Suggestion/TextFeedbackSuggestion.swift | 83 ++++++++++++-- .../Sources/CodeEditor/UXCodeTextView.swift | 3 +- Themis.xcodeproj/project.pbxproj | 2 - .../Robot.imageset/robot-solid.png | Bin 5840 -> 0 bytes .../Contents.json | 2 +- .../lightbulb-solid.png | Bin 0 -> 5574 bytes Themis/Extensions/ColorExtension.swift | 3 + .../Models/Feedback/AssessmentFeedback.swift | 6 +- Themis/Models/Result/AssessmentResult.swift | 4 + .../Services/Assessment/AthenaService.swift | 6 +- .../Assessment/AssessmentViewModel.swift | 6 +- .../Text/TextAssessmentViewModel.swift | 103 ++++++++++-------- .../Text/TextExerciseRendererViewModel.swift | 18 ++- .../EditFeedback/AddFeedbackView.swift | 2 - .../EditFeedback/EditFeedbackView.swift | 2 - .../EditFeedback/EditFeedbackViewBase.swift | 73 ++++++------- .../Text Exercise/TextAssessmentView.swift | 14 +-- 19 files changed, 230 insertions(+), 144 deletions(-) delete mode 100644 Themis/Assets.xcassets/Robot.imageset/robot-solid.png rename Themis/Assets.xcassets/{Robot.imageset => SuggestedFeedbackSymbol.imageset}/Contents.json (74%) create mode 100644 Themis/Assets.xcassets/SuggestedFeedbackSymbol.imageset/lightbulb-solid.png diff --git a/CodeEditor/Sources/CodeEditor/Models/Suggestion/FeedbackSuggestion.swift b/CodeEditor/Sources/CodeEditor/Models/Suggestion/FeedbackSuggestion.swift index 37e944ef..e5e8c6d8 100644 --- a/CodeEditor/Sources/CodeEditor/Models/Suggestion/FeedbackSuggestion.swift +++ b/CodeEditor/Sources/CodeEditor/Models/Suggestion/FeedbackSuggestion.swift @@ -6,14 +6,16 @@ // import Foundation +import SharedModels -public protocol FeedbackSuggestion: Equatable { - var id: UUID { get } - var text: String { get } +public protocol FeedbackSuggestion: Equatable, Decodable { + var id: Int { get } + var exerciseId: Int { get } + var submissionId: Int { get } + var title: String { get } + var description: String { get } var credits: Double { get } -} - -public extension FeedbackSuggestion { - var text: String { "" } - var credits: Double { 0.0 } + var gradingInstruction: GradingInstruction? { get } + + var associatedAssessmentFeedbackId: UUID? { get set } // not decoded } diff --git a/CodeEditor/Sources/CodeEditor/Models/Suggestion/ProgrammingFeedbackSuggestion.swift b/CodeEditor/Sources/CodeEditor/Models/Suggestion/ProgrammingFeedbackSuggestion.swift index 6b605c12..86bb9949 100644 --- a/CodeEditor/Sources/CodeEditor/Models/Suggestion/ProgrammingFeedbackSuggestion.swift +++ b/CodeEditor/Sources/CodeEditor/Models/Suggestion/ProgrammingFeedbackSuggestion.swift @@ -6,16 +6,31 @@ // import Foundation +import SharedModels public struct ProgrammingFeedbackSuggestion: FeedbackSuggestion, Decodable { - public let id = UUID() - public let exerciseId: Int + + public var id: Int + + public var exerciseId: Int + + public var submissionId: Int + + public var title: String + + public var description: String + + public var credits: Double + + public var gradingInstruction: GradingInstruction? + + public var associatedAssessmentFeedbackId: UUID? + + // TODO: rename/remove the fields below once programming suggestions are integrated into Athena public let participationId: Int public let srcFile: String public let fromLine: Int public let toLine: Int - public let text: String - public let credits: Double enum DecodingKeys: String, CodingKey { case exercise_id @@ -27,14 +42,18 @@ public struct ProgrammingFeedbackSuggestion: FeedbackSuggestion, Decodable { case credits } + // TODO: correct the decoding logic below once programming suggestions are integrated into Athena public init(from decoder: Decoder) throws { let values = try decoder.container(keyedBy: DecodingKeys.self) + id = Int.random(in: 1...999999) exerciseId = try values.decode(Int.self, forKey: .exercise_id) + submissionId = -1 + title = "Suggestion" participationId = try values.decode(Int.self, forKey: .participation_id) srcFile = try values.decode(String.self, forKey: .src_file) fromLine = try values.decode(Int.self, forKey: .from_line) toLine = try values.decode(Int.self, forKey: .to_line) - text = try values.decode(String.self, forKey: .text) + description = try values.decode(String.self, forKey: .text) credits = try values.decode(Double.self, forKey: .credits) } diff --git a/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift b/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift index 11e932df..b0067e17 100644 --- a/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift +++ b/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift @@ -6,27 +6,88 @@ // import Foundation +import SharedModels public struct TextFeedbackSuggestion: FeedbackSuggestion { - public let blockRef: TextBlockRef + public let id: Int - public var id: UUID { - blockRef.associatedAssessmentFeedbackId ?? UUID() + public let exerciseId: Int + + public let submissionId: Int + + public let title: String + + public let description: String + + public let credits: Double + + public let gradingInstruction: GradingInstruction? + + public var associatedAssessmentFeedbackId: UUID? + + public let indexStart: Int? + + public let indexEnd: Int? + + public static func == (lhs: TextFeedbackSuggestion, rhs: TextFeedbackSuggestion) -> Bool { + lhs.id == rhs.id + && lhs.exerciseId == rhs.exerciseId + && lhs.submissionId == rhs.submissionId + && lhs.title == rhs.title + && lhs.description == rhs.description + && lhs.credits == rhs.credits } - public var text: String { - blockRef.feedback.detailText ?? "" + public var textBlockContent: String? + + public var isReferenced: Bool { + indexStart != nil && indexEnd != nil } - public var credits: Double { - blockRef.feedback.credits ?? 0.0 + public var textBlock: TextBlock? { + guard isReferenced else { + return nil + } + return TextBlock(submissionId: submissionId, text: textBlockContent, startIndex: indexStart, endIndex: indexEnd) } - public init(blockRef: TextBlockRef) { - self.blockRef = blockRef + public var feedback: Feedback { + // TODO: grading instruction support + return Feedback(text: Feedback.feedbackSuggestionAcceptedIdentifier + title, + detailText: description, + reference: textBlock?.id, + credits: credits, + type: isReferenced ? .MANUAL : .MANUAL_UNREFERENCED, + positive: credits > 0) } - public static func == (lhs: TextFeedbackSuggestion, rhs: TextFeedbackSuggestion) -> Bool { - lhs.id == rhs.id && lhs.text == rhs.text && lhs.credits == rhs.credits + public mutating func setTextBlockContent(from submission: TextSubmission) { + guard let indexStart, + let indexEnd, + let text = submission.text else { + return + } + let nsRange = (indexStart ..< indexEnd).toNSRange() + if let indexRange = Range(nsRange, in: text) { + textBlockContent = String(text[indexRange]) + } + } +} + +public extension Feedback { + static let feedbackSuggestionIdentifier = "FeedbackSuggestion:" + static let feedbackSuggestionAcceptedIdentifier = "FeedbackSuggestion:accepted:" + static let feedbackSuggestionAdaptedIdentifier = "FeedbackSuggestion:adapted:" + + var isSuggested: Bool { + text?.hasPrefix(Self.feedbackSuggestionIdentifier) ?? false + } + + var isSuggestedAndAccepted: Bool { + text?.hasPrefix(Self.feedbackSuggestionAcceptedIdentifier) ?? false + } + + var isSuggestedAndAdapted: Bool { + text?.hasPrefix(Self.feedbackSuggestionAdaptedIdentifier) ?? false } } diff --git a/CodeEditor/Sources/CodeEditor/UXCodeTextView.swift b/CodeEditor/Sources/CodeEditor/UXCodeTextView.swift index ce4f969f..8ccd2905 100644 --- a/CodeEditor/Sources/CodeEditor/UXCodeTextView.swift +++ b/CodeEditor/Sources/CodeEditor/UXCodeTextView.swift @@ -409,7 +409,8 @@ final class UXCodeTextView: UXTextView, HighlightDelegate, UIScrollViewDelegate layoutManager.enumerateLineFragments(forGlyphRange: layoutManager.glyphRange(for: textContainer)) { rect, _, _, _, _ in let offset = self.calculateWrapOffsetOf(lineNumber) if let feedback = self.feedbackSuggestions.first(where: { $0.fromLine == lineNumber - offset }) { - if let lightbulb = self.buildLightbulbButton(rect: rect, feedbackId: feedback.id.uuidString) { + // TODO: get rid of the string interpolation once programming exercise suggestions are integrated into Athena + if let lightbulb = self.buildLightbulbButton(rect: rect, feedbackId: "\(feedback.id)") { self.lightBulbs.append(lightbulb) self.addSubview(lightbulb) } diff --git a/Themis.xcodeproj/project.pbxproj b/Themis.xcodeproj/project.pbxproj index af01577c..3e24320e 100644 --- a/Themis.xcodeproj/project.pbxproj +++ b/Themis.xcodeproj/project.pbxproj @@ -116,7 +116,6 @@ 65B9BD092AAE18570086F9EC /* FileUploadAssessmentViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = 65B9BD082AAE18570086F9EC /* FileUploadAssessmentViewModel.swift */; }; 65B9BD102AAE413B0086F9EC /* FileUploadSubmissionServiceImpl.swift in Sources */ = {isa = PBXBuildFile; fileRef = 65B9BD0F2AAE413B0086F9EC /* FileUploadSubmissionServiceImpl.swift */; }; 65BA61D72AE02B5C009FE11C /* PathStringExtension.swift in Sources */ = {isa = PBXBuildFile; fileRef = 65BA61D62AE02B5C009FE11C /* PathStringExtension.swift */; }; - 65BC14532A6FD9F30002D089 /* UMLBadgeSymbol.swift in Sources */ = {isa = PBXBuildFile; fileRef = 65BC14522A6FD9F30002D089 /* UMLBadgeSymbol.swift */; }; 65C2F7532A03E3D4001EDADC /* FormatterExtension.swift in Sources */ = {isa = PBXBuildFile; fileRef = 65C2F7522A03E3D4001EDADC /* FormatterExtension.swift */; }; 65C2F7552A03E428001EDADC /* JSONDecoderExtension.swift in Sources */ = {isa = PBXBuildFile; fileRef = 65C2F7542A03E428001EDADC /* JSONDecoderExtension.swift */; }; 65CAE5592A9E13060083CFDF /* ProgrammingAssessmentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = E21AF052292B9D4C0096ACFC /* ProgrammingAssessmentView.swift */; }; @@ -321,7 +320,6 @@ 65B9BD082AAE18570086F9EC /* FileUploadAssessmentViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FileUploadAssessmentViewModel.swift; sourceTree = ""; }; 65B9BD0F2AAE413B0086F9EC /* FileUploadSubmissionServiceImpl.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FileUploadSubmissionServiceImpl.swift; sourceTree = ""; }; 65BA61D62AE02B5C009FE11C /* PathStringExtension.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PathStringExtension.swift; sourceTree = ""; }; - 65BC14522A6FD9F30002D089 /* UMLBadgeSymbol.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UMLBadgeSymbol.swift; sourceTree = ""; }; 65C2F7522A03E3D4001EDADC /* FormatterExtension.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FormatterExtension.swift; sourceTree = ""; }; 65C2F7542A03E428001EDADC /* JSONDecoderExtension.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = JSONDecoderExtension.swift; sourceTree = ""; }; 65CAE5492A9E0FD50083CFDF /* UMLElementType.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = UMLElementType.swift; sourceTree = ""; }; diff --git a/Themis/Assets.xcassets/Robot.imageset/robot-solid.png b/Themis/Assets.xcassets/Robot.imageset/robot-solid.png deleted file mode 100644 index af5227233145149ec4e54e98a5f06a234aaacd6a..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 5840 zcmeHLc{G%7-@nHwS&J-5mPE*sl%?!hLb4{y6jI0@vZYyCkRL)(_8}oVg|Wm;n?muE zj2O%qG#E2uUuH4y_&x7=-}gMvbDs0O|GwwF&$++Xxjx_TwVwO?xjxsO>|l36knb=b z004s4R_0Csz=hzvz&gMhNxq8jh%lo>|jhnuH z{&$6vd z{T#-Q{2Cp@jZaMC2~))Bnct+j`33SKg-TmmURhmR-`L#RrZbqVoky|ZbxdD|rQkd_)EI#GWWe1<|Gx9>+~9T{Jzg z5lRVmVQ^`lq%oS#SE1V_K!5e1t ze@I&QpT(2oyGBU#E>AdSGrW+-Q0Zywaa6IXPN6}ZnpwBaAdrt!6qk!#h8gl}Hi)5_^YYbEKUnnhREb>mm{PkEL1F5l+?tWY zg-$_nTDHd_mClPQRLR_+Ji9iYwxF9;s%R@|?`hYzu$yxdn2>X6Eh|0(xdBCX)uZEQ zwHzIxcLDO&O|INI|8w7q2$PwcE5`N`gvylG_?b3~bCjLLjtfQl8Ace^xZ%o}8>PgD zDjQexgCE1kDT9#`h>lkHOf87_Byf<~sToH$qEyDWJ)EgjoCJ)zr4{U5;=o%c$4ys$ zMu!3He0{5;a|?om>r}b8s!1t~a!iJS*y`SfY!3cIW?D!D;6<1N3V;YG1F+&Y;gS>n zdjNC*v5RwHZWCZPd^Z{ZS#b(3P8)z*j?;7>CjykQj}s!#aVB$Mg?|eCE%0CW=qaT+ zrJv3oQq|@u74$z-_A57W?tPxCXY8W>G~<}cx<*lZ@N~Fn?)+AMA64>kKb~NT|V=t>`r( zGGP#3hQ1L88H={wVnQj(9k`)J@`vBF{*!JK<#q{+wQW5HtcIFjs1Gw?ccoHrSH7e! zZK+N_bk*T25EqAUF42&MTxjX1B-v9fEpM^7kA$!pl3GZR*CRqmo9Vk9|5ft+hlT7d zej5B2tLQOV_DS6RG)idpG!lOOIxc^9?cf0;Xc$zz^vDi?wcBV$8&Y9WANa7@4sF#t zp+uUoxu`6u9R1a^C3`Sbm+cYB+h-Hdn6jf0t z&sh?xwpmagVNSd+f73pE3UQD9s$nvw53lQX*<-qptF_3qtQo5acU?YS&`Q*RzJMvl z_Xf3@AwSmp+r;y0d=mb!D!T*hb24Hb75+tC;6n`Q&{VRIZa> z{`k_1ff!+4_2(pROuV~OpmPDRWngAQkyHX?e5@*^@4UyGM2kXv05q3V#smex_`iD1B9vvmB)*(FO&Rhv2o15Cc8V}v3R)bDNHlby`7rFYDR>NxqhRPGtO zaF%IWK8H=_`RXeFI5Z^|A$#iOjL>5qx4y?u^)+M&DwbJSAI2h>^#SH13Nmu+$UB2k zk9oc(z5&Z|l4Lg#{(bwODf6#0fz#z~qWzoJOt2x^Y?g_;kTxfkye54rkjV7vigUtJ zu?4b-%8@2&rlWh|K&JNHslM-8OYTuHpxo_V-#D|b|Eci}XcC1UtZxiBSIy2(P00Fx zI$f)?`j*B7Y-jyPPHN0=n>;*`SHl_&n%V~C+#@F`#=Lrea&&&6cNMHTYLr{REGa+I zoQ^Cb@pwP2rR4WkuDIfU4`5dzAC%dNJTK}d+ZhgKYQtc^-#X12R?8q?L!aSi!tn#h zg|5a#vyPTJPdY(x`}b0|23}G(+Qh3$Dxdz5ez((bN6iR6#^{dGKn4TX-!I1KGtU;k zH|jiBn546E8p0gC2eW6ipig&TQo3raht<1e{R^Q=uamdx+)M9&LLF-M^=FoRkcZhCk>^?|#r{{lh=4sSolf)g)zEX5bkl62qD zE}VYr481iS?YDyPEHZwwzH+g<6@%o*!C+<#s&{04_+1^m(j0tWaED--G zh28q-4u?fDEGrbYbL9IDdhlWF`3!_!Yw)clKW341Tf<(; zE9G7|6AW?Su`TWCjdXs%eT%)FMr9za4W`f~SNcG$*iZU~_1q+T%=ULG1V8Our+>Kl z2;!32B>_Ey2K$AJdgv*rz5U`&gfpZ`4$=4R-Ds!0GD4(3>@NP-8nsTxux<}1xb4$7 z#EZRaKE;n$rSQ5&wd9luY?UJD-q=qP6ecEMB23anhv9TP?(Gb|Rs^N)*VQJ>Cd)B< z^w9emMH*D%`io~Cq8QY7=;X$P9MG$WmStdzHhWzAuc-4AHopn#>F+~m_0#&gaPp+K z9}(*O-jVm)K&#+bMa!?-tWgxx;N)wiN1&?}0vmz9L40S?#*E`O`d$^U?fAi4k9$Dy ztT@E^e%esi;0L>rIMa(@z1|tbh_6g>#ww$kR%Is}Y-6zHOVgW>UO@XX%7;-bMD#s16EL^_PA)iz4e{dHHaq&&hoW(Us>gi^j%42W8H}bCb*n$N(t2)v(Sl_S(KI51zbr4iW8-W0vD&7Z7h* zOki>Tb5pqIM*ac*!^ISi&%Ur-hsCnQk}+?Y9B<=|0Cw6Yp>*uQqLDEWPbFlmZAh(@ zw8&Qv?8hk3k%RM#G2bi3o(6Ng&nk`5@H0whZUw#B;_zeDhJ{L3(eq}_a`oOx4jcAh z*Dv{^nc8D-f(<#r6e~VjwH@m+yAd#%2oG#AB6_Y4}8 zf3v_r>*u<&ja#S3gjw>Luw!+dt30Z@!vT)&kuboi_%DIBtJe0J{JZvl>bie%NO2B4 zjtu-?+JCMd|6vr)vBwdc|7Cl`5T^jA+l-c~ySs(uu<&jnb0r?IdQa)NI1FSwz{R9^ zfKB*9@qx{v2+{E_``1akPDptux7;37SYvNGIOD;XZc)WgJHW|-0V~e(5HjHl#Cs{1 z*?_D}DGUjl%6HL#lP05uq+0e zPtS!+9O@?HP+^ffvUQFWF3j|5eDy1GC&BzzUD$M&i+}woYagFm?<;=?RRWL0zNk!w8P3>SCS)%P5dWz3QdPKp!m&6Xrhpu5Td*Qj{LdbxjK&Y25j`A#s0OkY>$j-?NDwEwED1-OgxjBe^8J#dIf z1Xhl@9l~riQ8jd%6O@h`@GeL0o>uj`%T1by$WB4Mfps!u@6y_*A`Q2zs{G!~D3DhK zX9^$#$MIFO_8Lo3%xUPm#gUFdPTZa)bfVH~al{Fc2f)G}&%Mkp#t4eX`=x#7Ai(nCGq`ucz~|4v&x=1~mzFiJ;)M{)E(t1zO~eo%pk?AI6*hh2 zdTW;$5c7%|!!wF!SnV^prhlfBm1yxXoJ9b^wk8(O$arml<8J~#89zL3Z|yW(HTFhy zRQOkH+uy^!s`BFn)_Zs>ja&0er`li$Aiw&!Z(LlFmu6682MdS_4*w%;fn+4kKeP_! z`qdTZcYo#Twz3PKWAy-<`Rvy z(^Y(R`h*b{lwg&{mh%^Ysady^)`owrW zTQA;56_#KI44>PZjYW6Nq@x8gU6Y-GVR7;7Vg1l;XVmG+ny#rsCe68*6q|dr;Z+Ln zt0HVt!-b0Jx_;S1#vLSkL8Fg#PJu^eo`@M$hW;I;2g(N9A6| zCPy?ybZ7rLaBaOJV*99CF-6!PR27u>Yh)v@tLjLoI2bqSb)sTANJNbmfMQjShXe;l zLXJHFUzU))?i!rFhq@gx-1nHAVBS($8H|Y?3wJ8j4V6>%iJH} z`+AFBKTR>X*7j29$c{v9D_799PBlF!nC*4eD=40|bG0DKb#VOBse&zMa`$M$K60B; z40{^Xlmi>|We=5gHfG;p(O@fSHyTb2`G+xCS@O1Jp6GDj6l8o{-~DHTY05Uu-HBr? kT0La_mbXISUz$N83}YDvGe|~+7BiyA8j&!JUDk}OV;D5Zo~3Lp z))<2j5oJpyktDu-KacOl_v-f#{J!ULE}rN0I5+2Y@xC|-b~fffUNA2J003HAm|_6{ z7Iy%E6~N8*H%X$HJp};R0Cv_6$iK(`t^YOf|FeOkqoamY2Kn#f4+F6__5c=EHg*n9 zE^Z!PK7OEppb$t{L{v;%;@ENU2}!Awr=%e=vU2hYr=f~U$||aA>Kd9_XJBw`9o@5f z=kyKE8yXoSOiYnx<`$MHD{HiktsMqy@8EdB>7p~v<&vx0Wp@wHD_5^w_reprZ}|A$ zyyfSAJ0LJ9I3)B=SoqzD$a_)uqhn%;aq*;t#H8fZwDboVnOWI6xexQm`Hu<;i;7E1 zDP`313R-1V^<#QXZQYalhQ_Amme!|j?H!%Zo_BTk^!B}Y`RaB5z~IpE$eYnI#yE3g za_a5$%wTi^J+`Q_`kt?%1Ec6NX6{o4P1@aM4P8u7{BCRjYJ z%uSD|T;E>+03dHmQ)7o?$2YtA9;JLNv(6Iz*ec;0quYsw$@%;1J@VB~_ZkRR{{E`Ymo6RjmZNW6P|y24)ElxQ@OkVR zOY%kdrq+)_FT=O1UNFSD3qz4Vng{i?qd?9LNoH>@Iu3=hYdwR>V6#DuEh&1NdVQ`j(W8cB`%(euoGFW8GiliTBEIdkeicW zU^Zi8Yp3z@SoPF(gdxG?)8>`Zy`x|g&cKYjhkR6Pm7|__{HxsohiBT?*x$RDbog~n z3i?t5g3)UL9jo;-&vOJ^478-wf(~Mbv$~(Hd36!PMyktI5J~$z*G4W|Jbf&c$2s-Q zt%3Y~Iy*%25+vfsg@MYuC_ZTISZBCm?RRss31R-hIcUrGgX?xVE96^>&+8AYX@mtQ zhu$x&_a4=C=1+}t zLtv;&Vfe-)Ej@abaxD0m`-ja-`tX+7Et&}h^q%42k{V;QxvTjEC4pX_74`^ya zb8df>qKL=vUNZ^vQ%P?-ju~4TpoVSdIHxN$0Z_KH?t|p}|ki$<&Ij}9_t~O(M zOt`vt^MvE`7m|H0PNmq)rpN86mieCFWChi)wwepI>^f8>j2{2Ejq-5f^xQf5BK!1^S=KB9ddavW^5Af7DTV2i`~aF zA(u9SvIYuBxItlFv4NOSgJ7&$I9VDuuIMlvB%~G;T~z;65aVrc^5DP;(fH#*KD4Ow zIB8^Cw6gw`DQb8<_;<%`)EbK^wW)|T+Z9mF&~uHPYMe5J?v=E*W@srBQt{RmM%fZw zi9?4XE56)zVrkO6ryVhp_YC{!({(F{pfc=$i%k7dxh_0LsA60}P0-H39yG>to>!zp zz@K2~uR7sCPOvD)KGGza#BAio(H{Ggiv4+rd8uy;vYy8)DgTh^Xon84I4g8 z`WA_}^2s$Y$?i1s>mAJ7srd&CO_iD#(i^2PD&v}OS_Wr)aCIAgp^&&U0%$C7<@tNqnj{}Ae3UM&=nzi zK;kOOu)I`_P0`d?s}x3yR)Zf4ay9~a>&c?-@s83zl?b4#6?RrMUy=aj2<07H#Tf(nyC1kwf}V+hh$2a}7zuxNTM#`z z5gKYp73Ve3z2t>+nij@{lF7Pk@$K=G1~E{6irULXza)Zloq^1`U#HhUT#-~5HXocv zT(vx9e_{gdXJLp7IVp3(DuurXGLCmnLQ>X)*m(@Zo$ zOF%a}=c9Cnjj{KNwdA*OT$3dK>QVjhtYqokC$?g zt{&fk3I3*2ks&hCyO4DPTjVEtInqleI)UjxmoVbOIgpdNC$UbxEI%m7Bq3vS?+s5t z3!~JMhtfk8fHBW8;b+_oWitw)qzxX_zXvKPqhw70L`CkB~IX#OK}=8{S|oFX*u%A64v z9VKso=3(BO0gtKh2++KW_okIkG2BbG1Ie5T^C^X&<-R4>v@69+Kr6?#SPtpg#@xu}9wlFml_;KMo%R6y(e)2$~|or%4YnIygRT&U}vE7vES_b#8sE*LLoYQ~-W@WRf;+Ia}H39GWvp_a*8z zx|v_mQ?QSkllXAVAegZbbIb}hcVgi`{|x6Zg?G#_IY_KZn49`TQ>< zE5_^kE8o7^sPIUt5*;iXr<`Hwa=XgCT&aVmmEuR(KqxuSxHYW51BFRXL^OIOiH;V*EJ5m=l zibjQecm#A3;@n02S_25&!Quidie(Vy1;NyBlE9L}o(F0ah{G+(O5@|e`zWk`mJWd% zcEY^)3aWo0&vGXlWr_z`?{0+LSCzWzghLJsV7= z%9p4i&Zt}UjUCU8LrTa#RwrD1dLy;Cp=FL)2!-kN-o7-Sb+@djm8z-SpE>z9)J2>UxkdaTSIu#UCF9M<@z&!ob zJ+qth%$VA6qoMtA(#%IvuOt{lCWbb?HIY>cZxG6L!JEoUY;CEOfQZzSeQTnpQbY`8F0*=yeb9MU13>Q?5LSihpP9mR7n(i>>$wQ1Cf}rBYBz^HoZm~Qgt7lN zit2~0Z%Lb4?5CR*oVSS+a-JnHOX}p=*WA!c$KWd_51+tN{dM}bI?zi{K7AqD77lE_ zr^kk9nno^ZwAVx4o;?qKUb)V)7)QiQinf8YA6hRJ$GdaA$%jR5m2)-YFB&a*-C%fR zW(JQg)J)c)1r3{=A|O6lR+}kN^m0rJ2P5t|(-gRWMb9Q3vfMUso z-6oT0S2X=nT5R}z@Ew7K+(Xr-CzYC#qi2lJ?CP+-^*tn9!}^-e!K^HZ)hRU(q`95o&A2x>pJX4hF*GG~dBM$=zNPmlVnBPfvW*pi3 zR~w|y0F-J@$^CrpFyoaz$Kilex45M#wj1*L>m#9TvHcL}f~(R7nzk?`BqpZMh6+tf z4CC`&6)q)@+?Nr@SbjATGoY1i7bjnH7!{^bbi@KRMe8#LoH?@jD{j{iQP^gzrVF+> zE1dSQy(C9L-T_766g`uI>W}x1lqf^6PTP$jl$Kp-UI4M%ipT5G^jb>v0FDTKYIg%lkIMKh>D`{2aYC>usARdVOM}1n|J)|REGSjL z1P8PUM^iJZoInJzpy=XN)l{Z;aT^Xr=ll=B|FDEvRnqpC;Hv8Kf7AOvJg_S$!fu-c zMWTiUCw?iy4IP}2m5#Lv5a;eK~{Lw5Qz^plY9}8l*747q@k1&A8KQ;M7U*!0#O4-{R}}HVFoNJFeCV zuIxMV5*XaMj0ILf4DOC!Oy&(}B{_3}R~>^@?~$tilZ6d(Y3-wAF(`gXarw{^`kG%# zBxIn%@X|68wQ{Z1^5q58b@MZ=sINjF8K=~?g{fL>H#47P3r~;n*NmSwy~7{PH(U&J zBc8bCLy=1)cW3X%LgT9!u(fu)_lPdnZc^C)hE4l%s~yz~p0y1B+U{YkmN5_H@S-={ zYF)SA)8`D#nZ@8dVxS!+pmp9z53d3H2A0~^3ru&mi2v*(pnAzBp8vkWbq1)RIH`Pg zEox4+BxP_ds$}%_hIQ-osO1vSy6&r1N%pSl0Y{L9pPG1}p>d|?|qK;IvKsH1K%?b_oC`R3Ga0JcWC z7 [TextBlockRef] { + func getFeedbackSuggestions(exerciseId: Int, submissionId: Int) async throws -> [TextFeedbackSuggestion] { try await client.sendRequest(GetFeedbackSuggestionsRequest(exerciseId: exerciseId, submissionId: submissionId)).get().0 } } diff --git a/Themis/ViewModels/Assessment/AssessmentViewModel.swift b/Themis/ViewModels/Assessment/AssessmentViewModel.swift index 5692e442..318b8f04 100644 --- a/Themis/ViewModels/Assessment/AssessmentViewModel.swift +++ b/Themis/ViewModels/Assessment/AssessmentViewModel.swift @@ -218,7 +218,11 @@ class AssessmentViewModel: ObservableObject { func getManualFeedback(byId id: UUID) -> AssessmentFeedback? { let manualFeedbacks = assessmentResult.inlineFeedback + assessmentResult.generalFeedback - return manualFeedbacks.first(where: { $0.id == id }) + return manualFeedbacks.first(where: { !$0.isSuggested && $0.id == id }) + } + + func getSuggestedFeedback(byAssessmentFeedbackId id: UUID) -> AssessmentFeedback? { + assessmentResult.suggestedFeedback.first(where: { $0.id == id }) } } diff --git a/Themis/ViewModels/Assessment/Text/TextAssessmentViewModel.swift b/Themis/ViewModels/Assessment/Text/TextAssessmentViewModel.swift index 03839eb6..d15c1cf0 100644 --- a/Themis/ViewModels/Assessment/Text/TextAssessmentViewModel.swift +++ b/Themis/ViewModels/Assessment/Text/TextAssessmentViewModel.swift @@ -11,10 +11,10 @@ import SharedModels import CodeEditor class TextAssessmentViewModel: AssessmentViewModel { - private var suggestedBlockRefs = [TextBlockRef]() + private var suggestions = [TextFeedbackSuggestion]() private var shouldFetchSuggestions: Bool { - !readOnly && submission?.isAssessed == false && assessmentResult.automaticFeedback.isEmpty + !readOnly && submission?.isAssessed == false && assessmentResult.suggestedFeedback.isEmpty } @MainActor @@ -96,6 +96,7 @@ class TextAssessmentViewModel: AssessmentViewModel { } } + @MainActor private func fetchSuggestions() async { guard let exerciseId = participation?.exercise?.id, let submissionId = submission?.id else { @@ -104,71 +105,79 @@ class TextAssessmentViewModel: AssessmentViewModel { } do { - var blockRefs = try await AthenaService().getFeedbackSuggestions(exerciseId: exerciseId, submissionId: submissionId) - log.verbose("Fetched \(blockRefs.count) suggestions") + var fetchedSuggestions = try await AthenaService().getFeedbackSuggestions(exerciseId: exerciseId, submissionId: submissionId) + log.verbose("Fetched \(fetchedSuggestions.count) suggestions") - blockRefs = removeOverlappingRefs(blockRefs) + fetchedSuggestions = setTextBlockContent(of: fetchedSuggestions) + fetchedSuggestions = removeOverlappingSuggestions(fetchedSuggestions) - suggestedBlockRefs = blockRefs - await saveBlockRefsAsAssessmentFeedbacks() + self.suggestions = fetchedSuggestions + saveSuggestionsAsAssessmentFeedbacks() } catch { log.error(String(describing: error)) } } @MainActor - private func saveBlockRefsAsAssessmentFeedbacks() { - var suggestedFeedbacks = [AssessmentFeedback]() - - for index in 0 ..< suggestedBlockRefs.count { - let blockRef = suggestedBlockRefs[index] - let assessmentFeedback = AssessmentFeedback(baseFeedback: blockRef.feedback, - scope: .inline, - detail: TextFeedbackDetail(block: blockRef.block)) - suggestedFeedbacks.append(assessmentFeedback) - suggestedBlockRefs[index].associatedAssessmentFeedbackId = assessmentFeedback.id + private func setTextBlockContent(of suggestions: [TextFeedbackSuggestion]) -> [TextFeedbackSuggestion] { + guard let textSubmission = self.submission as? TextSubmission else { + log.error("Expected a TextSubmission but got \(type(of: self.submission)) instead") + return suggestions } - assessmentResult.computedFeedbacks += suggestedFeedbacks + var result = suggestions + for index in 0 ..< suggestions.count { + result[index].setTextBlockContent(from: textSubmission) + } + return result } - private func removeOverlappingRefs(_ blockRefs: [TextBlockRef]) -> [TextBlockRef] { - var rangeToBlockRef = [Range: TextBlockRef]() + @MainActor + private func saveSuggestionsAsAssessmentFeedbacks() { + var suggestedAssessmentFeedbacks = [AssessmentFeedback]() - for blockRef in blockRefs { - guard let startIndex = blockRef.block.startIndex, - let endIndex = blockRef.block.endIndex else { - continue + for index in 0 ..< suggestions.count { + let suggestion = suggestions[index] + + var feedbackDetail: TextFeedbackDetail? + if let suggestionTextBlock = suggestion.textBlock { + feedbackDetail = TextFeedbackDetail(block: suggestionTextBlock) } - let blockRefRange = startIndex.. TextFeedbackSuggestion? { - var suggestionBlockRef: TextBlockRef? + private func removeOverlappingSuggestions(_ suggestions: [TextFeedbackSuggestion]) -> [TextFeedbackSuggestion] { + var rangeToSuggestion = [Range: TextFeedbackSuggestion]() + var result = [TextFeedbackSuggestion]() - if let blockRef = suggestedBlockRefs.first(where: { $0.associatedAssessmentFeedbackId == id }) { - suggestionBlockRef = blockRef - } - // this happens when `fetchSuggestions()` is not called and automatic feedbacks embedded into the submission are used - else if let assessmentFeedback = assessmentResult.automaticFeedback.first(where: { $0.id == id }), - let textSubmission = submission as? TextSubmission, - let blocks = textSubmission.blocks, - let block = blocks.first(where: { $0.id == assessmentFeedback.baseFeedback.reference }) { - var blockRef = TextBlockRef(block: block, feedback: assessmentFeedback.baseFeedback) - blockRef.associatedAssessmentFeedbackId = id - suggestionBlockRef = blockRef + for suggestion in suggestions { + guard let startIndex = suggestion.indexStart, + let endIndex = suggestion.indexEnd else { + result.append(suggestion) // preserve unreferenced suggestion + continue + } + let range = startIndex.. simply add a new feedback for the given suggestion - if newFeedback == nil { - assessmentResult.addFeedback(feedback: feedback) - } - - feedbackDelegate?.onFeedbackSuggestionSelection(feedbackSuggestion, feedback) - } private func setStates() { if idForUpdate != nil { @@ -175,10 +174,6 @@ struct EditFeedbackViewBase: View { self.score = feedback.baseFeedback.credits ?? 0.0 } } - if let feedbackSuggestion = feedbackSuggestion { - self.detailText = feedbackSuggestion.text - self.score = feedbackSuggestion.credits - } } } @@ -189,8 +184,6 @@ struct EditFeedbackViewBase_Previews: PreviewProvider { static var previews: some View { EditFeedbackViewBase(assessmentResult: result, feedbackDelegate: cvm, - title: "Title", - isEditing: false, scope: .inline, gradingCriteria: [ .init(id: 1, structuredGradingInstructions: [ diff --git a/Themis/Views/Assessment/Text Exercise/TextAssessmentView.swift b/Themis/Views/Assessment/Text Exercise/TextAssessmentView.swift index 53b54d62..f6ad3356 100644 --- a/Themis/Views/Assessment/Text Exercise/TextAssessmentView.swift +++ b/Themis/Views/Assessment/Text Exercise/TextAssessmentView.swift @@ -75,23 +75,19 @@ struct TextAssessmentView: View { gradingCriteria: assessmentVM.gradingCriteria, showSheet: $textExerciseRendererVM.showEditFeedback ) - } else if let textAssessmentVM = assessmentVM as? TextAssessmentViewModel, - let suggestion = textAssessmentVM.getSuggestion(byAssessmentFeedbackId: textExerciseRendererVM.selectedFeedbackForEditingId) { - // The user tapped on a feedback suggestion. This is not actually an edit action, but is triggered as one - AddFeedbackView( + } else { + EditFeedbackView( assessmentResult: assessmentVM.assessmentResult, feedbackDelegate: textExerciseRendererVM, - incompleteFeedback: AssessmentFeedback(scope: .inline, - detail: TextFeedbackDetail(block: suggestion.blockRef.block)), - feedbackSuggestion: suggestion, scope: .inline, + idForUpdate: textExerciseRendererVM.selectedFeedbackForEditingId, gradingCriteria: assessmentVM.gradingCriteria, showSheet: $textExerciseRendererVM.showEditFeedback ) } } - .onChange(of: assessmentVM.fontSize, perform: { textExerciseRendererVM.fontSize = $0 }) - .onChange(of: assessmentVM.pencilModeDisabled, perform: { textExerciseRendererVM.pencilModeDisabled = $0 }) + .onChange(of: assessmentVM.fontSize) { textExerciseRendererVM.fontSize = $1 } + .onChange(of: assessmentVM.pencilModeDisabled) { textExerciseRendererVM.pencilModeDisabled = $1 } } private var correctionWithPlaceholder: some View { From 07a0b334d59e078b8bfb4ae8b58a4aa4d0e2c3a3 Mon Sep 17 00:00:00 2001 From: Tarlan Ismayilsoy Date: Thu, 26 Oct 2023 21:32:28 +0400 Subject: [PATCH 17/21] Add badge to feedback cell --- .../CorrectionSidebar/FeedbackCellView.swift | 50 +++++++++++++++++-- 1 file changed, 47 insertions(+), 3 deletions(-) diff --git a/Themis/Views/Assessment/CorrectionSidebar/FeedbackCellView.swift b/Themis/Views/Assessment/CorrectionSidebar/FeedbackCellView.swift index 3d8ba54b..8428f0a6 100644 --- a/Themis/Views/Assessment/CorrectionSidebar/FeedbackCellView.swift +++ b/Themis/Views/Assessment/CorrectionSidebar/FeedbackCellView.swift @@ -37,10 +37,9 @@ struct FeedbackCellView: View { let gradingCriteria: [GradingCriterion] var body: some View { - VStack(alignment: .leading, spacing: 5) { + VStack(alignment: .leading, spacing: 10) { HStack { - Text(feedbackText) - .foregroundColor(.getTextColor(forCredits: feedback.baseFeedback.credits ?? 0.0)) + feedbackTextLabel Spacer() @@ -90,6 +89,17 @@ struct FeedbackCellView: View { .scaleEffect(isTapped ? 1.05 : 1.0) } + @ViewBuilder + private var feedbackTextLabel: some View { + if feedback.isSuggested { + suggestionBadge + } else { + Text(feedbackText) + .foregroundColor(.getTextColor(forCredits: feedback.baseFeedback.credits ?? 0.0)) + } + } + + @ViewBuilder private var pointLabel: some View { Text(String(format: "%.1f", feedback.baseFeedback.credits ?? 0.0) + "P") .font(.headline) @@ -99,6 +109,7 @@ struct FeedbackCellView: View { .cornerRadius(5) } + @ViewBuilder private var editButton: some View { Button { showEditFeedback = true @@ -112,4 +123,37 @@ struct FeedbackCellView: View { .disabled(editingDisabled) .isHidden(feedback.baseFeedback.type?.isAutomatic ?? false) } + + @ViewBuilder + private var suggestionBadge: some View { + HStack(spacing: 4) { + Image("SuggestedFeedbackSymbol") + .renderingMode(.template) + .resizable() + .scaledToFit() + .frame(width: 15, height: 15) + + Text("SUGGESTED") + .font(.footnote) + .fontWeight(.bold) + } + .frame(height: 18) + .padding(8) + .foregroundStyle(.white) + .background(Color.feedbackSuggestionColor) + .cornerRadius(5) + } +} + +struct FeedbackCellView_Previews: PreviewProvider { + static var assessmentVM = AssessmentViewModel(exercise: .mockText, readOnly: false) + static var assessmentResult = TextAssessmentResult() + + static var previews: some View { + FeedbackCellView(assessmentVM: assessmentVM, + assessmentResult: assessmentResult, + feedback: AssessmentFeedback(scope: .inline), + gradingCriteria: []) + .padding(.horizontal, 200) + } } From b1bf578eac33d66312e1f008b4cfb9a9da6063f9 Mon Sep 17 00:00:00 2001 From: Tarlan Ismayilsoy Date: Fri, 27 Oct 2023 19:02:03 +0400 Subject: [PATCH 18/21] remove unrelated files --- docs/.idea/.gitignore | 8 -------- docs/.idea/docs.iml | 9 --------- docs/.idea/misc.xml | 6 ------ docs/.idea/modules.xml | 8 -------- docs/.idea/vcs.xml | 6 ------ docs/.vscode/ltex.dictionary.en-US.txt | 3 --- docs/.vscode/settings.json | 9 --------- 7 files changed, 49 deletions(-) delete mode 100644 docs/.idea/.gitignore delete mode 100644 docs/.idea/docs.iml delete mode 100644 docs/.idea/misc.xml delete mode 100644 docs/.idea/modules.xml delete mode 100644 docs/.idea/vcs.xml delete mode 100644 docs/.vscode/ltex.dictionary.en-US.txt delete mode 100644 docs/.vscode/settings.json diff --git a/docs/.idea/.gitignore b/docs/.idea/.gitignore deleted file mode 100644 index 13566b81..00000000 --- a/docs/.idea/.gitignore +++ /dev/null @@ -1,8 +0,0 @@ -# Default ignored files -/shelf/ -/workspace.xml -# Editor-based HTTP Client requests -/httpRequests/ -# Datasource local storage ignored files -/dataSources/ -/dataSources.local.xml diff --git a/docs/.idea/docs.iml b/docs/.idea/docs.iml deleted file mode 100644 index d6ebd480..00000000 --- a/docs/.idea/docs.iml +++ /dev/null @@ -1,9 +0,0 @@ - - - - - - - - - \ No newline at end of file diff --git a/docs/.idea/misc.xml b/docs/.idea/misc.xml deleted file mode 100644 index 639900d1..00000000 --- a/docs/.idea/misc.xml +++ /dev/null @@ -1,6 +0,0 @@ - - - - - - \ No newline at end of file diff --git a/docs/.idea/modules.xml b/docs/.idea/modules.xml deleted file mode 100644 index 6049cfe0..00000000 --- a/docs/.idea/modules.xml +++ /dev/null @@ -1,8 +0,0 @@ - - - - - - - - \ No newline at end of file diff --git a/docs/.idea/vcs.xml b/docs/.idea/vcs.xml deleted file mode 100644 index 6c0b8635..00000000 --- a/docs/.idea/vcs.xml +++ /dev/null @@ -1,6 +0,0 @@ - - - - - - \ No newline at end of file diff --git a/docs/.vscode/ltex.dictionary.en-US.txt b/docs/.vscode/ltex.dictionary.en-US.txt deleted file mode 100644 index e2cdeb96..00000000 --- a/docs/.vscode/ltex.dictionary.en-US.txt +++ /dev/null @@ -1,3 +0,0 @@ -Themis -includehidden -maxdepth diff --git a/docs/.vscode/settings.json b/docs/.vscode/settings.json deleted file mode 100644 index fbbc9860..00000000 --- a/docs/.vscode/settings.json +++ /dev/null @@ -1,9 +0,0 @@ -{ - "esbonio.sphinx.confDir": "${workspaceFolder}", - "cSpell.words": [ - "includehidden", - "maxdepth", - "themisml", - "toctree" - ] -} \ No newline at end of file From ff4002cb73ed5290646790c8e6f45cca6326986a Mon Sep 17 00:00:00 2001 From: Tarlan Ismayilsoy Date: Mon, 30 Oct 2023 21:55:40 +0400 Subject: [PATCH 19/21] Fix 2 swiftlint warnings --- .../CodeEditor/Models/Suggestion/FeedbackSuggestion.swift | 2 +- .../CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/CodeEditor/Sources/CodeEditor/Models/Suggestion/FeedbackSuggestion.swift b/CodeEditor/Sources/CodeEditor/Models/Suggestion/FeedbackSuggestion.swift index e5e8c6d8..df557344 100644 --- a/CodeEditor/Sources/CodeEditor/Models/Suggestion/FeedbackSuggestion.swift +++ b/CodeEditor/Sources/CodeEditor/Models/Suggestion/FeedbackSuggestion.swift @@ -17,5 +17,5 @@ public protocol FeedbackSuggestion: Equatable, Decodable { var credits: Double { get } var gradingInstruction: GradingInstruction? { get } - var associatedAssessmentFeedbackId: UUID? { get set } // not decoded + var associatedAssessmentFeedbackId: UUID? { get set } // not decoded } diff --git a/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift b/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift index b0067e17..42ec57ea 100644 --- a/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift +++ b/CodeEditor/Sources/CodeEditor/Models/Suggestion/TextFeedbackSuggestion.swift @@ -30,7 +30,7 @@ public struct TextFeedbackSuggestion: FeedbackSuggestion { public let indexEnd: Int? public static func == (lhs: TextFeedbackSuggestion, rhs: TextFeedbackSuggestion) -> Bool { - lhs.id == rhs.id + lhs.id == rhs.id && lhs.exerciseId == rhs.exerciseId && lhs.submissionId == rhs.submissionId && lhs.title == rhs.title From 7716c4d6fbb46a8a9cd26b6d8ede13cb5cecd527 Mon Sep 17 00:00:00 2001 From: Tarlan Ismayilsoy Date: Sun, 5 Nov 2023 23:19:46 +0400 Subject: [PATCH 20/21] Fix package version and merge issues --- CodeEditor/Package.swift | 6 +++--- .../EditFeedback/EditFeedbackViewBase.swift | 13 ------------- 2 files changed, 3 insertions(+), 16 deletions(-) diff --git a/CodeEditor/Package.swift b/CodeEditor/Package.swift index 31396d79..729b920b 100644 --- a/CodeEditor/Package.swift +++ b/CodeEditor/Package.swift @@ -1,4 +1,4 @@ -// swift-tools-version:5.7 +// swift-tools-version:5.9 import PackageDescription @@ -7,7 +7,7 @@ let package = Package( name: "CodeEditor", platforms: [ - .iOS(.v16) + .iOS(.v17) ], products: [ @@ -16,7 +16,7 @@ let package = Package( dependencies: [ .package(url: "https://github.com/raspu/Highlightr", from: "2.1.2"), - .package(url: "https://github.com/ls1intum/artemis-ios-core-modules", from: "3.3.2"), + .package(url: "https://github.com/ls1intum/artemis-ios-core-modules", from: "6.1.0"), ], targets: [ diff --git a/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/EditFeedbackViewBase.swift b/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/EditFeedbackViewBase.swift index fa046169..85f7e1b5 100644 --- a/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/EditFeedbackViewBase.swift +++ b/Themis/Views/Assessment/CorrectionSidebar/EditFeedback/EditFeedbackViewBase.swift @@ -92,19 +92,6 @@ struct EditFeedbackViewBase: View { .background(Color.getBackgroundColor(forCredits: score)) } - @ViewBuilder - private var textField: some View { - TextField("Enter your feedback here", text: $detailText, axis: .vertical) - .foregroundColor(Color.getTextColor(forCredits: score)) - .submitLabel(.return) - .lineLimit(10...40) - .padding() - .overlay(RoundedRectangle(cornerRadius: 5) - .stroke(lineWidth: 2) - .foregroundColor(.getTextColor(forCredits: score))) - .background(Color.getBackgroundColor(forCredits: score)) - } - private var gradingCriteriaList: some View { ScrollView(.vertical) { VStack { From 54c7a696fd44ed553d89f38609a36cf06b7c8ead Mon Sep 17 00:00:00 2001 From: Tarlan Ismayilsoy Date: Mon, 20 Nov 2023 22:15:29 +0400 Subject: [PATCH 21/21] Update CodeEditor's ArtemisCore dependency --- CodeEditor/Package.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CodeEditor/Package.swift b/CodeEditor/Package.swift index 729b920b..2c47b1fe 100644 --- a/CodeEditor/Package.swift +++ b/CodeEditor/Package.swift @@ -16,7 +16,7 @@ let package = Package( dependencies: [ .package(url: "https://github.com/raspu/Highlightr", from: "2.1.2"), - .package(url: "https://github.com/ls1intum/artemis-ios-core-modules", from: "6.1.0"), + .package(url: "https://github.com/ls1intum/artemis-ios-core-modules", from: "7.0.0"), ], targets: [