@@ -48,41 +48,57 @@ export function isPerfectCombo(score: SoloScoreJson) {
48
48
: score . is_perfect_combo ;
49
49
}
50
50
51
- interface AttributeData {
52
- attribute : SoloScoreStatisticsAttribute ;
51
+ interface AttributeDisplayMapping {
52
+ attributes : SoloScoreStatisticsAttribute [ ] ;
53
+ key : string ;
53
54
label : string ;
54
55
}
55
56
57
+ interface AttributeDisplayTotal {
58
+ key : string ;
59
+ label : string ;
60
+ total : number ;
61
+ }
62
+
56
63
const labelMiss = trans ( 'beatmapsets.show.scoreboard.headers.miss' ) ;
57
64
58
- export const modeAttributesMap : Record < GameMode , AttributeData [ ] > = {
65
+ export const modeAttributesMap : Record < GameMode , AttributeDisplayMapping [ ] > = {
59
66
fruits : [
60
- { attribute : 'great' , label : 'fruits' } ,
61
- { attribute : 'large_tick_hit' , label : 'ticks' } ,
62
- { attribute : 'small_tick_miss' , label : 'drp miss' } ,
63
- { attribute : 'miss' , label : labelMiss } ,
67
+ { attributes : [ 'great' ] , key : 'great' , label : 'fruits' } ,
68
+ { attributes : [ 'large_tick_hit' ] , key : 'ticks' , label : 'ticks' } ,
69
+ { attributes : [ 'small_tick_miss' ] , key : 'drp_miss' , label : 'drp miss' } ,
70
+ // legacy/stable scores merge miss and large_tick_miss into one number
71
+ { attributes : [ 'miss' , 'large_tick_miss' ] , key : 'miss' , label : labelMiss } ,
64
72
] ,
65
73
mania : [
66
- { attribute : 'perfect' , label : 'max' } ,
67
- { attribute : 'great' , label : '300' } ,
68
- { attribute : 'good' , label : '200' } ,
69
- { attribute : 'ok' , label : '100' } ,
70
- { attribute : 'meh' , label : '50' } ,
71
- { attribute : 'miss' , label : labelMiss } ,
74
+ { attributes : [ 'perfect' ] , key : 'perfect' , label : 'max' } ,
75
+ { attributes : [ 'great' ] , key : 'great' , label : '300' } ,
76
+ { attributes : [ 'good' ] , key : 'good' , label : '200' } ,
77
+ { attributes : [ 'ok' ] , key : 'ok' , label : '100' } ,
78
+ { attributes : [ 'meh' ] , key : 'meh' , label : '50' } ,
79
+ { attributes : [ 'miss' ] , key : 'miss' , label : labelMiss } ,
72
80
] ,
73
81
osu : [
74
- { attribute : 'great' , label : '300' } ,
75
- { attribute : 'ok' , label : '100' } ,
76
- { attribute : 'meh' , label : '50' } ,
77
- { attribute : 'miss' , label : labelMiss } ,
82
+ { attributes : [ 'great' ] , key : 'great' , label : '300' } ,
83
+ { attributes : [ 'ok' ] , key : 'ok' , label : '100' } ,
84
+ { attributes : [ 'meh' ] , key : 'meh' , label : '50' } ,
85
+ { attributes : [ 'miss' ] , key : 'miss' , label : labelMiss } ,
78
86
] ,
79
87
taiko : [
80
- { attribute : 'great' , label : 'great' } ,
81
- { attribute : 'ok' , label : 'good' } ,
82
- { attribute : 'miss' , label : labelMiss } ,
88
+ { attributes : [ 'great' ] , key : 'great' , label : 'great' } ,
89
+ { attributes : [ 'ok' ] , key : 'ok' , label : 'good' } ,
90
+ { attributes : [ 'miss' ] , key : 'miss' , label : labelMiss } ,
83
91
] ,
84
92
} ;
85
93
94
+ export function attributeDisplayTotals ( ruleset : GameMode , score : SoloScoreJson ) : AttributeDisplayTotal [ ] {
95
+ return modeAttributesMap [ ruleset ] . map ( ( mapping ) => ( {
96
+ key : mapping . key ,
97
+ label : mapping . label ,
98
+ total : mapping . attributes . reduce ( ( sum , attribute ) => sum + ( score . statistics [ attribute ] ?? 0 ) , 0 ) ,
99
+ } ) ) ;
100
+ }
101
+
86
102
export function rank ( score : SoloScoreJson ) {
87
103
return shouldReturnLegacyValue ( score )
88
104
? legacyAccuracyAndRank ( score ) . rank
0 commit comments