rust/lib-hedgewars-engine/src/world.rs
author alfadur
Sat, 06 Jul 2019 22:04:59 +0300
changeset 15223 b2c086629fb8
parent 15195 e2adb40c7988
child 15268 24828281c9c5
permissions -rw-r--r--
optimize 64-bit sqrt some more
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
14721
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
     1
use fpnum::{fp, FPNum, FPPoint};
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
     2
use hwphysics::{self as hwp, common::GearId};
14159
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
     3
use integral_geometry::{Point, Rect, Size};
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
     4
use land2d::Land2D;
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
     5
use landgen::{
14710
19122a329774 make world renderer optional + fmt
alfadur
parents: 14709
diff changeset
     6
    outline_template::OutlineTemplate, template_based::TemplatedLandGenerator,
19122a329774 make world renderer optional + fmt
alfadur
parents: 14709
diff changeset
     7
    LandGenerationParameters, LandGenerator,
14159
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
     8
};
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
     9
use lfprng::LaggedFibonacciPRNG;
14300
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
    10
15195
e2adb40c7988 fill the atlas with sprites
alfadur
parents: 14723
diff changeset
    11
use crate::render::{camera::Camera, GearRenderer, MapRenderer};
14707
29dbe9ce8b7d add basic map rendering with gl
fkaa
parents: 14378
diff changeset
    12
14300
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
    13
struct GameState {
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
    14
    land: Land2D<u32>,
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
    15
    physics: hwp::World,
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
    16
}
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
    17
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
    18
impl GameState {
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
    19
    fn new(land: Land2D<u32>, physics: hwp::World) -> Self {
14710
19122a329774 make world renderer optional + fmt
alfadur
parents: 14709
diff changeset
    20
        Self { land, physics }
14300
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
    21
    }
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
    22
}
14159
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    23
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    24
pub struct World {
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    25
    random_numbers_gen: LaggedFibonacciPRNG,
14378
4409344db447 Rework EngineInstance::generatePreview, add preview cleanup function in enginelib
unC0Rr
parents: 14300
diff changeset
    26
    preview: Option<Land2D<u8>>,
14300
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
    27
    game_state: Option<GameState>,
15195
e2adb40c7988 fill the atlas with sprites
alfadur
parents: 14723
diff changeset
    28
    map_renderer: Option<MapRenderer>,
e2adb40c7988 fill the atlas with sprites
alfadur
parents: 14723
diff changeset
    29
    gear_renderer: Option<GearRenderer>,
14710
19122a329774 make world renderer optional + fmt
alfadur
parents: 14709
diff changeset
    30
    camera: Camera,
14721
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
    31
    last_gear_id: GearId,
14159
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    32
}
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    33
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    34
impl World {
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    35
    pub fn new() -> Self {
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    36
        Self {
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    37
            random_numbers_gen: LaggedFibonacciPRNG::new(&[]),
14378
4409344db447 Rework EngineInstance::generatePreview, add preview cleanup function in enginelib
unC0Rr
parents: 14300
diff changeset
    38
            preview: None,
14300
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
    39
            game_state: None,
15195
e2adb40c7988 fill the atlas with sprites
alfadur
parents: 14723
diff changeset
    40
            map_renderer: None,
e2adb40c7988 fill the atlas with sprites
alfadur
parents: 14723
diff changeset
    41
            gear_renderer: None,
14710
19122a329774 make world renderer optional + fmt
alfadur
parents: 14709
diff changeset
    42
            camera: Camera::new(),
14721
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
    43
            last_gear_id: GearId::default(),
14159
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    44
        }
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    45
    }
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    46
14710
19122a329774 make world renderer optional + fmt
alfadur
parents: 14709
diff changeset
    47
    pub fn create_renderer(&mut self, width: u16, height: u16) {
14719
4df1aac5f090 a bit of cleanup
alfadur
parents: 14716
diff changeset
    48
        let land_tile_size = Size::square(512);
15195
e2adb40c7988 fill the atlas with sprites
alfadur
parents: 14723
diff changeset
    49
        self.map_renderer = Some(MapRenderer::new(land_tile_size));
e2adb40c7988 fill the atlas with sprites
alfadur
parents: 14723
diff changeset
    50
        self.gear_renderer = Some(GearRenderer::new());
14710
19122a329774 make world renderer optional + fmt
alfadur
parents: 14709
diff changeset
    51
        self.camera = Camera::with_size(Size::new(width as usize, height as usize));
14713
19358c313ebb fix texture initialization
alfadur
parents: 14710
diff changeset
    52
19358c313ebb fix texture initialization
alfadur
parents: 14710
diff changeset
    53
        use mapgen::{theme::Theme, MapGenerator};
19358c313ebb fix texture initialization
alfadur
parents: 14710
diff changeset
    54
        use std::path::Path;
19358c313ebb fix texture initialization
alfadur
parents: 14710
diff changeset
    55
19358c313ebb fix texture initialization
alfadur
parents: 14710
diff changeset
    56
        if let Some(ref state) = self.game_state {
14716
40809bfd44af center camera on land
alfadur
parents: 14715
diff changeset
    57
            self.camera.position = state.land.play_box().center();
14719
4df1aac5f090 a bit of cleanup
alfadur
parents: 14716
diff changeset
    58
14713
19358c313ebb fix texture initialization
alfadur
parents: 14710
diff changeset
    59
            let theme =
19358c313ebb fix texture initialization
alfadur
parents: 14710
diff changeset
    60
                Theme::load(Path::new("../../share/hedgewars/Data/Themes/Cheese/")).unwrap();
14715
946df0bb3b28 collapse mapgen back
alfadur
parents: 14714
diff changeset
    61
            let texture = MapGenerator::new().make_texture(&state.land, &theme);
15195
e2adb40c7988 fill the atlas with sprites
alfadur
parents: 14723
diff changeset
    62
            if let Some(ref mut renderer) = self.map_renderer {
14713
19358c313ebb fix texture initialization
alfadur
parents: 14710
diff changeset
    63
                renderer.init(&texture);
19358c313ebb fix texture initialization
alfadur
parents: 14710
diff changeset
    64
            }
19358c313ebb fix texture initialization
alfadur
parents: 14710
diff changeset
    65
        }
14710
19122a329774 make world renderer optional + fmt
alfadur
parents: 14709
diff changeset
    66
    }
19122a329774 make world renderer optional + fmt
alfadur
parents: 14709
diff changeset
    67
14277
3152d9fdb499 - Move EngineInstance into a separate module
unC0Rr
parents: 14171
diff changeset
    68
    pub fn set_seed(&mut self, seed: &[u8]) {
3152d9fdb499 - Move EngineInstance into a separate module
unC0Rr
parents: 14171
diff changeset
    69
        self.random_numbers_gen = LaggedFibonacciPRNG::new(seed);
3152d9fdb499 - Move EngineInstance into a separate module
unC0Rr
parents: 14171
diff changeset
    70
    }
3152d9fdb499 - Move EngineInstance into a separate module
unC0Rr
parents: 14171
diff changeset
    71
14378
4409344db447 Rework EngineInstance::generatePreview, add preview cleanup function in enginelib
unC0Rr
parents: 14300
diff changeset
    72
    pub fn preview(&self) -> &Option<Land2D<u8>> {
14159
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    73
        &self.preview
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    74
    }
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    75
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    76
    pub fn generate_preview(&mut self) {
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    77
        fn template() -> OutlineTemplate {
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    78
            let mut template = OutlineTemplate::new(Size::new(4096, 2048));
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    79
            template.islands = vec![vec![
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    80
                Rect::from_size_coords(100, 2050, 1, 1),
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    81
                Rect::from_size_coords(100, 500, 400, 1200),
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    82
                Rect::from_size_coords(3600, 500, 400, 1200),
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    83
                Rect::from_size_coords(3900, 2050, 1, 1),
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    84
            ]];
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    85
            template.fill_points = vec![Point::new(1, 0)];
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    86
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    87
            template
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    88
        }
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    89
14300
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
    90
        let params = LandGenerationParameters::new(0u8, u8::max_value(), 5, false, false);
14159
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    91
        let landgen = TemplatedLandGenerator::new(template());
14378
4409344db447 Rework EngineInstance::generatePreview, add preview cleanup function in enginelib
unC0Rr
parents: 14300
diff changeset
    92
        self.preview = Some(landgen.generate_land(&params, &mut self.random_numbers_gen));
4409344db447 Rework EngineInstance::generatePreview, add preview cleanup function in enginelib
unC0Rr
parents: 14300
diff changeset
    93
    }
4409344db447 Rework EngineInstance::generatePreview, add preview cleanup function in enginelib
unC0Rr
parents: 14300
diff changeset
    94
4409344db447 Rework EngineInstance::generatePreview, add preview cleanup function in enginelib
unC0Rr
parents: 14300
diff changeset
    95
    pub fn dispose_preview(&mut self) {
4409344db447 Rework EngineInstance::generatePreview, add preview cleanup function in enginelib
unC0Rr
parents: 14300
diff changeset
    96
        self.preview = None
14159
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
    97
    }
14300
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
    98
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
    99
    pub fn init(&mut self, template: OutlineTemplate) {
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
   100
        let physics = hwp::World::new(template.size);
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
   101
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
   102
        let params = LandGenerationParameters::new(0u32, u32::max_value(), 5, false, false);
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
   103
        let landgen = TemplatedLandGenerator::new(template);
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
   104
        let land = landgen.generate_land(&params, &mut self.random_numbers_gen);
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
   105
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
   106
        self.game_state = Some(GameState::new(land, physics));
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
   107
    }
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
   108
14709
12db7e435ea6 move camera into engine
alfadur
parents: 14707
diff changeset
   109
    pub fn move_camera(&mut self, position_shift: Point, zoom_shift: f32) {
12db7e435ea6 move camera into engine
alfadur
parents: 14707
diff changeset
   110
        self.camera.zoom += zoom_shift;
14714
65c971417780 fix zoom
alfadur
parents: 14713
diff changeset
   111
        self.camera.position += Point::new(
65c971417780 fix zoom
alfadur
parents: 14713
diff changeset
   112
            (position_shift.x as f32 / self.camera.zoom) as i32,
65c971417780 fix zoom
alfadur
parents: 14713
diff changeset
   113
            (position_shift.y as f32 / self.camera.zoom) as i32,
65c971417780 fix zoom
alfadur
parents: 14713
diff changeset
   114
        );
14709
12db7e435ea6 move camera into engine
alfadur
parents: 14707
diff changeset
   115
    }
12db7e435ea6 move camera into engine
alfadur
parents: 14707
diff changeset
   116
12db7e435ea6 move camera into engine
alfadur
parents: 14707
diff changeset
   117
    pub fn render(&mut self) {
15195
e2adb40c7988 fill the atlas with sprites
alfadur
parents: 14723
diff changeset
   118
        if let Some(ref mut renderer) = self.map_renderer {
14710
19122a329774 make world renderer optional + fmt
alfadur
parents: 14709
diff changeset
   119
            unsafe {
19122a329774 make world renderer optional + fmt
alfadur
parents: 14709
diff changeset
   120
                gl::ClearColor(0.4f32, 0f32, 0.2f32, 1f32);
19122a329774 make world renderer optional + fmt
alfadur
parents: 14709
diff changeset
   121
                gl::Clear(gl::COLOR_BUFFER_BIT);
19122a329774 make world renderer optional + fmt
alfadur
parents: 14709
diff changeset
   122
            }
14709
12db7e435ea6 move camera into engine
alfadur
parents: 14707
diff changeset
   123
14723
5915a199cb81 move projection matrix into camera
alfadur
parents: 14721
diff changeset
   124
            renderer.render(&self.camera);
14707
29dbe9ce8b7d add basic map rendering with gl
fkaa
parents: 14378
diff changeset
   125
        }
15195
e2adb40c7988 fill the atlas with sprites
alfadur
parents: 14723
diff changeset
   126
        if let Some(ref mut renderer) = self.gear_renderer {
e2adb40c7988 fill the atlas with sprites
alfadur
parents: 14723
diff changeset
   127
            renderer.render(&self.camera)
e2adb40c7988 fill the atlas with sprites
alfadur
parents: 14723
diff changeset
   128
        }
14707
29dbe9ce8b7d add basic map rendering with gl
fkaa
parents: 14378
diff changeset
   129
    }
29dbe9ce8b7d add basic map rendering with gl
fkaa
parents: 14378
diff changeset
   130
14721
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   131
    fn get_unused_gear_id(&mut self) -> GearId {
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   132
        let id = self.last_gear_id;
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   133
        self.last_gear_id += 1;
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   134
        id
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   135
    }
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   136
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   137
    fn create_gear(&mut self, position: Point) {
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   138
        let id = self.get_unused_gear_id();
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   139
        if let Some(ref mut state) = self.game_state {
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   140
            let fp_position = FPPoint::new(position.x.into(), position.y.into());
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   141
            state.physics.add_gear_data(
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   142
                id,
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   143
                hwp::physics::PhysicsData::new(fp_position, FPPoint::zero()),
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   144
            )
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   145
        }
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   146
    }
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   147
14300
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
   148
    pub fn step(&mut self) {
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
   149
        if let Some(ref mut state) = self.game_state {
14721
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   150
            let next = self.random_numbers_gen.next().unwrap();
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   151
            if next % 32 == 0 {
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   152
                let position = Point::new(
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   153
                    (self.random_numbers_gen.next().unwrap() % state.land.width() as u32) as i32,
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   154
                    0,
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   155
                );
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   156
                self.create_gear(position);
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   157
            }
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   158
        }
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   159
8e74d4eb89f5 add random falling stuff
alfadur
parents: 14720
diff changeset
   160
        if let Some(ref mut state) = self.game_state {
14300
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
   161
            state.physics.step(fp!(1), &state.land);
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
   162
        }
fd8e0e52d5bd add hwphysics to engine
alfadur
parents: 14277
diff changeset
   163
    }
14159
8354b390f1a2 Some refactoring of qmlfrontend. It now shows land preview generated by hedgewars-engine
unC0Rr
parents:
diff changeset
   164
}