group.rs 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282
  1. /*
  2. New code -
  3. AnyGroup
  4. Handles any size puzzles.
  5. The old code below only handles 3x3 puzzles only!
  6. */
  7. #[derive(Debug, Clone)]
  8. pub struct AnyGroup {
  9. pub size: u8,
  10. pub width: u8,
  11. pub max_index: usize,
  12. pub row: Vec<usize>,
  13. pub column: Vec<usize>,
  14. pub cell: Vec<usize>,
  15. }
  16. /// Find the number of digits needed to display given number.
  17. fn find_number_width(mut number: usize) -> usize {
  18. let mut size: usize = 1;
  19. number /= 10;
  20. while number > 0 {
  21. size += 1;
  22. number /= 10;
  23. }
  24. size
  25. }
  26. impl AnyGroup {
  27. pub fn new(board_size: u8) -> Self {
  28. let width = board_size as usize * board_size as usize;
  29. let mut g = AnyGroup {
  30. size: board_size,
  31. width: board_size * board_size,
  32. max_index: width * width,
  33. row: vec![0; width * width],
  34. column: vec![0; width * width],
  35. cell: vec![0; width * width],
  36. };
  37. g.calculate();
  38. g
  39. }
  40. pub fn pos(&self, x: u8, y: u8) -> usize {
  41. x as usize + y as usize * self.width as usize
  42. }
  43. pub fn xy(&self, index: usize) -> (u8, u8) {
  44. let width = self.width as usize;
  45. ((index % width) as u8, (index / width) as u8)
  46. }
  47. pub fn row(&self, row: u8) -> &[usize] {
  48. // return slice of row:
  49. let width = self.width as usize;
  50. let start = row as usize * width;
  51. &self.row[start..start + width]
  52. }
  53. pub fn column(&self, column: u8) -> &[usize] {
  54. // return slice of row:
  55. let width = self.width as usize;
  56. let start = column as usize * width;
  57. &self.column[start..start + width]
  58. }
  59. pub fn cell(&self, cell: u8) -> &[usize] {
  60. // return slice of cell:
  61. let width = self.width as usize;
  62. let start = cell as usize * width;
  63. &self.cell[start..start + width]
  64. }
  65. /// Which cell contains (x,y)?
  66. pub fn which_cell(&self, x: u8, y: u8) -> u8 {
  67. (x / self.size) + (y / self.size) * self.size
  68. }
  69. pub fn calculate(&mut self) {
  70. for y in 0..self.width {
  71. for x in 0..self.width {
  72. let index = x as usize + y as usize * self.width as usize;
  73. self.row[index] = self.pos(x, y);
  74. self.column[index] = self.pos(y, x);
  75. let x_off = x / self.size;
  76. let y_off = y / self.size;
  77. let x_mod = x % self.size;
  78. let y_mod = y % self.size;
  79. // x is correct:
  80. // x_mod + y_mod * self.size;
  81. // y is correct:
  82. // x_off + y_off * self.size
  83. self.cell[index] = self.pos(
  84. x_mod + y_mod * self.size,
  85. x_off + y_off * self.size, // x_mod + x_off * self.size,
  86. // y_off
  87. );
  88. }
  89. }
  90. }
  91. // Possibly update this so it shows +1 on x and y
  92. // to match actual positions (rather then starting at 0).
  93. pub fn display(&self) {
  94. let max_index_size = find_number_width(self.width as usize * self.width as usize);
  95. let max_pos_size = find_number_width(self.width as usize);
  96. let printer = |row: u8, data: &[usize]| {
  97. print!("[{:2}: ", row);
  98. for j in data {
  99. let xy = self.xy(*j);
  100. print!(
  101. "{0:1$}({3:2$},{4:2$}) ",
  102. j,
  103. max_index_size,
  104. max_pos_size,
  105. xy.0 + 1,
  106. xy.1 + 1
  107. );
  108. }
  109. println!("");
  110. };
  111. println!("rows:");
  112. // println!("{:?}", self.row);
  113. for i in 0..self.width {
  114. let r = self.row(i);
  115. /*
  116. print!("[{:2}]: ", i);
  117. for j in r {
  118. let xy = self.xy(*j);
  119. print!("{0:1$}({2:2},{3:2}) ", j, max_index_size, xy.0, xy.1);
  120. }
  121. println!("");
  122. */
  123. printer(i, r);
  124. }
  125. println!("columns:");
  126. // println!("{:?}", self.column);
  127. for i in 0..self.width {
  128. let r = self.column(i);
  129. /*
  130. print!("[{:2}]: ", i);
  131. for j in r {
  132. let xy = self.xy(*j);
  133. print!("{0:1$}({2:2},{3:2}) ", j, max_index_size, xy.0, xy.1);
  134. }
  135. println!("");
  136. */
  137. printer(i, r);
  138. }
  139. println!("cells:");
  140. // println!("{:?}", self.cell);
  141. for i in 0..self.width {
  142. let r = self.cell(i);
  143. /*
  144. print!("[{:2}]: ", i);
  145. for j in r {
  146. let xy = self.xy(*j);
  147. print!("{0:1$}({2:2},{3:2}) ", j, max_index_size, xy.0, xy.1);
  148. }
  149. println!("");
  150. */
  151. printer(i, r);
  152. }
  153. }
  154. }
  155. #[cfg(test)]
  156. mod tests {
  157. use crate::group::*;
  158. #[test]
  159. fn check_dynamic() {
  160. // Verify that each index (0..max_index) is defined in each group.
  161. // - Verify that it is used, and only once.
  162. for size in 3..=5 {
  163. let g = AnyGroup::new(size);
  164. let mut all = vec![0 as u8; g.max_index];
  165. // Verify the X,Y values are in the correct cell.
  166. all.fill(0);
  167. for idx in 0..g.width {
  168. let grp = g.cell(idx);
  169. // g.display();
  170. for indexes in grp {
  171. let (x, y) = g.xy(*indexes);
  172. assert!(
  173. all[*indexes] == 0,
  174. "Size {} Index [{}]({},{}) already seen",
  175. size,
  176. *indexes,
  177. x + 1,
  178. y + 1
  179. );
  180. all[*indexes] = 1;
  181. assert_eq!(g.which_cell(x, y), idx, "Verify [{}]({},{}) in cell {}", *indexes, x, y, idx);
  182. }
  183. }
  184. for indexes in 0..g.max_index {
  185. let (x, y) = g.xy(indexes);
  186. assert!(
  187. all[indexes] == 1,
  188. "Size {} Index [{}]({},{}) not seen",
  189. size,
  190. indexes,
  191. x + 1,
  192. y + 1
  193. );
  194. }
  195. all.fill(0);
  196. for idx in 0..g.width {
  197. let grp = g.row(idx);
  198. for indexes in grp {
  199. let (x, y) = g.xy(*indexes);
  200. assert_eq!(y, idx, "Expected row Y {} == {}", y, idx);
  201. assert!(
  202. all[*indexes] == 0,
  203. "Size {} row Index [{}]({},{}) already seen",
  204. size,
  205. *indexes,
  206. x + 1,
  207. y + 1
  208. );
  209. all[*indexes] = 1;
  210. }
  211. }
  212. for indexes in 0..g.max_index {
  213. let (x, y) = g.xy(indexes);
  214. assert!(
  215. all[indexes] == 1,
  216. "Size {} row Index [{}]({},{}) not seen",
  217. size,
  218. indexes,
  219. x + 1,
  220. y + 1
  221. );
  222. }
  223. all.fill(0);
  224. for idx in 0..g.width {
  225. let grp = g.column(idx);
  226. for indexes in grp {
  227. let (x, y) = g.xy(*indexes);
  228. assert_eq!(x, idx, "Expected column X {} == {}", x, idx);
  229. assert!(
  230. all[*indexes] == 0,
  231. "Size {} col Index [{}]({},{}) already seen",
  232. size,
  233. *indexes,
  234. x + 1,
  235. y + 1
  236. );
  237. all[*indexes] = 1;
  238. }
  239. }
  240. for indexes in 0..g.max_index {
  241. let (x, y) = g.xy(indexes);
  242. assert!(
  243. all[indexes] == 1,
  244. "Size {} col Index [{}]({},{}) not seen",
  245. size,
  246. indexes,
  247. x + 1,
  248. y + 1
  249. );
  250. }
  251. }
  252. }
  253. }