Decouple projects from rounds with RoundProject join table
Projects now exist at the program level instead of being locked to a single round. A new RoundProject join table enables many-to-many relationships with per-round status tracking. Rounds have sortOrder for configurable progression paths. - Add RoundProject model, programId on Project, sortOrder on Round - Migration preserves existing data (roundId -> RoundProject entries) - Update all routers to query through RoundProject join - Add assign/remove/advance/reorder round endpoints - Add Assign, Advance, Remove Projects dialogs on round detail page - Add round reorder controls (up/down arrows) on rounds list - Show all rounds on project detail page Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
@@ -80,18 +80,27 @@ export const learningResourceRouter = router({
|
||||
const assignments = await ctx.prisma.assignment.findMany({
|
||||
where: { userId: ctx.user.id },
|
||||
include: {
|
||||
project: { select: { status: true } },
|
||||
project: {
|
||||
select: {
|
||||
roundProjects: {
|
||||
select: { status: true },
|
||||
orderBy: { addedAt: 'desc' },
|
||||
take: 1,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
})
|
||||
|
||||
// Determine highest cohort level
|
||||
let userCohortLevel: 'ALL' | 'SEMIFINALIST' | 'FINALIST' = 'ALL'
|
||||
for (const assignment of assignments) {
|
||||
if (assignment.project.status === 'FINALIST') {
|
||||
const rpStatus = assignment.project.roundProjects[0]?.status
|
||||
if (rpStatus === 'FINALIST') {
|
||||
userCohortLevel = 'FINALIST'
|
||||
break
|
||||
}
|
||||
if (assignment.project.status === 'SEMIFINALIST') {
|
||||
if (rpStatus === 'SEMIFINALIST') {
|
||||
userCohortLevel = 'SEMIFINALIST'
|
||||
}
|
||||
}
|
||||
@@ -155,17 +164,26 @@ export const learningResourceRouter = router({
|
||||
const assignments = await ctx.prisma.assignment.findMany({
|
||||
where: { userId: ctx.user.id },
|
||||
include: {
|
||||
project: { select: { status: true } },
|
||||
project: {
|
||||
select: {
|
||||
roundProjects: {
|
||||
select: { status: true },
|
||||
orderBy: { addedAt: 'desc' as const },
|
||||
take: 1,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
})
|
||||
|
||||
let userCohortLevel: 'ALL' | 'SEMIFINALIST' | 'FINALIST' = 'ALL'
|
||||
for (const assignment of assignments) {
|
||||
if (assignment.project.status === 'FINALIST') {
|
||||
const rpStatus = assignment.project.roundProjects[0]?.status
|
||||
if (rpStatus === 'FINALIST') {
|
||||
userCohortLevel = 'FINALIST'
|
||||
break
|
||||
}
|
||||
if (assignment.project.status === 'SEMIFINALIST') {
|
||||
if (rpStatus === 'SEMIFINALIST') {
|
||||
userCohortLevel = 'SEMIFINALIST'
|
||||
}
|
||||
}
|
||||
@@ -220,16 +238,27 @@ export const learningResourceRouter = router({
|
||||
// Check cohort level access
|
||||
const assignments = await ctx.prisma.assignment.findMany({
|
||||
where: { userId: ctx.user.id },
|
||||
include: { project: { select: { status: true } } },
|
||||
include: {
|
||||
project: {
|
||||
select: {
|
||||
roundProjects: {
|
||||
select: { status: true },
|
||||
orderBy: { addedAt: 'desc' as const },
|
||||
take: 1,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
})
|
||||
|
||||
let userCohortLevel: 'ALL' | 'SEMIFINALIST' | 'FINALIST' = 'ALL'
|
||||
for (const assignment of assignments) {
|
||||
if (assignment.project.status === 'FINALIST') {
|
||||
const rpStatus = assignment.project.roundProjects[0]?.status
|
||||
if (rpStatus === 'FINALIST') {
|
||||
userCohortLevel = 'FINALIST'
|
||||
break
|
||||
}
|
||||
if (assignment.project.status === 'SEMIFINALIST') {
|
||||
if (rpStatus === 'SEMIFINALIST') {
|
||||
userCohortLevel = 'SEMIFINALIST'
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user