@@ -71,6 +71,31 @@ impl Tile {
71
71
amount
72
72
}
73
73
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
+ }
74
99
75
100
// will save a pixels in a tile as an image
76
101
#[ allow( dead_code) ] // debug function
@@ -228,8 +253,6 @@ impl Key {
228
253
229
254
// encodes Key into a log file
230
255
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)?;
233
256
fs:: write ( & path, "" ) ?;
234
257
let mut log = fs:: OpenOptions :: new ( )
235
258
. append ( true )
@@ -248,13 +271,11 @@ impl Key {
248
271
Ok ( ( ) )
249
272
}
250
273
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
253
274
// decodes the log file and returns the checksum and the Key
254
275
fn read_log < P : AsRef < Path > > ( & self , path : P ) -> Option < ( String , Key ) > {
255
276
let log = fs:: read_to_string ( & path) . ok ( ) ?. parse :: < String > ( ) . ok ( ) ?;
256
277
let mut values = log. lines ( ) ;
257
- let checksum = values. next ( ) . unwrap ( ) . to_owned ( ) ;
278
+ let checksum = values. next ( ) ? . to_owned ( ) ;
258
279
259
280
Some ( ( checksum,
260
281
Key {
@@ -455,7 +476,7 @@ impl Key {
455
476
. iter ( )
456
477
. position ( |& p| p != self . background && p != self . grid )
457
478
. unwrap_or ( 64 ) , // TODO: dont hardcode this
458
- y
479
+ y
459
480
) } )
460
481
. min ( ) . unwrap ( ) ;
461
482
@@ -475,10 +496,10 @@ impl Key {
475
496
token,
476
497
colour : key[ 0 ] [ 0 ] ,
477
498
499
+ // fields values are from leftmost
478
500
width_left : ( first_pixel. 0 as i16 - leftmost_pixel. 0 as i16 ) . abs ( ) as u8 ,
479
501
width_right : ( width - ( first_pixel. 0 as i16 - leftmost_pixel. 0 as i16 ) ) . abs ( ) as u8 ,
480
502
481
- // TODO: this ignores hollow in height which causes wrong height if theres gaps in the middle (y wise) of keys
482
503
height_up : ( leftmost_pixel. 1 as i16 - first_pixel. 1 as i16 ) . abs ( ) as u8 ,
483
504
height_down : key. len ( ) as u8 - ( leftmost_pixel. 1 as i16 - first_pixel. 1 as i16 ) . abs ( ) as u8 ,
484
505
@@ -489,11 +510,13 @@ impl Key {
489
510
// read each 64x64 "tile" and apply the colour inside to the key structure
490
511
fn read_keys ( & mut self , image : & image:: DynamicImage ) {
491
512
self . identify_background ( image) ;
492
-
493
513
let tiles = self . image_to_tiles ( image) ;
494
514
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
+ }
497
520
assign_key ! ( & self , self . zero, & tiles, 0 ) ;
498
521
assign_key ! ( & self , self . increment, & tiles, 1 ) ;
499
522
assign_key ! ( & self , self . decrement, & tiles, 2 ) ;
@@ -519,32 +542,6 @@ impl Lexer {
519
542
520
543
// TODO: instead of passing around background we should keep a background field to change and read that whenever, like another self.key.background
521
544
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
-
548
545
// returns the first keys token from a 1d index onwards
549
546
// TODO: wont get the first, will get the heighest
550
547
// TODO: optimise this with ignore map
@@ -625,8 +622,7 @@ impl Lexer {
625
622
max_height
626
623
}
627
624
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?
630
626
// TODO: panics when variables are referenced with rectangular symbols/names
631
627
// TODO: dont duplicate code in analyse(), make a generic loop with a higher order function or something
632
628
// tokenizes a scope
@@ -662,15 +658,10 @@ impl Lexer {
662
658
frame. x = init_x;
663
659
while frame. x < scope. tile . x + scope. tile . width as usize {
664
660
' 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 } ) ;
668
662
669
663
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 } ) ;
674
665
675
666
if pixels[ y + frame. y ] [ x + frame. x ] == scope. colour {
676
667
continue ;
@@ -752,7 +743,7 @@ impl Lexer {
752
743
753
744
// if the pixel is unknown then it could be a scope
754
745
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) ;
756
747
// rectangle is big enough to be a scope
757
748
if scope. width > 64 && scope. height > 64 {
758
749
self . analyse_scope ( & Scope {
@@ -895,6 +886,7 @@ pub fn deserialize(key: &String, source: &String) -> Result<Vec<Lexeme>, image::
895
886
}
896
887
897
888
// 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
898
890
#[ cfg( test) ]
899
891
mod tests {
900
892
use super :: * ;
@@ -960,6 +952,20 @@ mod tests {
960
952
961
953
}
962
954
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
+
963
969
// Key tests
964
970
struct KeySetup {
965
971
img : image:: DynamicImage ,
0 commit comments