@ethlete/cdk 4.45.0 → 4.46.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +6 -0
- package/esm2022/lib/components/bracket/components/new-bracket/bracket-new.mjs +782 -0
- package/esm2022/lib/components/bracket/components/new-bracket/draw-man.mjs +66 -0
- package/esm2022/lib/components/bracket/components/new-bracket/grid-definitions.mjs +47 -0
- package/esm2022/lib/components/bracket/components/new-bracket/grid-placements.mjs +97 -0
- package/esm2022/lib/components/bracket/components/new-bracket/index.mjs +8 -0
- package/esm2022/lib/components/bracket/components/new-bracket/new-bracket-default-match.component.mjs +17 -0
- package/esm2022/lib/components/bracket/components/new-bracket/new-bracket-default-round-header.component.mjs +16 -0
- package/esm2022/lib/components/bracket/components/new-bracket/new-bracket.component.mjs +54 -0
- package/esm2022/lib/components/bracket/public-api/index.mjs +2 -1
- package/fesm2022/ethlete-cdk.mjs +1096 -3
- package/fesm2022/ethlete-cdk.mjs.map +1 -1
- package/lib/components/bracket/components/new-bracket/bracket-new.d.ts +238 -0
- package/lib/components/bracket/components/new-bracket/draw-man.d.ts +12 -0
- package/lib/components/bracket/components/new-bracket/grid-definitions.d.ts +9 -0
- package/lib/components/bracket/components/new-bracket/grid-placements.d.ts +56 -0
- package/lib/components/bracket/components/new-bracket/index.d.ts +7 -0
- package/lib/components/bracket/components/new-bracket/new-bracket-default-match.component.d.ts +8 -0
- package/lib/components/bracket/components/new-bracket/new-bracket-default-round-header.component.d.ts +7 -0
- package/lib/components/bracket/components/new-bracket/new-bracket.component.d.ts +26 -0
- package/lib/components/bracket/public-api/index.d.ts +1 -0
- package/package.json +1 -1
|
@@ -0,0 +1,782 @@
|
|
|
1
|
+
export const FALLBACK_MATCH_POSITION = -1;
|
|
2
|
+
export const TOURNAMENT_MODE = {
|
|
3
|
+
SINGLE_ELIMINATION: 'single-elimination',
|
|
4
|
+
DOUBLE_ELIMINATION: 'double-elimination',
|
|
5
|
+
GROUP: 'group',
|
|
6
|
+
SWISS: 'swiss',
|
|
7
|
+
SWISS_WITH_ELIMINATION: 'swiss-with-elimination',
|
|
8
|
+
};
|
|
9
|
+
export const COMMON_BRACKET_ROUND_TYPE = {
|
|
10
|
+
THIRD_PLACE: 'third-place',
|
|
11
|
+
FINAL: 'final',
|
|
12
|
+
};
|
|
13
|
+
export const SINGLE_ELIMINATION_BRACKET_ROUND_TYPE = {
|
|
14
|
+
SINGLE_ELIMINATION_BRACKET: 'single-elimination-bracket',
|
|
15
|
+
};
|
|
16
|
+
export const DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE = {
|
|
17
|
+
UPPER_BRACKET: 'upper-bracket',
|
|
18
|
+
LOWER_BRACKET: 'lower-bracket',
|
|
19
|
+
REVERSE_FINAL: 'reverse-final',
|
|
20
|
+
};
|
|
21
|
+
export const SWISS_BRACKET_ROUND_TYPE = {
|
|
22
|
+
SWISS: 'swiss',
|
|
23
|
+
};
|
|
24
|
+
export const GROUP_BRACKET_ROUND_TYPE = {
|
|
25
|
+
GROUP: 'group',
|
|
26
|
+
};
|
|
27
|
+
export const generateRoundTypeFromEthleteRoundType = (type, tournamentMode) => {
|
|
28
|
+
switch (type) {
|
|
29
|
+
case 'normal':
|
|
30
|
+
switch (tournamentMode) {
|
|
31
|
+
case 'single-elimination':
|
|
32
|
+
return SINGLE_ELIMINATION_BRACKET_ROUND_TYPE.SINGLE_ELIMINATION_BRACKET;
|
|
33
|
+
case 'group':
|
|
34
|
+
return GROUP_BRACKET_ROUND_TYPE.GROUP;
|
|
35
|
+
case 'swiss':
|
|
36
|
+
return SWISS_BRACKET_ROUND_TYPE.SWISS;
|
|
37
|
+
default:
|
|
38
|
+
throw new Error(`Unsupported tournament mode for a normal type round: ${tournamentMode}`);
|
|
39
|
+
}
|
|
40
|
+
case 'third_place':
|
|
41
|
+
return COMMON_BRACKET_ROUND_TYPE.THIRD_PLACE;
|
|
42
|
+
case 'final':
|
|
43
|
+
return COMMON_BRACKET_ROUND_TYPE.FINAL;
|
|
44
|
+
case 'reverse_final':
|
|
45
|
+
return DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE.REVERSE_FINAL;
|
|
46
|
+
case 'winner_bracket':
|
|
47
|
+
return DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE.UPPER_BRACKET;
|
|
48
|
+
case 'loser_bracket':
|
|
49
|
+
return DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE.LOWER_BRACKET;
|
|
50
|
+
}
|
|
51
|
+
};
|
|
52
|
+
export const generateTournamentModeFormEthleteRounds = (source) => {
|
|
53
|
+
const firstRound = source[0];
|
|
54
|
+
const firstMatch = firstRound?.matches[0];
|
|
55
|
+
if (!firstRound)
|
|
56
|
+
throw new Error('No rounds found');
|
|
57
|
+
if (!firstMatch)
|
|
58
|
+
throw new Error('No matches found');
|
|
59
|
+
switch (firstMatch.matchType) {
|
|
60
|
+
case 'groups':
|
|
61
|
+
return TOURNAMENT_MODE.GROUP;
|
|
62
|
+
case 'fifa_swiss': {
|
|
63
|
+
const lastRound = source[source.length - 1];
|
|
64
|
+
if (!lastRound)
|
|
65
|
+
throw new Error('No last round found');
|
|
66
|
+
if (lastRound.matches.length !== firstRound.matches.length) {
|
|
67
|
+
return TOURNAMENT_MODE.SWISS_WITH_ELIMINATION;
|
|
68
|
+
}
|
|
69
|
+
else {
|
|
70
|
+
return TOURNAMENT_MODE.SWISS;
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
case 'double_elimination':
|
|
74
|
+
return TOURNAMENT_MODE.DOUBLE_ELIMINATION;
|
|
75
|
+
case 'single_elimination':
|
|
76
|
+
return TOURNAMENT_MODE.SINGLE_ELIMINATION;
|
|
77
|
+
default:
|
|
78
|
+
throw new Error(`Unsupported tournament mode: ${firstMatch.matchType}`);
|
|
79
|
+
}
|
|
80
|
+
};
|
|
81
|
+
export const generateBracketDataForEthlete = (source) => {
|
|
82
|
+
const tournamentMode = generateTournamentModeFormEthleteRounds(source);
|
|
83
|
+
const bracketData = {
|
|
84
|
+
rounds: new Map(),
|
|
85
|
+
matches: new Map(),
|
|
86
|
+
participants: new Map(),
|
|
87
|
+
mode: tournamentMode,
|
|
88
|
+
};
|
|
89
|
+
let currentUpperBracketIndex = 0;
|
|
90
|
+
let currentLowerBracketIndex = 0;
|
|
91
|
+
for (const currentItem of source) {
|
|
92
|
+
if (bracketData.rounds.has(currentItem.round.id)) {
|
|
93
|
+
throw new Error(`Round with id ${currentItem.round.id} already exists in the bracket data.`);
|
|
94
|
+
}
|
|
95
|
+
const roundType = generateRoundTypeFromEthleteRoundType(currentItem.round.type, tournamentMode);
|
|
96
|
+
const isLowerBracket = roundType === DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE.LOWER_BRACKET;
|
|
97
|
+
const bracketRound = {
|
|
98
|
+
type: roundType,
|
|
99
|
+
id: currentItem.round.id,
|
|
100
|
+
index: isLowerBracket ? currentLowerBracketIndex : currentUpperBracketIndex,
|
|
101
|
+
data: currentItem.round,
|
|
102
|
+
position: currentItem.round.number,
|
|
103
|
+
name: currentItem.round.name || currentItem.round.type,
|
|
104
|
+
matchCount: currentItem.matches.length,
|
|
105
|
+
matches: new Map(),
|
|
106
|
+
};
|
|
107
|
+
bracketData.rounds.set(currentItem.round.id, bracketRound);
|
|
108
|
+
for (const [matchIndex, match] of currentItem.matches.entries()) {
|
|
109
|
+
if (bracketData.matches.has(match.id)) {
|
|
110
|
+
throw new Error(`Match with id ${match.id} already exists in the bracket data.`);
|
|
111
|
+
}
|
|
112
|
+
const bracketMatch = {
|
|
113
|
+
id: match.id,
|
|
114
|
+
indexInRound: matchIndex,
|
|
115
|
+
position: (matchIndex + 1),
|
|
116
|
+
data: match,
|
|
117
|
+
round: bracketRound,
|
|
118
|
+
home: match.home?.id || null,
|
|
119
|
+
away: match.away?.id || null,
|
|
120
|
+
winner: match.winningSide,
|
|
121
|
+
status: match.status === 'published' ? 'completed' : 'pending',
|
|
122
|
+
};
|
|
123
|
+
bracketData.matches.set(match.id, bracketMatch);
|
|
124
|
+
bracketRound.matches.set(match.id, bracketMatch);
|
|
125
|
+
const participants = [match.home, match.away];
|
|
126
|
+
for (const participant of participants) {
|
|
127
|
+
if (!participant)
|
|
128
|
+
continue;
|
|
129
|
+
const participantId = participant.id;
|
|
130
|
+
if (!bracketData.participants.has(participantId)) {
|
|
131
|
+
bracketData.participants.set(participantId, {
|
|
132
|
+
id: participantId,
|
|
133
|
+
name: participant.name || 'Unknown',
|
|
134
|
+
matches: new Map(),
|
|
135
|
+
});
|
|
136
|
+
}
|
|
137
|
+
const participantData = bracketData.participants.get(participantId);
|
|
138
|
+
if (!participantData.matches.has(match.id)) {
|
|
139
|
+
participantData.matches.set(match.id, {
|
|
140
|
+
side: match.home?.id === participantId ? 'home' : 'away',
|
|
141
|
+
bracketMatch,
|
|
142
|
+
});
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
if (isLowerBracket) {
|
|
147
|
+
currentLowerBracketIndex++;
|
|
148
|
+
}
|
|
149
|
+
else {
|
|
150
|
+
currentUpperBracketIndex++;
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
return bracketData;
|
|
154
|
+
};
|
|
155
|
+
export const generateMatchPositionMaps = (bracketData) => {
|
|
156
|
+
const matchPositionMaps = new Map();
|
|
157
|
+
for (const round of bracketData.rounds.values()) {
|
|
158
|
+
const matchMap = new Map([...round.matches.values()].map((m) => [m.position, m]));
|
|
159
|
+
matchPositionMaps.set(round.id, matchMap);
|
|
160
|
+
}
|
|
161
|
+
return matchPositionMaps;
|
|
162
|
+
};
|
|
163
|
+
export const generateBracketRoundTypeMap = (bracketData) => {
|
|
164
|
+
const roundAmountMap = new Map();
|
|
165
|
+
for (const round of bracketData.rounds.values()) {
|
|
166
|
+
if (!roundAmountMap.has(round.type)) {
|
|
167
|
+
roundAmountMap.set(round.type, new Map([[round.id, round]]));
|
|
168
|
+
}
|
|
169
|
+
else {
|
|
170
|
+
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
|
|
171
|
+
roundAmountMap.set(round.type, roundAmountMap.get(round.type).set(round.id, round));
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
return roundAmountMap;
|
|
175
|
+
};
|
|
176
|
+
const UPPER_LOOP_ORDER = [
|
|
177
|
+
SINGLE_ELIMINATION_BRACKET_ROUND_TYPE.SINGLE_ELIMINATION_BRACKET,
|
|
178
|
+
DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE.UPPER_BRACKET,
|
|
179
|
+
SWISS_BRACKET_ROUND_TYPE.SWISS,
|
|
180
|
+
GROUP_BRACKET_ROUND_TYPE.GROUP,
|
|
181
|
+
COMMON_BRACKET_ROUND_TYPE.FINAL,
|
|
182
|
+
DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE.REVERSE_FINAL,
|
|
183
|
+
COMMON_BRACKET_ROUND_TYPE.THIRD_PLACE,
|
|
184
|
+
];
|
|
185
|
+
const LOWER_LOOP_ORDER = [DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE.LOWER_BRACKET];
|
|
186
|
+
export const generateRoundRelations = (bracketData) => {
|
|
187
|
+
const roundRelations = new Map();
|
|
188
|
+
const sortedRoundsByType = new Map();
|
|
189
|
+
for (const round of bracketData.rounds.values()) {
|
|
190
|
+
if (!sortedRoundsByType.has(round.type)) {
|
|
191
|
+
sortedRoundsByType.set(round.type, [round]);
|
|
192
|
+
}
|
|
193
|
+
else {
|
|
194
|
+
sortedRoundsByType.get(round.type)?.push(round);
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
for (const rounds of sortedRoundsByType.values()) {
|
|
198
|
+
if (rounds.length === 1)
|
|
199
|
+
continue;
|
|
200
|
+
rounds.sort((a, b) => a.position - b.position);
|
|
201
|
+
}
|
|
202
|
+
const sortedUpperRounds = UPPER_LOOP_ORDER.map((t) => sortedRoundsByType.get(t))
|
|
203
|
+
.filter((r) => !!r)
|
|
204
|
+
.flat();
|
|
205
|
+
const sortedLowerRounds = LOWER_LOOP_ORDER.map((t) => sortedRoundsByType.get(t))
|
|
206
|
+
.filter((r) => !!r)
|
|
207
|
+
.flat();
|
|
208
|
+
const firstUpperRound = sortedUpperRounds[0] || null;
|
|
209
|
+
const firstLowerRound = sortedLowerRounds[0] || null;
|
|
210
|
+
if (!firstUpperRound)
|
|
211
|
+
throw new Error('firstUpperRound is null');
|
|
212
|
+
const hasLowerRounds = sortedLowerRounds.length > 0;
|
|
213
|
+
const lastLowerRound = sortedLowerRounds[sortedLowerRounds.length - 1] || null;
|
|
214
|
+
for (const [currentUpperRoundIndex, currentUpperRound] of sortedUpperRounds.entries()) {
|
|
215
|
+
const previousUpperRound = sortedUpperRounds[currentUpperRoundIndex - 1] || null;
|
|
216
|
+
const nextUpperRound = sortedUpperRounds[currentUpperRoundIndex + 1] || null;
|
|
217
|
+
const currentLowerRound = sortedLowerRounds[currentUpperRoundIndex] || null;
|
|
218
|
+
const previousLowerRound = sortedLowerRounds[currentUpperRoundIndex - 1] || null;
|
|
219
|
+
const nextLowerRound = sortedLowerRounds[currentUpperRoundIndex + 1] || null;
|
|
220
|
+
const isFirstRound = currentUpperRoundIndex === 0;
|
|
221
|
+
const isLastUpperRound = currentUpperRoundIndex === sortedUpperRounds.length - 1;
|
|
222
|
+
const isFinal = currentUpperRound.type === COMMON_BRACKET_ROUND_TYPE.FINAL;
|
|
223
|
+
if (isFinal && hasLowerRounds) {
|
|
224
|
+
// two to one relation
|
|
225
|
+
if (!lastLowerRound || !currentLowerRound || !previousUpperRound)
|
|
226
|
+
throw new Error('lastLowerRound or currentLowerRound or previousUpperRound is null');
|
|
227
|
+
// in a sync double elimination bracket, the final round has the same index as the last lower round
|
|
228
|
+
// in an async one, there is always one more round in the lower bracket since we only display a section of the whole tournament
|
|
229
|
+
const isAsyncBracket = currentLowerRound.id !== lastLowerRound.id;
|
|
230
|
+
const finalLowerRound = isAsyncBracket ? nextLowerRound : currentLowerRound;
|
|
231
|
+
if (!finalLowerRound)
|
|
232
|
+
throw new Error('finalLowerRound is null');
|
|
233
|
+
if (!firstLowerRound)
|
|
234
|
+
throw new Error('firstLowerRound is null');
|
|
235
|
+
if (finalLowerRound.id !== lastLowerRound.id)
|
|
236
|
+
throw new Error('finalLowerRound is not the last lower round');
|
|
237
|
+
// if we have a reverse final
|
|
238
|
+
if (nextUpperRound) {
|
|
239
|
+
// for the final
|
|
240
|
+
roundRelations.set(currentUpperRound.id, {
|
|
241
|
+
type: 'two-to-one',
|
|
242
|
+
previousUpperRound: previousUpperRound,
|
|
243
|
+
previousLowerRound: finalLowerRound,
|
|
244
|
+
nextRound: nextUpperRound,
|
|
245
|
+
currentRound: currentUpperRound,
|
|
246
|
+
nextRoundMatchFactor: nextUpperRound.matchCount / currentUpperRound.matchCount,
|
|
247
|
+
previousUpperRoundMatchFactor: previousUpperRound.matchCount / currentUpperRound.matchCount,
|
|
248
|
+
previousLowerRoundMatchFactor: finalLowerRound.matchCount / currentUpperRound.matchCount,
|
|
249
|
+
upperRootRoundMatchFactor: firstUpperRound.matchCount / currentUpperRound.matchCount,
|
|
250
|
+
lowerRootRoundMatchFactor: firstLowerRound.matchCount / currentUpperRound.matchCount,
|
|
251
|
+
});
|
|
252
|
+
}
|
|
253
|
+
else {
|
|
254
|
+
// no reverse final means the final is the last round
|
|
255
|
+
// for the final
|
|
256
|
+
roundRelations.set(currentUpperRound.id, {
|
|
257
|
+
type: 'two-to-nothing',
|
|
258
|
+
previousUpperRound: previousUpperRound,
|
|
259
|
+
previousLowerRound: finalLowerRound,
|
|
260
|
+
currentRound: currentUpperRound,
|
|
261
|
+
previousLowerRoundMatchFactor: finalLowerRound.matchCount / currentUpperRound.matchCount,
|
|
262
|
+
previousUpperRoundMatchFactor: previousUpperRound.matchCount / currentUpperRound.matchCount,
|
|
263
|
+
lowerRootRoundMatchFactor: firstLowerRound.matchCount / currentUpperRound.matchCount,
|
|
264
|
+
upperRootRoundMatchFactor: firstUpperRound.matchCount / currentUpperRound.matchCount,
|
|
265
|
+
});
|
|
266
|
+
}
|
|
267
|
+
if (isAsyncBracket) {
|
|
268
|
+
// if this is an async bracket, we need to set the relations for the 2 last lower rounds since they will be skipped by the default one to one logic
|
|
269
|
+
const preFinalLowerRound = sortedLowerRounds[sortedLowerRounds.length - 2] || null;
|
|
270
|
+
const prePreFinalLowerRound = sortedLowerRounds[sortedLowerRounds.length - 3] || null;
|
|
271
|
+
if (!preFinalLowerRound)
|
|
272
|
+
throw new Error('preFinalLowerRound is null');
|
|
273
|
+
if (!firstLowerRound)
|
|
274
|
+
throw new Error('firstLowerRound is null');
|
|
275
|
+
// for the last lower round
|
|
276
|
+
roundRelations.set(finalLowerRound.id, {
|
|
277
|
+
type: 'one-to-one',
|
|
278
|
+
previousRound: preFinalLowerRound,
|
|
279
|
+
nextRound: currentUpperRound,
|
|
280
|
+
currentRound: finalLowerRound,
|
|
281
|
+
nextRoundMatchFactor: currentUpperRound.matchCount / finalLowerRound.matchCount,
|
|
282
|
+
previousRoundMatchFactor: preFinalLowerRound.matchCount / finalLowerRound.matchCount,
|
|
283
|
+
rootRoundMatchFactor: firstLowerRound.matchCount / finalLowerRound.matchCount,
|
|
284
|
+
});
|
|
285
|
+
if (prePreFinalLowerRound) {
|
|
286
|
+
// for the pre final lower round
|
|
287
|
+
roundRelations.set(preFinalLowerRound.id, {
|
|
288
|
+
type: 'one-to-one',
|
|
289
|
+
previousRound: prePreFinalLowerRound,
|
|
290
|
+
nextRound: finalLowerRound,
|
|
291
|
+
currentRound: preFinalLowerRound,
|
|
292
|
+
nextRoundMatchFactor: finalLowerRound.matchCount / preFinalLowerRound.matchCount,
|
|
293
|
+
previousRoundMatchFactor: prePreFinalLowerRound.matchCount / preFinalLowerRound.matchCount,
|
|
294
|
+
rootRoundMatchFactor: firstLowerRound.matchCount / preFinalLowerRound.matchCount,
|
|
295
|
+
});
|
|
296
|
+
}
|
|
297
|
+
else {
|
|
298
|
+
// for the first lower round
|
|
299
|
+
roundRelations.set(preFinalLowerRound.id, {
|
|
300
|
+
type: 'nothing-to-one',
|
|
301
|
+
nextRound: finalLowerRound,
|
|
302
|
+
currentRound: preFinalLowerRound,
|
|
303
|
+
nextRoundMatchFactor: finalLowerRound.matchCount / preFinalLowerRound.matchCount,
|
|
304
|
+
});
|
|
305
|
+
}
|
|
306
|
+
}
|
|
307
|
+
else {
|
|
308
|
+
// this is a sync bracket, we only need to set the relation for the last lower round
|
|
309
|
+
if (!previousLowerRound)
|
|
310
|
+
throw new Error('previousLowerRound is null');
|
|
311
|
+
if (!firstLowerRound)
|
|
312
|
+
throw new Error('firstLowerRound is null');
|
|
313
|
+
// for the last lower round
|
|
314
|
+
roundRelations.set(finalLowerRound.id, {
|
|
315
|
+
type: 'one-to-one',
|
|
316
|
+
previousRound: previousLowerRound,
|
|
317
|
+
nextRound: currentUpperRound,
|
|
318
|
+
currentRound: finalLowerRound,
|
|
319
|
+
nextRoundMatchFactor: currentUpperRound.matchCount / finalLowerRound.matchCount,
|
|
320
|
+
previousRoundMatchFactor: previousLowerRound.matchCount / finalLowerRound.matchCount,
|
|
321
|
+
rootRoundMatchFactor: firstLowerRound.matchCount / finalLowerRound.matchCount,
|
|
322
|
+
});
|
|
323
|
+
}
|
|
324
|
+
}
|
|
325
|
+
else if (isLastUpperRound) {
|
|
326
|
+
// one to nothing relation
|
|
327
|
+
if (!previousUpperRound)
|
|
328
|
+
throw new Error('previousUpperRound is null');
|
|
329
|
+
roundRelations.set(currentUpperRound.id, {
|
|
330
|
+
type: 'one-to-nothing',
|
|
331
|
+
previousRound: previousUpperRound,
|
|
332
|
+
currentRound: currentUpperRound,
|
|
333
|
+
previousRoundMatchFactor: previousUpperRound.matchCount / currentUpperRound.matchCount,
|
|
334
|
+
rootRoundMatchFactor: firstUpperRound.matchCount / currentUpperRound.matchCount,
|
|
335
|
+
});
|
|
336
|
+
}
|
|
337
|
+
else if (isFirstRound) {
|
|
338
|
+
// nothing to one relation
|
|
339
|
+
if (!nextUpperRound)
|
|
340
|
+
throw new Error('nextUpperRound is null');
|
|
341
|
+
roundRelations.set(currentUpperRound.id, {
|
|
342
|
+
type: 'nothing-to-one',
|
|
343
|
+
nextRound: nextUpperRound,
|
|
344
|
+
currentRound: currentUpperRound,
|
|
345
|
+
nextRoundMatchFactor: nextUpperRound.matchCount / currentUpperRound.matchCount,
|
|
346
|
+
});
|
|
347
|
+
if (currentLowerRound) {
|
|
348
|
+
if (!nextLowerRound)
|
|
349
|
+
throw new Error('nextLowerRound is null');
|
|
350
|
+
roundRelations.set(currentLowerRound.id, {
|
|
351
|
+
type: 'nothing-to-one',
|
|
352
|
+
nextRound: nextLowerRound,
|
|
353
|
+
currentRound: currentLowerRound,
|
|
354
|
+
nextRoundMatchFactor: nextLowerRound.matchCount / currentLowerRound.matchCount,
|
|
355
|
+
});
|
|
356
|
+
}
|
|
357
|
+
}
|
|
358
|
+
else {
|
|
359
|
+
// one to one relation
|
|
360
|
+
if (!previousUpperRound)
|
|
361
|
+
throw new Error('previousUpperRound is null');
|
|
362
|
+
if (!nextUpperRound)
|
|
363
|
+
throw new Error('nextUpperRound is null');
|
|
364
|
+
roundRelations.set(currentUpperRound.id, {
|
|
365
|
+
type: 'one-to-one',
|
|
366
|
+
previousRound: previousUpperRound,
|
|
367
|
+
nextRound: nextUpperRound,
|
|
368
|
+
currentRound: currentUpperRound,
|
|
369
|
+
nextRoundMatchFactor: nextUpperRound.matchCount / currentUpperRound.matchCount,
|
|
370
|
+
previousRoundMatchFactor: previousUpperRound.matchCount / currentUpperRound.matchCount,
|
|
371
|
+
rootRoundMatchFactor: firstUpperRound.matchCount / currentUpperRound.matchCount,
|
|
372
|
+
});
|
|
373
|
+
// we only want to set lower rounds here until the special merging point of the final.
|
|
374
|
+
// lower bracket rounds after and including the final will be set in the final round block
|
|
375
|
+
if (currentLowerRound && currentUpperRound.type === DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE.UPPER_BRACKET) {
|
|
376
|
+
if (!previousLowerRound)
|
|
377
|
+
throw new Error('previousLowerRound is null');
|
|
378
|
+
if (!nextLowerRound)
|
|
379
|
+
throw new Error('nextLowerRound is null');
|
|
380
|
+
if (!firstLowerRound)
|
|
381
|
+
throw new Error('firstLowerRound is null');
|
|
382
|
+
roundRelations.set(currentLowerRound.id, {
|
|
383
|
+
type: 'one-to-one',
|
|
384
|
+
previousRound: previousLowerRound,
|
|
385
|
+
nextRound: nextLowerRound,
|
|
386
|
+
currentRound: currentLowerRound,
|
|
387
|
+
nextRoundMatchFactor: nextLowerRound.matchCount / currentLowerRound.matchCount,
|
|
388
|
+
previousRoundMatchFactor: previousLowerRound.matchCount / currentLowerRound.matchCount,
|
|
389
|
+
rootRoundMatchFactor: firstLowerRound.matchCount / currentLowerRound.matchCount,
|
|
390
|
+
});
|
|
391
|
+
}
|
|
392
|
+
}
|
|
393
|
+
}
|
|
394
|
+
return roundRelations;
|
|
395
|
+
};
|
|
396
|
+
export const generateMatchRelations = (bracketData, roundRelations, matchPositionMaps) => {
|
|
397
|
+
const matchRelations = new Map();
|
|
398
|
+
for (const match of bracketData.matches.values()) {
|
|
399
|
+
const currentRelation = roundRelations.get(match.round.id);
|
|
400
|
+
if (!currentRelation)
|
|
401
|
+
throw new Error('Match round not found');
|
|
402
|
+
const { nextRoundMatchPosition, previousLowerRoundMatchPosition, previousUpperRoundMatchPosition } = generateMatchRelationPositions(currentRelation, match);
|
|
403
|
+
switch (currentRelation.type) {
|
|
404
|
+
case 'nothing-to-one': {
|
|
405
|
+
const nextMatch = matchPositionMaps.get(currentRelation.nextRound.id)?.get(nextRoundMatchPosition);
|
|
406
|
+
if (!nextMatch)
|
|
407
|
+
throw new Error('Next round match not found');
|
|
408
|
+
// means left is nothing. right is one
|
|
409
|
+
matchRelations.set(match.id, {
|
|
410
|
+
type: 'nothing-to-one',
|
|
411
|
+
currentMatch: match,
|
|
412
|
+
currentRound: currentRelation.currentRound,
|
|
413
|
+
nextRound: currentRelation.nextRound,
|
|
414
|
+
nextMatch,
|
|
415
|
+
});
|
|
416
|
+
break;
|
|
417
|
+
}
|
|
418
|
+
case 'one-to-nothing': {
|
|
419
|
+
const previousUpperMatch = matchPositionMaps
|
|
420
|
+
.get(currentRelation.previousRound.id)
|
|
421
|
+
?.get(previousUpperRoundMatchPosition);
|
|
422
|
+
const previousLowerMatch = matchPositionMaps
|
|
423
|
+
.get(currentRelation.previousRound.id)
|
|
424
|
+
?.get(previousLowerRoundMatchPosition);
|
|
425
|
+
if (!previousUpperMatch)
|
|
426
|
+
throw new Error('Previous round match not found');
|
|
427
|
+
if (previousUpperRoundMatchPosition !== previousLowerRoundMatchPosition) {
|
|
428
|
+
// means left is two. right is one
|
|
429
|
+
if (!previousLowerMatch)
|
|
430
|
+
throw new Error('Previous lower round match not found');
|
|
431
|
+
matchRelations.set(match.id, {
|
|
432
|
+
type: 'two-to-nothing',
|
|
433
|
+
currentMatch: match,
|
|
434
|
+
currentRound: currentRelation.currentRound,
|
|
435
|
+
previousUpperMatch,
|
|
436
|
+
previousUpperRound: currentRelation.previousRound,
|
|
437
|
+
previousLowerMatch,
|
|
438
|
+
previousLowerRound: currentRelation.previousRound,
|
|
439
|
+
});
|
|
440
|
+
}
|
|
441
|
+
else {
|
|
442
|
+
// means left is one. right is nothing
|
|
443
|
+
matchRelations.set(match.id, {
|
|
444
|
+
type: 'one-to-nothing',
|
|
445
|
+
currentMatch: match,
|
|
446
|
+
currentRound: currentRelation.currentRound,
|
|
447
|
+
previousMatch: previousUpperMatch,
|
|
448
|
+
previousRound: currentRelation.previousRound,
|
|
449
|
+
});
|
|
450
|
+
}
|
|
451
|
+
break;
|
|
452
|
+
}
|
|
453
|
+
case 'one-to-one': {
|
|
454
|
+
const nextMatch = matchPositionMaps.get(currentRelation.nextRound.id)?.get(nextRoundMatchPosition);
|
|
455
|
+
const previousUpperMatch = matchPositionMaps
|
|
456
|
+
.get(currentRelation.previousRound.id)
|
|
457
|
+
?.get(previousUpperRoundMatchPosition);
|
|
458
|
+
const previousLowerMatch = matchPositionMaps
|
|
459
|
+
.get(currentRelation.previousRound.id)
|
|
460
|
+
?.get(previousLowerRoundMatchPosition);
|
|
461
|
+
if (!nextMatch)
|
|
462
|
+
throw new Error('Next round match not found');
|
|
463
|
+
if (!previousUpperMatch)
|
|
464
|
+
throw new Error(`Previous upper round match not found`);
|
|
465
|
+
if (!previousLowerMatch)
|
|
466
|
+
throw new Error('Previous lower round match not found');
|
|
467
|
+
// can be either one to one or two to one
|
|
468
|
+
const isLeftOne = previousUpperRoundMatchPosition === previousLowerRoundMatchPosition;
|
|
469
|
+
if (isLeftOne) {
|
|
470
|
+
// one-to-one
|
|
471
|
+
matchRelations.set(match.id, {
|
|
472
|
+
type: 'one-to-one',
|
|
473
|
+
currentMatch: match,
|
|
474
|
+
currentRound: currentRelation.currentRound,
|
|
475
|
+
previousMatch: previousUpperMatch,
|
|
476
|
+
previousRound: currentRelation.previousRound,
|
|
477
|
+
nextMatch,
|
|
478
|
+
nextRound: currentRelation.nextRound,
|
|
479
|
+
});
|
|
480
|
+
}
|
|
481
|
+
else {
|
|
482
|
+
// two-to-one
|
|
483
|
+
matchRelations.set(match.id, {
|
|
484
|
+
type: 'two-to-one',
|
|
485
|
+
currentMatch: match,
|
|
486
|
+
currentRound: currentRelation.currentRound,
|
|
487
|
+
previousUpperMatch,
|
|
488
|
+
previousUpperRound: currentRelation.previousRound,
|
|
489
|
+
previousLowerMatch,
|
|
490
|
+
previousLowerRound: currentRelation.previousRound,
|
|
491
|
+
nextMatch,
|
|
492
|
+
nextRound: currentRelation.nextRound,
|
|
493
|
+
});
|
|
494
|
+
}
|
|
495
|
+
break;
|
|
496
|
+
}
|
|
497
|
+
case 'two-to-one': {
|
|
498
|
+
const nextMatch = matchPositionMaps.get(currentRelation.nextRound.id)?.get(nextRoundMatchPosition);
|
|
499
|
+
const previousUpperMatch = matchPositionMaps
|
|
500
|
+
.get(currentRelation.previousUpperRound.id)
|
|
501
|
+
?.get(previousUpperRoundMatchPosition);
|
|
502
|
+
const previousLowerMatch = matchPositionMaps
|
|
503
|
+
.get(currentRelation.previousLowerRound.id)
|
|
504
|
+
?.get(previousLowerRoundMatchPosition);
|
|
505
|
+
if (!nextMatch)
|
|
506
|
+
throw new Error('Next round match not found');
|
|
507
|
+
if (!previousUpperMatch)
|
|
508
|
+
throw new Error(`Previous upper round match not found`);
|
|
509
|
+
if (!previousLowerMatch)
|
|
510
|
+
throw new Error('Previous lower round match not found');
|
|
511
|
+
matchRelations.set(match.id, {
|
|
512
|
+
type: 'two-to-one',
|
|
513
|
+
currentMatch: match,
|
|
514
|
+
currentRound: currentRelation.currentRound,
|
|
515
|
+
previousUpperMatch,
|
|
516
|
+
previousUpperRound: currentRelation.previousUpperRound,
|
|
517
|
+
previousLowerMatch,
|
|
518
|
+
previousLowerRound: currentRelation.previousLowerRound,
|
|
519
|
+
nextMatch,
|
|
520
|
+
nextRound: currentRelation.nextRound,
|
|
521
|
+
});
|
|
522
|
+
break;
|
|
523
|
+
}
|
|
524
|
+
case 'two-to-nothing': {
|
|
525
|
+
const previousUpperMatch = matchPositionMaps
|
|
526
|
+
.get(currentRelation.previousUpperRound.id)
|
|
527
|
+
?.get(previousUpperRoundMatchPosition);
|
|
528
|
+
const previousLowerMatch = matchPositionMaps
|
|
529
|
+
.get(currentRelation.previousUpperRound.id)
|
|
530
|
+
?.get(previousLowerRoundMatchPosition);
|
|
531
|
+
if (!previousUpperMatch)
|
|
532
|
+
throw new Error(`Previous upper round match not found`);
|
|
533
|
+
if (!previousLowerMatch)
|
|
534
|
+
throw new Error('Previous lower round match not found');
|
|
535
|
+
matchRelations.set(match.id, {
|
|
536
|
+
type: 'two-to-nothing',
|
|
537
|
+
currentMatch: match,
|
|
538
|
+
currentRound: currentRelation.currentRound,
|
|
539
|
+
previousUpperMatch,
|
|
540
|
+
previousUpperRound: currentRelation.previousUpperRound,
|
|
541
|
+
previousLowerMatch,
|
|
542
|
+
previousLowerRound: currentRelation.previousUpperRound,
|
|
543
|
+
});
|
|
544
|
+
break;
|
|
545
|
+
}
|
|
546
|
+
}
|
|
547
|
+
}
|
|
548
|
+
return matchRelations;
|
|
549
|
+
};
|
|
550
|
+
export const generateMatchRelationPositions = (currentRelation, match) => {
|
|
551
|
+
switch (currentRelation.type) {
|
|
552
|
+
case 'nothing-to-one': {
|
|
553
|
+
return {
|
|
554
|
+
nextRoundMatchPosition: generateMatchPosition(match, currentRelation.nextRoundMatchFactor),
|
|
555
|
+
previousUpperRoundMatchPosition: FALLBACK_MATCH_POSITION,
|
|
556
|
+
previousLowerRoundMatchPosition: FALLBACK_MATCH_POSITION,
|
|
557
|
+
};
|
|
558
|
+
}
|
|
559
|
+
case 'one-to-nothing': {
|
|
560
|
+
const previousRoundHasDoubleTheMatchCount = currentRelation.previousRoundMatchFactor === 2;
|
|
561
|
+
const doubleUpperMatchCountShift = previousRoundHasDoubleTheMatchCount ? 1 : 0;
|
|
562
|
+
return {
|
|
563
|
+
nextRoundMatchPosition: FALLBACK_MATCH_POSITION,
|
|
564
|
+
previousUpperRoundMatchPosition: (generateMatchPosition(match, currentRelation.previousRoundMatchFactor) -
|
|
565
|
+
doubleUpperMatchCountShift),
|
|
566
|
+
previousLowerRoundMatchPosition: generateMatchPosition(match, currentRelation.previousRoundMatchFactor),
|
|
567
|
+
};
|
|
568
|
+
}
|
|
569
|
+
case 'one-to-one': {
|
|
570
|
+
const previousRoundHasDoubleTheMatchCount = currentRelation.previousRoundMatchFactor === 2;
|
|
571
|
+
const doubleUpperMatchCountShift = previousRoundHasDoubleTheMatchCount ? 1 : 0;
|
|
572
|
+
return {
|
|
573
|
+
nextRoundMatchPosition: generateMatchPosition(match, currentRelation.nextRoundMatchFactor),
|
|
574
|
+
previousUpperRoundMatchPosition: (generateMatchPosition(match, currentRelation.previousRoundMatchFactor) -
|
|
575
|
+
doubleUpperMatchCountShift),
|
|
576
|
+
previousLowerRoundMatchPosition: generateMatchPosition(match, currentRelation.previousRoundMatchFactor),
|
|
577
|
+
};
|
|
578
|
+
}
|
|
579
|
+
case 'two-to-one': {
|
|
580
|
+
return {
|
|
581
|
+
nextRoundMatchPosition: generateMatchPosition(match, currentRelation.nextRoundMatchFactor),
|
|
582
|
+
previousUpperRoundMatchPosition: generateMatchPosition(match, currentRelation.previousUpperRoundMatchFactor),
|
|
583
|
+
previousLowerRoundMatchPosition: generateMatchPosition(match, currentRelation.previousLowerRoundMatchFactor),
|
|
584
|
+
};
|
|
585
|
+
}
|
|
586
|
+
case 'two-to-nothing': {
|
|
587
|
+
return {
|
|
588
|
+
nextRoundMatchPosition: FALLBACK_MATCH_POSITION,
|
|
589
|
+
previousUpperRoundMatchPosition: generateMatchPosition(match, currentRelation.previousUpperRoundMatchFactor),
|
|
590
|
+
previousLowerRoundMatchPosition: generateMatchPosition(match, currentRelation.previousLowerRoundMatchFactor),
|
|
591
|
+
};
|
|
592
|
+
}
|
|
593
|
+
}
|
|
594
|
+
};
|
|
595
|
+
export const generateMatchPosition = (match, factor) => {
|
|
596
|
+
return Math.ceil(match.position * factor);
|
|
597
|
+
};
|
|
598
|
+
export const logRoundRelations = (roundRelations, bracketData) => {
|
|
599
|
+
for (const [roundId, relation] of roundRelations.entries()) {
|
|
600
|
+
const round = bracketData.rounds.get(roundId);
|
|
601
|
+
if (!round) {
|
|
602
|
+
console.error(`Round with id ${roundId} not found in bracket data. The bracket will be malformed.`);
|
|
603
|
+
continue;
|
|
604
|
+
}
|
|
605
|
+
switch (relation.type) {
|
|
606
|
+
case 'nothing-to-one':
|
|
607
|
+
console.log(`START: ${round.name} -> ${relation.nextRound.name} (F: ${relation.nextRoundMatchFactor})`);
|
|
608
|
+
break;
|
|
609
|
+
case 'one-to-nothing':
|
|
610
|
+
console.log(`${relation.previousRound.name} (F: ${relation.previousRoundMatchFactor}) <- ENDING: ${round.name}`);
|
|
611
|
+
break;
|
|
612
|
+
case 'one-to-one':
|
|
613
|
+
console.log(`${relation.previousRound.name} (F: ${relation.previousRoundMatchFactor}) <- ${round.name} -> ${relation.nextRound.name} (F: ${relation.nextRoundMatchFactor})`);
|
|
614
|
+
break;
|
|
615
|
+
case 'two-to-one':
|
|
616
|
+
console.log(`MERGER: ${relation.previousUpperRound.name} (F: ${relation.previousUpperRoundMatchFactor}) AND ${relation.previousLowerRound.name} (F: ${relation.previousLowerRoundMatchFactor}) <- ${round.name} -> ${relation.nextRound.name} (F: ${relation.nextRoundMatchFactor})`);
|
|
617
|
+
break;
|
|
618
|
+
case 'two-to-nothing':
|
|
619
|
+
console.log(`MERGER: ${relation.previousUpperRound.name} (F: ${relation.previousUpperRoundMatchFactor}) AND ${relation.previousLowerRound.name} (F: ${relation.previousLowerRoundMatchFactor}) <- ENDING: ${round.name}`);
|
|
620
|
+
break;
|
|
621
|
+
}
|
|
622
|
+
}
|
|
623
|
+
};
|
|
624
|
+
export const generateMatchParticipantMap = (bracketData) => {
|
|
625
|
+
const matchParticipantMap = new Map();
|
|
626
|
+
const hasElimination = bracketData.mode === TOURNAMENT_MODE.SINGLE_ELIMINATION ||
|
|
627
|
+
bracketData.mode === TOURNAMENT_MODE.DOUBLE_ELIMINATION ||
|
|
628
|
+
bracketData.mode === TOURNAMENT_MODE.SWISS_WITH_ELIMINATION;
|
|
629
|
+
for (const participant of bracketData.participants.values()) {
|
|
630
|
+
let winsTilNow = 0;
|
|
631
|
+
let lossesTilNow = 0;
|
|
632
|
+
let tiesTilNow = 0;
|
|
633
|
+
for (const matchParticipantMatch of participant.matches.values()) {
|
|
634
|
+
const isWinner = matchParticipantMatch.bracketMatch.winner === matchParticipantMatch.side;
|
|
635
|
+
const isLooser = matchParticipantMatch.bracketMatch.winner &&
|
|
636
|
+
matchParticipantMatch.bracketMatch.winner !== matchParticipantMatch.side;
|
|
637
|
+
const isTie = matchParticipantMatch.bracketMatch.status === 'completed' && !matchParticipantMatch.bracketMatch.winner;
|
|
638
|
+
if (isWinner) {
|
|
639
|
+
winsTilNow++;
|
|
640
|
+
}
|
|
641
|
+
else if (isLooser) {
|
|
642
|
+
lossesTilNow++;
|
|
643
|
+
}
|
|
644
|
+
else if (isTie) {
|
|
645
|
+
tiesTilNow++;
|
|
646
|
+
}
|
|
647
|
+
let isEliminated = false;
|
|
648
|
+
let isEliminationMatch = false;
|
|
649
|
+
if (hasElimination) {
|
|
650
|
+
// TODO: Implement elimination logic
|
|
651
|
+
// Means the current match is loss and it's the last match of the participant
|
|
652
|
+
isEliminated = false;
|
|
653
|
+
// Always true for single elimination, never for e.g. groups, depends on the round for double elimination, depends on the loss count for swiss with elimination
|
|
654
|
+
isEliminationMatch = false;
|
|
655
|
+
}
|
|
656
|
+
// TODO: Implement round logic
|
|
657
|
+
// true if its the first round of the bracket
|
|
658
|
+
const isFirstRound = false;
|
|
659
|
+
// true if its the last round of the bracket (eg. final for single elimination)
|
|
660
|
+
const isLastRound = false;
|
|
661
|
+
const participantMatchData = {
|
|
662
|
+
bracketMatch: matchParticipantMatch.bracketMatch,
|
|
663
|
+
result: isWinner ? 'win' : isLooser ? 'loss' : isTie ? 'tie' : null,
|
|
664
|
+
isEliminated,
|
|
665
|
+
isEliminationMatch,
|
|
666
|
+
isFirstRound,
|
|
667
|
+
isLastRound,
|
|
668
|
+
tieCount: tiesTilNow,
|
|
669
|
+
winCount: winsTilNow,
|
|
670
|
+
lossCount: lossesTilNow,
|
|
671
|
+
};
|
|
672
|
+
if (!matchParticipantMap.has(participant.id)) {
|
|
673
|
+
matchParticipantMap.set(participant.id, {
|
|
674
|
+
id: participant.id,
|
|
675
|
+
name: participant.name,
|
|
676
|
+
matches: new Map(),
|
|
677
|
+
});
|
|
678
|
+
}
|
|
679
|
+
const participantData = matchParticipantMap.get(participant.id);
|
|
680
|
+
participantData.matches.set(matchParticipantMatch.bracketMatch.id, participantMatchData);
|
|
681
|
+
}
|
|
682
|
+
}
|
|
683
|
+
return matchParticipantMap;
|
|
684
|
+
};
|
|
685
|
+
const factorialCache = new Map();
|
|
686
|
+
export const getAvailableSwissGroupsForRound = (roundNumber, totalMatchesInRound) => {
|
|
687
|
+
const ADVANCE_WINS = 3;
|
|
688
|
+
const ELIMINATE_LOSSES = 3;
|
|
689
|
+
// Cache factorial calculations
|
|
690
|
+
const getFactorial = (n) => {
|
|
691
|
+
if (n <= 1)
|
|
692
|
+
return 1;
|
|
693
|
+
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
|
|
694
|
+
if (factorialCache.has(n))
|
|
695
|
+
return factorialCache.get(n);
|
|
696
|
+
const result = n * getFactorial(n - 1);
|
|
697
|
+
factorialCache.set(n, result);
|
|
698
|
+
return result;
|
|
699
|
+
};
|
|
700
|
+
// Pre-calculate roundFactorial
|
|
701
|
+
const roundFact = getFactorial(roundNumber);
|
|
702
|
+
let totalCombinations = 0;
|
|
703
|
+
const validGroups = [];
|
|
704
|
+
// Single loop to gather valid groups and total combinations
|
|
705
|
+
for (let wins = roundNumber; wins >= 0; wins--) {
|
|
706
|
+
const losses = roundNumber - wins;
|
|
707
|
+
const remainingGames = ADVANCE_WINS + ELIMINATE_LOSSES - (wins + losses) - 1;
|
|
708
|
+
const notYetEliminated = losses < ELIMINATE_LOSSES;
|
|
709
|
+
const canStillAdvance = wins < ADVANCE_WINS && remainingGames >= 0;
|
|
710
|
+
if (!canStillAdvance || !notYetEliminated)
|
|
711
|
+
continue;
|
|
712
|
+
const combinations = roundFact / (getFactorial(wins) * getFactorial(losses));
|
|
713
|
+
totalCombinations += combinations;
|
|
714
|
+
validGroups.push({ wins, losses, combinations });
|
|
715
|
+
}
|
|
716
|
+
// Create final groups with calculated proportions
|
|
717
|
+
return validGroups.map(({ wins, losses, combinations }) => ({
|
|
718
|
+
id: `${wins}-${losses}`,
|
|
719
|
+
name: `${wins}-${losses}`,
|
|
720
|
+
matchesInGroup: Math.round((combinations / totalCombinations) * totalMatchesInRound),
|
|
721
|
+
}));
|
|
722
|
+
};
|
|
723
|
+
export const generateBracketRoundSwissGroupMaps = (bracketData, matchParticipantMap) => {
|
|
724
|
+
if (bracketData.mode !== TOURNAMENT_MODE.SWISS_WITH_ELIMINATION) {
|
|
725
|
+
return null;
|
|
726
|
+
}
|
|
727
|
+
const roundsWithSwissGroups = new Map();
|
|
728
|
+
let roundNumber = 0;
|
|
729
|
+
for (const bracketRound of bracketData.rounds.values()) {
|
|
730
|
+
const availableGroups = getAvailableSwissGroupsForRound(roundNumber, bracketRound.matchCount);
|
|
731
|
+
const roundSwissData = {
|
|
732
|
+
groups: new Map(),
|
|
733
|
+
};
|
|
734
|
+
for (const group of availableGroups) {
|
|
735
|
+
const subGroup = {
|
|
736
|
+
id: group.id,
|
|
737
|
+
name: group.name,
|
|
738
|
+
matches: new Map(),
|
|
739
|
+
allowedMatchCount: group.matchesInGroup,
|
|
740
|
+
};
|
|
741
|
+
roundSwissData.groups.set(group.id, subGroup);
|
|
742
|
+
}
|
|
743
|
+
const emptyMatchIds = [];
|
|
744
|
+
for (const match of bracketRound.matches.values()) {
|
|
745
|
+
const participantHome = match.home ? (matchParticipantMap.get(match.home) ?? null) : null;
|
|
746
|
+
const participantAway = match.away ? (matchParticipantMap.get(match.away) ?? null) : null;
|
|
747
|
+
const anyParticipant = participantHome || participantAway;
|
|
748
|
+
if (!anyParticipant) {
|
|
749
|
+
emptyMatchIds.push(match.id);
|
|
750
|
+
continue;
|
|
751
|
+
}
|
|
752
|
+
const matchParticipantMatch = anyParticipant.matches.get(match.id);
|
|
753
|
+
if (!matchParticipantMatch)
|
|
754
|
+
throw new Error('Match participant match not found');
|
|
755
|
+
const wins = matchParticipantMatch.winCount;
|
|
756
|
+
const losses = matchParticipantMatch.lossCount;
|
|
757
|
+
const group = roundSwissData.groups.get(`${wins}-${losses}`);
|
|
758
|
+
if (!group)
|
|
759
|
+
throw new Error('Group not found for match: ' + match.id);
|
|
760
|
+
group.matches.set(match.id, match);
|
|
761
|
+
}
|
|
762
|
+
for (const emptyMatchId of emptyMatchIds) {
|
|
763
|
+
const match = bracketRound.matches.get(emptyMatchId);
|
|
764
|
+
if (!match)
|
|
765
|
+
throw new Error('Empty match not found');
|
|
766
|
+
let groupFound = false;
|
|
767
|
+
for (const group of roundSwissData.groups.values()) {
|
|
768
|
+
if (group.matches.size < group.allowedMatchCount) {
|
|
769
|
+
group.matches.set(match.id, match);
|
|
770
|
+
groupFound = true;
|
|
771
|
+
break;
|
|
772
|
+
}
|
|
773
|
+
}
|
|
774
|
+
if (!groupFound) {
|
|
775
|
+
throw new Error('No group found for empty match');
|
|
776
|
+
}
|
|
777
|
+
}
|
|
778
|
+
roundNumber++;
|
|
779
|
+
}
|
|
780
|
+
return roundsWithSwissGroups;
|
|
781
|
+
};
|
|
782
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bracket-new.js","sourceRoot":"","sources":["../../../../../../../../../libs/cdk/src/lib/components/bracket/components/new-bracket/bracket-new.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  MatchListViewUnion,\n  OpponentSide,\n  RoundStageStructureView,\n  RoundStageStructureWithMatchesView,\n  RoundType,\n} from '@ethlete/types';\n\nexport type BracketRoundId = string & { __brand: 'BracketRoundId' };\nexport type BracketMatchId = string & { __brand: 'BracketMatchId' };\nexport type BracketRoundPosition = number & { __brand: 'BracketRoundPosition' };\nexport type BracketMatchPosition = number & { __brand: 'BracketMatchPosition' };\nexport type MatchParticipantId = string & { __brand: 'MatchParticipantId' };\nexport type BracketRoundSwissGroupId = string & { __brand: 'BracketRoundSwissGroupId' };\n\n// Will usually be 0.5, 1 or 2. In Swiss this value will be gibberish\nexport type MatchFactor = number;\n\nexport const FALLBACK_MATCH_POSITION = -1 as BracketMatchPosition;\n\nexport const TOURNAMENT_MODE = {\n  SINGLE_ELIMINATION: 'single-elimination',\n  DOUBLE_ELIMINATION: 'double-elimination',\n  GROUP: 'group',\n  SWISS: 'swiss',\n  SWISS_WITH_ELIMINATION: 'swiss-with-elimination',\n} as const;\n\nexport type TournamentMode = (typeof TOURNAMENT_MODE)[keyof typeof TOURNAMENT_MODE];\n\nexport const COMMON_BRACKET_ROUND_TYPE = {\n  THIRD_PLACE: 'third-place',\n  FINAL: 'final',\n} as const;\n\nexport const SINGLE_ELIMINATION_BRACKET_ROUND_TYPE = {\n  SINGLE_ELIMINATION_BRACKET: 'single-elimination-bracket',\n} as const;\n\nexport const DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE = {\n  UPPER_BRACKET: 'upper-bracket',\n  LOWER_BRACKET: 'lower-bracket',\n  REVERSE_FINAL: 'reverse-final',\n} as const;\n\nexport const SWISS_BRACKET_ROUND_TYPE = {\n  SWISS: 'swiss',\n} as const;\n\nexport const GROUP_BRACKET_ROUND_TYPE = {\n  GROUP: 'group',\n} as const;\n\nexport type CommonBracketRoundType = (typeof COMMON_BRACKET_ROUND_TYPE)[keyof typeof COMMON_BRACKET_ROUND_TYPE];\nexport type SingleEliminationBracketRoundType =\n  | CommonBracketRoundType\n  | (typeof SINGLE_ELIMINATION_BRACKET_ROUND_TYPE)[keyof typeof SINGLE_ELIMINATION_BRACKET_ROUND_TYPE];\nexport type DoubleEliminationBracketRoundType =\n  | CommonBracketRoundType\n  | (typeof DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE)[keyof typeof DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE];\nexport type SwissBracketRoundType = (typeof SWISS_BRACKET_ROUND_TYPE)[keyof typeof SWISS_BRACKET_ROUND_TYPE];\nexport type GroupBracketRoundType = (typeof GROUP_BRACKET_ROUND_TYPE)[keyof typeof GROUP_BRACKET_ROUND_TYPE];\n\nexport type BracketRoundType =\n  | SingleEliminationBracketRoundType\n  | DoubleEliminationBracketRoundType\n  | SwissBracketRoundType\n  | GroupBracketRoundType;\n\nexport type BracketRound<TRoundData, TMatchData> = {\n  // NOTE: This is the logical index inside the bracket.\n  // In a double elimination bracket, the first round of the lower bracket will have the same index as the final round of the upper bracket\n  index: number;\n  type: BracketRoundType;\n  id: BracketRoundId;\n  data: TRoundData;\n  position: BracketRoundPosition;\n  name: string;\n  matchCount: number;\n  matches: BracketMatchMap<TRoundData, TMatchData>;\n};\n\nexport type BracketMatchStatus = 'completed' | 'pending';\n\nexport type BracketMatch<TRoundData, TMatchData> = {\n  data: TMatchData;\n  indexInRound: number;\n  id: BracketMatchId;\n  round: BracketRound<TRoundData, TMatchData>;\n  position: BracketMatchPosition;\n  home: MatchParticipantId | null;\n  away: MatchParticipantId | null;\n  winner: OpponentSide | null;\n  status: BracketMatchStatus;\n};\n\nexport type BracketRoundMap<TRoundData, TMatchData> = Map<BracketRoundId, BracketRound<TRoundData, TMatchData>>;\nexport type BracketMatchMap<TRoundData, TMatchData> = Map<BracketMatchId, BracketMatch<TRoundData, TMatchData>>;\n\nexport type BracketData<TRoundData, TMatchData> = {\n  rounds: BracketRoundMap<TRoundData, TMatchData>;\n  matches: BracketMatchMap<TRoundData, TMatchData>;\n  participants: BracketParticipantMap<TRoundData, TMatchData>;\n  mode: TournamentMode;\n};\n\nexport type AnyBracketData = BracketData<unknown, unknown>;\n\n// One match has one previous match and one next match (this would be the case, if a group tournament is being displayed via a bracket)\nexport type BracketMatchRelationOneToOne<TRoundData, TMatchData> = {\n  type: 'one-to-one';\n  currentMatch: BracketMatch<TRoundData, TMatchData>;\n  currentRound: BracketRound<TRoundData, TMatchData>;\n  previousMatch: BracketMatch<TRoundData, TMatchData>;\n  previousRound: BracketRound<TRoundData, TMatchData>;\n  nextMatch: BracketMatch<TRoundData, TMatchData>;\n  nextRound: BracketRound<TRoundData, TMatchData>;\n};\n\n// The match has no previous match but a single next match (eg. the start of the bracket)\nexport type BracketMatchRelationNothingToOne<TRoundData, TMatchData> = {\n  type: 'nothing-to-one';\n  currentMatch: BracketMatch<TRoundData, TMatchData>;\n  currentRound: BracketRound<TRoundData, TMatchData>;\n  nextMatch: BracketMatch<TRoundData, TMatchData>;\n  nextRound: BracketRound<TRoundData, TMatchData>;\n};\n\n// The match has no next match but a single previous match (eg. reverse finals)\nexport type BracketMatchRelationOneToNothing<TRoundData, TMatchData> = {\n  type: 'one-to-nothing';\n  currentMatch: BracketMatch<TRoundData, TMatchData>;\n  currentRound: BracketRound<TRoundData, TMatchData>;\n  previousMatch: BracketMatch<TRoundData, TMatchData>;\n  previousRound: BracketRound<TRoundData, TMatchData>;\n};\n\n// The match has two previous matches and one next match (eg. a normal match in the bracket that is neither the start nor the end)\nexport type BracketMatchRelationTwoToOne<TRoundData, TMatchData> = {\n  type: 'two-to-one';\n  currentMatch: BracketMatch<TRoundData, TMatchData>;\n  currentRound: BracketRound<TRoundData, TMatchData>;\n  previousUpperMatch: BracketMatch<TRoundData, TMatchData>;\n  previousUpperRound: BracketRound<TRoundData, TMatchData>;\n  previousLowerMatch: BracketMatch<TRoundData, TMatchData>;\n  previousLowerRound: BracketRound<TRoundData, TMatchData>;\n  nextMatch: BracketMatch<TRoundData, TMatchData>;\n  nextRound: BracketRound<TRoundData, TMatchData>;\n};\n\n// The match has two previous matches and no next match (eg. the finals in a single elimination bracket)\nexport type BracketMatchRelationTwoToNothing<TRoundData, TMatchData> = {\n  type: 'two-to-nothing';\n  currentMatch: BracketMatch<TRoundData, TMatchData>;\n  currentRound: BracketRound<TRoundData, TMatchData>;\n  previousUpperMatch: BracketMatch<TRoundData, TMatchData>;\n  previousUpperRound: BracketRound<TRoundData, TMatchData>;\n  previousLowerMatch: BracketMatch<TRoundData, TMatchData>;\n  previousLowerRound: BracketRound<TRoundData, TMatchData>;\n};\n\nexport type BracketMatchRelation<TRoundData, TMatchData> =\n  | BracketMatchRelationOneToOne<TRoundData, TMatchData>\n  | BracketMatchRelationTwoToOne<TRoundData, TMatchData>\n  | BracketMatchRelationNothingToOne<TRoundData, TMatchData>\n  | BracketMatchRelationOneToNothing<TRoundData, TMatchData>\n  | BracketMatchRelationTwoToNothing<TRoundData, TMatchData>;\n\nexport type BracketMatchRelationsMap<TRoundData, TMatchData> = Map<\n  BracketMatchId,\n  BracketMatchRelation<TRoundData, TMatchData>\n>;\n\n// One round has one next round (the first round of the bracket)\nexport type BracketRoundRelationNothingToOne<TRoundData, TMatchData> = {\n  type: 'nothing-to-one';\n  currentRound: BracketRound<TRoundData, TMatchData>;\n  nextRound: BracketRound<TRoundData, TMatchData>;\n  nextRoundMatchFactor: number;\n};\n\n// One round has one previous round (eg. the finals round of the bracket in case of a single elimination bracket)\nexport type BracketRoundRelationOneToNothing<TRoundData, TMatchData> = {\n  type: 'one-to-nothing';\n  currentRound: BracketRound<TRoundData, TMatchData>;\n  previousRound: BracketRound<TRoundData, TMatchData>;\n  previousRoundMatchFactor: number;\n  rootRoundMatchFactor: number;\n};\n\n// One round has one previous round and one next round (eg. a normal round in the bracket that is neither the start nor the end)\nexport type BracketRoundRelationOneToOne<TRoundData, TMatchData> = {\n  type: 'one-to-one';\n  currentRound: BracketRound<TRoundData, TMatchData>;\n  previousRound: BracketRound<TRoundData, TMatchData>;\n  nextRound: BracketRound<TRoundData, TMatchData>;\n  nextRoundMatchFactor: number;\n  previousRoundMatchFactor: number;\n  rootRoundMatchFactor: number;\n};\n\n// One round has two previous rounds and one next round (eg. the finals round in a double elimination bracket when the reverse finals is also being played)\nexport type BracketRoundRelationTwoToOne<TRoundData, TMatchData> = {\n  type: 'two-to-one';\n  currentRound: BracketRound<TRoundData, TMatchData>;\n  previousUpperRound: BracketRound<TRoundData, TMatchData>;\n  previousLowerRound: BracketRound<TRoundData, TMatchData>;\n  nextRound: BracketRound<TRoundData, TMatchData>;\n  nextRoundMatchFactor: number;\n  previousUpperRoundMatchFactor: number;\n  previousLowerRoundMatchFactor: number;\n  upperRootRoundMatchFactor: number;\n  lowerRootRoundMatchFactor: number;\n};\n\n// One round has two previous rounds and no next round (eg. the finals round in a double elimination bracket when the reverse finals is not being played)\nexport type BracketRoundRelationTwoToNothing<TRoundData, TMatchData> = {\n  type: 'two-to-nothing';\n  currentRound: BracketRound<TRoundData, TMatchData>;\n  previousUpperRound: BracketRound<TRoundData, TMatchData>;\n  previousLowerRound: BracketRound<TRoundData, TMatchData>;\n  previousUpperRoundMatchFactor: number;\n  previousLowerRoundMatchFactor: number;\n  upperRootRoundMatchFactor: number;\n  lowerRootRoundMatchFactor: number;\n};\n\nexport type BracketRoundRelation<TRoundData, TMatchData> =\n  | BracketRoundRelationNothingToOne<TRoundData, TMatchData>\n  | BracketRoundRelationOneToNothing<TRoundData, TMatchData>\n  | BracketRoundRelationOneToOne<TRoundData, TMatchData>\n  | BracketRoundRelationTwoToOne<TRoundData, TMatchData>\n  | BracketRoundRelationTwoToNothing<TRoundData, TMatchData>;\n\nexport type BracketRoundRelations<TRoundData, TMatchData> = Map<\n  BracketRoundId,\n  BracketRoundRelation<TRoundData, TMatchData>\n>;\n\nexport type MatchPositionMaps<TRoundData, TMatchData> = Map<\n  BracketRoundId,\n  Map<BracketMatchPosition, BracketMatch<TRoundData, TMatchData>>\n>;\n\nexport type ParticipantMatchResult = 'win' | 'loss' | 'tie';\nexport type ParticipantMatchType = BracketRoundType;\n\nexport type MatchParticipantMatch<TRoundData, TMatchData> = {\n  bracketMatch: BracketMatch<TRoundData, TMatchData>;\n  result: ParticipantMatchResult | null;\n  isEliminated: boolean;\n  isEliminationMatch: boolean;\n  isFirstRound: boolean;\n  isLastRound: boolean;\n  tieCount: number;\n  winCount: number;\n  lossCount: number;\n};\n\nexport type MatchParticipant<TRoundData, TMatchData> = {\n  id: MatchParticipantId;\n  name: string;\n  matches: Map<BracketMatchId, MatchParticipantMatch<TRoundData, TMatchData>>;\n};\n\nexport type BracketParticipantMatch<TRoundData, TMatchData> = {\n  side: OpponentSide;\n  bracketMatch: BracketMatch<TRoundData, TMatchData>;\n};\n\nexport type BracketParticipant<TRoundData, TMatchData> = {\n  id: MatchParticipantId;\n  name: string;\n  matches: Map<BracketMatchId, BracketParticipantMatch<TRoundData, TMatchData>>;\n};\n\nexport type BracketParticipantMap<TRoundData, TMatchData> = Map<\n  MatchParticipantId,\n  BracketParticipant<TRoundData, TMatchData>\n>;\n\nexport type MatchParticipantMap<TRoundData, TMatchData> = Map<\n  MatchParticipantId,\n  MatchParticipant<TRoundData, TMatchData>\n>;\n\nexport type BracketRoundSwissGroup<TRoundData, TMatchData> = {\n  id: BracketRoundSwissGroupId;\n  name: string;\n  matches: BracketMatchMap<TRoundData, TMatchData>;\n  allowedMatchCount: number;\n};\n\nexport type BracketRoundSwissGroupMap<TRoundData, TMatchData> = Map<\n  BracketRoundSwissGroupId,\n  BracketRoundSwissGroup<TRoundData, TMatchData>\n>;\n\nexport type BracketRoundSwissData<TRoundData, TMatchData> = {\n  groups: BracketRoundSwissGroupMap<TRoundData, TMatchData>;\n};\n\nexport type BracketRoundMapWithSwissData<TRoundData, TMatchData> = Map<\n  BracketRoundId,\n  BracketRoundSwissData<TRoundData, TMatchData>\n>;\n\nexport type BracketRoundTypeMap<TRoundData, TMatchData> = Map<\n  BracketRoundType,\n  BracketRoundMap<TRoundData, TMatchData>\n>;\n\nexport const generateRoundTypeFromEthleteRoundType = (\n  type: RoundType,\n  tournamentMode: TournamentMode,\n): BracketRoundType => {\n  switch (type) {\n    case 'normal':\n      switch (tournamentMode) {\n        case 'single-elimination':\n          return SINGLE_ELIMINATION_BRACKET_ROUND_TYPE.SINGLE_ELIMINATION_BRACKET;\n        case 'group':\n          return GROUP_BRACKET_ROUND_TYPE.GROUP;\n        case 'swiss':\n          return SWISS_BRACKET_ROUND_TYPE.SWISS;\n        default:\n          throw new Error(`Unsupported tournament mode for a normal type round: ${tournamentMode}`);\n      }\n    case 'third_place':\n      return COMMON_BRACKET_ROUND_TYPE.THIRD_PLACE;\n    case 'final':\n      return COMMON_BRACKET_ROUND_TYPE.FINAL;\n    case 'reverse_final':\n      return DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE.REVERSE_FINAL;\n    case 'winner_bracket':\n      return DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE.UPPER_BRACKET;\n    case 'loser_bracket':\n      return DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE.LOWER_BRACKET;\n  }\n};\n\nexport const generateTournamentModeFormEthleteRounds = (\n  source: RoundStageStructureWithMatchesView[],\n): TournamentMode => {\n  const firstRound = source[0];\n  const firstMatch = firstRound?.matches[0];\n\n  if (!firstRound) throw new Error('No rounds found');\n  if (!firstMatch) throw new Error('No matches found');\n\n  switch (firstMatch.matchType) {\n    case 'groups':\n      return TOURNAMENT_MODE.GROUP;\n    case 'fifa_swiss': {\n      const lastRound = source[source.length - 1];\n\n      if (!lastRound) throw new Error('No last round found');\n\n      if (lastRound.matches.length !== firstRound.matches.length) {\n        return TOURNAMENT_MODE.SWISS_WITH_ELIMINATION;\n      } else {\n        return TOURNAMENT_MODE.SWISS;\n      }\n    }\n    case 'double_elimination':\n      return TOURNAMENT_MODE.DOUBLE_ELIMINATION;\n    case 'single_elimination':\n      return TOURNAMENT_MODE.SINGLE_ELIMINATION;\n    default:\n      throw new Error(`Unsupported tournament mode: ${firstMatch.matchType}`);\n  }\n};\n\nexport const generateBracketDataForEthlete = (source: RoundStageStructureWithMatchesView[]) => {\n  const tournamentMode = generateTournamentModeFormEthleteRounds(source);\n\n  const bracketData: BracketData<RoundStageStructureView, MatchListViewUnion> = {\n    rounds: new Map(),\n    matches: new Map(),\n    participants: new Map(),\n    mode: tournamentMode,\n  };\n\n  let currentUpperBracketIndex = 0;\n  let currentLowerBracketIndex = 0;\n\n  for (const currentItem of source) {\n    if (bracketData.rounds.has(currentItem.round.id as BracketRoundId)) {\n      throw new Error(`Round with id ${currentItem.round.id} already exists in the bracket data.`);\n    }\n\n    const roundType = generateRoundTypeFromEthleteRoundType(currentItem.round.type, tournamentMode);\n    const isLowerBracket = roundType === DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE.LOWER_BRACKET;\n\n    const bracketRound: BracketRound<RoundStageStructureView, MatchListViewUnion> = {\n      type: roundType,\n      id: currentItem.round.id as BracketRoundId,\n      index: isLowerBracket ? currentLowerBracketIndex : currentUpperBracketIndex,\n      data: currentItem.round,\n      position: currentItem.round.number as BracketRoundPosition,\n      name: currentItem.round.name || currentItem.round.type,\n      matchCount: currentItem.matches.length,\n      matches: new Map(),\n    };\n\n    bracketData.rounds.set(currentItem.round.id as BracketRoundId, bracketRound);\n\n    for (const [matchIndex, match] of currentItem.matches.entries()) {\n      if (bracketData.matches.has(match.id as BracketMatchId)) {\n        throw new Error(`Match with id ${match.id} already exists in the bracket data.`);\n      }\n\n      const bracketMatch: BracketMatch<RoundStageStructureView, MatchListViewUnion> = {\n        id: match.id as BracketMatchId,\n        indexInRound: matchIndex,\n        position: (matchIndex + 1) as BracketMatchPosition,\n        data: match,\n        round: bracketRound,\n        home: (match.home?.id as MatchParticipantId) || null,\n        away: (match.away?.id as MatchParticipantId) || null,\n        winner: match.winningSide,\n        status: match.status === 'published' ? 'completed' : 'pending',\n      };\n\n      bracketData.matches.set(match.id as BracketMatchId, bracketMatch);\n      bracketRound.matches.set(match.id as BracketMatchId, bracketMatch);\n\n      const participants = [match.home, match.away];\n\n      for (const participant of participants) {\n        if (!participant) continue;\n\n        const participantId = participant.id as MatchParticipantId;\n\n        if (!bracketData.participants.has(participantId)) {\n          bracketData.participants.set(participantId, {\n            id: participantId,\n            name: participant.name || 'Unknown',\n            matches: new Map(),\n          });\n        }\n\n        const participantData = bracketData.participants.get(participantId) as BracketParticipant<\n          RoundStageStructureView,\n          MatchListViewUnion\n        >;\n\n        if (!participantData.matches.has(match.id as BracketMatchId)) {\n          participantData.matches.set(match.id as BracketMatchId, {\n            side: match.home?.id === participantId ? 'home' : 'away',\n            bracketMatch,\n          });\n        }\n      }\n    }\n\n    if (isLowerBracket) {\n      currentLowerBracketIndex++;\n    } else {\n      currentUpperBracketIndex++;\n    }\n  }\n\n  return bracketData;\n};\n\nexport const generateMatchPositionMaps = <TRoundData, TMatchData>(bracketData: BracketData<TRoundData, TMatchData>) => {\n  const matchPositionMaps: MatchPositionMaps<TRoundData, TMatchData> = new Map();\n\n  for (const round of bracketData.rounds.values()) {\n    const matchMap = new Map([...round.matches.values()].map((m) => [m.position, m]));\n\n    matchPositionMaps.set(round.id, matchMap);\n  }\n\n  return matchPositionMaps;\n};\n\nexport const generateBracketRoundTypeMap = <TRoundData, TMatchData>(\n  bracketData: BracketData<TRoundData, TMatchData>,\n) => {\n  const roundAmountMap: BracketRoundTypeMap<TRoundData, TMatchData> = new Map();\n\n  for (const round of bracketData.rounds.values()) {\n    if (!roundAmountMap.has(round.type)) {\n      roundAmountMap.set(round.type, new Map([[round.id, round]]));\n    } else {\n      // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n      roundAmountMap.set(round.type, roundAmountMap.get(round.type)!.set(round.id, round));\n    }\n  }\n\n  return roundAmountMap;\n};\n\nconst UPPER_LOOP_ORDER: BracketRoundType[] = [\n  SINGLE_ELIMINATION_BRACKET_ROUND_TYPE.SINGLE_ELIMINATION_BRACKET,\n  DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE.UPPER_BRACKET,\n  SWISS_BRACKET_ROUND_TYPE.SWISS,\n  GROUP_BRACKET_ROUND_TYPE.GROUP,\n  COMMON_BRACKET_ROUND_TYPE.FINAL,\n  DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE.REVERSE_FINAL,\n  COMMON_BRACKET_ROUND_TYPE.THIRD_PLACE,\n];\n\nconst LOWER_LOOP_ORDER: BracketRoundType[] = [DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE.LOWER_BRACKET];\n\nexport const generateRoundRelations = <TRoundData, TMatchData>(bracketData: BracketData<TRoundData, TMatchData>) => {\n  const roundRelations: BracketRoundRelations<TRoundData, TMatchData> = new Map();\n\n  const sortedRoundsByType = new Map<BracketRoundType, BracketRound<TRoundData, TMatchData>[]>();\n\n  for (const round of bracketData.rounds.values()) {\n    if (!sortedRoundsByType.has(round.type)) {\n      sortedRoundsByType.set(round.type, [round]);\n    } else {\n      sortedRoundsByType.get(round.type)?.push(round);\n    }\n  }\n\n  for (const rounds of sortedRoundsByType.values()) {\n    if (rounds.length === 1) continue;\n\n    rounds.sort((a, b) => a.position - b.position);\n  }\n\n  const sortedUpperRounds = UPPER_LOOP_ORDER.map((t) => sortedRoundsByType.get(t))\n    .filter((r) => !!r)\n    .flat();\n  const sortedLowerRounds = LOWER_LOOP_ORDER.map((t) => sortedRoundsByType.get(t))\n    .filter((r) => !!r)\n    .flat();\n\n  const firstUpperRound = sortedUpperRounds[0] || null;\n  const firstLowerRound = sortedLowerRounds[0] || null;\n\n  if (!firstUpperRound) throw new Error('firstUpperRound is null');\n\n  const hasLowerRounds = sortedLowerRounds.length > 0;\n  const lastLowerRound = sortedLowerRounds[sortedLowerRounds.length - 1] || null;\n\n  for (const [currentUpperRoundIndex, currentUpperRound] of sortedUpperRounds.entries()) {\n    const previousUpperRound = sortedUpperRounds[currentUpperRoundIndex - 1] || null;\n    const nextUpperRound = sortedUpperRounds[currentUpperRoundIndex + 1] || null;\n\n    const currentLowerRound = sortedLowerRounds[currentUpperRoundIndex] || null;\n    const previousLowerRound = sortedLowerRounds[currentUpperRoundIndex - 1] || null;\n    const nextLowerRound = sortedLowerRounds[currentUpperRoundIndex + 1] || null;\n\n    const isFirstRound = currentUpperRoundIndex === 0;\n    const isLastUpperRound = currentUpperRoundIndex === sortedUpperRounds.length - 1;\n\n    const isFinal = currentUpperRound.type === COMMON_BRACKET_ROUND_TYPE.FINAL;\n\n    if (isFinal && hasLowerRounds) {\n      // two to one relation\n\n      if (!lastLowerRound || !currentLowerRound || !previousUpperRound)\n        throw new Error('lastLowerRound or currentLowerRound or previousUpperRound is null');\n\n      // in a sync double elimination bracket, the final round has the same index as the last lower round\n      // in an async one, there is always one more round in the lower bracket since we only display a section of the whole tournament\n      const isAsyncBracket = currentLowerRound.id !== lastLowerRound.id;\n      const finalLowerRound = isAsyncBracket ? nextLowerRound : currentLowerRound;\n\n      if (!finalLowerRound) throw new Error('finalLowerRound is null');\n      if (!firstLowerRound) throw new Error('firstLowerRound is null');\n\n      if (finalLowerRound.id !== lastLowerRound.id) throw new Error('finalLowerRound is not the last lower round');\n\n      // if we have a reverse final\n      if (nextUpperRound) {\n        // for the final\n        roundRelations.set(currentUpperRound.id, {\n          type: 'two-to-one',\n          previousUpperRound: previousUpperRound,\n          previousLowerRound: finalLowerRound,\n          nextRound: nextUpperRound,\n          currentRound: currentUpperRound,\n          nextRoundMatchFactor: nextUpperRound.matchCount / currentUpperRound.matchCount,\n          previousUpperRoundMatchFactor: previousUpperRound.matchCount / currentUpperRound.matchCount,\n          previousLowerRoundMatchFactor: finalLowerRound.matchCount / currentUpperRound.matchCount,\n          upperRootRoundMatchFactor: firstUpperRound.matchCount / currentUpperRound.matchCount,\n          lowerRootRoundMatchFactor: firstLowerRound.matchCount / currentUpperRound.matchCount,\n        });\n      } else {\n        // no reverse final means the final is the last round\n\n        // for the final\n        roundRelations.set(currentUpperRound.id, {\n          type: 'two-to-nothing',\n          previousUpperRound: previousUpperRound,\n          previousLowerRound: finalLowerRound,\n          currentRound: currentUpperRound,\n          previousLowerRoundMatchFactor: finalLowerRound.matchCount / currentUpperRound.matchCount,\n          previousUpperRoundMatchFactor: previousUpperRound.matchCount / currentUpperRound.matchCount,\n          lowerRootRoundMatchFactor: firstLowerRound.matchCount / currentUpperRound.matchCount,\n          upperRootRoundMatchFactor: firstUpperRound.matchCount / currentUpperRound.matchCount,\n        });\n      }\n\n      if (isAsyncBracket) {\n        // if this is an async bracket, we need to set the relations for the 2 last lower rounds since they will be skipped by the default one to one logic\n        const preFinalLowerRound = sortedLowerRounds[sortedLowerRounds.length - 2] || null;\n        const prePreFinalLowerRound = sortedLowerRounds[sortedLowerRounds.length - 3] || null;\n\n        if (!preFinalLowerRound) throw new Error('preFinalLowerRound is null');\n\n        if (!firstLowerRound) throw new Error('firstLowerRound is null');\n\n        // for the last lower round\n        roundRelations.set(finalLowerRound.id, {\n          type: 'one-to-one',\n          previousRound: preFinalLowerRound,\n          nextRound: currentUpperRound,\n          currentRound: finalLowerRound,\n          nextRoundMatchFactor: currentUpperRound.matchCount / finalLowerRound.matchCount,\n          previousRoundMatchFactor: preFinalLowerRound.matchCount / finalLowerRound.matchCount,\n          rootRoundMatchFactor: firstLowerRound.matchCount / finalLowerRound.matchCount,\n        });\n\n        if (prePreFinalLowerRound) {\n          // for the pre final lower round\n          roundRelations.set(preFinalLowerRound.id, {\n            type: 'one-to-one',\n            previousRound: prePreFinalLowerRound,\n            nextRound: finalLowerRound,\n            currentRound: preFinalLowerRound,\n            nextRoundMatchFactor: finalLowerRound.matchCount / preFinalLowerRound.matchCount,\n            previousRoundMatchFactor: prePreFinalLowerRound.matchCount / preFinalLowerRound.matchCount,\n            rootRoundMatchFactor: firstLowerRound.matchCount / preFinalLowerRound.matchCount,\n          });\n        } else {\n          // for the first lower round\n          roundRelations.set(preFinalLowerRound.id, {\n            type: 'nothing-to-one',\n            nextRound: finalLowerRound,\n            currentRound: preFinalLowerRound,\n            nextRoundMatchFactor: finalLowerRound.matchCount / preFinalLowerRound.matchCount,\n          });\n        }\n      } else {\n        // this is a sync bracket, we only need to set the relation for the last lower round\n        if (!previousLowerRound) throw new Error('previousLowerRound is null');\n        if (!firstLowerRound) throw new Error('firstLowerRound is null');\n\n        // for the last lower round\n        roundRelations.set(finalLowerRound.id, {\n          type: 'one-to-one',\n          previousRound: previousLowerRound,\n          nextRound: currentUpperRound,\n          currentRound: finalLowerRound,\n          nextRoundMatchFactor: currentUpperRound.matchCount / finalLowerRound.matchCount,\n          previousRoundMatchFactor: previousLowerRound.matchCount / finalLowerRound.matchCount,\n          rootRoundMatchFactor: firstLowerRound.matchCount / finalLowerRound.matchCount,\n        });\n      }\n    } else if (isLastUpperRound) {\n      // one to nothing relation\n\n      if (!previousUpperRound) throw new Error('previousUpperRound is null');\n\n      roundRelations.set(currentUpperRound.id, {\n        type: 'one-to-nothing',\n        previousRound: previousUpperRound,\n        currentRound: currentUpperRound,\n        previousRoundMatchFactor: previousUpperRound.matchCount / currentUpperRound.matchCount,\n        rootRoundMatchFactor: firstUpperRound.matchCount / currentUpperRound.matchCount,\n      });\n    } else if (isFirstRound) {\n      // nothing to one relation\n\n      if (!nextUpperRound) throw new Error('nextUpperRound is null');\n\n      roundRelations.set(currentUpperRound.id, {\n        type: 'nothing-to-one',\n        nextRound: nextUpperRound,\n        currentRound: currentUpperRound,\n        nextRoundMatchFactor: nextUpperRound.matchCount / currentUpperRound.matchCount,\n      });\n\n      if (currentLowerRound) {\n        if (!nextLowerRound) throw new Error('nextLowerRound is null');\n\n        roundRelations.set(currentLowerRound.id, {\n          type: 'nothing-to-one',\n          nextRound: nextLowerRound,\n          currentRound: currentLowerRound,\n          nextRoundMatchFactor: nextLowerRound.matchCount / currentLowerRound.matchCount,\n        });\n      }\n    } else {\n      // one to one relation\n\n      if (!previousUpperRound) throw new Error('previousUpperRound is null');\n      if (!nextUpperRound) throw new Error('nextUpperRound is null');\n\n      roundRelations.set(currentUpperRound.id, {\n        type: 'one-to-one',\n        previousRound: previousUpperRound,\n        nextRound: nextUpperRound,\n        currentRound: currentUpperRound,\n        nextRoundMatchFactor: nextUpperRound.matchCount / currentUpperRound.matchCount,\n        previousRoundMatchFactor: previousUpperRound.matchCount / currentUpperRound.matchCount,\n        rootRoundMatchFactor: firstUpperRound.matchCount / currentUpperRound.matchCount,\n      });\n\n      // we only want to set lower rounds here until the special merging point of the final.\n      // lower bracket rounds after and including the final will be set in the final round block\n      if (currentLowerRound && currentUpperRound.type === DOUBLE_ELIMINATION_BRACKET_ROUND_TYPE.UPPER_BRACKET) {\n        if (!previousLowerRound) throw new Error('previousLowerRound is null');\n        if (!nextLowerRound) throw new Error('nextLowerRound is null');\n        if (!firstLowerRound) throw new Error('firstLowerRound is null');\n\n        roundRelations.set(currentLowerRound.id, {\n          type: 'one-to-one',\n          previousRound: previousLowerRound,\n          nextRound: nextLowerRound,\n          currentRound: currentLowerRound,\n          nextRoundMatchFactor: nextLowerRound.matchCount / currentLowerRound.matchCount,\n          previousRoundMatchFactor: previousLowerRound.matchCount / currentLowerRound.matchCount,\n          rootRoundMatchFactor: firstLowerRound.matchCount / currentLowerRound.matchCount,\n        });\n      }\n    }\n  }\n\n  return roundRelations;\n};\n\nexport const generateMatchRelations = <TRoundData, TMatchData>(\n  bracketData: BracketData<TRoundData, TMatchData>,\n  roundRelations: BracketRoundRelations<TRoundData, TMatchData>,\n  matchPositionMaps: MatchPositionMaps<TRoundData, TMatchData>,\n) => {\n  const matchRelations: BracketMatchRelationsMap<TRoundData, TMatchData> = new Map();\n\n  for (const match of bracketData.matches.values()) {\n    const currentRelation = roundRelations.get(match.round.id);\n\n    if (!currentRelation) throw new Error('Match round not found');\n\n    const { nextRoundMatchPosition, previousLowerRoundMatchPosition, previousUpperRoundMatchPosition } =\n      generateMatchRelationPositions(currentRelation, match);\n\n    switch (currentRelation.type) {\n      case 'nothing-to-one': {\n        const nextMatch = matchPositionMaps.get(currentRelation.nextRound.id)?.get(nextRoundMatchPosition);\n\n        if (!nextMatch) throw new Error('Next round match not found');\n\n        // means left is nothing. right is one\n        matchRelations.set(match.id, {\n          type: 'nothing-to-one',\n          currentMatch: match,\n          currentRound: currentRelation.currentRound,\n          nextRound: currentRelation.nextRound,\n          nextMatch,\n        });\n\n        break;\n      }\n      case 'one-to-nothing': {\n        const previousUpperMatch = matchPositionMaps\n          .get(currentRelation.previousRound.id)\n          ?.get(previousUpperRoundMatchPosition);\n        const previousLowerMatch = matchPositionMaps\n          .get(currentRelation.previousRound.id)\n          ?.get(previousLowerRoundMatchPosition);\n\n        if (!previousUpperMatch) throw new Error('Previous round match not found');\n\n        if (previousUpperRoundMatchPosition !== previousLowerRoundMatchPosition) {\n          // means left is two. right is one\n\n          if (!previousLowerMatch) throw new Error('Previous lower round match not found');\n\n          matchRelations.set(match.id, {\n            type: 'two-to-nothing',\n            currentMatch: match,\n            currentRound: currentRelation.currentRound,\n            previousUpperMatch,\n            previousUpperRound: currentRelation.previousRound,\n            previousLowerMatch,\n            previousLowerRound: currentRelation.previousRound,\n          });\n        } else {\n          // means left is one. right is nothing\n          matchRelations.set(match.id, {\n            type: 'one-to-nothing',\n            currentMatch: match,\n            currentRound: currentRelation.currentRound,\n            previousMatch: previousUpperMatch,\n            previousRound: currentRelation.previousRound,\n          });\n        }\n\n        break;\n      }\n\n      case 'one-to-one': {\n        const nextMatch = matchPositionMaps.get(currentRelation.nextRound.id)?.get(nextRoundMatchPosition);\n        const previousUpperMatch = matchPositionMaps\n          .get(currentRelation.previousRound.id)\n          ?.get(previousUpperRoundMatchPosition);\n        const previousLowerMatch = matchPositionMaps\n          .get(currentRelation.previousRound.id)\n          ?.get(previousLowerRoundMatchPosition);\n\n        if (!nextMatch) throw new Error('Next round match not found');\n        if (!previousUpperMatch) throw new Error(`Previous upper round match not found`);\n        if (!previousLowerMatch) throw new Error('Previous lower round match not found');\n\n        // can be either one to one or two to one\n        const isLeftOne = previousUpperRoundMatchPosition === previousLowerRoundMatchPosition;\n\n        if (isLeftOne) {\n          // one-to-one\n          matchRelations.set(match.id, {\n            type: 'one-to-one',\n            currentMatch: match,\n            currentRound: currentRelation.currentRound,\n            previousMatch: previousUpperMatch,\n            previousRound: currentRelation.previousRound,\n            nextMatch,\n            nextRound: currentRelation.nextRound,\n          });\n        } else {\n          // two-to-one\n          matchRelations.set(match.id, {\n            type: 'two-to-one',\n            currentMatch: match,\n            currentRound: currentRelation.currentRound,\n            previousUpperMatch,\n            previousUpperRound: currentRelation.previousRound,\n            previousLowerMatch,\n            previousLowerRound: currentRelation.previousRound,\n            nextMatch,\n            nextRound: currentRelation.nextRound,\n          });\n        }\n\n        break;\n      }\n      case 'two-to-one': {\n        const nextMatch = matchPositionMaps.get(currentRelation.nextRound.id)?.get(nextRoundMatchPosition);\n\n        const previousUpperMatch = matchPositionMaps\n          .get(currentRelation.previousUpperRound.id)\n          ?.get(previousUpperRoundMatchPosition);\n        const previousLowerMatch = matchPositionMaps\n          .get(currentRelation.previousLowerRound.id)\n          ?.get(previousLowerRoundMatchPosition);\n\n        if (!nextMatch) throw new Error('Next round match not found');\n        if (!previousUpperMatch) throw new Error(`Previous upper round match not found`);\n        if (!previousLowerMatch) throw new Error('Previous lower round match not found');\n\n        matchRelations.set(match.id, {\n          type: 'two-to-one',\n          currentMatch: match,\n          currentRound: currentRelation.currentRound,\n          previousUpperMatch,\n          previousUpperRound: currentRelation.previousUpperRound,\n          previousLowerMatch,\n          previousLowerRound: currentRelation.previousLowerRound,\n          nextMatch,\n          nextRound: currentRelation.nextRound,\n        });\n\n        break;\n      }\n      case 'two-to-nothing': {\n        const previousUpperMatch = matchPositionMaps\n          .get(currentRelation.previousUpperRound.id)\n          ?.get(previousUpperRoundMatchPosition);\n        const previousLowerMatch = matchPositionMaps\n          .get(currentRelation.previousUpperRound.id)\n          ?.get(previousLowerRoundMatchPosition);\n\n        if (!previousUpperMatch) throw new Error(`Previous upper round match not found`);\n        if (!previousLowerMatch) throw new Error('Previous lower round match not found');\n\n        matchRelations.set(match.id, {\n          type: 'two-to-nothing',\n          currentMatch: match,\n          currentRound: currentRelation.currentRound,\n          previousUpperMatch,\n          previousUpperRound: currentRelation.previousUpperRound,\n          previousLowerMatch,\n          previousLowerRound: currentRelation.previousUpperRound,\n        });\n\n        break;\n      }\n    }\n  }\n\n  return matchRelations;\n};\n\nexport const generateMatchRelationPositions = <TRoundData, TMatchData>(\n  currentRelation: BracketRoundRelation<TRoundData, TMatchData>,\n  match: BracketMatch<TRoundData, TMatchData>,\n) => {\n  switch (currentRelation.type) {\n    case 'nothing-to-one': {\n      return {\n        nextRoundMatchPosition: generateMatchPosition(match, currentRelation.nextRoundMatchFactor),\n        previousUpperRoundMatchPosition: FALLBACK_MATCH_POSITION,\n        previousLowerRoundMatchPosition: FALLBACK_MATCH_POSITION,\n      };\n    }\n    case 'one-to-nothing': {\n      const previousRoundHasDoubleTheMatchCount = currentRelation.previousRoundMatchFactor === 2;\n      const doubleUpperMatchCountShift = previousRoundHasDoubleTheMatchCount ? 1 : 0;\n\n      return {\n        nextRoundMatchPosition: FALLBACK_MATCH_POSITION,\n        previousUpperRoundMatchPosition: (generateMatchPosition(match, currentRelation.previousRoundMatchFactor) -\n          doubleUpperMatchCountShift) as BracketMatchPosition,\n        previousLowerRoundMatchPosition: generateMatchPosition(match, currentRelation.previousRoundMatchFactor),\n      };\n    }\n    case 'one-to-one': {\n      const previousRoundHasDoubleTheMatchCount = currentRelation.previousRoundMatchFactor === 2;\n      const doubleUpperMatchCountShift = previousRoundHasDoubleTheMatchCount ? 1 : 0;\n\n      return {\n        nextRoundMatchPosition: generateMatchPosition(match, currentRelation.nextRoundMatchFactor),\n        previousUpperRoundMatchPosition: (generateMatchPosition(match, currentRelation.previousRoundMatchFactor) -\n          doubleUpperMatchCountShift) as BracketMatchPosition,\n        previousLowerRoundMatchPosition: generateMatchPosition(match, currentRelation.previousRoundMatchFactor),\n      };\n    }\n    case 'two-to-one': {\n      return {\n        nextRoundMatchPosition: generateMatchPosition(match, currentRelation.nextRoundMatchFactor),\n        previousUpperRoundMatchPosition: generateMatchPosition(match, currentRelation.previousUpperRoundMatchFactor),\n        previousLowerRoundMatchPosition: generateMatchPosition(match, currentRelation.previousLowerRoundMatchFactor),\n      };\n    }\n    case 'two-to-nothing': {\n      return {\n        nextRoundMatchPosition: FALLBACK_MATCH_POSITION,\n        previousUpperRoundMatchPosition: generateMatchPosition(match, currentRelation.previousUpperRoundMatchFactor),\n        previousLowerRoundMatchPosition: generateMatchPosition(match, currentRelation.previousLowerRoundMatchFactor),\n      };\n    }\n  }\n};\n\nexport const generateMatchPosition = (match: BracketMatch<unknown, unknown>, factor: MatchFactor) => {\n  return Math.ceil(match.position * factor) as BracketMatchPosition;\n};\n\nexport const logRoundRelations = (\n  roundRelations: BracketRoundRelations<unknown, unknown>,\n  bracketData: BracketData<unknown, unknown>,\n) => {\n  for (const [roundId, relation] of roundRelations.entries()) {\n    const round = bracketData.rounds.get(roundId);\n\n    if (!round) {\n      console.error(`Round with id ${roundId} not found in bracket data. The bracket will be malformed.`);\n      continue;\n    }\n\n    switch (relation.type) {\n      case 'nothing-to-one':\n        console.log(`START: ${round.name} -> ${relation.nextRound.name} (F: ${relation.nextRoundMatchFactor})`);\n        break;\n      case 'one-to-nothing':\n        console.log(\n          `${relation.previousRound.name} (F: ${relation.previousRoundMatchFactor}) <- ENDING: ${round.name}`,\n        );\n        break;\n      case 'one-to-one':\n        console.log(\n          `${relation.previousRound.name} (F: ${relation.previousRoundMatchFactor}) <- ${round.name} -> ${relation.nextRound.name} (F: ${relation.nextRoundMatchFactor})`,\n        );\n        break;\n      case 'two-to-one':\n        console.log(\n          `MERGER: ${relation.previousUpperRound.name} (F: ${relation.previousUpperRoundMatchFactor}) AND ${relation.previousLowerRound.name} (F: ${relation.previousLowerRoundMatchFactor}) <- ${round.name} -> ${relation.nextRound.name} (F: ${relation.nextRoundMatchFactor})`,\n        );\n        break;\n      case 'two-to-nothing':\n        console.log(\n          `MERGER: ${relation.previousUpperRound.name} (F: ${relation.previousUpperRoundMatchFactor}) AND ${relation.previousLowerRound.name} (F: ${relation.previousLowerRoundMatchFactor}) <- ENDING: ${round.name}`,\n        );\n        break;\n    }\n  }\n};\n\nexport const generateMatchParticipantMap = <TRoundData, TMatchData>(\n  bracketData: BracketData<TRoundData, TMatchData>,\n) => {\n  const matchParticipantMap: MatchParticipantMap<TRoundData, TMatchData> = new Map();\n\n  const hasElimination =\n    bracketData.mode === TOURNAMENT_MODE.SINGLE_ELIMINATION ||\n    bracketData.mode === TOURNAMENT_MODE.DOUBLE_ELIMINATION ||\n    bracketData.mode === TOURNAMENT_MODE.SWISS_WITH_ELIMINATION;\n\n  for (const participant of bracketData.participants.values()) {\n    let winsTilNow = 0;\n    let lossesTilNow = 0;\n    let tiesTilNow = 0;\n\n    for (const matchParticipantMatch of participant.matches.values()) {\n      const isWinner = matchParticipantMatch.bracketMatch.winner === matchParticipantMatch.side;\n      const isLooser =\n        matchParticipantMatch.bracketMatch.winner &&\n        matchParticipantMatch.bracketMatch.winner !== matchParticipantMatch.side;\n      const isTie =\n        matchParticipantMatch.bracketMatch.status === 'completed' && !matchParticipantMatch.bracketMatch.winner;\n\n      if (isWinner) {\n        winsTilNow++;\n      } else if (isLooser) {\n        lossesTilNow++;\n      } else if (isTie) {\n        tiesTilNow++;\n      }\n\n      let isEliminated = false;\n      let isEliminationMatch = false;\n\n      if (hasElimination) {\n        // TODO: Implement elimination logic\n\n        // Means the current match is loss and it's the last match of the participant\n        isEliminated = false;\n\n        // Always true for single elimination, never for e.g. groups, depends on the round for double elimination, depends on the loss count for swiss with elimination\n        isEliminationMatch = false;\n      }\n\n      // TODO: Implement round logic\n      // true if its the first round of the bracket\n      const isFirstRound = false;\n\n      // true if its the last round of the bracket (eg. final for single elimination)\n      const isLastRound = false;\n\n      const participantMatchData: MatchParticipantMatch<TRoundData, TMatchData> = {\n        bracketMatch: matchParticipantMatch.bracketMatch,\n        result: isWinner ? 'win' : isLooser ? 'loss' : isTie ? 'tie' : null,\n        isEliminated,\n        isEliminationMatch,\n        isFirstRound,\n        isLastRound,\n        tieCount: tiesTilNow,\n        winCount: winsTilNow,\n        lossCount: lossesTilNow,\n      };\n\n      if (!matchParticipantMap.has(participant.id)) {\n        matchParticipantMap.set(participant.id, {\n          id: participant.id,\n          name: participant.name,\n          matches: new Map(),\n        });\n      }\n\n      const participantData = matchParticipantMap.get(participant.id) as MatchParticipant<TRoundData, TMatchData>;\n      participantData.matches.set(matchParticipantMatch.bracketMatch.id, participantMatchData);\n    }\n  }\n\n  return matchParticipantMap;\n};\n\nconst factorialCache = new Map<number, number>();\n\nexport const getAvailableSwissGroupsForRound = (roundNumber: number, totalMatchesInRound: number) => {\n  const ADVANCE_WINS = 3;\n  const ELIMINATE_LOSSES = 3;\n\n  // Cache factorial calculations\n  const getFactorial = (n: number): number => {\n    if (n <= 1) return 1;\n    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n    if (factorialCache.has(n)) return factorialCache.get(n)!;\n\n    const result = n * getFactorial(n - 1);\n    factorialCache.set(n, result);\n    return result;\n  };\n\n  // Pre-calculate roundFactorial\n  const roundFact = getFactorial(roundNumber);\n\n  let totalCombinations = 0;\n  const validGroups: { wins: number; losses: number; combinations: number }[] = [];\n\n  // Single loop to gather valid groups and total combinations\n  for (let wins = roundNumber; wins >= 0; wins--) {\n    const losses = roundNumber - wins;\n    const remainingGames = ADVANCE_WINS + ELIMINATE_LOSSES - (wins + losses) - 1;\n    const notYetEliminated = losses < ELIMINATE_LOSSES;\n    const canStillAdvance = wins < ADVANCE_WINS && remainingGames >= 0;\n\n    if (!canStillAdvance || !notYetEliminated) continue;\n\n    const combinations = roundFact / (getFactorial(wins) * getFactorial(losses));\n    totalCombinations += combinations;\n    validGroups.push({ wins, losses, combinations });\n  }\n\n  // Create final groups with calculated proportions\n  return validGroups.map(({ wins, losses, combinations }) => ({\n    id: `${wins}-${losses}` as BracketRoundSwissGroupId,\n    name: `${wins}-${losses}`,\n    matchesInGroup: Math.round((combinations / totalCombinations) * totalMatchesInRound),\n  }));\n};\n\nexport const generateBracketRoundSwissGroupMaps = <TRoundData, TMatchData>(\n  bracketData: BracketData<TRoundData, TMatchData>,\n  matchParticipantMap: MatchParticipantMap<TRoundData, TMatchData>,\n) => {\n  if (bracketData.mode !== TOURNAMENT_MODE.SWISS_WITH_ELIMINATION) {\n    return null;\n  }\n\n  const roundsWithSwissGroups: BracketRoundMapWithSwissData<TRoundData, TMatchData> = new Map();\n\n  let roundNumber = 0;\n  for (const bracketRound of bracketData.rounds.values()) {\n    const availableGroups = getAvailableSwissGroupsForRound(roundNumber, bracketRound.matchCount);\n\n    const roundSwissData: BracketRoundSwissData<TRoundData, TMatchData> = {\n      groups: new Map(),\n    };\n\n    for (const group of availableGroups) {\n      const subGroup: BracketRoundSwissGroup<TRoundData, TMatchData> = {\n        id: group.id,\n        name: group.name,\n        matches: new Map(),\n        allowedMatchCount: group.matchesInGroup,\n      };\n\n      roundSwissData.groups.set(group.id, subGroup);\n    }\n\n    const emptyMatchIds: BracketMatchId[] = [];\n\n    for (const match of bracketRound.matches.values()) {\n      const participantHome = match.home ? (matchParticipantMap.get(match.home) ?? null) : null;\n      const participantAway = match.away ? (matchParticipantMap.get(match.away) ?? null) : null;\n\n      const anyParticipant = participantHome || participantAway;\n\n      if (!anyParticipant) {\n        emptyMatchIds.push(match.id);\n        continue;\n      }\n\n      const matchParticipantMatch = anyParticipant.matches.get(match.id);\n\n      if (!matchParticipantMatch) throw new Error('Match participant match not found');\n\n      const wins = matchParticipantMatch.winCount;\n      const losses = matchParticipantMatch.lossCount;\n\n      const group = roundSwissData.groups.get(`${wins}-${losses}` as BracketRoundSwissGroupId);\n\n      if (!group) throw new Error('Group not found for match: ' + match.id);\n\n      group.matches.set(match.id, match);\n    }\n\n    for (const emptyMatchId of emptyMatchIds) {\n      const match = bracketRound.matches.get(emptyMatchId);\n\n      if (!match) throw new Error('Empty match not found');\n\n      let groupFound = false;\n      for (const group of roundSwissData.groups.values()) {\n        if (group.matches.size < group.allowedMatchCount) {\n          group.matches.set(match.id, match);\n          groupFound = true;\n          break;\n        }\n      }\n\n      if (!groupFound) {\n        throw new Error('No group found for empty match');\n      }\n    }\n\n    roundNumber++;\n  }\n\n  return roundsWithSwissGroups;\n};\n"]}
|