Skip to content

Commit 0aa68ac

Browse files
authored
Merge pull request #11062 from bdach/catch-miss-display-fix
Fix inconsistency in displayed miss count between stable & lazer catch scores
2 parents 466936e + 49b9462 commit 0aa68ac

File tree

6 files changed

+52
-36
lines changed

6 files changed

+52
-36
lines changed

resources/css/bem/beatmap-scoreboard-table.less

+1-1
Original file line numberDiff line numberDiff line change
@@ -75,7 +75,7 @@
7575
white-space: nowrap;
7676
}
7777

78-
&--hitstat-count_miss {
78+
&--hitstat-miss {
7979
width: 50px;
8080
}
8181

resources/js/beatmapsets-show/scoreboard/table-row.tsx

+5-5
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ import PpValue from 'scores/pp-value';
1616
import { classWithModifiers, Modifiers } from 'utils/css';
1717
import { formatNumber } from 'utils/html';
1818
import { trans } from 'utils/lang';
19-
import { accuracy, filterMods, hasMenu, isPerfectCombo, modeAttributesMap, rank, scoreUrl, totalScore } from 'utils/score-helper';
19+
import { accuracy, filterMods, hasMenu, isPerfectCombo, attributeDisplayTotals, rank, scoreUrl, totalScore } from 'utils/score-helper';
2020

2121
const bn = 'beatmap-scoreboard-table';
2222

@@ -128,13 +128,13 @@ export default class ScoreboardTableRow extends React.Component<Props> {
128128
{`${formatNumber(score.max_combo)}x`}
129129
</TdLink>
130130

131-
{modeAttributesMap[this.props.beatmap.mode].map((stat) => (
131+
{attributeDisplayTotals(this.props.beatmap.mode, score).map((stat) => (
132132
<TdLink
133-
key={stat.attribute}
133+
key={stat.key}
134134
href={this.scoreUrl}
135-
modifiers={{ zero: (score.statistics[stat.attribute] ?? 0) === 0 }}
135+
modifiers={{ zero: stat.total === 0 }}
136136
>
137-
{formatNumber(score.statistics[stat.attribute] ?? 0)}
137+
{formatNumber(stat.total)}
138138
</TdLink>
139139
))}
140140

resources/js/beatmapsets-show/scoreboard/table.tsx

+2-2
Original file line numberDiff line numberDiff line change
@@ -64,8 +64,8 @@ export default class Table extends React.Component<Props> {
6464
</th>
6565
{modeAttributesMap[this.props.controller.beatmap.mode].map((stat) => (
6666
<th
67-
key={stat.attribute}
68-
className={classWithModifiers(`${bn}__header`, ['hitstat', `hitstat-${stat.attribute}`])}
67+
key={stat.key}
68+
className={classWithModifiers(`${bn}__header`, ['hitstat', `hitstat-${stat.key}`])}
6969
>
7070
{stat.label}
7171
</th>

resources/js/beatmapsets-show/scoreboard/top-card.tsx

+4-4
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ import { rulesetName, shouldShowPp } from 'utils/beatmap-helper';
1919
import { classWithModifiers, Modifiers } from 'utils/css';
2020
import { formatNumber } from 'utils/html';
2121
import { trans } from 'utils/lang';
22-
import { accuracy, filterMods, isPerfectCombo, modeAttributesMap, rank, scoreUrl, totalScore } from 'utils/score-helper';
22+
import { accuracy, filterMods, isPerfectCombo, attributeDisplayTotals, rank, scoreUrl, totalScore } from 'utils/score-helper';
2323

2424
interface Props {
2525
beatmap: BeatmapJson;
@@ -147,13 +147,13 @@ export default class TopCard extends React.PureComponent<Props> {
147147
</div>
148148

149149
<div className='beatmap-score-top__stats beatmap-score-top__stats--wrappable'>
150-
{modeAttributesMap[ruleset].map((attr) => (
151-
<div key={attr.attribute} className='beatmap-score-top__stat'>
150+
{attributeDisplayTotals(ruleset, this.props.score).map((attr) => (
151+
<div key={attr.key} className='beatmap-score-top__stat'>
152152
<div className='beatmap-score-top__stat-header'>
153153
{attr.label}
154154
</div>
155155
<div className='beatmap-score-top__stat-value beatmap-score-top__stat-value--smaller'>
156-
{formatNumber(this.props.score.statistics[attr.attribute] ?? 0)}
156+
{formatNumber(attr.total)}
157157
</div>
158158
</div>
159159
))}

resources/js/scores-show/stats.tsx

+4-4
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@ import { rulesetName, shouldShowPp } from 'utils/beatmap-helper';
1010
import { classWithModifiers } from 'utils/css';
1111
import { formatNumber } from 'utils/html';
1212
import { trans } from 'utils/lang';
13-
import { accuracy, isPerfectCombo, modeAttributesMap } from 'utils/score-helper';
13+
import { accuracy, isPerfectCombo, attributeDisplayTotals } from 'utils/score-helper';
1414

1515
interface Props {
1616
beatmap: BeatmapJson;
@@ -58,13 +58,13 @@ export default function Stats(props: Props) {
5858
)}
5959
</div>
6060
<div className='score-stats__group-row'>
61-
{modeAttributesMap[rulesetName(props.score.ruleset_id)].map((attr) => (
62-
<div key={attr.attribute} className='score-stats__stat'>
61+
{attributeDisplayTotals(rulesetName(props.score.ruleset_id), props.score).map((attr) => (
62+
<div key={attr.key} className='score-stats__stat'>
6363
<div className='score-stats__stat-row score-stats__stat-row--label'>
6464
{attr.label}
6565
</div>
6666
<div className='score-stats__stat-row'>
67-
{formatNumber(props.score.statistics[attr.attribute] ?? 0)}
67+
{formatNumber(attr.total)}
6868
</div>
6969
</div>
7070
))}

resources/js/utils/score-helper.ts

+36-20
Original file line numberDiff line numberDiff line change
@@ -48,41 +48,57 @@ export function isPerfectCombo(score: SoloScoreJson) {
4848
: score.is_perfect_combo;
4949
}
5050

51-
interface AttributeData {
52-
attribute: SoloScoreStatisticsAttribute;
51+
interface AttributeDisplayMapping {
52+
attributes: SoloScoreStatisticsAttribute[];
53+
key: string;
5354
label: string;
5455
}
5556

57+
interface AttributeDisplayTotal {
58+
key: string;
59+
label: string;
60+
total: number;
61+
}
62+
5663
const labelMiss = trans('beatmapsets.show.scoreboard.headers.miss');
5764

58-
export const modeAttributesMap: Record<GameMode, AttributeData[]> = {
65+
export const modeAttributesMap: Record<GameMode, AttributeDisplayMapping[]> = {
5966
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 },
6472
],
6573
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 },
7280
],
7381
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 },
7886
],
7987
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 },
8391
],
8492
};
8593

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+
86102
export function rank(score: SoloScoreJson) {
87103
return shouldReturnLegacyValue(score)
88104
? legacyAccuracyAndRank(score).rank

0 commit comments

Comments
 (0)