2021-10-18 21:13:32 +00:00
|
|
|
#![allow(unsafe_code)]
|
|
|
|
|
2021-12-26 20:21:28 +00:00
|
|
|
use std::collections::HashMap;
|
|
|
|
|
2021-10-18 21:13:32 +00:00
|
|
|
use egui::{
|
|
|
|
emath::Rect,
|
|
|
|
epaint::{Color32, Mesh, Vertex},
|
|
|
|
};
|
2021-11-03 18:17:07 +00:00
|
|
|
use glow::HasContext;
|
2021-12-26 20:21:28 +00:00
|
|
|
use memoffset::offset_of;
|
2021-10-18 21:13:32 +00:00
|
|
|
|
2021-11-03 18:17:07 +00:00
|
|
|
use crate::misc_util::{
|
2022-01-09 22:04:00 +00:00
|
|
|
as_u8_slice, check_for_gl_error, compile_shader, glow_print, link_program, srgb_texture2d,
|
2021-11-03 18:17:07 +00:00
|
|
|
};
|
|
|
|
use crate::post_process::PostProcess;
|
|
|
|
use crate::shader_version::ShaderVersion;
|
|
|
|
use crate::vao_emulate;
|
2021-10-18 21:13:32 +00:00
|
|
|
|
2021-12-26 20:21:28 +00:00
|
|
|
pub use glow::Context;
|
|
|
|
|
2021-10-18 21:13:32 +00:00
|
|
|
const VERT_SRC: &str = include_str!("shader/vertex.glsl");
|
|
|
|
const FRAG_SRC: &str = include_str!("shader/fragment.glsl");
|
|
|
|
|
|
|
|
/// OpenGL painter
|
|
|
|
///
|
|
|
|
/// This struct must be destroyed with [`Painter::destroy`] before dropping, to ensure OpenGL
|
|
|
|
/// objects have been properly deleted and are not leaked.
|
|
|
|
pub struct Painter {
|
2021-11-03 18:17:07 +00:00
|
|
|
program: glow::Program,
|
2021-10-18 21:13:32 +00:00
|
|
|
u_screen_size: glow::UniformLocation,
|
|
|
|
u_sampler: glow::UniformLocation,
|
2021-11-03 18:17:07 +00:00
|
|
|
egui_texture: Option<glow::Texture>,
|
2021-10-18 21:13:32 +00:00
|
|
|
egui_texture_version: Option<u64>,
|
2021-11-03 18:17:07 +00:00
|
|
|
is_webgl_1: bool,
|
|
|
|
is_embedded: bool,
|
|
|
|
vertex_array: crate::misc_util::VAO,
|
|
|
|
srgb_support: bool,
|
2022-01-06 16:09:53 +00:00
|
|
|
/// The filter used for subsequent textures.
|
|
|
|
texture_filter: TextureFilter,
|
2021-11-03 18:17:07 +00:00
|
|
|
post_process: Option<PostProcess>,
|
|
|
|
vertex_buffer: glow::Buffer,
|
|
|
|
element_array_buffer: glow::Buffer,
|
2021-10-18 21:13:32 +00:00
|
|
|
|
2021-12-26 20:21:28 +00:00
|
|
|
/// Index is the same as in [`egui::TextureId::User`].
|
|
|
|
user_textures: HashMap<u64, glow::Texture>,
|
2021-12-28 12:22:01 +00:00
|
|
|
|
|
|
|
#[cfg(feature = "epi")]
|
|
|
|
next_native_tex_id: u64, // TODO: 128-bit texture space?
|
|
|
|
|
2021-12-26 20:21:28 +00:00
|
|
|
/// Stores outdated OpenGL textures that are yet to be deleted
|
|
|
|
textures_to_destroy: Vec<glow::Texture>,
|
2021-10-18 21:13:32 +00:00
|
|
|
|
2021-12-28 13:24:59 +00:00
|
|
|
/// Used to make sure we are destroyed correctly.
|
2021-12-26 20:21:28 +00:00
|
|
|
destroyed: bool,
|
2021-10-18 21:13:32 +00:00
|
|
|
}
|
|
|
|
|
2022-01-06 16:09:53 +00:00
|
|
|
#[derive(Copy, Clone)]
|
|
|
|
pub enum TextureFilter {
|
|
|
|
Linear,
|
|
|
|
Nearest,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for TextureFilter {
|
|
|
|
fn default() -> Self {
|
|
|
|
TextureFilter::Linear
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TextureFilter {
|
|
|
|
pub(crate) fn glow_code(&self) -> u32 {
|
|
|
|
match self {
|
|
|
|
TextureFilter::Linear => glow::LINEAR,
|
|
|
|
TextureFilter::Nearest => glow::NEAREST,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-18 21:13:32 +00:00
|
|
|
impl Painter {
|
2021-11-03 18:17:07 +00:00
|
|
|
/// Create painter.
|
|
|
|
///
|
|
|
|
/// Set `pp_fb_extent` to the framebuffer size to enable `sRGB` support on OpenGL ES and WebGL.
|
2021-11-13 11:32:01 +00:00
|
|
|
///
|
2021-11-13 11:55:48 +00:00
|
|
|
/// Set `shader_prefix` if you want to turn on shader workaround e.g. `"#define APPLY_BRIGHTENING_GAMMA\n"`
|
|
|
|
/// (see <https://github.com/emilk/egui/issues/794>).
|
|
|
|
///
|
2021-11-03 18:17:07 +00:00
|
|
|
/// # Errors
|
|
|
|
/// will return `Err` below cases
|
|
|
|
/// * failed to compile shader
|
|
|
|
/// * failed to create postprocess on webgl with `sRGB` support
|
|
|
|
/// * failed to create buffer
|
2021-11-13 11:32:01 +00:00
|
|
|
pub fn new(
|
|
|
|
gl: &glow::Context,
|
|
|
|
pp_fb_extent: Option<[i32; 2]>,
|
|
|
|
shader_prefix: &str,
|
|
|
|
) -> Result<Painter, String> {
|
2022-01-09 22:04:00 +00:00
|
|
|
check_for_gl_error(gl, "before Painter::new");
|
|
|
|
|
2021-11-13 11:55:48 +00:00
|
|
|
let support_vao = crate::misc_util::supports_vao(gl);
|
2021-11-03 18:17:07 +00:00
|
|
|
let shader_version = ShaderVersion::get(gl);
|
|
|
|
let is_webgl_1 = shader_version == ShaderVersion::Es100;
|
|
|
|
let header = shader_version.version();
|
2022-01-09 22:04:00 +00:00
|
|
|
glow_print(format!("Shader header: {:?}", header));
|
2021-11-03 18:17:07 +00:00
|
|
|
let srgb_support = gl.supported_extensions().contains("EXT_sRGB");
|
2021-11-13 11:32:01 +00:00
|
|
|
|
2021-11-03 18:17:07 +00:00
|
|
|
let (post_process, srgb_support_define) = match (shader_version, srgb_support) {
|
|
|
|
//WebGL2 support sRGB default
|
|
|
|
(ShaderVersion::Es300, _) | (ShaderVersion::Es100, true) => unsafe {
|
|
|
|
//Add sRGB support marker for fragment shader
|
|
|
|
if let Some([width, height]) = pp_fb_extent {
|
2022-01-09 22:04:00 +00:00
|
|
|
glow_print("WebGL with sRGB enabled so turn on post process");
|
2021-11-03 18:17:07 +00:00
|
|
|
//install post process to correct sRGB color
|
|
|
|
(
|
|
|
|
Some(PostProcess::new(
|
|
|
|
gl,
|
2021-11-13 11:32:01 +00:00
|
|
|
shader_prefix,
|
|
|
|
support_vao,
|
2021-11-03 18:17:07 +00:00
|
|
|
is_webgl_1,
|
|
|
|
width,
|
|
|
|
height,
|
|
|
|
)?),
|
|
|
|
"#define SRGB_SUPPORTED",
|
|
|
|
)
|
|
|
|
} else {
|
2022-01-09 22:04:00 +00:00
|
|
|
glow_print("WebGL or OpenGL ES detected but PostProcess disabled because dimension is None");
|
2021-11-03 18:17:07 +00:00
|
|
|
(None, "")
|
|
|
|
}
|
|
|
|
},
|
|
|
|
//WebGL1 without sRGB support disable postprocess and use fallback shader
|
|
|
|
(ShaderVersion::Es100, false) => (None, ""),
|
|
|
|
//OpenGL 2.1 or above always support sRGB so add sRGB support marker
|
|
|
|
_ => (None, "#define SRGB_SUPPORTED"),
|
|
|
|
};
|
2021-10-20 10:29:35 +00:00
|
|
|
|
2021-10-18 21:13:32 +00:00
|
|
|
unsafe {
|
2021-11-03 18:17:07 +00:00
|
|
|
let vert = compile_shader(
|
|
|
|
gl,
|
|
|
|
glow::VERTEX_SHADER,
|
|
|
|
&format!(
|
2021-11-13 11:32:01 +00:00
|
|
|
"{}\n{}\n{}\n{}",
|
2021-11-03 18:17:07 +00:00
|
|
|
header,
|
2021-11-13 11:32:01 +00:00
|
|
|
shader_prefix,
|
2021-11-03 18:17:07 +00:00
|
|
|
shader_version.is_new_shader_interface(),
|
|
|
|
VERT_SRC
|
|
|
|
),
|
|
|
|
)?;
|
|
|
|
let frag = compile_shader(
|
|
|
|
gl,
|
|
|
|
glow::FRAGMENT_SHADER,
|
|
|
|
&format!(
|
2021-11-13 11:32:01 +00:00
|
|
|
"{}\n{}\n{}\n{}\n{}",
|
2021-11-03 18:17:07 +00:00
|
|
|
header,
|
2021-11-13 11:32:01 +00:00
|
|
|
shader_prefix,
|
2021-11-03 18:17:07 +00:00
|
|
|
srgb_support_define,
|
|
|
|
shader_version.is_new_shader_interface(),
|
|
|
|
FRAG_SRC
|
|
|
|
),
|
|
|
|
)?;
|
|
|
|
let program = link_program(gl, [vert, frag].iter())?;
|
2021-10-20 10:29:35 +00:00
|
|
|
gl.detach_shader(program, vert);
|
|
|
|
gl.detach_shader(program, frag);
|
|
|
|
gl.delete_shader(vert);
|
|
|
|
gl.delete_shader(frag);
|
2021-10-18 21:13:32 +00:00
|
|
|
let u_screen_size = gl.get_uniform_location(program, "u_screen_size").unwrap();
|
|
|
|
let u_sampler = gl.get_uniform_location(program, "u_sampler").unwrap();
|
2021-11-03 18:17:07 +00:00
|
|
|
let vertex_buffer = gl.create_buffer()?;
|
|
|
|
let element_array_buffer = gl.create_buffer()?;
|
2021-10-18 21:13:32 +00:00
|
|
|
gl.bind_buffer(glow::ARRAY_BUFFER, Some(vertex_buffer));
|
|
|
|
let a_pos_loc = gl.get_attrib_location(program, "a_pos").unwrap();
|
|
|
|
let a_tc_loc = gl.get_attrib_location(program, "a_tc").unwrap();
|
|
|
|
let a_srgba_loc = gl.get_attrib_location(program, "a_srgba").unwrap();
|
2021-11-13 11:32:01 +00:00
|
|
|
let mut vertex_array = if support_vao {
|
2021-11-03 18:17:07 +00:00
|
|
|
crate::misc_util::VAO::native(gl)
|
2021-11-13 11:32:01 +00:00
|
|
|
} else {
|
|
|
|
crate::misc_util::VAO::emulated()
|
2021-11-03 18:17:07 +00:00
|
|
|
};
|
|
|
|
vertex_array.bind_vertex_array(gl);
|
|
|
|
vertex_array.bind_buffer(gl, &vertex_buffer);
|
|
|
|
let stride = std::mem::size_of::<Vertex>() as i32;
|
|
|
|
let position_buffer_info = vao_emulate::BufferInfo {
|
|
|
|
location: a_pos_loc,
|
|
|
|
vector_size: 2,
|
|
|
|
data_type: glow::FLOAT,
|
|
|
|
normalized: false,
|
|
|
|
stride,
|
|
|
|
offset: offset_of!(Vertex, pos) as i32,
|
|
|
|
};
|
|
|
|
let tex_coord_buffer_info = vao_emulate::BufferInfo {
|
|
|
|
location: a_tc_loc,
|
|
|
|
vector_size: 2,
|
|
|
|
data_type: glow::FLOAT,
|
|
|
|
normalized: false,
|
|
|
|
stride,
|
|
|
|
offset: offset_of!(Vertex, uv) as i32,
|
|
|
|
};
|
|
|
|
let color_buffer_info = vao_emulate::BufferInfo {
|
|
|
|
location: a_srgba_loc,
|
|
|
|
vector_size: 4,
|
|
|
|
data_type: glow::UNSIGNED_BYTE,
|
|
|
|
normalized: false,
|
|
|
|
stride,
|
|
|
|
offset: offset_of!(Vertex, color) as i32,
|
|
|
|
};
|
|
|
|
vertex_array.add_new_attribute(gl, position_buffer_info);
|
|
|
|
vertex_array.add_new_attribute(gl, tex_coord_buffer_info);
|
|
|
|
vertex_array.add_new_attribute(gl, color_buffer_info);
|
2022-01-09 22:04:00 +00:00
|
|
|
check_for_gl_error(gl, "after Painter::new");
|
2021-10-18 21:13:32 +00:00
|
|
|
|
2021-11-03 18:17:07 +00:00
|
|
|
Ok(Painter {
|
2021-10-18 21:13:32 +00:00
|
|
|
program,
|
|
|
|
u_screen_size,
|
|
|
|
u_sampler,
|
|
|
|
egui_texture: None,
|
|
|
|
egui_texture_version: None,
|
2021-11-03 18:17:07 +00:00
|
|
|
is_webgl_1,
|
|
|
|
is_embedded: matches!(shader_version, ShaderVersion::Es100 | ShaderVersion::Es300),
|
2021-10-18 21:13:32 +00:00
|
|
|
vertex_array,
|
2021-11-03 18:17:07 +00:00
|
|
|
srgb_support,
|
2022-01-06 16:09:53 +00:00
|
|
|
texture_filter: Default::default(),
|
2021-11-03 18:17:07 +00:00
|
|
|
post_process,
|
2021-10-18 21:13:32 +00:00
|
|
|
vertex_buffer,
|
|
|
|
element_array_buffer,
|
2021-12-26 20:21:28 +00:00
|
|
|
user_textures: Default::default(),
|
2021-12-28 12:22:01 +00:00
|
|
|
#[cfg(feature = "epi")]
|
2021-12-26 20:21:28 +00:00
|
|
|
next_native_tex_id: 1 << 32,
|
|
|
|
textures_to_destroy: Vec::new(),
|
2021-10-18 21:13:32 +00:00
|
|
|
destroyed: false,
|
2021-11-03 18:17:07 +00:00
|
|
|
})
|
2021-10-18 21:13:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-28 20:02:23 +00:00
|
|
|
pub fn upload_egui_texture(&mut self, gl: &glow::Context, font_image: &egui::FontImage) {
|
2021-10-18 21:13:32 +00:00
|
|
|
self.assert_not_destroyed();
|
|
|
|
|
2021-12-28 20:02:23 +00:00
|
|
|
if self.egui_texture_version == Some(font_image.version) {
|
2021-10-18 21:13:32 +00:00
|
|
|
return; // No change
|
|
|
|
}
|
2021-11-03 18:17:07 +00:00
|
|
|
let gamma = if self.is_embedded && self.post_process.is_none() {
|
|
|
|
1.0 / 2.2
|
|
|
|
} else {
|
|
|
|
1.0
|
|
|
|
};
|
2021-12-28 20:02:23 +00:00
|
|
|
let pixels: Vec<u8> = font_image
|
2021-11-03 18:17:07 +00:00
|
|
|
.srgba_pixels(gamma)
|
|
|
|
.flat_map(|a| Vec::from(a.to_array()))
|
2021-10-18 21:13:32 +00:00
|
|
|
.collect();
|
|
|
|
|
|
|
|
if let Some(old_tex) = std::mem::replace(
|
|
|
|
&mut self.egui_texture,
|
2022-01-09 22:04:00 +00:00
|
|
|
Some(srgb_texture2d(
|
2021-11-03 18:17:07 +00:00
|
|
|
gl,
|
|
|
|
self.is_webgl_1,
|
|
|
|
self.srgb_support,
|
2022-01-06 16:09:53 +00:00
|
|
|
self.texture_filter,
|
2021-11-03 18:17:07 +00:00
|
|
|
&pixels,
|
2021-12-28 20:02:23 +00:00
|
|
|
font_image.width,
|
|
|
|
font_image.height,
|
2021-11-03 18:17:07 +00:00
|
|
|
)),
|
2021-10-18 21:13:32 +00:00
|
|
|
) {
|
|
|
|
unsafe {
|
|
|
|
gl.delete_texture(old_tex);
|
|
|
|
}
|
|
|
|
}
|
2021-12-28 20:02:23 +00:00
|
|
|
self.egui_texture_version = Some(font_image.version);
|
2021-10-18 21:13:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn prepare_painting(
|
|
|
|
&mut self,
|
2021-11-03 18:17:07 +00:00
|
|
|
[width_in_pixels, height_in_pixels]: [u32; 2],
|
2021-10-18 21:13:32 +00:00
|
|
|
gl: &glow::Context,
|
|
|
|
pixels_per_point: f32,
|
|
|
|
) -> (u32, u32) {
|
|
|
|
gl.enable(glow::SCISSOR_TEST);
|
2021-10-20 10:29:35 +00:00
|
|
|
// egui outputs mesh in both winding orders
|
2021-10-18 21:13:32 +00:00
|
|
|
gl.disable(glow::CULL_FACE);
|
|
|
|
|
|
|
|
gl.enable(glow::BLEND);
|
|
|
|
gl.blend_equation(glow::FUNC_ADD);
|
|
|
|
gl.blend_func_separate(
|
|
|
|
// egui outputs colors with premultiplied alpha:
|
|
|
|
glow::ONE,
|
|
|
|
glow::ONE_MINUS_SRC_ALPHA,
|
|
|
|
// Less important, but this is technically the correct alpha blend function
|
|
|
|
// when you want to make use of the framebuffer alpha (for screenshots, compositing, etc).
|
|
|
|
glow::ONE_MINUS_DST_ALPHA,
|
|
|
|
glow::ONE,
|
|
|
|
);
|
|
|
|
|
|
|
|
let width_in_points = width_in_pixels as f32 / pixels_per_point;
|
|
|
|
let height_in_points = height_in_pixels as f32 / pixels_per_point;
|
|
|
|
|
|
|
|
gl.viewport(0, 0, width_in_pixels as i32, height_in_pixels as i32);
|
|
|
|
gl.use_program(Some(self.program));
|
|
|
|
|
|
|
|
gl.uniform_2_f32(Some(&self.u_screen_size), width_in_points, height_in_points);
|
|
|
|
gl.uniform_1_i32(Some(&self.u_sampler), 0);
|
|
|
|
gl.active_texture(glow::TEXTURE0);
|
2021-11-03 18:17:07 +00:00
|
|
|
self.vertex_array.bind_vertex_array(gl);
|
2021-10-18 21:13:32 +00:00
|
|
|
|
|
|
|
gl.bind_buffer(glow::ELEMENT_ARRAY_BUFFER, Some(self.element_array_buffer));
|
|
|
|
|
|
|
|
(width_in_pixels, height_in_pixels)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Main entry-point for painting a frame.
|
|
|
|
/// You should call `target.clear_color(..)` before
|
|
|
|
/// and `target.finish()` after this.
|
|
|
|
///
|
|
|
|
/// The following OpenGL features will be set:
|
|
|
|
/// - Scissor test will be enabled
|
|
|
|
/// - Cull face will be disabled
|
|
|
|
/// - Blend will be enabled
|
|
|
|
///
|
|
|
|
/// The scissor area and blend parameters will be changed.
|
|
|
|
///
|
2021-11-03 18:17:07 +00:00
|
|
|
/// As well as this, the following objects will be unset:
|
2021-10-18 21:13:32 +00:00
|
|
|
/// - Vertex Buffer
|
|
|
|
/// - Element Buffer
|
|
|
|
/// - Texture (and active texture will be set to 0)
|
|
|
|
/// - Program
|
|
|
|
///
|
|
|
|
/// Please be mindful of these effects when integrating into your program, and also be mindful
|
|
|
|
/// of the effects your program might have on this code. Look at the source if in doubt.
|
|
|
|
pub fn paint_meshes(
|
|
|
|
&mut self,
|
|
|
|
gl: &glow::Context,
|
2021-12-26 20:21:28 +00:00
|
|
|
inner_size: [u32; 2],
|
2021-10-18 21:13:32 +00:00
|
|
|
pixels_per_point: f32,
|
|
|
|
clipped_meshes: Vec<egui::ClippedMesh>,
|
|
|
|
) {
|
|
|
|
self.assert_not_destroyed();
|
|
|
|
|
2021-11-03 18:17:07 +00:00
|
|
|
if let Some(ref mut post_process) = self.post_process {
|
|
|
|
unsafe {
|
|
|
|
post_process.begin(gl, inner_size[0] as i32, inner_size[1] as i32);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let size_in_pixels = unsafe { self.prepare_painting(inner_size, gl, pixels_per_point) };
|
2021-10-18 21:13:32 +00:00
|
|
|
for egui::ClippedMesh(clip_rect, mesh) in clipped_meshes {
|
2021-10-20 14:43:40 +00:00
|
|
|
self.paint_mesh(gl, size_in_pixels, pixels_per_point, clip_rect, &mesh);
|
2021-10-18 21:13:32 +00:00
|
|
|
}
|
2021-11-03 18:17:07 +00:00
|
|
|
unsafe {
|
|
|
|
self.vertex_array.unbind_vertex_array(gl);
|
|
|
|
gl.bind_buffer(glow::ELEMENT_ARRAY_BUFFER, None);
|
2021-10-18 21:13:32 +00:00
|
|
|
|
2021-11-03 18:17:07 +00:00
|
|
|
if let Some(ref post_process) = self.post_process {
|
|
|
|
post_process.end(gl);
|
|
|
|
}
|
2021-11-27 10:44:23 +00:00
|
|
|
|
|
|
|
gl.disable(glow::SCISSOR_TEST);
|
|
|
|
|
2022-01-09 22:04:00 +00:00
|
|
|
check_for_gl_error(gl, "painting");
|
2021-11-03 18:17:07 +00:00
|
|
|
}
|
2021-10-18 21:13:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline(never)] // Easier profiling
|
|
|
|
fn paint_mesh(
|
|
|
|
&mut self,
|
|
|
|
gl: &glow::Context,
|
|
|
|
size_in_pixels: (u32, u32),
|
|
|
|
pixels_per_point: f32,
|
|
|
|
clip_rect: Rect,
|
|
|
|
mesh: &Mesh,
|
|
|
|
) {
|
|
|
|
debug_assert!(mesh.is_valid());
|
|
|
|
if let Some(texture) = self.get_texture(mesh.texture_id) {
|
|
|
|
unsafe {
|
2021-11-03 18:17:07 +00:00
|
|
|
gl.bind_buffer(glow::ARRAY_BUFFER, Some(self.vertex_buffer));
|
2021-10-18 21:13:32 +00:00
|
|
|
gl.buffer_data_u8_slice(
|
|
|
|
glow::ARRAY_BUFFER,
|
|
|
|
as_u8_slice(mesh.vertices.as_slice()),
|
|
|
|
glow::STREAM_DRAW,
|
|
|
|
);
|
|
|
|
|
2021-11-03 18:17:07 +00:00
|
|
|
gl.bind_buffer(glow::ELEMENT_ARRAY_BUFFER, Some(self.element_array_buffer));
|
2021-10-18 21:13:32 +00:00
|
|
|
gl.buffer_data_u8_slice(
|
|
|
|
glow::ELEMENT_ARRAY_BUFFER,
|
|
|
|
as_u8_slice(mesh.indices.as_slice()),
|
|
|
|
glow::STREAM_DRAW,
|
|
|
|
);
|
|
|
|
|
|
|
|
gl.bind_texture(glow::TEXTURE_2D, Some(texture));
|
|
|
|
}
|
|
|
|
// Transform clip rect to physical pixels:
|
|
|
|
let clip_min_x = pixels_per_point * clip_rect.min.x;
|
|
|
|
let clip_min_y = pixels_per_point * clip_rect.min.y;
|
|
|
|
let clip_max_x = pixels_per_point * clip_rect.max.x;
|
|
|
|
let clip_max_y = pixels_per_point * clip_rect.max.y;
|
|
|
|
|
|
|
|
// Make sure clip rect can fit within a `u32`:
|
|
|
|
let clip_min_x = clip_min_x.clamp(0.0, size_in_pixels.0 as f32);
|
|
|
|
let clip_min_y = clip_min_y.clamp(0.0, size_in_pixels.1 as f32);
|
|
|
|
let clip_max_x = clip_max_x.clamp(clip_min_x, size_in_pixels.0 as f32);
|
|
|
|
let clip_max_y = clip_max_y.clamp(clip_min_y, size_in_pixels.1 as f32);
|
|
|
|
|
|
|
|
let clip_min_x = clip_min_x.round() as i32;
|
|
|
|
let clip_min_y = clip_min_y.round() as i32;
|
|
|
|
let clip_max_x = clip_max_x.round() as i32;
|
|
|
|
let clip_max_y = clip_max_y.round() as i32;
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
gl.scissor(
|
|
|
|
clip_min_x,
|
|
|
|
size_in_pixels.1 as i32 - clip_max_y,
|
|
|
|
clip_max_x - clip_min_x,
|
|
|
|
clip_max_y - clip_min_y,
|
|
|
|
);
|
|
|
|
gl.draw_elements(
|
|
|
|
glow::TRIANGLES,
|
|
|
|
mesh.indices.len() as i32,
|
|
|
|
glow::UNSIGNED_INT,
|
|
|
|
0,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-06 16:09:53 +00:00
|
|
|
// Set the filter to be used for any subsequent textures loaded via
|
|
|
|
// [`Self::set_texture`].
|
|
|
|
pub fn set_texture_filter(&mut self, texture_filter: TextureFilter) {
|
|
|
|
self.texture_filter = texture_filter;
|
|
|
|
}
|
|
|
|
|
2021-10-18 21:13:32 +00:00
|
|
|
// ------------------------------------------------------------------------
|
|
|
|
|
2021-12-28 12:22:01 +00:00
|
|
|
#[cfg(feature = "epi")]
|
2021-12-26 20:21:28 +00:00
|
|
|
pub fn set_texture(&mut self, gl: &glow::Context, tex_id: u64, image: &epi::Image) {
|
2021-10-18 21:13:32 +00:00
|
|
|
self.assert_not_destroyed();
|
|
|
|
|
2021-12-26 20:21:28 +00:00
|
|
|
assert_eq!(
|
|
|
|
image.size[0] * image.size[1],
|
|
|
|
image.pixels.len(),
|
|
|
|
"Mismatch between texture size and texel count"
|
|
|
|
);
|
2021-10-18 21:13:32 +00:00
|
|
|
|
2021-12-26 20:21:28 +00:00
|
|
|
// TODO: optimize
|
|
|
|
let pixels: Vec<u8> = image
|
|
|
|
.pixels
|
|
|
|
.iter()
|
|
|
|
.flat_map(|srgba| Vec::from(srgba.to_array()))
|
|
|
|
.collect();
|
2021-10-18 21:13:32 +00:00
|
|
|
|
2022-01-09 22:04:00 +00:00
|
|
|
let gl_texture = srgb_texture2d(
|
2021-12-26 20:21:28 +00:00
|
|
|
gl,
|
|
|
|
self.is_webgl_1,
|
|
|
|
self.srgb_support,
|
2022-01-06 16:09:53 +00:00
|
|
|
self.texture_filter,
|
2021-12-26 20:21:28 +00:00
|
|
|
&pixels,
|
|
|
|
image.size[0],
|
|
|
|
image.size[1],
|
2021-10-18 21:13:32 +00:00
|
|
|
);
|
|
|
|
|
2021-12-26 20:21:28 +00:00
|
|
|
if let Some(old_tex) = self.user_textures.insert(tex_id, gl_texture) {
|
|
|
|
self.textures_to_destroy.push(old_tex);
|
2021-10-18 21:13:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-26 20:21:28 +00:00
|
|
|
pub fn free_texture(&mut self, tex_id: u64) {
|
|
|
|
self.user_textures.remove(&tex_id);
|
2021-10-18 21:13:32 +00:00
|
|
|
}
|
|
|
|
|
2021-12-26 20:21:28 +00:00
|
|
|
fn get_texture(&self, texture_id: egui::TextureId) -> Option<glow::Texture> {
|
2021-10-18 21:13:32 +00:00
|
|
|
self.assert_not_destroyed();
|
|
|
|
|
|
|
|
match texture_id {
|
|
|
|
egui::TextureId::Egui => self.egui_texture,
|
2021-12-26 20:21:28 +00:00
|
|
|
egui::TextureId::User(id) => self.user_textures.get(&id).copied(),
|
2021-10-18 21:13:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn destroy_gl(&self, gl: &glow::Context) {
|
|
|
|
gl.delete_program(self.program);
|
|
|
|
if let Some(tex) = self.egui_texture {
|
|
|
|
gl.delete_texture(tex);
|
|
|
|
}
|
2021-12-26 20:21:28 +00:00
|
|
|
for tex in self.user_textures.values() {
|
|
|
|
gl.delete_texture(*tex);
|
2021-10-18 21:13:32 +00:00
|
|
|
}
|
|
|
|
gl.delete_buffer(self.vertex_buffer);
|
|
|
|
gl.delete_buffer(self.element_array_buffer);
|
2021-12-26 20:21:28 +00:00
|
|
|
for t in &self.textures_to_destroy {
|
2021-10-18 21:13:32 +00:00
|
|
|
gl.delete_texture(*t);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// This function must be called before Painter is dropped, as Painter has some OpenGL objects
|
|
|
|
/// that should be deleted.
|
|
|
|
|
2021-11-03 18:17:07 +00:00
|
|
|
pub fn destroy(&mut self, gl: &glow::Context) {
|
2021-12-28 13:24:59 +00:00
|
|
|
if !self.destroyed {
|
|
|
|
unsafe {
|
|
|
|
self.destroy_gl(gl);
|
|
|
|
if let Some(ref post_process) = self.post_process {
|
|
|
|
post_process.destroy(gl);
|
|
|
|
}
|
2021-11-03 18:17:07 +00:00
|
|
|
}
|
2021-12-28 13:24:59 +00:00
|
|
|
self.destroyed = true;
|
2021-10-18 21:13:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn assert_not_destroyed(&self) {
|
2021-12-28 13:24:59 +00:00
|
|
|
assert!(!self.destroyed, "the egui glow has already been destroyed!");
|
2021-10-18 21:13:32 +00:00
|
|
|
}
|
|
|
|
}
|
2021-11-03 18:35:20 +00:00
|
|
|
|
2021-11-03 18:17:07 +00:00
|
|
|
pub fn clear(gl: &glow::Context, dimension: [u32; 2], clear_color: egui::Rgba) {
|
|
|
|
unsafe {
|
|
|
|
gl.disable(glow::SCISSOR_TEST);
|
|
|
|
|
|
|
|
gl.viewport(0, 0, dimension[0] as i32, dimension[1] as i32);
|
2021-10-18 21:13:32 +00:00
|
|
|
|
2021-11-03 18:17:07 +00:00
|
|
|
let clear_color: Color32 = clear_color.into();
|
|
|
|
gl.clear_color(
|
|
|
|
clear_color[0] as f32 / 255.0,
|
|
|
|
clear_color[1] as f32 / 255.0,
|
|
|
|
clear_color[2] as f32 / 255.0,
|
|
|
|
clear_color[3] as f32 / 255.0,
|
|
|
|
);
|
|
|
|
gl.clear(glow::COLOR_BUFFER_BIT);
|
|
|
|
}
|
|
|
|
}
|
2021-11-03 18:35:20 +00:00
|
|
|
|
2021-10-18 21:13:32 +00:00
|
|
|
impl Drop for Painter {
|
|
|
|
fn drop(&mut self) {
|
2021-12-28 13:24:59 +00:00
|
|
|
if !self.destroyed {
|
|
|
|
eprintln!(
|
|
|
|
"You forgot to call destroy() on the egui glow painter. Resources will leak!"
|
|
|
|
);
|
|
|
|
}
|
2021-10-18 21:13:32 +00:00
|
|
|
}
|
|
|
|
}
|
2021-11-03 18:17:07 +00:00
|
|
|
|
2021-11-03 18:35:20 +00:00
|
|
|
#[cfg(feature = "epi")]
|
2021-11-03 18:17:07 +00:00
|
|
|
impl epi::NativeTexture for Painter {
|
|
|
|
type Texture = glow::Texture;
|
|
|
|
|
2021-11-03 18:35:20 +00:00
|
|
|
fn register_native_texture(&mut self, native: Self::Texture) -> egui::TextureId {
|
2021-12-26 20:21:28 +00:00
|
|
|
self.assert_not_destroyed();
|
|
|
|
|
|
|
|
let id = self.next_native_tex_id;
|
|
|
|
self.next_native_tex_id += 1;
|
|
|
|
|
|
|
|
self.user_textures.insert(id, native);
|
|
|
|
|
|
|
|
egui::TextureId::User(id as u64)
|
2021-11-03 18:17:07 +00:00
|
|
|
}
|
|
|
|
|
2021-11-03 18:35:20 +00:00
|
|
|
fn replace_native_texture(&mut self, id: egui::TextureId, replacing: Self::Texture) {
|
2021-11-03 18:17:07 +00:00
|
|
|
if let egui::TextureId::User(id) = id {
|
2021-12-26 20:21:28 +00:00
|
|
|
if let Some(old_tex) = self.user_textures.insert(id, replacing) {
|
|
|
|
self.textures_to_destroy.push(old_tex);
|
2021-11-03 18:17:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|