display objective awards in participant tables

This commit is contained in:
Maxime Réaux 2026-03-03 15:39:30 +01:00
parent 53b1fc916c
commit f55106c260
8 changed files with 160 additions and 24 deletions

View file

@ -64,15 +64,30 @@ class IconName(str, Enum):
VPNTHDRAW = auto() VPNTHDRAW = auto()
VPNTHBREAK = auto() VPNTHBREAK = auto()
VPNTHTIEDRAW = auto() VPNTHTIEDRAW = auto()
NP1STDRAW = auto()
NP1STBREAK = auto()
NP1STTIEDRAW = auto()
NP2NDDRAW = auto()
NP2NDBREAK = auto()
NP2NDTIEDRAW = auto()
NP3RDDRAW = auto()
NP3RDBREAK = auto()
NP3RDTIEDRAW = auto()
RANK_TO_ICON = { VP_RANK_TO_ICON = {
1: IconName.VP1ST, 1: IconName.VP1ST,
2: IconName.VP2ND, 2: IconName.VP2ND,
3: IconName.VP3RD, 3: IconName.VP3RD,
4: IconName.VPNTH, 4: IconName.VPNTH,
} }
NP_RANK_TO_ICON = {
1: IconName.NP1ST,
2: IconName.NP2ND,
3: IconName.NP3RD,
}
class Icons: class Icons:
_icon_cache: Dict[IconName, QIcon] = {} _icon_cache: Dict[IconName, QIcon] = {}
@ -194,6 +209,54 @@ class Icons:
cls.get_pixmap(IconName.DRAW), cls.get_pixmap(IconName.DRAW),
cls.get_pixmap(IconName.TOKEN), cls.get_pixmap(IconName.TOKEN),
) )
elif name == IconName.NP1STDRAW:
pix = cls._compose(
cls.get_pixmap(IconName.NP1ST),
cls.get_pixmap(IconName.DRAW),
)
elif name == IconName.NP1STBREAK:
pix = cls._compose(
cls.get_pixmap(IconName.NP1ST),
cls.get_pixmap(IconName.TOKEN),
)
elif name == IconName.NP1STTIEDRAW:
pix = cls._compose(
cls.get_pixmap(IconName.NP1ST),
cls.get_pixmap(IconName.DRAW),
cls.get_pixmap(IconName.TOKEN),
)
elif name == IconName.NP2NDDRAW:
pix = cls._compose(
cls.get_pixmap(IconName.NP2ND),
cls.get_pixmap(IconName.DRAW),
)
elif name == IconName.NP2NDBREAK:
pix = cls._compose(
cls.get_pixmap(IconName.NP2ND),
cls.get_pixmap(IconName.TOKEN),
)
elif name == IconName.NP2NDTIEDRAW:
pix = cls._compose(
cls.get_pixmap(IconName.NP2ND),
cls.get_pixmap(IconName.DRAW),
cls.get_pixmap(IconName.TOKEN),
)
elif name == IconName.NP3RDDRAW:
pix = cls._compose(
cls.get_pixmap(IconName.NP3RD),
cls.get_pixmap(IconName.DRAW),
)
elif name == IconName.NP3RDBREAK:
pix = cls._compose(
cls.get_pixmap(IconName.NP3RD),
cls.get_pixmap(IconName.TOKEN),
)
elif name == IconName.NP3RDTIEDRAW:
pix = cls._compose(
cls.get_pixmap(IconName.NP3RD),
cls.get_pixmap(IconName.DRAW),
cls.get_pixmap(IconName.TOKEN),
)
else: else:
path = RESOURCES_DIR / cls._paths[name] path = RESOURCES_DIR / cls._paths[name]
pix = QPixmap(path.as_posix()) pix = QPixmap(path.as_posix())

View file

@ -1,6 +1,7 @@
from typing import List, Dict, Tuple, TYPE_CHECKING from typing import List, Dict, Tuple, TYPE_CHECKING
from PyQt6.QtWidgets import QMessageBox, QDialog from PyQt6.QtWidgets import QMessageBox, QDialog
from PyQt6.QtGui import QIcon
from warchron.constants import ( from warchron.constants import (
RefreshScope, RefreshScope,
@ -57,16 +58,30 @@ class CampaignController:
self.app.view.display_campaign_sectors(sectors_for_display) self.app.view.display_campaign_sectors(sectors_for_display)
scores = ScoreService.compute_scores(war, ContextType.CAMPAIGN, campaign_id) scores = ScoreService.compute_scores(war, ContextType.CAMPAIGN, campaign_id)
rows: List[CampaignParticipantScoreDTO] = [] rows: List[CampaignParticipantScoreDTO] = []
icon_map = {} vp_icon_map: Dict[str, QIcon] = {}
objective_icon_maps: Dict[str, Dict[str, QIcon]] = {}
if camp.is_over: if camp.is_over:
icon_map = RankingIcon.compute_icons( vp_icon_map = RankingIcon.compute_icons(
war, ContextType.CAMPAIGN, campaign_id, scores war, ContextType.CAMPAIGN, campaign_id, scores
) )
for obj in war.get_all_objectives():
objective_icon_maps[obj.id] = RankingIcon.compute_icons(
war,
ContextType.CAMPAIGN,
f"{camp.id}:{obj.id}",
scores,
objective_id=obj.id,
)
for camp_part in camp.get_all_campaign_participants(): for camp_part in camp.get_all_campaign_participants():
war_part_id = camp_part.war_participant_id war_part_id = camp_part.war_participant_id
war_part = war.get_war_participant(war_part_id) war_part = war.get_war_participant(war_part_id)
player_name = self.app.model.get_player_name(war_part.player_id) player_name = self.app.model.get_player_name(war_part.player_id)
score = scores[war_part_id] score = scores[war_part_id]
objective_icons = {
obj_id: icon_map[war_part_id]
for obj_id, icon_map in objective_icon_maps.items()
if war_part_id in icon_map
}
rows.append( rows.append(
CampaignParticipantScoreDTO( CampaignParticipantScoreDTO(
campaign_participant_id=camp_part.id, campaign_participant_id=camp_part.id,
@ -77,7 +92,8 @@ class CampaignController:
victory_points=score.victory_points, victory_points=score.victory_points,
narrative_points=dict(score.narrative_points), narrative_points=dict(score.narrative_points),
tokens=war.get_influence_tokens(war_part.id), tokens=war.get_influence_tokens(war_part.id),
rank_icon=icon_map.get(war_part_id), rank_icon=vp_icon_map.get(war_part_id),
objective_icons=objective_icons,
) )
) )
objectives = [ objectives = [

View file

@ -1,5 +1,5 @@
from typing import List, Dict from typing import List, Dict
from dataclasses import dataclass from dataclasses import dataclass, field
from PyQt6.QtGui import QIcon from PyQt6.QtGui import QIcon
@ -118,6 +118,7 @@ class CampaignParticipantScoreDTO:
narrative_points: Dict[str, int] narrative_points: Dict[str, int]
tokens: int tokens: int
rank_icon: QIcon | None = None rank_icon: QIcon | None = None
objective_icons: Dict[str, QIcon] = field(default_factory=dict)
@dataclass(frozen=True, slots=True) @dataclass(frozen=True, slots=True)
@ -130,3 +131,4 @@ class WarParticipantScoreDTO:
narrative_points: Dict[str, int] narrative_points: Dict[str, int]
tokens: int tokens: int
rank_icon: QIcon | None = None rank_icon: QIcon | None = None
objective_icons: Dict[str, QIcon] = field(default_factory=dict)

View file

@ -6,7 +6,8 @@ from warchron.constants import (
ContextType, ContextType,
Icons, Icons,
IconName, IconName,
RANK_TO_ICON, VP_RANK_TO_ICON,
NP_RANK_TO_ICON,
) )
from warchron.model.war import War from warchron.model.war import War
from warchron.model.score_service import ParticipantScore from warchron.model.score_service import ParticipantScore
@ -20,16 +21,37 @@ class RankingIcon:
context_type: ContextType, context_type: ContextType,
context_id: str, context_id: str,
scores: Dict[str, ParticipantScore], scores: Dict[str, ParticipantScore],
*,
objective_id: str | None = None,
) -> Dict[str, QIcon]: ) -> Dict[str, QIcon]:
if objective_id is None:
def value_getter(score: ParticipantScore) -> int:
return score.victory_points
icon_ranking = VP_RANK_TO_ICON
else:
def value_getter(score: ParticipantScore) -> int:
return score.narrative_points.get(objective_id, 0)
icon_ranking = NP_RANK_TO_ICON
ranking = ResultChecker.get_effective_ranking( ranking = ResultChecker.get_effective_ranking(
war, context_type, context_id, scores war, context_type, context_id, scores, value_getter=value_getter
) )
icon_map = {} icon_map: Dict[str, QIcon] = {}
for rank, group, token_map in ranking: for rank, group, token_map in ranking:
base_icon = RANK_TO_ICON.get(rank, IconName.VPNTH) if objective_id and rank not in icon_ranking:
vp = scores[group[0]].victory_points continue
base_icon = icon_ranking.get(
rank, IconName.VPNTH if objective_id is None else None
)
if base_icon is None:
continue
value = value_getter(scores[group[0]])
original_group_size = sum( original_group_size = sum(
1 for s in scores.values() if s.victory_points == vp 1 for s in scores.values() if value_getter(s) == value
) )
for pid in group: for pid in group:
spent = token_map.get(pid, 0) spent = token_map.get(pid, 0)

View file

@ -1,6 +1,7 @@
from typing import List, Tuple, TYPE_CHECKING, Dict from typing import List, Tuple, TYPE_CHECKING, Dict
from PyQt6.QtWidgets import QMessageBox, QDialog from PyQt6.QtWidgets import QMessageBox, QDialog
from PyQt6.QtGui import QIcon
from warchron.constants import ( from warchron.constants import (
RefreshScope, RefreshScope,
@ -53,12 +54,28 @@ class WarController:
self.app.view.display_war_objectives(objectives_for_display) self.app.view.display_war_objectives(objectives_for_display)
scores = ScoreService.compute_scores(war, ContextType.WAR, war.id) scores = ScoreService.compute_scores(war, ContextType.WAR, war.id)
rows: List[WarParticipantScoreDTO] = [] rows: List[WarParticipantScoreDTO] = []
icon_map = {} vp_icon_map: dict[str, QIcon] = {}
objective_icon_maps: Dict[str, Dict[str, QIcon]] = {}
if war.is_over: if war.is_over:
icon_map = RankingIcon.compute_icons(war, ContextType.WAR, war_id, scores) vp_icon_map = RankingIcon.compute_icons(
war, ContextType.WAR, war_id, scores
)
for obj in war.get_all_objectives():
objective_icon_maps[obj.id] = RankingIcon.compute_icons(
war,
ContextType.WAR,
f"{war.id}:{obj.id}",
scores,
objective_id=obj.id,
)
for war_part in war.get_all_war_participants(): for war_part in war.get_all_war_participants():
player_name = self.app.model.get_player_name(war_part.player_id) player_name = self.app.model.get_player_name(war_part.player_id)
score = scores[war_part.id] score = scores[war_part.id]
objective_icons = {
obj_id: icon_map[war_part.id]
for obj_id, icon_map in objective_icon_maps.items()
if war_part.id in icon_map
}
rows.append( rows.append(
WarParticipantScoreDTO( WarParticipantScoreDTO(
war_participant_id=war_part.id, war_participant_id=war_part.id,
@ -68,7 +85,8 @@ class WarController:
victory_points=score.victory_points, victory_points=score.victory_points,
narrative_points=dict(score.narrative_points), narrative_points=dict(score.narrative_points),
tokens=war.get_influence_tokens(war_part.id), tokens=war.get_influence_tokens(war_part.id),
rank_icon=icon_map.get(war_part.id), rank_icon=vp_icon_map.get(war_part.id),
objective_icons=objective_icons,
) )
) )
self.app.view.display_war_participants(rows, objectives_for_display) self.app.view.display_war_participants(rows, objectives_for_display)
@ -133,6 +151,7 @@ class WarController:
RefreshScope.WARS_TREE, item_type=ItemType.WAR, item_id=war_id RefreshScope.WARS_TREE, item_type=ItemType.WAR, item_id=war_id
) )
# FIXME tie dialog with all participant even without tie
def resolve_ties( def resolve_ties(
self, war: War, contexts: List[TieContext] self, war: War, contexts: List[TieContext]
) -> Dict[Tuple[ContextType, str, int | None], Dict[str, bool]]: ) -> Dict[Tuple[ContextType, str, int | None], Dict[str, bool]]:

View file

@ -1,5 +1,5 @@
from __future__ import annotations from __future__ import annotations
from typing import List, Tuple, Dict, TYPE_CHECKING from typing import List, Tuple, Dict, TYPE_CHECKING, Callable
from collections import defaultdict from collections import defaultdict
from warchron.constants import ContextType from warchron.constants import ContextType
@ -36,15 +36,16 @@ class ResultChecker:
context_type: ContextType, context_type: ContextType,
context_id: str, context_id: str,
scores: Dict[str, ParticipantScore], scores: Dict[str, ParticipantScore],
value_getter: Callable[[ParticipantScore], int],
) -> List[Tuple[int, List[str], Dict[str, int]]]: ) -> List[Tuple[int, List[str], Dict[str, int]]]:
vp_buckets: Dict[int, List[str]] = defaultdict(list) buckets: Dict[int, List[str]] = defaultdict(list)
for pid, score in scores.items(): for pid, score in scores.items():
vp_buckets[score.victory_points].append(pid) buckets[value_getter(score)].append(pid)
sorted_vps = sorted(vp_buckets.keys(), reverse=True) sorted_vps = sorted(buckets.keys(), reverse=True)
ranking: List[Tuple[int, List[str], Dict[str, int]]] = [] ranking: List[Tuple[int, List[str], Dict[str, int]]] = []
current_rank = 1 current_rank = 1
for vp in sorted_vps: for value in sorted_vps:
participants = vp_buckets[vp] participants = buckets[value]
if context_type == ContextType.WAR and len(participants) > 1: if context_type == ContextType.WAR and len(participants) > 1:
subgroups = ResultChecker._secondary_sorting_war(war, participants) subgroups = ResultChecker._secondary_sorting_war(war, participants)
for subgroup in subgroups: for subgroup in subgroups:
@ -57,7 +58,7 @@ class ResultChecker:
continue continue
# normal tie-break if tie persists # normal tie-break if tie persists
if not TieResolver.is_tie_resolved( if not TieResolver.is_tie_resolved(
war, context_type, context_id, vp war, context_type, context_id, value
): ):
ranking.append( ranking.append(
(current_rank, subgroup, {pid: 0 for pid in subgroup}) (current_rank, subgroup, {pid: 0 for pid in subgroup})
@ -80,7 +81,7 @@ class ResultChecker:
continue continue
# no tie # no tie
if len(participants) == 1 or not TieResolver.is_tie_resolved( if len(participants) == 1 or not TieResolver.is_tie_resolved(
war, context_type, context_id, vp war, context_type, context_id, value
): ):
ranking.append( ranking.append(
(current_rank, participants, {pid: 0 for pid in participants}) (current_rank, participants, {pid: 0 for pid in participants})
@ -118,7 +119,11 @@ class ResultChecker:
war, ContextType.CAMPAIGN, campaign.id war, ContextType.CAMPAIGN, campaign.id
) )
ranking = ResultChecker.get_effective_ranking( ranking = ResultChecker.get_effective_ranking(
war, ContextType.CAMPAIGN, campaign.id, scores war,
ContextType.CAMPAIGN,
campaign.id,
scores,
lambda s: s.victory_points,
) )
for rank, group, _ in ranking: for rank, group, _ in ranking:
if pid in group: if pid in group:

View file

@ -146,7 +146,11 @@ class TieResolver:
scores = ScoreService.compute_scores(war, ContextType.WAR, war.id) scores = ScoreService.compute_scores(war, ContextType.WAR, war.id)
ranking = ResultChecker.get_effective_ranking( ranking = ResultChecker.get_effective_ranking(
war, ContextType.WAR, war.id, scores war,
ContextType.WAR,
war.id,
scores,
value_getter=lambda s: s.victory_points,
) )
ties: List[TieContext] = [] ties: List[TieContext] = []
for _, group, _ in ranking: for _, group, _ in ranking:
@ -191,6 +195,7 @@ class TieResolver:
ContextType.OBJECTIVE, ContextType.OBJECTIVE,
f"{war.id}:{objective_id}", f"{war.id}:{objective_id}",
scores, scores,
value_getter=lambda s: s.narrative_points.get(objective_id, 0),
) )
ties: List[TieContext] = [] ties: List[TieContext] = []
for _, group, _ in ranking: for _, group, _ in ranking:

View file

@ -413,6 +413,8 @@ class View(QtWidgets.QMainWindow, Ui_MainWindow):
for obj in objectives: for obj in objectives:
value = part.narrative_points.get(obj.id, 0) value = part.narrative_points.get(obj.id, 0)
NP_item = QtWidgets.QTableWidgetItem(str(value)) NP_item = QtWidgets.QTableWidgetItem(str(value))
if part.objective_icons.get(obj.id):
NP_item.setIcon(part.objective_icons[obj.id])
table.setItem(row, col, NP_item) table.setItem(row, col, NP_item)
col += 1 col += 1
table.setItem(row, col, token_item) table.setItem(row, col, token_item)
@ -545,6 +547,8 @@ class View(QtWidgets.QMainWindow, Ui_MainWindow):
for obj in objectives: for obj in objectives:
value = part.narrative_points.get(obj.id, 0) value = part.narrative_points.get(obj.id, 0)
NP_item = QtWidgets.QTableWidgetItem(str(value)) NP_item = QtWidgets.QTableWidgetItem(str(value))
if part.objective_icons.get(obj.id):
NP_item.setIcon(part.objective_icons[obj.id])
table.setItem(row, col, NP_item) table.setItem(row, col, NP_item)
col += 1 col += 1
table.setItem(row, col, token_item) table.setItem(row, col, token_item)