use std::{borrow::Cow, iter::{self, repeat_n}, mem}; use itertools::Itertools; use ratatui::{style::{Color, Style, Stylize}, text::Span}; use crate::{BYTES_PER_CHUNK, BYTES_PER_LINE, CHUNKS_PER_LINE, buffer::{Buffer, Mode, PartialAction}, utilities::{CustomGreys, empty_span, HasCardinality}, cursor::InCursor}; impl Buffer { pub fn render_chunks( &self, address: usize, bytes: &[u8; BYTES_PER_LINE] ) -> impl Iterator> { let (chunks, remainder) = bytes.as_chunks::(); assert!(remainder.is_empty(), "BYTES_PER_LINE should be a multiple of BYTES_PER_CHUNK"); #[allow(unstable_name_collisions)] chunks .iter() .copied() .zip((address..).step_by(BYTES_PER_CHUNK)) .flat_map(|(chunk, address)| { self.render_chunk(address, &chunk).collect::>() }) } pub fn render_partial_chunks( &self, address: usize, bytes: &[u8] ) -> impl Iterator> { let (chunks, remainder) = bytes.as_chunks::(); let remainder_address = address + chunks.len() * BYTES_PER_CHUNK; #[allow(clippy::if_not_else)] let remainder_chunks: Option> = if !remainder.is_empty() { Some(self.render_partial_chunk(remainder_address, remainder).collect()) } else { None }; let chunks_rendered = chunks.len() + remainder_chunks.iter().len(); let chunks_not_rendered = CHUNKS_PER_LINE - chunks_rendered; let spaces_per_chunk = BYTES_PER_CHUNK - 1 + 2; let bytes_not_rendered = BYTES_PER_LINE - bytes.len(); let padding_width = 2 * bytes_not_rendered + spaces_per_chunk * chunks_not_rendered; #[allow(unstable_name_collisions)] chunks .iter() .copied() .zip((address..).step_by(BYTES_PER_CHUNK)) .map(|(chunk, address)| self.render_chunk(address, &chunk).collect()) .chain(remainder_chunks) .flatten() .chain(repeat_n(" ".into(), padding_width)) } fn render_chunk( &self, address: usize, bytes: &[u8; BYTES_PER_CHUNK] ) -> impl Iterator> { iter::once(self.render_large_space_before(address)) .chain( bytes .iter() .copied() .zip(address..) .map(|(byte, address)| self.render_byte_at(address, byte)) .interleave( (address..) .take(BYTES_PER_CHUNK) .skip(1) .map(|address| self.render_space_before(address)) ) ) } fn render_partial_chunk( &self, address: usize, bytes: &[u8] ) -> impl Iterator> { iter::once(self.render_large_space_before(address)) .chain( bytes .iter() .copied() .zip(address..) .map(|(byte, address)| self.render_byte_at(address, byte)) .interleave( (address..) .take(BYTES_PER_CHUNK) .skip(1) .map(|address| self.render_space_before(address)) ) ) } fn render_byte_at( &self, address: usize, byte: u8 ) -> Span<'static> { // TODO: checking this with lots of selections is slow if self.partial_action == Some(PartialAction::Replace) && iter::once(&self.primary_cursor) .chain(&self.cursors) .any(|cursor| cursor.contains(address).is_some()) { let replaced_byte = self.partial_replace.unwrap_or(0) << 4; self.render_byte_without_replace_preview(address, replaced_byte) .black() } else { self.render_byte_without_replace_preview(address, byte) } } fn render_byte_without_replace_preview( &self, address: usize, byte: u8 ) -> Span<'static> { const SPAN_FOR_BYTE: [Span; u8::CARDINALITY] = create_byte_lookup_table(); let span = SPAN_FOR_BYTE[byte as usize].clone(); if let Some(place_in_cursor) = self.primary_cursor.contains(address) { let head_color = match self.mode { Mode::Select => Color::Yellow, Mode::Normal => Color::Gray }; match place_in_cursor { InCursor::Head => span.bg(head_color), InCursor::Rest => span.bg(Color::selection_tail_grey()), } } else { match self.cursors .iter() .find_map(|cursor| cursor.contains(address)) { Some(InCursor::Head) => span.bg(Color::secondary_selection_head_grey()), Some(InCursor::Rest) => span.bg(Color::selection_tail_grey()), None => span, } } } fn render_large_space_before(&self, address: usize) -> Span<'static> { let span: Span = if self.marks.contains(&address) { " →".into() } else { " ".into() }; if !address.is_multiple_of(BYTES_PER_LINE) && iter::once(&self.primary_cursor) .chain(&self.cursors) .any(|cursor| cursor.contains_space_before(address)) { span.bg(Color::selection_tail_grey()) } else { span } } fn render_space_before(&self, address: usize) -> Span<'static> { let span: Span = if self.marks.contains(&address) { "→".into() } else { " ".into() }; if iter::once(&self.primary_cursor) .chain(&self.cursors) .any(|cursor| cursor.contains_space_before(address)) { span.bg(Color::selection_tail_grey()) } else { span } } } const fn create_byte_lookup_table() -> [Span<'static>; u8::CARDINALITY] { let mut result = [const { empty_span() }; u8::CARDINALITY]; let mut index = 0; while index < u8::CARDINALITY { #[allow(clippy::cast_possible_truncation)] let byte = index as u8; result[index].style = style_for_byte(byte); mem::forget(mem::replace(&mut result[index].content, content_for_byte(byte))); index += 1; } result } const fn style_for_byte(byte: u8) -> Style { Style::new().fg(fg_for_byte(byte)) } const fn fg_for_byte(byte: u8) -> Color { match byte { 0x00 => Color::Rgb(0x80, 0x80, 0x80), // grey 0x01..0x10 => Color::Rgb(0xFF, 0x71, 0xA9), // red 0x10..0x20 => Color::Rgb(0xFF, 0x7A, 0x78), // salmon 0x20..0x30 => Color::Rgb(0xFF, 0x81, 0x23), // red-orange 0x30..0x40 => Color::Rgb(0xF7, 0x93, 0x00), // yellow-orange 0x40..0x50 => Color::Rgb(0xE6, 0x9F, 0x00), // yellow 0x50..0x60 => Color::Rgb(0xC1, 0xB2, 0x00), // green-yellow 0x60..0x70 => Color::Rgb(0x82, 0xC6, 0x00), // lime 0x70..0x80 => Color::Rgb(0x00, 0xD5, 0x00), // green 0x80..0x90 => Color::Rgb(0x00, 0xD4, 0x59), // clover 0x90..0xA0 => Color::Rgb(0x00, 0xD0, 0x91), // teal 0xA0..0xB0 => Color::Rgb(0x00, 0xCC, 0xBB), // cyan 0xB0..0xC0 => Color::Rgb(0x00, 0xC7, 0xDE), // light blue 0xC0..0xD0 => Color::Rgb(0x00, 0xBE, 0xFF), // blue 0xD0..0xE0 => Color::Rgb(0x6C, 0xAF, 0xFF), // blurple 0xE0..0xF0 => Color::Rgb(0xB2, 0x98, 0xFF), // purple 0xF0..0xFF => Color::Rgb(0xFF, 0x4D, 0xFF), // pink 0xFF => Color::White } } const fn content_for_byte(byte: u8) -> Cow<'static, str> { Cow::Borrowed(hex_for_byte(byte)) } const fn hex_for_byte(byte: u8) -> &'static str { const LOOK_UP_TABLE: [&str; u8::CARDINALITY] = ["00", "01", "02", "03", "04", "05", "06", "07", "08", "09", "0a", "0b", "0c", "0d", "0e", "0f", "10", "11", "12", "13", "14", "15", "16", "17", "18", "19", "1a", "1b", "1c", "1d", "1e", "1f", "20", "21", "22", "23", "24", "25", "26", "27", "28", "29", "2a", "2b", "2c", "2d", "2e", "2f", "30", "31", "32", "33", "34", "35", "36", "37", "38", "39", "3a", "3b", "3c", "3d", "3e", "3f", "40", "41", "42", "43", "44", "45", "46", "47", "48", "49", "4a", "4b", "4c", "4d", "4e", "4f", "50", "51", "52", "53", "54", "55", "56", "57", "58", "59", "5a", "5b", "5c", "5d", "5e", "5f", "60", "61", "62", "63", "64", "65", "66", "67", "68", "69", "6a", "6b", "6c", "6d", "6e", "6f", "70", "71", "72", "73", "74", "75", "76", "77", "78", "79", "7a", "7b", "7c", "7d", "7e", "7f", "80", "81", "82", "83", "84", "85", "86", "87", "88", "89", "8a", "8b", "8c", "8d", "8e", "8f", "90", "91", "92", "93", "94", "95", "96", "97", "98", "99", "9a", "9b", "9c", "9d", "9e", "9f", "a0", "a1", "a2", "a3", "a4", "a5", "a6", "a7", "a8", "a9", "aa", "ab", "ac", "ad", "ae", "af", "b0", "b1", "b2", "b3", "b4", "b5", "b6", "b7", "b8", "b9", "ba", "bb", "bc", "bd", "be", "bf", "c0", "c1", "c2", "c3", "c4", "c5", "c6", "c7", "c8", "c9", "ca", "cb", "cc", "cd", "ce", "cf", "d0", "d1", "d2", "d3", "d4", "d5", "d6", "d7", "d8", "d9", "da", "db", "dc", "dd", "de", "df", "e0", "e1", "e2", "e3", "e4", "e5", "e6", "e7", "e8", "e9", "ea", "eb", "ec", "ed", "ee", "ef", "f0", "f1", "f2", "f3", "f4", "f5", "f6", "f7", "f8", "f9", "fa", "fb", "fc", "fd", "fe", "ff"]; LOOK_UP_TABLE[byte as usize] }