15776
|
1 |
use std::time::Duration;
|
|
2 |
|
|
3 |
use futures::executor::block_on;
|
14187
|
4 |
use glutin::{
|
14705
|
5 |
dpi, ContextTrait, DeviceEvent, ElementState, Event, EventsLoop, GlProfile, GlRequest,
|
15776
|
6 |
MouseButton, MouseScrollDelta, Window, WindowBuilder, WindowEvent, WindowedContext,
|
|
7 |
};
|
|
8 |
use hedgewars_engine::instance::EngineInstance;
|
|
9 |
use integral_geometry::Point;
|
|
10 |
use std::error::Error;
|
|
11 |
use wgpu::{
|
|
12 |
Adapter, BackendBit, Color, CommandEncoderDescriptor, Device, DeviceDescriptor, Features,
|
|
13 |
LoadOp, Operations, PowerPreference, PresentMode, Queue, RenderPassColorAttachmentDescriptor,
|
|
14 |
RenderPassDescriptor, RequestAdapterOptions, Surface, SwapChain, SwapChainDescriptor,
|
|
15 |
TextureFormat, TextureUsage,
|
14187
|
16 |
};
|
|
17 |
|
15776
|
18 |
type HwGlRendererContext = WindowedContext;
|
|
19 |
|
|
20 |
struct HwWgpuRenderingContext {
|
|
21 |
window: Window,
|
|
22 |
surface: Surface,
|
|
23 |
adapter: Adapter,
|
|
24 |
device: Device,
|
|
25 |
queue: Queue,
|
|
26 |
swap_chain: SwapChain,
|
|
27 |
}
|
|
28 |
|
|
29 |
enum HwRendererContext {
|
|
30 |
Gl(HwGlRendererContext),
|
|
31 |
Wgpu(HwWgpuRenderingContext),
|
|
32 |
}
|
|
33 |
|
|
34 |
struct ErrorStub;
|
|
35 |
|
|
36 |
impl<T: Error> From<T> for ErrorStub {
|
|
37 |
fn from(_: T) -> Self {
|
|
38 |
ErrorStub
|
|
39 |
}
|
|
40 |
}
|
14704
|
41 |
|
15776
|
42 |
impl HwRendererContext {
|
|
43 |
pub fn window(&self) -> &Window {
|
|
44 |
match self {
|
|
45 |
HwRendererContext::Gl(gl) => &gl.window(),
|
|
46 |
HwRendererContext::Wgpu(wgpu) => &wgpu.window,
|
|
47 |
}
|
|
48 |
}
|
|
49 |
|
|
50 |
pub fn update(&mut self, size: dpi::LogicalSize) {
|
|
51 |
let phys = size.to_physical(self.window().get_hidpi_factor());
|
|
52 |
match self {
|
|
53 |
HwRendererContext::Gl(context) => unsafe {
|
|
54 |
gl::Viewport(0, 0, phys.width as i32, phys.height as i32);
|
|
55 |
},
|
|
56 |
HwRendererContext::Wgpu(context) => {
|
|
57 |
context.swap_chain = context.device.create_swap_chain(
|
|
58 |
&context.surface,
|
|
59 |
&SwapChainDescriptor {
|
|
60 |
usage: TextureUsage::OUTPUT_ATTACHMENT,
|
|
61 |
format: TextureFormat::Bgra8Unorm,
|
|
62 |
width: phys.width as u32,
|
|
63 |
height: phys.height as u32,
|
|
64 |
present_mode: PresentMode::Fifo,
|
|
65 |
},
|
|
66 |
);
|
|
67 |
}
|
|
68 |
}
|
|
69 |
}
|
|
70 |
|
|
71 |
pub fn present(&mut self) -> Result<(), ErrorStub> {
|
|
72 |
match self {
|
|
73 |
HwRendererContext::Gl(context) => context.swap_buffers()?,
|
|
74 |
HwRendererContext::Wgpu(context) => {
|
|
75 |
let frame_view = &context.swap_chain.get_current_frame()?.output.view;
|
14188
|
76 |
|
15776
|
77 |
let mut encoder =
|
|
78 |
context
|
|
79 |
.device
|
|
80 |
.create_command_encoder(&CommandEncoderDescriptor {
|
|
81 |
label: Some("Main encoder"),
|
|
82 |
});
|
|
83 |
encoder.begin_render_pass(&RenderPassDescriptor {
|
|
84 |
color_attachments: &[RenderPassColorAttachmentDescriptor {
|
|
85 |
attachment: &frame_view,
|
|
86 |
resolve_target: None,
|
|
87 |
ops: Operations {
|
|
88 |
load: LoadOp::Clear(Color::BLUE),
|
|
89 |
store: false,
|
|
90 |
},
|
|
91 |
}],
|
|
92 |
depth_stencil_attachment: None,
|
|
93 |
});
|
|
94 |
let buffer = encoder.finish();
|
|
95 |
context.queue.submit(std::iter::once(buffer));
|
|
96 |
}
|
|
97 |
}
|
|
98 |
Ok(())
|
|
99 |
}
|
|
100 |
}
|
|
101 |
|
|
102 |
fn init_wgpu(event_loop: &EventsLoop, size: dpi::LogicalSize) -> HwWgpuRenderingContext {
|
|
103 |
let builder = WindowBuilder::new()
|
|
104 |
.with_title("hwengine")
|
|
105 |
.with_dimensions(size);
|
|
106 |
let window = builder.build(event_loop).unwrap();
|
|
107 |
|
|
108 |
let instance = wgpu::Instance::new(BackendBit::PRIMARY);
|
|
109 |
|
|
110 |
let surface = unsafe { instance.create_surface(&window) };
|
14187
|
111 |
|
15776
|
112 |
let adapter = block_on(instance.request_adapter(&RequestAdapterOptions {
|
|
113 |
power_preference: PowerPreference::HighPerformance,
|
|
114 |
compatible_surface: Some(&surface),
|
|
115 |
}))
|
|
116 |
.unwrap();
|
|
117 |
|
|
118 |
let (device, queue) = block_on(adapter.request_device(&Default::default(), None)).unwrap();
|
|
119 |
|
|
120 |
let size = window.get_inner_size().unwrap();
|
|
121 |
|
|
122 |
let phys = size.to_physical(window.get_hidpi_factor());
|
|
123 |
|
|
124 |
let mut swap_chain = device.create_swap_chain(
|
|
125 |
&surface,
|
|
126 |
&SwapChainDescriptor {
|
|
127 |
usage: TextureUsage::OUTPUT_ATTACHMENT,
|
|
128 |
format: TextureFormat::Bgra8Unorm,
|
|
129 |
width: phys.width as u32,
|
|
130 |
height: phys.height as u32,
|
|
131 |
present_mode: PresentMode::Fifo,
|
|
132 |
},
|
|
133 |
);
|
|
134 |
|
|
135 |
HwWgpuRenderingContext {
|
|
136 |
window,
|
|
137 |
surface,
|
|
138 |
adapter,
|
|
139 |
device,
|
|
140 |
queue,
|
|
141 |
swap_chain,
|
|
142 |
}
|
|
143 |
}
|
|
144 |
|
|
145 |
fn init_gl(event_loop: &EventsLoop, size: dpi::LogicalSize) -> HwGlRendererContext {
|
|
146 |
use glutin::ContextBuilder;
|
|
147 |
|
|
148 |
let builder = WindowBuilder::new()
|
14187
|
149 |
.with_title("hwengine")
|
|
150 |
.with_dimensions(size);
|
|
151 |
|
15776
|
152 |
let context = ContextBuilder::new()
|
14702
|
153 |
.with_gl(GlRequest::Latest)
|
|
154 |
.with_gl_profile(GlProfile::Core)
|
15776
|
155 |
.build_windowed(builder, &event_loop)
|
14705
|
156 |
.ok()
|
|
157 |
.unwrap();
|
14702
|
158 |
|
|
159 |
unsafe {
|
15776
|
160 |
context.make_current().unwrap();
|
|
161 |
gl::load_with(|ptr| context.get_proc_address(ptr) as *const _);
|
14705
|
162 |
|
15776
|
163 |
if let Some(sz) = context.get_inner_size() {
|
|
164 |
let phys = sz.to_physical(context.get_hidpi_factor());
|
14705
|
165 |
|
14702
|
166 |
gl::Viewport(0, 0, phys.width as i32, phys.height as i32);
|
|
167 |
}
|
|
168 |
}
|
|
169 |
|
15776
|
170 |
context
|
|
171 |
}
|
|
172 |
|
|
173 |
fn init(event_loop: &EventsLoop, size: dpi::LogicalSize, use_wgpu: bool) -> HwRendererContext {
|
|
174 |
if use_wgpu {
|
|
175 |
HwRendererContext::Wgpu(init_wgpu(event_loop, size))
|
|
176 |
} else {
|
|
177 |
HwRendererContext::Gl(init_gl(event_loop, size))
|
|
178 |
}
|
14187
|
179 |
}
|
|
180 |
|
|
181 |
fn main() {
|
15776
|
182 |
let use_wgpu = false;
|
14187
|
183 |
let mut event_loop = EventsLoop::new();
|
14702
|
184 |
let (w, h) = (1024.0, 768.0);
|
15776
|
185 |
|
|
186 |
let mut context = init(&event_loop, dpi::LogicalSize::new(w, h), use_wgpu);
|
14187
|
187 |
|
14702
|
188 |
let mut engine = EngineInstance::new();
|
15776
|
189 |
if !use_wgpu {
|
|
190 |
engine.world.create_renderer(w as u16, h as u16);
|
|
191 |
}
|
14188
|
192 |
|
14702
|
193 |
let mut dragging = false;
|
14188
|
194 |
|
14702
|
195 |
use std::time::Instant;
|
14188
|
196 |
|
14702
|
197 |
let mut now = Instant::now();
|
15776
|
198 |
let mut update_time = Instant::now();
|
|
199 |
let mut render_time = Instant::now();
|
14705
|
200 |
|
14187
|
201 |
let mut is_running = true;
|
15766
|
202 |
|
14187
|
203 |
while is_running {
|
15776
|
204 |
let current_time = Instant::now();
|
|
205 |
let delta = current_time - now;
|
|
206 |
now = current_time;
|
14702
|
207 |
let ms = delta.as_secs() as f64 * 1000.0 + delta.subsec_millis() as f64;
|
15776
|
208 |
context.window().set_title(&format!("hwengine {:.3}ms", ms));
|
14705
|
209 |
|
15776
|
210 |
if update_time.elapsed() > Duration::from_millis(10) {
|
|
211 |
update_time = current_time;
|
14716
|
212 |
engine.world.step()
|
|
213 |
}
|
|
214 |
|
14705
|
215 |
event_loop.poll_events(|event| match event {
|
|
216 |
Event::WindowEvent { event, .. } => match event {
|
|
217 |
WindowEvent::CloseRequested => {
|
|
218 |
is_running = false;
|
|
219 |
}
|
15776
|
220 |
WindowEvent::Resized(size) => context.update(size),
|
|
221 |
|
14705
|
222 |
WindowEvent::MouseInput { button, state, .. } => {
|
|
223 |
if let MouseButton::Right = button {
|
14709
|
224 |
dragging = state == ElementState::Pressed;
|
14702
|
225 |
}
|
14705
|
226 |
}
|
14709
|
227 |
|
14705
|
228 |
WindowEvent::MouseWheel { delta, .. } => {
|
|
229 |
let zoom_change = match delta {
|
|
230 |
MouseScrollDelta::LineDelta(x, y) => y as f32 * 0.1f32,
|
|
231 |
MouseScrollDelta::PixelDelta(delta) => {
|
15776
|
232 |
let physical = delta.to_physical(context.window().get_hidpi_factor());
|
14705
|
233 |
physical.y as f32 * 0.1f32
|
|
234 |
}
|
|
235 |
};
|
|
236 |
engine.world.move_camera(Point::ZERO, zoom_change);
|
|
237 |
}
|
|
238 |
_ => (),
|
|
239 |
},
|
|
240 |
Event::DeviceEvent { event, .. } => match event {
|
|
241 |
DeviceEvent::MouseMotion { delta } => {
|
|
242 |
if dragging {
|
|
243 |
engine
|
|
244 |
.world
|
|
245 |
.move_camera(Point::new(delta.0 as i32, delta.1 as i32), 0.0)
|
14702
|
246 |
}
|
|
247 |
}
|
14705
|
248 |
_ => {}
|
|
249 |
},
|
|
250 |
_ => (),
|
14188
|
251 |
});
|
|
252 |
|
15776
|
253 |
if render_time.elapsed() > Duration::from_millis(16) {
|
|
254 |
render_time = current_time;
|
|
255 |
if !use_wgpu {
|
|
256 |
engine.render();
|
|
257 |
}
|
|
258 |
context.present();
|
15766
|
259 |
}
|
14187
|
260 |
}
|
|
261 |
}
|