mirror of
https://github.com/emilk/egui.git
synced 2026-06-26 22:53:14 -04:00
Add IdInfo and only enable this with debug_assertions
This commit is contained in:
@@ -306,13 +306,13 @@ impl ScrollArea {
|
||||
/// A source for the unique [`Id`], e.g. `.id_source("second_scroll_area")` or `.id_source(loop_index)`.
|
||||
#[inline]
|
||||
#[deprecated = "Renamed id_salt"]
|
||||
pub fn id_source(self, id_salt: impl crate::IdTrait) -> Self {
|
||||
pub fn id_source(self, id_salt: impl crate::AsId) -> Self {
|
||||
self.id_salt(id_salt)
|
||||
}
|
||||
|
||||
/// A source for the unique [`Id`], e.g. `.id_salt("second_scroll_area")` or `.id_salt(loop_index)`.
|
||||
#[inline]
|
||||
pub fn id_salt(mut self, id_salt: impl crate::IdTrait) -> Self {
|
||||
pub fn id_salt(mut self, id_salt: impl crate::AsId) -> Self {
|
||||
self.id_salt = Some(Id::new(id_salt));
|
||||
self
|
||||
}
|
||||
|
||||
@@ -1,11 +1,6 @@
|
||||
// TODO(emilk): have separate types `PositionId` and `UniqueId`. ?
|
||||
|
||||
use ahash::HashMap;
|
||||
use epaint::mutex::{Mutex, RwLock};
|
||||
use std::any::TypeId;
|
||||
use std::hash::Hasher;
|
||||
use std::num::NonZeroU64;
|
||||
use std::sync::LazyLock;
|
||||
|
||||
/// egui tracks widgets frame-to-frame using [`Id`]s.
|
||||
///
|
||||
@@ -38,21 +33,10 @@ use std::sync::LazyLock;
|
||||
#[cfg_attr(feature = "serde", derive(serde::Deserialize, serde::Serialize))]
|
||||
pub struct Id(NonZeroU64);
|
||||
|
||||
enum IdSource {
|
||||
Id(Id),
|
||||
Other(String),
|
||||
}
|
||||
|
||||
static ID_MAP: LazyLock<RwLock<HashMap<Id, (IdSource, Option<Id>)>>> = LazyLock::new(|| {
|
||||
let mut map = HashMap::default();
|
||||
map.insert(Id::NULL, (IdSource::Other("Id::NULL".to_owned()), None));
|
||||
RwLock::new(map)
|
||||
});
|
||||
|
||||
impl nohash_hasher::IsEnabled for Id {}
|
||||
|
||||
pub trait IdTrait: std::hash::Hash + std::fmt::Debug {}
|
||||
impl<T: std::hash::Hash + std::fmt::Debug> IdTrait for T {}
|
||||
pub trait AsId: std::hash::Hash + std::fmt::Debug {}
|
||||
impl<T: std::hash::Hash + std::fmt::Debug> AsId for T {}
|
||||
|
||||
impl Id {
|
||||
/// A special [`Id`], in particular as a key to [`crate::Memory::data`]
|
||||
@@ -71,82 +55,26 @@ impl Id {
|
||||
}
|
||||
}
|
||||
|
||||
/// Checks if [`T`] is a [`Id`].
|
||||
///
|
||||
/// If it is, it returns `IdSource::Id`, otherwise it returns `IdSource::Other`.
|
||||
fn get_source<T: IdTrait>(t: T) -> IdSource {
|
||||
/// Ugly hack to try to determine if T is an Id or not.
|
||||
struct FakeHasher {
|
||||
val: Option<u64>,
|
||||
first: bool,
|
||||
}
|
||||
|
||||
impl Hasher for FakeHasher {
|
||||
fn finish(&self) -> u64 {
|
||||
unreachable!()
|
||||
}
|
||||
|
||||
fn write(&mut self, bytes: &[u8]) {
|
||||
self.first = false;
|
||||
}
|
||||
|
||||
fn write_u64(&mut self, i: u64) {
|
||||
if self.first {
|
||||
self.val = Some(i);
|
||||
self.first = false;
|
||||
} else {
|
||||
self.val = None;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
let mut hasher = FakeHasher {
|
||||
val: None,
|
||||
first: true,
|
||||
};
|
||||
|
||||
t.hash(&mut hasher);
|
||||
|
||||
let maybe_source_id = hasher.val.map(Id::from_hash);
|
||||
|
||||
// Ideally we would just implement IdTriat for Id with specialization, but that's not
|
||||
// a thing yet :( So we check if the hash is already in the map, if so, the source must be
|
||||
// an Id.
|
||||
if let Some(maybe_source_id) = maybe_source_id {
|
||||
if ID_MAP.read().contains_key(&maybe_source_id) {
|
||||
IdSource::Id(maybe_source_id)
|
||||
} else {
|
||||
IdSource::Other(format!("{:?}", t))
|
||||
}
|
||||
} else {
|
||||
IdSource::Other(format!("{:?}", t))
|
||||
}
|
||||
}
|
||||
|
||||
/// Generate a new [`Id`] by hashing some source (e.g. a string or integer).
|
||||
pub fn new<T: IdTrait>(source: T) -> Self {
|
||||
pub fn new<T: AsId>(source: T) -> Self {
|
||||
let id = Self::from_hash(ahash::RandomState::with_seeds(1, 2, 3, 4).hash_one(&source));
|
||||
|
||||
if !ID_MAP.read().contains_key(&id) {
|
||||
let source = Self::get_source(source);
|
||||
ID_MAP.write().insert(id, (source, None));
|
||||
}
|
||||
#[cfg(debug_assertions)]
|
||||
id_source::maybe_insert(id, source, None);
|
||||
|
||||
id
|
||||
}
|
||||
|
||||
/// Generate a new [`Id`] by hashing the parent [`Id`] and the given argument.
|
||||
pub fn with(self, child: impl std::hash::Hash + std::fmt::Debug) -> Self {
|
||||
pub fn with(self, child: impl AsId) -> Self {
|
||||
use std::hash::{BuildHasher, Hasher};
|
||||
let mut hasher = ahash::RandomState::with_seeds(1, 2, 3, 4).build_hasher();
|
||||
hasher.write_u64(self.0.get());
|
||||
(&child).hash(&mut hasher);
|
||||
let id = Self::from_hash(hasher.finish());
|
||||
|
||||
if !ID_MAP.read().contains_key(&id) {
|
||||
let source = Self::get_source(child);
|
||||
ID_MAP.write().insert(id, (source, Some(self)));
|
||||
}
|
||||
#[cfg(debug_assertions)]
|
||||
id_source::maybe_insert(id, &child, Some(self));
|
||||
|
||||
id
|
||||
}
|
||||
@@ -168,22 +96,152 @@ impl Id {
|
||||
pub(crate) fn accesskit_id(&self) -> accesskit::NodeId {
|
||||
self.value().into()
|
||||
}
|
||||
|
||||
// TODO: Nice debug ui
|
||||
// pub fn ui(self, ui: &mut crate::Ui) -> crate::Response {
|
||||
// ui.code(self.short_debug_format()).on_hover_ui(|ui| {
|
||||
// let data = self.info();
|
||||
// })
|
||||
// }
|
||||
}
|
||||
|
||||
#[cfg(debug_assertions)]
|
||||
mod id_source {
|
||||
use crate::{AsId, Id};
|
||||
use ahash::HashMap;
|
||||
use epaint::mutex::RwLock;
|
||||
use std::hash::Hasher;
|
||||
use std::sync::LazyLock;
|
||||
|
||||
#[derive(Clone)]
|
||||
pub struct IdInfo {
|
||||
/// What was this Id generated from?
|
||||
pub source: IdSource,
|
||||
/// If the Id was crated via [`Id::with`], what was the parent Id?
|
||||
pub parent: Option<Id>,
|
||||
}
|
||||
|
||||
#[derive(Clone)]
|
||||
pub enum IdSource {
|
||||
Id(Id),
|
||||
Other(String),
|
||||
}
|
||||
|
||||
static ID_MAP: LazyLock<RwLock<HashMap<Id, IdInfo>>> = LazyLock::new(|| {
|
||||
let mut map = HashMap::default();
|
||||
map.insert(
|
||||
Id::NULL,
|
||||
IdInfo {
|
||||
source: IdSource::Other("Id::NULL".to_owned()),
|
||||
parent: None,
|
||||
},
|
||||
);
|
||||
RwLock::new(map)
|
||||
});
|
||||
|
||||
/// Ugly hack to try to determine if T is an Id or not.
|
||||
#[derive(Default)]
|
||||
struct ExtractIdHasher {
|
||||
val: Option<u64>,
|
||||
not_id: bool,
|
||||
}
|
||||
|
||||
impl ExtractIdHasher {
|
||||
fn id(&self) -> Option<Id> {
|
||||
self.val.map(Id::from_hash)
|
||||
}
|
||||
}
|
||||
|
||||
impl Hasher for ExtractIdHasher {
|
||||
fn finish(&self) -> u64 {
|
||||
unreachable!()
|
||||
}
|
||||
|
||||
fn write(&mut self, _bytes: &[u8]) {
|
||||
self.not_id = true;
|
||||
self.val = None;
|
||||
}
|
||||
|
||||
fn write_u64(&mut self, i: u64) {
|
||||
if !self.not_id && !self.val.is_some() {
|
||||
self.val = Some(i);
|
||||
} else {
|
||||
self.not_id = true;
|
||||
self.val = None;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// Checks if [`T`] is a [`Id`].
|
||||
///
|
||||
/// If it is, it returns `IdSource::Id`, otherwise it returns `IdSource::Other`.
|
||||
fn get_source<T: AsId>(t: T) -> IdSource {
|
||||
let mut hasher = ExtractIdHasher::default();
|
||||
|
||||
t.hash(&mut hasher);
|
||||
|
||||
let maybe_source_id = hasher.id();
|
||||
|
||||
// Ideally we would just implement IdTriat for Id with specialization, but that's not
|
||||
// a thing yet :( So we check if the hash is already in the map, if so, the source must be
|
||||
// an Id.
|
||||
if let Some(maybe_source_id) = maybe_source_id {
|
||||
if ID_MAP.read().contains_key(&maybe_source_id) {
|
||||
IdSource::Id(maybe_source_id)
|
||||
} else {
|
||||
IdSource::Other(format!("{:?}", t))
|
||||
}
|
||||
} else {
|
||||
IdSource::Other(format!("{:?}", t))
|
||||
}
|
||||
}
|
||||
|
||||
pub(super) fn maybe_insert(id: Id, source: impl AsId, parent: Option<Id>) {
|
||||
if !ID_MAP.read().contains_key(&id) {
|
||||
let source1 = get_source(source);
|
||||
ID_MAP.write().insert(
|
||||
id,
|
||||
IdInfo {
|
||||
source: source1,
|
||||
parent,
|
||||
},
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
impl Id {
|
||||
pub fn info(&self) -> Option<IdInfo> {
|
||||
ID_MAP.read().get(self).cloned()
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_fake_hasher() {
|
||||
use std::hash::Hash;
|
||||
let mut hasher = ExtractIdHasher::default();
|
||||
|
||||
let id = Id::new("test");
|
||||
id.hash(&mut hasher);
|
||||
|
||||
assert_eq!(hasher.id(), Some(id));
|
||||
}
|
||||
}
|
||||
|
||||
impl std::fmt::Debug for Id {
|
||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||
write!(f, "{:04X}", self.value() as u16)?;
|
||||
let lock = ID_MAP.read();
|
||||
if let Some((source, parent)) = lock.get(self) {
|
||||
match source {
|
||||
IdSource::Id(source_id) => {
|
||||
|
||||
#[cfg(debug_assertions)]
|
||||
if let Some(info) = self.info() {
|
||||
match info.source {
|
||||
id_source::IdSource::Id(source_id) => {
|
||||
write!(f, "({:?})", source_id)?;
|
||||
}
|
||||
IdSource::Other(label) => {
|
||||
id_source::IdSource::Other(label) => {
|
||||
write!(f, " ({})", label)?;
|
||||
}
|
||||
}
|
||||
if let Some(parent) = parent {
|
||||
if let Some(parent) = info.parent {
|
||||
// Let's hope there are no cycles!
|
||||
write!(f, " <- {:?}", parent)?;
|
||||
}
|
||||
|
||||
@@ -492,7 +492,7 @@ pub use self::{
|
||||
drag_and_drop::DragAndDrop,
|
||||
epaint::text::TextWrapMode,
|
||||
grid::Grid,
|
||||
id::{Id, IdMap, IdTrait},
|
||||
id::{AsId, Id, IdMap},
|
||||
input_state::{InputState, MultiTouchInfo, PointerState},
|
||||
layers::{LayerId, Order},
|
||||
layout::*,
|
||||
|
||||
@@ -131,7 +131,7 @@ impl ViewportId {
|
||||
pub const ROOT: Self = Self(Id::NULL);
|
||||
|
||||
#[inline]
|
||||
pub fn from_hash_of(source: impl crate::IdTrait) -> Self {
|
||||
pub fn from_hash_of(source: impl crate::AsId) -> Self {
|
||||
Self(Id::new(source))
|
||||
}
|
||||
}
|
||||
|
||||
@@ -166,13 +166,13 @@ impl<'t> TextEdit<'t> {
|
||||
|
||||
/// A source for the unique [`Id`], e.g. `.id_source("second_text_edit_field")` or `.id_source(loop_index)`.
|
||||
#[inline]
|
||||
pub fn id_source(self, id_salt: impl crate::IdTrait) -> Self {
|
||||
pub fn id_source(self, id_salt: impl crate::AsId) -> Self {
|
||||
self.id_salt(id_salt)
|
||||
}
|
||||
|
||||
/// A source for the unique [`Id`], e.g. `.id_salt("second_text_edit_field")` or `.id_salt(loop_index)`.
|
||||
#[inline]
|
||||
pub fn id_salt(mut self, id_salt: impl crate::IdTrait) -> Self {
|
||||
pub fn id_salt(mut self, id_salt: impl crate::AsId) -> Self {
|
||||
self.id_salt = Some(Id::new(id_salt));
|
||||
self
|
||||
}
|
||||
|
||||
@@ -275,7 +275,7 @@ impl<'a> TableBuilder<'a> {
|
||||
/// This is required if you have multiple tables in the same [`Ui`].
|
||||
#[inline]
|
||||
#[deprecated = "Renamed id_salt"]
|
||||
pub fn id_source(self, id_salt: impl egui::IdTrait) -> Self {
|
||||
pub fn id_source(self, id_salt: impl egui::AsId) -> Self {
|
||||
self.id_salt(id_salt)
|
||||
}
|
||||
|
||||
@@ -283,7 +283,7 @@ impl<'a> TableBuilder<'a> {
|
||||
///
|
||||
/// This is required if you have multiple tables in the same [`Ui`].
|
||||
#[inline]
|
||||
pub fn id_salt(mut self, id_salt: impl egui::IdTrait) -> Self {
|
||||
pub fn id_salt(mut self, id_salt: impl egui::AsId) -> Self {
|
||||
self.id_salt = Id::new(id_salt);
|
||||
self
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user