Skip to content

Commit 6e3c825

Browse files
committed
LEXER: refactor & add simple detect_rectangle() test
1 parent c0708a2 commit 6e3c825

File tree

2 files changed

+52
-48
lines changed

2 files changed

+52
-48
lines changed

SPEC.md

-2
Original file line numberDiff line numberDiff line change
@@ -32,8 +32,6 @@ the background colour
3232
the background colour of the image, which is ignored (both in source and key files), is defined by the most common colour in the key file as a whole (includes what would usually be ignored colours such as grid colour)
3333

3434
a few quirks of key files currently:
35-
- the grid colour is found from the very first pixel (top left corner) of the image
36-
- this grid colour is ignore in the key file
3735
- non rectangular objects are tokenized from a rectangular tile (like a bounding box in video games)
3836
- if the amount of pixels in the tokens tile matches the amount of pixels in the keys tile, then we deem it a match
3937
- a keys pixels can be arranged in any way withing the bounding box

lexer/src/lib.rs

+52-46
Original file line numberDiff line numberDiff line change
@@ -71,6 +71,31 @@ impl Tile {
7171
amount
7272
}
7373

74+
// detects solid rectangles for scopes
75+
// returns the tile that encampasses the rectangle
76+
fn detect_rectangle(begin: (usize, usize), image: &image::DynamicImage) -> Self {
77+
let pixels: Vec<Rgb<u8>> = image.to_rgb8().pixels().copied().collect();
78+
let pixels: Vec<Vec<Rgb<u8>>> = pixels.chunks_exact(image.width() as usize).map(|chunk| chunk.to_vec()).collect();
79+
let background = pixels[begin.1][begin.0];
80+
81+
Self {
82+
x: begin.0,
83+
y: begin.1,
84+
85+
width: pixels[begin.1][begin.0..]
86+
.iter()
87+
.position(|p| *p != background)
88+
.unwrap_or(image.width() as usize) as u32,
89+
90+
height: pixels
91+
.iter()
92+
.map(|row| row[begin.0])
93+
.collect::<Vec<Rgb<u8>>>()[begin.1..]
94+
.iter()
95+
.position(|&p| p != background)
96+
.unwrap_or(image.height() as usize) as u32
97+
}
98+
}
7499

75100
// will save a pixels in a tile as an image
76101
#[allow(dead_code)] // debug function
@@ -228,8 +253,6 @@ impl Key {
228253

229254
// encodes Key into a log file
230255
fn write_log<P: AsRef<Path>>(&self, checksum: &String, path: P) -> std::io::Result<()> {
231-
// TODO: check if file exists
232-
// fs::write(path, checksum)?;
233256
fs::write(&path, "")?;
234257
let mut log = fs::OpenOptions::new()
235258
.append(true)
@@ -248,13 +271,11 @@ impl Key {
248271
Ok(())
249272
}
250273

251-
// TODO: propagates some errors but panics others
252-
// TODO: unwrap() is not fine even though we hardcode it the data it could be corrupted
253274
// decodes the log file and returns the checksum and the Key
254275
fn read_log<P: AsRef<Path>>(&self, path: P) -> Option<(String, Key)> {
255276
let log = fs::read_to_string(&path).ok()?.parse::<String>().ok()?;
256277
let mut values = log.lines();
257-
let checksum = values.next().unwrap().to_owned();
278+
let checksum = values.next()?.to_owned();
258279

259280
Some((checksum,
260281
Key {
@@ -455,7 +476,7 @@ impl Key {
455476
.iter()
456477
.position(|&p| p != self.background && p != self.grid)
457478
.unwrap_or(64), // TODO: dont hardcode this
458-
y
479+
y
459480
)})
460481
.min().unwrap();
461482

@@ -475,10 +496,10 @@ impl Key {
475496
token,
476497
colour: key[0][0],
477498

499+
// fields values are from leftmost
478500
width_left: (first_pixel.0 as i16 - leftmost_pixel.0 as i16).abs() as u8,
479501
width_right: (width - (first_pixel.0 as i16 - leftmost_pixel.0 as i16)).abs() as u8,
480502

481-
// TODO: this ignores hollow in height which causes wrong height if theres gaps in the middle (y wise) of keys
482503
height_up: (leftmost_pixel.1 as i16 - first_pixel.1 as i16).abs() as u8,
483504
height_down: key.len() as u8 - (leftmost_pixel.1 as i16 - first_pixel.1 as i16).abs() as u8,
484505

@@ -489,11 +510,13 @@ impl Key {
489510
// read each 64x64 "tile" and apply the colour inside to the key structure
490511
fn read_keys(&mut self, image: &image::DynamicImage) {
491512
self.identify_background(image);
492-
493513
let tiles = self.image_to_tiles(image);
494514

495-
// TODO: find better way of finding key grid colour like detect rectangles or something
496-
self.grid = tiles[0][0][0];
515+
let grid = Tile::detect_rectangle((0, 0), image);
516+
if grid.width == image.width() &&
517+
grid.height == image.height() {
518+
self.grid = tiles[0][0][0];
519+
}
497520
assign_key!(&self, self.zero, &tiles, 0);
498521
assign_key!(&self, self.increment, &tiles, 1);
499522
assign_key!(&self, self.decrement, &tiles, 2);
@@ -519,32 +542,6 @@ impl Lexer {
519542

520543
// TODO: instead of passing around background we should keep a background field to change and read that whenever, like another self.key.background
521544

522-
// detects solid rectangles for scopes
523-
// returns the tile that encampasses the rectangle
524-
fn detect_rectangle(&self, begin: (usize, usize), image: &image::DynamicImage) -> Tile {
525-
let pixels: Vec<Rgb<u8>> = image.to_rgb8().pixels().copied().collect();
526-
let pixels: Vec<Vec<Rgb<u8>>> = pixels.chunks_exact(image.width() as usize).map(|chunk| chunk.to_vec()).collect();
527-
let background = pixels[begin.1][begin.0];
528-
529-
Tile {
530-
x: begin.0,
531-
y: begin.1,
532-
533-
width: pixels[begin.1][begin.0..]
534-
.iter()
535-
.position(|p| *p != background)
536-
.unwrap_or(image.width() as usize) as u32,
537-
538-
height: pixels
539-
.iter()
540-
.map(|row| row[begin.0])
541-
.collect::<Vec<Rgb<u8>>>()[begin.1..]
542-
.iter()
543-
.position(|&p| p != background)
544-
.unwrap_or(image.height() as usize) as u32
545-
}
546-
}
547-
548545
// returns the first keys token from a 1d index onwards
549546
// TODO: wont get the first, will get the heighest
550547
// TODO: optimise this with ignore map
@@ -625,8 +622,7 @@ impl Lexer {
625622
max_height
626623
}
627624

628-
// TODO: should multiple analysis functions change self.tokens
629-
// or should they each return Vec<Lexeme> to concantenate together in one place?
625+
// TODO: should multiple analysis functions change self.tokens or should they each return Vec<Lexeme> to concantenate together in one place?
630626
// TODO: panics when variables are referenced with rectangular symbols/names
631627
// TODO: dont duplicate code in analyse(), make a generic loop with a higher order function or something
632628
// tokenizes a scope
@@ -662,15 +658,10 @@ impl Lexer {
662658
frame.x = init_x;
663659
while frame.x < scope.tile.x + scope.tile.width as usize {
664660
'frame: for x in 0..frame.width as usize {
665-
if x + frame.x >= image.width() as usize {
666-
break;
667-
}
661+
bounds_check!(x + frame.x, image.width(), {break});
668662

669663
for y in 0..frame.height as usize {
670-
// TODO: better way to do this bounds checking
671-
if y + frame.y >= image.height() as usize {
672-
break;
673-
}
664+
bounds_check!(y + frame.y, image.height(), {break});
674665

675666
if pixels[y + frame.y][x + frame.x] == scope.colour {
676667
continue;
@@ -752,7 +743,7 @@ impl Lexer {
752743

753744
// if the pixel is unknown then it could be a scope
754745
if self.key.data_from_colour(pixels[y][x]).is_empty() {
755-
let scope = self.detect_rectangle((x, y), image);
746+
let scope = Tile::detect_rectangle((x, y), image);
756747
// rectangle is big enough to be a scope
757748
if scope.width > 64 && scope.height > 64 {
758749
self.analyse_scope(&Scope {
@@ -895,6 +886,7 @@ pub fn deserialize(key: &String, source: &String) -> Result<Vec<Lexeme>, image::
895886
}
896887

897888
// TODO: maybe use a special test key instead of official default key so we can test for weirder shapes
889+
// TODO: do more extensive tests and test multiple cases
898890
#[cfg(test)]
899891
mod tests {
900892
use super::*;
@@ -960,6 +952,20 @@ mod tests {
960952

961953
}
962954

955+
#[test]
956+
fn tile_detect_rectangle() {
957+
let img = ImageReader::open("../test/100x100.png").unwrap().decode().unwrap();
958+
959+
let test = Tile::detect_rectangle((0, 0), &img);
960+
let expected = Tile {
961+
x: 0, y: 0,
962+
width: img.width(),
963+
height: img.height()
964+
};
965+
966+
assert_eq!(test, expected);
967+
}
968+
963969
// Key tests
964970
struct KeySetup {
965971
img: image::DynamicImage,

0 commit comments

Comments
 (0)