Skip to content

Commit 800da80

Browse files
authored
New draft (#149)
1 parent 181dc6d commit 800da80

File tree

62 files changed

+606
-463
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

62 files changed

+606
-463
lines changed

client/Cargo.lock

+1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

client/src/advance_ui.rs

+2-5
Original file line numberDiff line numberDiff line change
@@ -29,10 +29,7 @@ pub enum AdvanceState {
2929
}
3030

3131
fn new_advance_payment(rc: &RenderContext, name: &str) -> Payment {
32-
let p = rc.shown_player;
33-
let options = &p.advance_cost(name);
34-
let available = &p.resources;
35-
Payment::new(options, available, name, false)
32+
rc.new_payment(&rc.shown_player.advance_cost(name), name, false)
3633
}
3734

3835
pub fn show_paid_advance_menu(rc: &RenderContext) -> StateUpdate {
@@ -214,7 +211,7 @@ pub fn pay_advance_dialog(ap: &Payment, rc: &RenderContext) -> StateUpdate {
214211
// select a different advance
215212
return update;
216213
};
217-
payment_dialog(rc, ap, ActiveDialog::AdvancePayment, |payment| {
214+
payment_dialog(rc, ap, true, ActiveDialog::AdvancePayment, |payment| {
218215
StateUpdate::Execute(Action::Playing(PlayingAction::Advance {
219216
advance: ap.name.to_string(),
220217
payment,

client/src/client.rs

+1
Original file line numberDiff line numberDiff line change
@@ -146,6 +146,7 @@ fn render_active_dialog(rc: &RenderContext) -> StateUpdate {
146146
}
147147
ActiveDialog::ExploreResolution(r) => explore_dialog(rc, r),
148148
ActiveDialog::MoveUnits(_) => move_ui::move_units_dialog(rc),
149+
ActiveDialog::MovePayment(p) => move_ui::move_payment_dialog(rc, p),
149150

150151
//status phase
151152
ActiveDialog::FreeAdvance => show_free_advance_menu(rc),

client/src/client_state.rs

+19-16
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,3 @@
1-
use macroquad::prelude::*;
2-
use server::ability_initializer::EventOrigin;
3-
use server::action::Action;
4-
use server::city::{City, MoodState};
5-
use server::combat::{active_attackers, active_defenders, CombatPhase};
6-
use server::content::advances::{NAVIGATION, ROADS};
7-
use server::content::custom_phase_actions::CustomPhaseRequest;
8-
use server::game::{CulturalInfluenceResolution, CurrentMove, Game, GameState};
9-
use server::position::Position;
10-
use server::status_phase::{StatusPhaseAction, StatusPhaseState};
11-
121
use crate::assets::Assets;
132
use crate::client::{Features, GameSyncRequest};
143
use crate::collect_ui::CollectResources;
@@ -18,11 +7,21 @@ use crate::happiness_ui::IncreaseHappinessConfig;
187
use crate::layout_ui::FONT_SIZE;
198
use crate::log_ui::{add_advance_help, advance_help};
209
use crate::map_ui::ExploreResolutionConfig;
21-
use crate::move_ui::{MoveDestination, MoveIntent, MoveSelection};
10+
use crate::move_ui::{MoveDestination, MoveIntent, MovePayment, MoveSelection};
2211
use crate::payment_ui::Payment;
2312
use crate::recruit_unit_ui::{RecruitAmount, RecruitSelection};
2413
use crate::render_context::RenderContext;
2514
use crate::status_phase_ui::ChooseAdditionalAdvances;
15+
use macroquad::prelude::*;
16+
use server::ability_initializer::EventOrigin;
17+
use server::action::Action;
18+
use server::city::{City, MoodState};
19+
use server::combat::{active_attackers, active_defenders, CombatPhase};
20+
use server::content::advances::{NAVIGATION, ROADS};
21+
use server::content::custom_phase_actions::CustomPhaseRequest;
22+
use server::game::{CulturalInfluenceResolution, CurrentMove, Game, GameState};
23+
use server::position::Position;
24+
use server::status_phase::{StatusPhaseAction, StatusPhaseState};
2625

2726
#[derive(Clone)]
2827
pub enum ActiveDialog {
@@ -40,6 +39,7 @@ pub enum ActiveDialog {
4039
RecruitUnitSelection(RecruitAmount),
4140
ReplaceUnits(RecruitSelection),
4241
MoveUnits(MoveSelection),
42+
MovePayment(MovePayment),
4343
CulturalInfluence,
4444
CulturalInfluenceResolution(CulturalInfluenceResolution),
4545
ExploreResolution(ExploreResolutionConfig),
@@ -78,6 +78,7 @@ impl ActiveDialog {
7878
ActiveDialog::RecruitUnitSelection(_) => "recruit unit selection",
7979
ActiveDialog::ReplaceUnits(_) => "replace units",
8080
ActiveDialog::MoveUnits(_) => "move units",
81+
ActiveDialog::MovePayment(_) => "move payment",
8182
ActiveDialog::CulturalInfluence => "cultural influence",
8283
ActiveDialog::CulturalInfluenceResolution(_) => "cultural influence resolution",
8384
ActiveDialog::ExploreResolution(_) => "explore resolution",
@@ -109,7 +110,9 @@ impl ActiveDialog {
109110
"Click on a city to increase happiness".to_string(),
110111
]
111112
}
112-
ActiveDialog::AdvancePayment(_) | ActiveDialog::ConstructionPayment(_) => {
113+
ActiveDialog::AdvancePayment(_)
114+
| ActiveDialog::ConstructionPayment(_)
115+
| ActiveDialog::MovePayment(_) => {
113116
vec!["Pay resources".to_string()]
114117
}
115118
ActiveDialog::CollectResources(collect) => collect.help_text(rc.game),
@@ -144,7 +147,7 @@ impl ActiveDialog {
144147
vec!["Click on a building to influence its culture".to_string()]
145148
}
146149
ActiveDialog::CulturalInfluenceResolution(c) => vec![format!(
147-
"Pay {} culture tokens to influence {}",
150+
"Pay {} to influence {}",
148151
c.roll_boost_cost,
149152
c.city_piece.name()
150153
)],
@@ -509,7 +512,7 @@ impl State {
509512
r.iter()
510513
.map(|p| {
511514
Payment::new(
512-
&p.options,
515+
&p.cost,
513516
&game.get_player(game.active_player()).resources,
514517
&p.name,
515518
p.optional,
@@ -518,7 +521,7 @@ impl State {
518521
.collect(),
519522
),
520523
CustomPhaseRequest::Reward(r) => {
521-
ActiveDialog::CustomPhaseRewardRequest(Payment::new_gain(&r.options, &r.name))
524+
ActiveDialog::CustomPhaseRewardRequest(Payment::new_gain(&r.reward, &r.name))
522525
}
523526
};
524527
}

client/src/construct_ui.rs

+12-15
Original file line numberDiff line numberDiff line change
@@ -7,10 +7,8 @@ use server::city::City;
77
use server::city_pieces::Building;
88
use server::content::custom_actions::CustomAction;
99
use server::map::Terrain;
10-
use server::payment::PaymentOptions;
1110
use server::playing_actions::{Construct, PlayingAction, Recruit};
1211
use server::position::Position;
13-
use server::unit::UnitType;
1412

1513
pub fn new_building_positions(
1614
building: Building,
@@ -40,6 +38,7 @@ pub fn pay_construction_dialog(rc: &RenderContext, cp: &ConstructionPayment) ->
4038
payment_dialog(
4139
rc,
4240
&cp.payment.clone(),
41+
true,
4342
|p| {
4443
let mut new = cp.clone();
4544
new.payment = p;
@@ -69,7 +68,7 @@ pub fn pay_construction_dialog(rc: &RenderContext, cp: &ConstructionPayment) ->
6968
ConstructionProject::Units(r) => StateUpdate::execute_activation(
7069
Action::Playing(PlayingAction::Recruit(Recruit {
7170
city_position: cp.city_position,
72-
units: r.amount.units.clone().to_vec(),
71+
units: r.amount.units.clone(),
7372
payment,
7473
replaced_units: r.replaced_units.clone(),
7574
leader_name: r.amount.leader_name.clone(),
@@ -113,20 +112,18 @@ impl ConstructionPayment {
113112
.unwrap()
114113
.cost
115114
.clone(),
116-
ConstructionProject::Units(sel) => PaymentOptions::resources(
117-
sel.amount
118-
.units
119-
.clone()
120-
.to_vec()
121-
.iter()
122-
.map(UnitType::cost)
123-
.sum(),
124-
),
115+
ConstructionProject::Units(sel) => rc
116+
.shown_player
117+
.recruit_cost(
118+
&sel.amount.units,
119+
city.position,
120+
sel.amount.leader_name.as_ref(),
121+
&sel.replaced_units,
122+
)
123+
.unwrap(),
125124
};
126125

127-
let available = &rc.shown_player.resources;
128-
let payment = Payment::new(&cost, available, name, false);
129-
126+
let payment = rc.new_payment(&cost, name, false);
130127
ConstructionPayment {
131128
player_index: city.player_index,
132129
city_position: city.position,

client/src/custom_actions_ui.rs

+1
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ pub fn reward_dialog(rc: &RenderContext, payment: &Payment) -> StateUpdate {
88
payment_dialog(
99
rc,
1010
payment,
11+
false,
1112
|p| ActiveDialog::CustomPhaseRewardRequest(p.clone()),
1213
|p| {
1314
StateUpdate::Execute(Action::CustomPhaseEvent(CustomPhaseEventAction::Reward(

client/src/happiness_ui.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -40,8 +40,7 @@ impl IncreaseHappinessConfig {
4040
})
4141
.unwrap();
4242

43-
let available = &p.resources;
44-
Payment::new(&payment, available, "Increase happiness", false)
43+
Payment::new(&payment, &p.resources, "Increase happiness", false)
4544
}
4645
}
4746

@@ -132,6 +131,7 @@ pub fn increase_happiness_menu(rc: &RenderContext, h: &IncreaseHappinessConfig)
132131
payment_dialog(
133132
rc,
134133
&h.payment,
134+
true,
135135
|payment| {
136136
ActiveDialog::IncreaseHappiness(IncreaseHappinessConfig {
137137
steps: h.steps.clone(),

client/src/influence_ui.rs

+3-4
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,6 @@ use server::player::Player;
1616
use server::playing_actions::{InfluenceCultureAttempt, PlayingAction};
1717
use server::position::Position;
1818
use server::resource::ResourceType;
19-
use server::resource_pile::ResourcePile;
2019

2120
fn closest_city(player: &Player, position: Position) -> Position {
2221
player
@@ -32,8 +31,8 @@ pub fn cultural_influence_resolution_dialog(
3231
r: &CulturalInfluenceResolution,
3332
) -> StateUpdate {
3433
let name = r.city_piece.name();
35-
let pile = ResourcePile::culture_tokens(r.roll_boost_cost);
36-
show_resource_pile(rc, &pile, &[ResourceType::CultureTokens]);
34+
let pile = &r.roll_boost_cost;
35+
show_resource_pile(rc, pile, &[ResourceType::CultureTokens]);
3736
if ok_button(rc, OkTooltip::Valid(format!("Influence {name} for {pile}"))) {
3837
return StateUpdate::Execute(Action::CulturalInfluenceResolution(true));
3938
}
@@ -77,7 +76,7 @@ fn show_city(rc: &RenderContext, mouse_pos: Vec2, city: &City) -> Option<StateUp
7776
city.position,
7877
*b,
7978
) {
80-
if player.can_afford_resources(&cost) {
79+
if player.can_afford(&cost) {
8180
let name = b.name();
8281
let _ = rc.with_camera(CameraMode::World, |rc| {
8382
draw_circle_lines(center.x, center.y, BUILDING_SIZE, 1., WHITE);

client/src/local_client/bin/main.rs

+4-8
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ use server::game::{Game, GameData};
99
use server::map::Terrain;
1010
use server::position::Position;
1111
use server::resource_pile::ResourcePile;
12-
use server::unit::UnitType;
12+
use server::unit::{UnitType, Units};
1313
use std::env;
1414
use std::fs::File;
1515
use std::io::BufReader;
@@ -75,13 +75,9 @@ pub fn setup_local_game() -> Game {
7575
game.round = 6;
7676
game.dice_roll_outcomes = vec![1, 1, 10, 10, 10, 10, 10, 10, 10, 10];
7777
let add_unit = |game: &mut Game, pos: &str, player_index: usize, unit_type: UnitType| {
78-
game.recruit(
79-
player_index,
80-
vec![unit_type],
81-
Position::from_offset(pos),
82-
None,
83-
&[],
84-
);
78+
let mut units = Units::empty();
79+
units += &unit_type;
80+
game.recruit(player_index, units, Position::from_offset(pos), None, &[]);
8581
};
8682

8783
let player_index1 = 0;

client/src/map_ui.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -146,7 +146,7 @@ fn overlay_color(rc: &RenderContext, pos: Position) -> Color {
146146
} else if s
147147
.destinations
148148
.iter()
149-
.any(|d| matches!(d, MoveDestination::Tile(p) if *p == pos))
149+
.any(|d| matches!(d, MoveDestination::Tile((p, _)) if *p == pos))
150150
{
151151
MOVE_DESTINATION
152152
} else {

client/src/move_ui.rs

+50-9
Original file line numberDiff line numberDiff line change
@@ -2,12 +2,15 @@ use macroquad::math::{u32, Vec2};
22
use macroquad::prelude::Texture2D;
33
use server::action::Action;
44
use server::game::{CurrentMove, Game, GameState};
5+
use server::payment::PaymentOptions;
56
use server::player::Player;
67
use server::position::Position;
7-
use server::unit::{MovementAction, Unit, UnitType};
8+
use server::resource_pile::ResourcePile;
9+
use server::unit::{MoveUnits, MovementAction, Unit, UnitType};
810

911
use crate::client_state::{ActiveDialog, StateUpdate};
1012
use crate::dialog_ui::cancel_button_with_tooltip;
13+
use crate::payment_ui::{payment_dialog, Payment};
1114
use crate::render_context::RenderContext;
1215
use crate::unit_ui::{click_unit, unit_selection_clicked};
1316

@@ -18,6 +21,12 @@ pub enum MoveIntent {
1821
Disembark,
1922
}
2023

24+
#[derive(Clone)]
25+
pub struct MovePayment {
26+
pub action: MovementAction,
27+
pub payment: Payment,
28+
}
29+
2130
impl MoveIntent {
2231
pub fn to_predicate(self) -> impl Fn(&Unit) -> bool {
2332
match self {
@@ -56,7 +65,7 @@ pub fn possible_destinations(
5665
.move_units_destinations(game, units, start, None)
5766
.unwrap_or_default()
5867
.into_iter()
59-
.map(|route| MoveDestination::Tile(route.destination))
68+
.map(|route| MoveDestination::Tile((route.destination, route.cost)))
6069
.collect::<Vec<_>>();
6170

6271
player.units.iter().for_each(|u| {
@@ -75,10 +84,12 @@ fn move_destination(
7584
dest: &MoveDestination,
7685
pos: Position,
7786
unit: Option<u32>,
78-
) -> Option<(Position, Option<u32>)> {
87+
) -> Option<(Position, Option<u32>, Option<PaymentOptions>)> {
7988
match dest {
80-
MoveDestination::Tile(p) if p == &pos => Some((*p, None)),
81-
MoveDestination::Carrier(id) if unit.is_some_and(|u| u == *id) => Some((pos, Some(*id))),
89+
MoveDestination::Tile((p, cost)) if p == &pos => Some((*p, None, Some(cost.clone()))),
90+
MoveDestination::Carrier(id) if unit.is_some_and(|u| u == *id) => {
91+
Some((pos, Some(*id), None))
92+
}
8293
_ => None,
8394
}
8495
}
@@ -87,17 +98,26 @@ pub fn click(rc: &RenderContext, pos: Position, s: &MoveSelection, mouse_pos: Ve
8798
let game = rc.game;
8899
let p = game.get_player(s.player_index);
89100
let carrier = click_unit(rc, pos, mouse_pos, p, false);
90-
if let Some((destination, embark_carrier_id)) = s
101+
if let Some((destination, embark_carrier_id, cost)) = s
91102
.destinations
92103
.iter()
93104
.find_map(|d| move_destination(d, pos, carrier))
94105
{
95106
let units = s.units.clone();
96-
StateUpdate::execute(Action::Movement(MovementAction::Move {
107+
let action = MovementAction::Move(MoveUnits {
97108
units,
98109
destination,
99110
embark_carrier_id,
100-
}))
111+
payment: ResourcePile::empty(),
112+
});
113+
114+
if let Some(cost) = cost {
115+
return StateUpdate::OpenDialog(ActiveDialog::MovePayment(MovePayment {
116+
action,
117+
payment: rc.new_payment(&cost, "Move units", true),
118+
}));
119+
}
120+
StateUpdate::execute(Action::Movement(action))
101121
} else if s.start.is_some_and(|p| p != pos) {
102122
// first need to deselect units
103123
StateUpdate::None
@@ -168,7 +188,7 @@ pub fn movable_units(
168188

169189
#[derive(Clone, Debug)]
170190
pub enum MoveDestination {
171-
Tile(Position),
191+
Tile((Position, PaymentOptions)),
172192
Carrier(u32),
173193
}
174194

@@ -242,3 +262,24 @@ pub(crate) fn move_units_dialog(rc: &RenderContext) -> StateUpdate {
242262
}
243263
StateUpdate::None
244264
}
265+
266+
pub(crate) fn move_payment_dialog(rc: &RenderContext, mp: &MovePayment) -> StateUpdate {
267+
payment_dialog(
268+
rc,
269+
&mp.payment.clone(),
270+
true,
271+
|p| {
272+
let mut new = mp.clone();
273+
new.payment = p;
274+
ActiveDialog::MovePayment(new)
275+
},
276+
|payment| {
277+
if let MovementAction::Move(mut m) = mp.action.clone() {
278+
m.payment = payment;
279+
StateUpdate::execute(Action::Movement(MovementAction::Move(m)))
280+
} else {
281+
panic!("Unexpected action");
282+
}
283+
},
284+
)
285+
}

0 commit comments

Comments
 (0)