1
Fork 0

difficulty color recognition

Signed-off-by: prescientmoon <git@moonythm.dev>
This commit is contained in:
prescientmoon 2024-07-31 17:51:03 +02:00
parent d7930cba5d
commit 6556e81433
Signed by: prescientmoon
SSH key fingerprint: SHA256:UUF9JT2s8Xfyv76b8ZuVL7XrmimH4o49p4b+iexbVH4
3 changed files with 190 additions and 29 deletions

View file

@ -37,6 +37,14 @@ impl Color {
self.3 = a; self.3 = a;
self self
} }
#[inline]
pub fn distance(self, other: Self) -> f32 {
let dr = self.0 as f32 - other.0 as f32;
let dg = self.1 as f32 - other.1 as f32;
let db = self.2 as f32 - other.2 as f32;
(dr * dr + dg * dg + db * db).sqrt()
}
} }
// }}} // }}}
// {{{ Rect // {{{ Rect

View file

@ -123,7 +123,8 @@ pub async fn magic(
.content(format!("Image {}: reading difficulty", i + 1)); .content(format!("Image {}: reading difficulty", i + 1));
handle.edit(ctx, edited).await?; handle.edit(ctx, edited).await?;
let difficulty = match cropper.read_difficulty(&ocr_image, kind) { // Do not use `ocr_image` because this reads the colors
let difficulty = match cropper.read_difficulty(&image, kind) {
// {{{ OCR error handling // {{{ OCR error handling
Err(err) => { Err(err) => {
error_with_image( error_with_image(
@ -140,6 +141,8 @@ pub async fn magic(
// }}} // }}}
Ok(d) => d, Ok(d) => d,
}; };
println!("{difficulty:?}");
// }}} // }}}
// {{{ Jacket & distribution // {{{ Jacket & distribution
let mut jacket_rect = None; let mut jacket_rect = None;

View file

@ -14,7 +14,7 @@ use poise::serenity_prelude::{
}; };
use tesseract::{PageSegMode, Tesseract}; use tesseract::{PageSegMode, Tesseract};
use crate::bitmap::Rect; use crate::bitmap::{Color, Rect};
use crate::chart::{Chart, Difficulty, Song, SongCache}; use crate::chart::{Chart, Difficulty, Song, SongCache};
use crate::context::{Error, UserContext}; use crate::context::{Error, UserContext};
use crate::image::rotate; use crate::image::rotate;
@ -785,49 +785,108 @@ impl RelativeRect {
self.dimensions, self.dimensions,
) )
} }
}
// }}}
// {{{ AbsolutePoint
#[derive(Debug, Clone, Copy)]
pub struct AbsolutePoint {
pub x: u32,
pub y: u32,
pub dimensions: ImageDimensions,
}
pub fn from_aspect_ratio( impl AbsolutePoint {
dimensions: ImageDimensions, #[inline]
datapoints: &[RelativeRect], pub fn new(x: u32, y: u32, dimensions: ImageDimensions) -> Self {
) -> Option<Self> { Self { x, y, dimensions }
let aspect_ratio = dimensions.aspect_ratio(); }
for i in 0..(datapoints.len() - 1) { #[inline]
let low = datapoints[i]; pub fn to_relative(&self) -> RelativePoint {
let high = datapoints[i + 1]; RelativePoint::new(
self.x as f32 / self.dimensions.width as f32,
self.y as f32 / self.dimensions.height as f32,
self.dimensions,
)
}
}
// }}}
// {{{ RelativePoint
#[derive(Debug, Clone, Copy)]
pub struct RelativePoint {
pub x: f32,
pub y: f32,
pub dimensions: ImageDimensions,
}
let low_ratio = low.dimensions.aspect_ratio(); impl RelativePoint {
let high_ratio = high.dimensions.aspect_ratio(); #[inline]
pub fn new(x: f32, y: f32, dimensions: ImageDimensions) -> Self {
Self { x, y, dimensions }
}
if (i == 0 || low_ratio <= aspect_ratio) #[inline]
&& (aspect_ratio <= high_ratio || i == datapoints.len() - 2) pub fn to_absolute(&self) -> AbsolutePoint {
{ AbsolutePoint::new(
let p = (aspect_ratio - low_ratio) / (high_ratio - low_ratio); (self.x * self.dimensions.width as f32) as u32,
return Some(Self::new( (self.y * self.dimensions.height as f32) as u32,
lerp(p, low.x, high.x), self.dimensions,
lerp(p, low.y, high.y), )
lerp(p, low.width, high.width),
lerp(p, low.height, high.height),
dimensions,
));
}
}
None
} }
} }
// }}} // }}}
// }}} // }}}
// {{{ Data points // {{{ Data points
// {{{ Trait // {{{ Trait
trait UIDataPoint { trait UIDataPoint: Sized + Copy {
fn aspect_ratio(&self) -> f32; fn aspect_ratio(&self) -> f32;
fn lerp(low: &Self, high: &Self, p: f32, dimensions: ImageDimensions) -> Self;
fn from_aspect_ratio(dimensions: ImageDimensions, datapoints: &[Self]) -> Option<Self> {
let aspect_ratio = dimensions.aspect_ratio();
for i in 0..(datapoints.len() - 1) {
let low = datapoints[i];
let high = datapoints[i + 1];
let low_ratio = low.aspect_ratio();
let high_ratio = high.aspect_ratio();
if (i == 0 || low_ratio <= aspect_ratio)
&& (aspect_ratio <= high_ratio || i == datapoints.len() - 2)
{
let p = (aspect_ratio - low_ratio) / (high_ratio - low_ratio);
return Some(Self::lerp(&low, &high, p, dimensions));
}
}
None
}
} }
impl UIDataPoint for RelativeRect { impl UIDataPoint for RelativeRect {
fn aspect_ratio(&self) -> f32 { fn aspect_ratio(&self) -> f32 {
self.dimensions.aspect_ratio() self.dimensions.aspect_ratio()
} }
fn lerp(low: &Self, high: &Self, p: f32, dimensions: ImageDimensions) -> Self {
Self::new(
lerp(p, low.x, high.x),
lerp(p, low.y, high.y),
lerp(p, low.width, high.width),
lerp(p, low.height, high.height),
dimensions,
)
}
}
impl UIDataPoint for RelativePoint {
fn aspect_ratio(&self) -> f32 {
self.dimensions.aspect_ratio()
}
fn lerp(low: &Self, high: &Self, p: f32, dimensions: ImageDimensions) -> Self {
Self::new(lerp(p, low.x, high.x), lerp(p, low.y, high.y), dimensions)
}
} }
// }}} // }}}
// {{{ Processing // {{{ Processing
@ -1011,6 +1070,65 @@ fn score_kind_rects() -> &'static [RelativeRect] {
}) })
} }
// }}} // }}}
// {{{ Difficulty pixel locations
fn pst_pixel() -> &'static [RelativePoint] {
static CELL: OnceLock<Vec<RelativePoint>> = OnceLock::new();
CELL.get_or_init(|| {
let mut points: Vec<RelativePoint> = vec![
AbsolutePoint::new(89, 153, ImageDimensions::new(2532, 1170)).to_relative(),
AbsolutePoint::new(12, 159, ImageDimensions::new(2160, 1620)).to_relative(),
];
process_datapoints(&mut points);
points
})
}
fn prs_pixel() -> &'static [RelativePoint] {
static CELL: OnceLock<Vec<RelativePoint>> = OnceLock::new();
CELL.get_or_init(|| {
let mut points: Vec<RelativePoint> = vec![
AbsolutePoint::new(269, 153, ImageDimensions::new(2532, 1170)).to_relative(),
AbsolutePoint::new(199, 159, ImageDimensions::new(2160, 1620)).to_relative(),
];
process_datapoints(&mut points);
points
})
}
fn ftr_pixel() -> &'static [RelativePoint] {
static CELL: OnceLock<Vec<RelativePoint>> = OnceLock::new();
CELL.get_or_init(|| {
let mut points: Vec<RelativePoint> = vec![
AbsolutePoint::new(452, 153, ImageDimensions::new(2532, 1170)).to_relative(),
AbsolutePoint::new(389, 159, ImageDimensions::new(2160, 1620)).to_relative(),
];
process_datapoints(&mut points);
points
})
}
fn byd_etr_pixel() -> &'static [RelativePoint] {
static CELL: OnceLock<Vec<RelativePoint>> = OnceLock::new();
CELL.get_or_init(|| {
let mut points: Vec<RelativePoint> = vec![
AbsolutePoint::new(638, 153, ImageDimensions::new(2532, 1170)).to_relative(),
AbsolutePoint::new(579, 159, ImageDimensions::new(2160, 1620)).to_relative(),
];
process_datapoints(&mut points);
points
})
}
fn difficulty_pixel(difficulty: Difficulty) -> &'static [RelativePoint] {
match difficulty {
Difficulty::PST => pst_pixel(),
Difficulty::PRS => prs_pixel(),
Difficulty::FTR => ftr_pixel(),
Difficulty::ETR => byd_etr_pixel(),
Difficulty::BYD => byd_etr_pixel(),
}
}
// }}}
// }}} // }}}
// {{{ Recognise chart // {{{ Recognise chart
fn strip_case_insensitive_suffix<'a>(string: &'a str, suffix: &str) -> Option<&'a str> { fn strip_case_insensitive_suffix<'a>(string: &'a str, suffix: &str) -> Option<&'a str> {
@ -1291,7 +1409,39 @@ impl ImageCropper {
kind: ScoreKind, kind: ScoreKind,
) -> Result<Difficulty, Error> { ) -> Result<Difficulty, Error> {
if kind == ScoreKind::SongSelect { if kind == ScoreKind::SongSelect {
return Ok(Difficulty::FTR); let colors = [
Color::BLACK,
Color::BLACK,
Color::from_rgb_int(0xCB74AB),
Color::from_rgb_int(0xC4B7D3),
Color::BLACK,
];
let dimensions = ImageDimensions::from_image(image);
let min = colors
.iter()
.zip(Difficulty::DIFFICULTIES)
.min_by_key(|(c, d)| {
let points = difficulty_pixel(*d);
let point = RelativePoint::from_aspect_ratio(dimensions, points)
.ok_or_else(|| "Could not find difficulty pixel in picture")
.unwrap_or(RelativePoint::new(0.0, 0.0, dimensions))
.to_absolute();
let image_color = image.get_pixel(point.x, point.y);
let image_color = Color(
image_color[0],
image_color[1],
image_color[2],
image_color[3],
);
let distance = c.distance(image_color);
(distance * 10000.0) as u32
});
return Ok(min.unwrap().1);
} }
self.crop_image_to_bytes( self.crop_image_to_bytes(