group.rs 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344
  1. // use std::fmt;
  2. use strum::EnumIter;
  3. use strum::IntoEnumIterator;
  4. // use strum_macros::EnumIter;
  5. #[derive(Debug, EnumIter, PartialEq, Copy, Clone)]
  6. pub enum Groups {
  7. Row,
  8. Column,
  9. Cell,
  10. }
  11. #[derive(Debug, Clone)]
  12. pub struct AnyGroup {
  13. pub size: u8,
  14. pub width: u8,
  15. pub max_index: usize,
  16. pub groups: [Vec<usize>; 3],
  17. pub row_cell: Vec<usize>,
  18. pub col_cell: Vec<usize>,
  19. }
  20. /// Find the number of digits needed to display given number.
  21. fn find_number_width(mut number: usize) -> usize {
  22. let mut size: usize = 1;
  23. number /= 10;
  24. while number > 0 {
  25. size += 1;
  26. number /= 10;
  27. }
  28. size
  29. }
  30. impl AnyGroup {
  31. pub fn new(board_size: u8) -> Self {
  32. let b_width = board_size as usize * board_size as usize;
  33. let size = b_width * b_width;
  34. let mut g = AnyGroup {
  35. size: board_size,
  36. width: board_size * board_size,
  37. max_index: size,
  38. groups: [vec![0; size], vec![0; size], vec![0; size]],
  39. row_cell: vec![0; size],
  40. col_cell: vec![0; size],
  41. /*
  42. row: vec![0; b_width * b_width],
  43. column: vec![0; size],
  44. cell: vec![0; size],
  45. */
  46. };
  47. g.calculate();
  48. g
  49. }
  50. fn calculate(&mut self) {
  51. let size = self.size;
  52. for y in 0..self.width {
  53. self.row_cell[y as usize] = y as usize / size as usize;
  54. self.col_cell[y as usize] = (y as usize/ size as usize) * size as usize;
  55. for x in 0..self.width {
  56. let index = self.pos(x,y);
  57. // x as usize + y as usize * self.width as usize;
  58. self.groups[Groups::Row as usize][index] = self.pos(x, y);
  59. self.groups[Groups::Column as usize][index] = self.pos(y, x);
  60. // self.row[index] = self.pos(x, y);
  61. // self.column[index] = self.pos(y, x);
  62. let x_off = x / size;
  63. let y_off = y / size;
  64. let x_mod = x % size;
  65. let y_mod = y % size;
  66. // x is correct:
  67. // x_mod + y_mod * self.size;
  68. // y is correct:
  69. // x_off + y_off * self.size
  70. self.groups[Groups::Cell as usize][index] =
  71. self.pos(x_mod + y_mod * size, x_off + y_off * size);
  72. }
  73. }
  74. }
  75. #[must_use]
  76. #[inline]
  77. /// Convert (X,Y) to index.
  78. pub fn pos(&self, x: u8, y: u8) -> usize {
  79. x as usize + y as usize * self.width as usize
  80. }
  81. #[must_use]
  82. #[inline]
  83. /// Convert index to (X,Y).
  84. pub fn xy(&self, index: usize) -> (u8, u8) {
  85. let width = self.width as usize;
  86. ((index % width) as u8, (index / width) as u8)
  87. }
  88. /// Indexes for that Group (Row,Column,Cell)
  89. pub fn group(&self, g: Groups, row: u8) -> &[usize] {
  90. let width = self.width as usize;
  91. let start = row as usize * width;
  92. &self.groups[g as usize][start..start + width]
  93. }
  94. /// Indexes for that row
  95. pub fn row(&self, row: u8) -> &[usize] {
  96. // return slice of row:
  97. let width = self.width as usize;
  98. let start = row as usize * width;
  99. &self.groups[Groups::Row as usize][start..start + width]
  100. }
  101. /// Indexes for that column
  102. pub fn column(&self, column: u8) -> &[usize] {
  103. // return slice of row:
  104. let width = self.width as usize;
  105. let start = column as usize * width;
  106. &self.groups[Groups::Column as usize][start..start + width]
  107. }
  108. // Indexes for that cell.
  109. pub fn cell(&self, cell: u8) -> &[usize] {
  110. // return slice of cell:
  111. let width = self.width as usize;
  112. let start = cell as usize * width;
  113. &self.groups[Groups::Cell as usize][start..start + width]
  114. }
  115. #[must_use]
  116. /// Which cell contains (x,y)?
  117. pub fn which_cell(&self, x: u8, y: u8) -> u8 {
  118. (x / self.size) + (y / self.size) * self.size
  119. }
  120. /*
  121. #[deprecated="Use xy() instead"]
  122. #[must_use]
  123. /// Convert index to x,y offsets.
  124. pub fn cell_offset(&self, idx:u8) -> (u8,u8) {
  125. (idx % self.size, idx / self.size)
  126. }
  127. */
  128. #[must_use]
  129. /// Which index for given cell and (x,y)?
  130. pub fn which_cell_index(&self, cell_index:u8, x:u8, y:u8) -> usize {
  131. let (sx,sy) = self.cell_start(cell_index);
  132. self.pos(sx + x, sy+y)
  133. }
  134. #[must_use]
  135. /// Where does a given cell index start?
  136. pub fn cell_start(&self, cell_index:u8) -> (u8,u8) {
  137. ((cell_index % self.size) * self.size, (cell_index/self.size) * self.size)
  138. }
  139. #[must_use]
  140. /// Return index of cell (x,y)
  141. /// - This uses the groups to locate the index of the group.
  142. pub fn cell_index(&self, index: u8, x: u8, y: u8) -> usize {
  143. debug_assert!( x < self.size);
  144. debug_assert!( y < self.size);
  145. let result: usize =
  146. self.groups[Groups::Cell as usize][index as usize * self.width as usize];
  147. result + (y * self.width + x) as usize
  148. }
  149. // Possibly update this so it shows +1 on x and y
  150. // to match actual positions (rather then starting at 0).
  151. /// display Group
  152. pub fn display(&self) {
  153. let max_index_size = find_number_width(self.width as usize * self.width as usize);
  154. let max_pos_size = find_number_width(self.width as usize);
  155. let printer = |row: u8, data: &[usize]| {
  156. print!("[{:2}: ", row);
  157. for j in data {
  158. let xy = self.xy(*j);
  159. print!(
  160. "{0:1$}({3:2$},{4:2$}) ",
  161. j,
  162. max_index_size,
  163. max_pos_size,
  164. xy.0 + 1,
  165. xy.1 + 1
  166. );
  167. }
  168. println!("");
  169. };
  170. for g in Groups::iter() {
  171. println!("{:?}:", g);
  172. for i in 0..self.width {
  173. let r = self.group(g, i);
  174. /*
  175. print!("[{:2}]: ", i);
  176. for j in r {
  177. let xy = self.xy(*j);
  178. print!("{0:1$}({2:2},{3:2}) ", j, max_index_size, xy.0, xy.1);
  179. }
  180. println!("");
  181. */
  182. printer(i, r);
  183. }
  184. }
  185. }
  186. }
  187. #[cfg(test)]
  188. mod tests {
  189. use crate::group::*;
  190. #[test]
  191. /// Verify that each index (0..max_index) is defined in each group.
  192. /// - Verify that it is used, and only once.
  193. fn check_dynamic() {
  194. for size in 3..=5 {
  195. let g = AnyGroup::new(size);
  196. // g.display();
  197. let mut all = vec![0 as u8; g.max_index];
  198. // Verify the X,Y values are in the correct cell.
  199. for gr in Groups::iter() {
  200. all.fill(0);
  201. for idx in 0..g.width {
  202. let grp = g.group(gr, idx); // cell(idx);
  203. // g.display();
  204. for indexes in grp {
  205. let (x, y) = g.xy(*indexes);
  206. assert!(
  207. all[*indexes] == 0,
  208. "{:?} Size {} Index [{}]({},{}) already seen",
  209. gr,
  210. size,
  211. *indexes,
  212. x + 1,
  213. y + 1
  214. );
  215. all[*indexes] = 1;
  216. if gr == Groups::Cell {
  217. assert_eq!(
  218. g.which_cell(x, y),
  219. idx,
  220. "Verify {:?} [{}]({},{}) in cell {}",
  221. gr,
  222. *indexes,
  223. x,
  224. y,
  225. idx
  226. );
  227. }
  228. }
  229. }
  230. for indexes in 0..g.max_index {
  231. let (x, y) = g.xy(indexes);
  232. assert!(
  233. all[indexes] == 1,
  234. "{:?} Size {} Index [{}]({},{}) not seen",
  235. gr,
  236. size,
  237. indexes,
  238. x + 1,
  239. y + 1
  240. );
  241. }
  242. }
  243. /*
  244. all.fill(0);
  245. for idx in 0..g.width {
  246. let grp = g.row(idx);
  247. for indexes in grp {
  248. let (x, y) = g.xy(*indexes);
  249. assert_eq!(y, idx, "Expected row Y {} == {}", y, idx);
  250. assert!(
  251. all[*indexes] == 0,
  252. "Size {} row Index [{}]({},{}) already seen",
  253. size,
  254. *indexes,
  255. x + 1,
  256. y + 1
  257. );
  258. all[*indexes] = 1;
  259. }
  260. }
  261. for indexes in 0..g.max_index {
  262. let (x, y) = g.xy(indexes);
  263. assert!(
  264. all[indexes] == 1,
  265. "Size {} row Index [{}]({},{}) not seen",
  266. size,
  267. indexes,
  268. x + 1,
  269. y + 1
  270. );
  271. }
  272. all.fill(0);
  273. for idx in 0..g.width {
  274. let grp = g.column(idx);
  275. for indexes in grp {
  276. let (x, y) = g.xy(*indexes);
  277. assert_eq!(x, idx, "Expected column X {} == {}", x, idx);
  278. assert!(
  279. all[*indexes] == 0,
  280. "Size {} col Index [{}]({},{}) already seen",
  281. size,
  282. *indexes,
  283. x + 1,
  284. y + 1
  285. );
  286. all[*indexes] = 1;
  287. }
  288. }
  289. for indexes in 0..g.max_index {
  290. let (x, y) = g.xy(indexes);
  291. assert!(
  292. all[indexes] == 1,
  293. "Size {} col Index [{}]({},{}) not seen",
  294. size,
  295. indexes,
  296. x + 1,
  297. y + 1
  298. );
  299. }
  300. */
  301. }
  302. }
  303. #[test]
  304. fn check_cells() {
  305. let size = 3;
  306. let g = AnyGroup::new(size);
  307. assert_eq!(g.cell_index(0, 0, 0), 0);
  308. }
  309. }