Skip to content

Commit 93aa305

Browse files
Internal: Fix vote handling & course ranking updates
1 parent 5bbca45 commit 93aa305

13 files changed

+396
-466
lines changed

assets/vue/services/trackCourseRankingService.js

-26
This file was deleted.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,100 @@
1+
import baseService from "./baseService"
2+
3+
/**
4+
* Saves a new vote for a course in the catalog.
5+
*
6+
* @param {string} courseIri - IRI of the course
7+
* @param {number} userId - ID of the user who votes
8+
* @param {number} vote - Rating given by the user (1-5)
9+
* @param {number} sessionId - Session ID (optional)
10+
* @param {number} urlId - Access URL ID
11+
* @returns {Promise<Object>}
12+
*/
13+
export async function saveVote({ courseIri, userId, vote, sessionId = null, urlId }) {
14+
return await baseService.post("/api/user_rel_course_votes", {
15+
course: courseIri,
16+
user: `/api/users/${userId}`,
17+
vote,
18+
session: sessionId ? `/api/sessions/${sessionId}` : null,
19+
url: `/api/access_urls/${urlId}`,
20+
})
21+
}
22+
23+
/**
24+
* Updates an existing vote for a course.
25+
*
26+
* @param {string} iri - IRI of the vote to update
27+
* @param {number} vote - New rating from the user (1-5)
28+
* @param sessionId
29+
* @param urlId
30+
* @returns {Promise<Object>}
31+
*/
32+
export async function updateVote({ iri, vote, sessionId = null, urlId }) {
33+
try {
34+
if (!iri) {
35+
throw new Error("Cannot update vote because IRI is missing.")
36+
}
37+
38+
let payload = { vote }
39+
if (sessionId) payload.session = `/api/sessions/${sessionId}`
40+
if (urlId) payload.url = `/api/access_urls/${urlId}`
41+
42+
return await baseService.put(iri, payload)
43+
} catch (error) {
44+
console.error("Error updating user vote:", error)
45+
throw error
46+
}
47+
}
48+
49+
/**
50+
* Retrieves the user's vote for a specific course.
51+
*
52+
* @param {number} userId - ID of the user
53+
* @param {number} courseId - ID of the course
54+
* @param sessionId
55+
* @param urlId
56+
* @returns {Promise<Object|null>} - Returns the vote object if found, otherwise null
57+
*/
58+
export async function getUserVote({ userId, courseId, sessionId = null, urlId }) {
59+
try {
60+
let query = `/api/user_rel_course_votes?user.id=${userId}&course.id=${courseId}`
61+
if (urlId) query += `&url.id=${urlId}`
62+
63+
// Remove session.id if null
64+
if (sessionId) {
65+
query += `&session.id=${sessionId}`
66+
}
67+
68+
const response = await baseService.get(query)
69+
70+
if (response && response["hydra:member"] && response["hydra:member"].length > 0) {
71+
return response["hydra:member"][0]
72+
}
73+
74+
return null
75+
// eslint-disable-next-line no-unused-vars
76+
} catch (error) {
77+
return null
78+
}
79+
}
80+
81+
/**
82+
* Retrieves all votes of a user for different courses.
83+
*
84+
* @param {number} userId - User ID
85+
* @param {number} urlId - Access URL ID
86+
* @returns {Promise<Array>} - List of user votes
87+
*/
88+
export async function getUserVotes({ userId, urlId }) {
89+
try {
90+
let query = `/api/user_rel_course_votes?user.id=${userId}`
91+
if (urlId) query += `&url.id=${urlId}`
92+
93+
const response = await baseService.get(query)
94+
95+
return response && response["hydra:member"] ? response["hydra:member"] : []
96+
} catch (error) {
97+
console.error("Error retrieving user votes:", error)
98+
return []
99+
}
100+
}

assets/vue/views/course/CatalogueCourses.vue

+60-37
Original file line numberDiff line numberDiff line change
@@ -132,16 +132,16 @@
132132
<Column
133133
:header="$t('Ranking')"
134134
:sortable="true"
135-
field="trackCourseRanking.realTotalScore"
135+
field="userVote.vote"
136136
style="min-width: 10rem; text-align: center"
137137
>
138138
<template #body="{ data }">
139139
<Rating
140140
:cancel="false"
141-
:model-value="data.trackCourseRanking ? data.trackCourseRanking.realTotalScore : 0"
141+
:model-value="data.userVote ? data.userVote.vote : 0"
142142
:stars="5"
143143
class="pointer-events: none"
144-
@change="onRatingChange($event, data.trackCourseRanking, data.id)"
144+
@change="onRatingChange($event, data.userVote, data.id)"
145145
/>
146146
</template>
147147
</Column>
@@ -204,10 +204,9 @@ import { usePlatformConfig } from "../../store/platformConfig"
204204
import { useSecurityStore } from "../../store/securityStore"
205205
206206
import courseService from "../../services/courseService"
207-
import * as trackCourseRanking from "../../services/trackCourseRankingService"
208-
209207
import { useNotification } from "../../composables/notification"
210208
import { useLanguage } from "../../composables/language"
209+
import * as userRelCourseVoteService from "../../services/userRelCourseVoteService"
211210
212211
const { showErrorNotification } = useNotification()
213212
const { findByIsoCode: findLanguageByIsoCode } = useLanguage()
@@ -227,31 +226,43 @@ async function load() {
227226
try {
228227
const { items } = await courseService.listAll()
229228
230-
courses.value = items.map((course) => ({
231-
...course,
232-
courseLanguage: findLanguageByIsoCode(course.courseLanguage)?.originalName,
233-
}))
229+
const votes = await userRelCourseVoteService.getUserVotes({
230+
userId: currentUserId,
231+
urlId: window.access_url_id,
232+
})
233+
234+
courses.value = items.map((course) => {
235+
const userVote = votes.find((vote) => vote.course === `/api/courses/${course.id}`)
236+
237+
return {
238+
...course,
239+
courseLanguage: findLanguageByIsoCode(course.courseLanguage)?.originalName,
240+
userVote: userVote ? { ...userVote } : { vote: 0 },
241+
}
242+
})
234243
} catch (error) {
235244
showErrorNotification(error)
236245
} finally {
237246
status.value = false
238247
}
239248
}
240249
241-
async function updateRating(id, value) {
250+
async function updateRating(voteIri, value) {
242251
status.value = true
243252
244253
try {
245-
const response = await trackCourseRanking.updateRanking({
246-
iri: `/api/track_course_rankings/${id}`,
247-
totalScore: value,
254+
await userRelCourseVoteService.updateVote({
255+
iri: voteIri,
256+
vote: value,
257+
sessionId: window.session_id,
258+
urlId: window.access_url_id,
248259
})
249260
250-
courses.value.forEach((course) => {
251-
if (course.trackCourseRanking && course.trackCourseRanking.id === id) {
252-
course.trackCourseRanking.realTotalScore = response.realTotalScore
253-
}
254-
})
261+
courses.value = courses.value.map((course) =>
262+
course.userVote && course.userVote["@id"] === voteIri
263+
? { ...course, userVote: { ...course.userVote, vote: value } }
264+
: course,
265+
)
255266
} catch (e) {
256267
showErrorNotification(e)
257268
} finally {
@@ -263,25 +274,47 @@ const newRating = async function (courseId, value) {
263274
status.value = true
264275
265276
try {
266-
const response = await trackCourseRanking.saveRanking({
267-
totalScore: value,
268-
courseIri: `/api/courses/${courseId}`,
277+
const existingVote = await userRelCourseVoteService.getUserVote({
278+
userId: currentUserId,
279+
courseId,
280+
sessionId: window.session_id || null,
269281
urlId: window.access_url_id,
270-
sessionId: 0,
271282
})
272283
273-
courses.value.forEach((course) => {
274-
if (course.id === courseId) {
275-
course.trackCourseRanking = response
276-
}
277-
})
284+
if (existingVote) {
285+
await updateRating(existingVote["@id"], value)
286+
} else {
287+
await userRelCourseVoteService.saveVote({
288+
vote: value,
289+
courseIri: `/api/courses/${courseId}`,
290+
userId: currentUserId,
291+
sessionId: window.session_id || null,
292+
urlId: window.access_url_id,
293+
})
294+
}
295+
296+
await load()
278297
} catch (e) {
279298
showErrorNotification(e)
280299
} finally {
281300
status.value = false
282301
}
283302
}
284303
304+
const onRatingChange = function (event, userVote, courseId) {
305+
let { value } = event
306+
307+
if (value > 0) {
308+
if (userVote && userVote["@id"]) {
309+
updateRating(userVote["@id"], value)
310+
} else {
311+
newRating(courseId, value)
312+
}
313+
} else {
314+
event.preventDefault()
315+
}
316+
}
317+
285318
const isUserInCourse = (course) => {
286319
return course.users.some((user) => user.user.id === currentUserId)
287320
}
@@ -296,16 +329,6 @@ const initFilters = function () {
296329
}
297330
}
298331
299-
const onRatingChange = function (event, trackCourseRanking, courseId) {
300-
let { value } = event
301-
if (value > 0) {
302-
if (trackCourseRanking) updateRating(trackCourseRanking.id, value)
303-
else newRating(courseId, value)
304-
} else {
305-
event.preventDefault()
306-
}
307-
}
308-
309332
load()
310333
initFilters()
311334
</script>

0 commit comments

Comments
 (0)