use super::BACKEND;
use crate::hal::native::{shader_strings, WrappedContext};
use crate::hal::scaler::ScreenScaler;
use crate::hal::{setup_quad, Framebuffer, Shader};
use crate::prelude::{BTerm, InitHints, BACKEND_INTERNAL};
use crate::BResult;
use glow::HasContext;
use glutin::{event_loop::EventLoop, window::WindowBuilder, ContextBuilder};
pub fn init_raw<S: ToString>(
width_pixels: u32,
height_pixels: u32,
window_title: S,
platform_hints: InitHints,
) -> BResult<BTerm> {
let mut scaler = ScreenScaler::new(platform_hints.desired_gutter, width_pixels, height_pixels);
let el = EventLoop::new();
let window_size = scaler.new_window_size();
let window_size = glutin::dpi::LogicalSize::new(window_size.width, window_size.height);
let wb = WindowBuilder::new()
.with_title(window_title.to_string())
.with_resizable(platform_hints.fitscreen)
.with_min_inner_size(window_size)
.with_inner_size(window_size);
let windowed_context = ContextBuilder::new()
.with_gl(platform_hints.gl_version)
.with_gl_profile(platform_hints.gl_profile)
.with_hardware_acceleration(Some(true))
.with_vsync(platform_hints.vsync)
.with_srgb(platform_hints.srgb)
.build_windowed(wb, &el)?;
let windowed_context = unsafe { windowed_context.make_current().unwrap() };
if platform_hints.fullscreen {
if let Some(mh) = el.available_monitors().next() {
windowed_context
.window()
.set_fullscreen(Some(glutin::window::Fullscreen::Borderless(Some(mh))));
} else {
return Err("No available monitor found".into());
}
}
let gl = unsafe {
glow::Context::from_loader_function(|ptr| {
windowed_context.get_proc_address(ptr) as *const _
})
};
#[cfg(debug_assertions)]
unsafe {
let gl_version = gl.get_parameter_string(glow::VERSION);
let shader_version = gl.get_parameter_string(glow::SHADING_LANGUAGE_VERSION);
println!(
"Initialized OpenGL with: {}, Shader Language Version: {}",
gl_version, shader_version
);
}
let mut shaders: Vec<Shader> = Vec::new();
shaders.push(Shader::new(
&gl,
shader_strings::CONSOLE_WITH_BG_VS,
shader_strings::CONSOLE_WITH_BG_FS,
));
shaders.push(Shader::new(
&gl,
shader_strings::CONSOLE_NO_BG_VS,
shader_strings::CONSOLE_NO_BG_FS,
));
shaders.push(Shader::new(
&gl,
shader_strings::BACKING_VS,
shader_strings::BACKING_FS,
));
shaders.push(Shader::new(
&gl,
shader_strings::SCANLINES_VS,
shader_strings::SCANLINES_FS,
));
shaders.push(Shader::new(
&gl,
shader_strings::FANCY_CONSOLE_VS,
shader_strings::FANCY_CONSOLE_FS,
));
shaders.push(Shader::new(
&gl,
shader_strings::SPRITE_CONSOLE_VS,
shader_strings::SPRITE_CONSOLE_FS,
));
let initial_dpi_factor = windowed_context.window().scale_factor();
scaler.change_logical_size(width_pixels, height_pixels, initial_dpi_factor as f32);
let backing_fbo = Framebuffer::build_fbo(
&gl,
scaler.logical_size.0 as i32,
scaler.logical_size.1 as i32,
)?;
let quad_vao = setup_quad(&gl);
let mut be = BACKEND.lock();
be.gl = Some(gl);
be.quad_vao = Some(quad_vao);
be.context_wrapper = Some(WrappedContext {
el,
wc: windowed_context,
});
be.backing_buffer = Some(backing_fbo);
be.frame_sleep_time = crate::hal::convert_fps_to_wait(platform_hints.frame_sleep_time);
be.resize_scaling = platform_hints.resize_scaling;
be.screen_scaler = scaler;
BACKEND_INTERNAL.lock().shaders = shaders;
let bterm = BTerm {
width_pixels,
height_pixels,
original_width_pixels: width_pixels,
original_height_pixels: height_pixels,
fps: 0.0,
frame_time_ms: 0.0,
active_console: 0,
key: None,
mouse_pos: (0, 0),
left_click: false,
shift: false,
control: false,
alt: false,
web_button: None,
quitting: false,
post_scanlines: false,
post_screenburn: false,
screen_burn_color: bracket_color::prelude::RGB::from_f32(0.0, 1.0, 1.0),
mouse_visible: true,
};
Ok(bterm)
}