feat: 灯条段排序。

This commit is contained in:
Ivan Li 2023-04-02 16:08:28 +08:00
parent 535f731770
commit 86e9b072bc
8 changed files with 89 additions and 186 deletions

View File

@ -17,14 +17,6 @@ pub enum Border {
Right, Right,
} }
#[derive(Clone, Copy, Serialize, Deserialize, Debug)]
pub struct LedStripConfigOfBorders {
pub top: Option<LedStripConfig>,
pub bottom: Option<LedStripConfig>,
pub left: Option<LedStripConfig>,
pub right: Option<LedStripConfig>,
}
#[derive(Clone, Copy, Serialize, Deserialize, Debug)] #[derive(Clone, Copy, Serialize, Deserialize, Debug)]
pub struct LedStripConfig { pub struct LedStripConfig {
pub index: usize, pub index: usize,
@ -119,6 +111,7 @@ impl LedStripConfigGroup {
mappers.push(SamplePointMapper { mappers.push(SamplePointMapper {
start: (j + i * 4) * 30, start: (j + i * 4) * 30,
end: (j + i * 4 + 1) * 30, end: (j + i * 4 + 1) * 30,
pos: (j + i * 4) * 30,
}) })
} }
} }
@ -126,124 +119,11 @@ impl LedStripConfigGroup {
} }
} }
#[derive(Clone, Copy, Serialize, Deserialize, Debug)]
pub struct LedStripConfigOfDisplays {
pub id: u32,
pub index_of_display: usize,
pub led_strip_of_borders: LedStripConfigOfBorders,
}
impl LedStripConfigOfBorders {
pub fn default() -> Self {
Self {
top: None,
bottom: None,
left: None,
right: None,
}
}
}
impl LedStripConfigOfDisplays {
pub fn default(id: u32, index_of_display: usize) -> Self {
Self {
id,
index_of_display,
led_strip_of_borders: LedStripConfigOfBorders::default(),
}
}
pub async fn read_from_disk() -> anyhow::Result<Self> {
let path = config_dir()
.unwrap_or(current_dir().unwrap())
.join("led_strip_config_of_displays.toml");
let exists = tokio::fs::try_exists(path.clone())
.await
.map_err(|e| anyhow::anyhow!("Failed to check config file exists: {}", e))?;
if exists {
let config = tokio::fs::read_to_string(path).await?;
let config: Self = toml::from_str(&config)
.map_err(|e| anyhow::anyhow!("Failed to parse config file: {}", e))?;
Ok(config)
} else {
info!("config file not exist, fallback to default config");
Ok(Self::get_default_config().await?)
}
}
pub async fn write_to_disk(&self) -> anyhow::Result<()> {
let path = config_dir()
.unwrap_or(current_dir().unwrap())
.join("led_strip_config_of_displays.toml");
let config = toml::to_string(self).map_err(|e| {
anyhow::anyhow!("Failed to parse config file: {}. config: {:?}", e, self)
})?;
tokio::fs::write(&path, config).await.map_err(|e| {
anyhow::anyhow!("Failed to write config file: {}. path: {:?}", e, &path)
})?;
Ok(())
}
pub async fn get_default_config() -> anyhow::Result<Self> {
let displays = display_info::DisplayInfo::all().map_err(|e| {
error!("can not list display info: {}", e);
anyhow::anyhow!("can not list display info: {}", e)
})?;
let mut configs = Vec::new();
for (i, display) in displays.iter().enumerate() {
let config = Self {
id: display.id,
index_of_display: i,
led_strip_of_borders: LedStripConfigOfBorders {
top: Some(LedStripConfig {
index: i * 4 * 30,
display_id: display.id,
border: Border::Top,
start_pos: i * 4 * 30,
len: 30,
}),
bottom: Some(LedStripConfig {
index: i * 4 * 30 + 30,
display_id: display.id,
border: Border::Bottom,
start_pos: i * 4 * 30 + 30,
len: 30,
}),
left: Some(LedStripConfig {
index: i * 4 * 30 + 60,
display_id: display.id,
border: Border::Left,
start_pos: i * 4 * 30 + 60,
len: 30,
}),
right: Some(LedStripConfig {
index: i * 4 * 30 + 90,
display_id: display.id,
border: Border::Right,
start_pos: i * 4 * 30 + 90,
len: 30,
}),
},
};
configs.push(config);
}
Ok(configs[0])
}
}
#[derive(Debug, Clone, Serialize, Deserialize)] #[derive(Debug, Clone, Serialize, Deserialize)]
pub struct SamplePointMapper { pub struct SamplePointMapper {
pub start: usize, pub start: usize,
pub end: usize, pub end: usize,
pub pos: usize,
} }
#[derive(Debug, Clone, Serialize, Deserialize)] #[derive(Debug, Clone, Serialize, Deserialize)]

View File

@ -122,7 +122,8 @@ impl ConfigManager {
log::info!("mapper: {:?}", mapper); log::info!("mapper: {:?}", mapper);
} }
} }
log::info!("mapper: {:?}", config.mappers[4]);
Self::rebuild_mappers(&mut config);
let cloned_config = config.clone(); let cloned_config = config.clone();
@ -138,17 +139,31 @@ impl ConfigManager {
} }
fn rebuild_mappers(config: &mut LedStripConfigGroup) { fn rebuild_mappers(config: &mut LedStripConfigGroup) {
let mut prev_end = 0; let mut prev_pos_end = 0;
let mappers: Vec<SamplePointMapper> = config let mappers: Vec<SamplePointMapper> = config
.strips .strips
.iter() .iter()
.map(|strip| { .enumerate()
let mapper = SamplePointMapper { .map(|(index, strip)| {
start: prev_end, let mapper = &config.mappers[index];
end: prev_end + strip.len,
}; if mapper.start < mapper.end {
prev_end = mapper.end; let mapper = SamplePointMapper {
mapper start: mapper.start,
end: mapper.start + strip.len,
pos: prev_pos_end,
};
prev_pos_end = prev_pos_end + strip.len;
mapper
} else {
let mapper = SamplePointMapper {
end: mapper.end,
start: mapper.end + strip.len,
pos: prev_pos_end,
};
prev_pos_end = prev_pos_end + strip.len;
mapper
}
}) })
.collect(); .collect();

View File

@ -236,22 +236,33 @@ impl ScreenshotManager {
pub async fn get_sorted_colors(colors: &Vec<LedColor>, mappers: &Vec<SamplePointMapper>) -> Vec<u8> { pub async fn get_sorted_colors(colors: &Vec<LedColor>, mappers: &Vec<SamplePointMapper>) -> Vec<u8> {
let total_leds = mappers let total_leds = mappers
.iter() .iter()
.map(|mapper| mapper.end) .map(|mapper| usize::max(mapper.start, mapper.end))
.max() .max()
.unwrap_or(0) as usize; .unwrap_or(0) as usize;
let mut global_colors = vec![0u8; total_leds * 3]; let mut global_colors = vec![0u8; total_leds * 3];
let mut color_index = 0; let mut color_index = 0;
mappers.iter().for_each(|group| { mappers.iter().for_each(|group| {
if group.end > colors.len() || group.start > colors.len() { if group.end > global_colors.len() || group.start > global_colors.len() {
warn!( warn!(
"get_sorted_colors: group out of range. start: {}, end: {}, colors.len(): {}", "get_sorted_colors: group out of range. start: {}, end: {}, global_colors.len(): {}",
group.start, group.start,
group.end, group.end,
global_colors.len()
);
return;
}
if color_index + group.start.abs_diff(group.end) > colors.len() {
warn!(
"get_sorted_colors: color_index out of range. color_index: {}, strip len: {}, colors.len(): {}",
color_index,
group.start.abs_diff(group.end),
colors.len() colors.len()
); );
return; return;
} }
if group.end > group.start { if group.end > group.start {
for i in group.start..group.end { for i in group.start..group.end {
let rgb = colors[color_index].get_rgb(); let rgb = colors[color_index].get_rgb();

View File

@ -44,7 +44,7 @@ function App() {
// listen to led_colors_changed event // listen to led_colors_changed event
createEffect(() => { createEffect(() => {
const unlisten = listen<Uint8ClampedArray>('led_colors_changed', (event) => { const unlisten = listen<Array<string>>('led_colors_changed', (event) => {
const colors = event.payload; const colors = event.payload;
setLedStripStore({ setLedStripStore({
@ -59,7 +59,7 @@ function App() {
// listen to led_sorted_colors_changed event // listen to led_sorted_colors_changed event
createEffect(() => { createEffect(() => {
const unlisten = listen<Array<string>>('led_sorted_colors_changed', (event) => { const unlisten = listen<Uint8ClampedArray>('led_sorted_colors_changed', (event) => {
const sortedColors = event.payload; const sortedColors = event.payload;
setLedStripStore({ setLedStripStore({

View File

@ -16,6 +16,7 @@ import { useTippy } from 'solid-tippy';
import { followCursor } from 'tippy.js'; import { followCursor } from 'tippy.js';
import { LedStripConfig } from '../models/led-strip-config'; import { LedStripConfig } from '../models/led-strip-config';
import { LedStripConfigurationContext } from '../contexts/led-strip-configuration.context'; import { LedStripConfigurationContext } from '../contexts/led-strip-configuration.context';
import { ledStripStore } from '../stores/led-strip.store';
type LedStripPartProps = { type LedStripPartProps = {
config?: LedStripConfig | null; config?: LedStripConfig | null;
@ -55,44 +56,31 @@ export const LedStripPart: Component<LedStripPartProps> = (props) => {
const [ledSamplePoints, setLedSamplePoints] = createSignal(); const [ledSamplePoints, setLedSamplePoints] = createSignal();
const [colors, setColors] = createSignal<string[]>([]); const [colors, setColors] = createSignal<string[]>([]);
// get led strip colors when screenshot updated // update led strip colors from global store
createEffect(() => { createEffect(() => {
const samplePoints = ledSamplePoints(); if (!localProps.config) {
if (!localProps.config || !samplePoints) {
return; return;
} }
let pendingCount = 0; const index = ledStripStore.strips.findIndex(
const unlisten = listen<{ (s) =>
base64_image: string; s.display_id === localProps.config!.display_id &&
display_id: number; s.border === localProps.config!.border,
height: number; );
width: number;
}>('encoded-screenshot-updated', (event) => {
if (event.payload.display_id !== localProps.config!.display_id) {
return;
}
if (pendingCount >= 1) {
return;
}
pendingCount++;
invoke<string[]>('get_one_edge_colors', { if (index === -1) {
samplePoints, return;
displayId: event.payload.display_id, }
})
.then((colors) => {
setColors(colors);
})
.finally(() => {
pendingCount--;
});
});
subscribeScreenshotUpdate(localProps.config.display_id);
onCleanup(() => { const mapper = ledStripStore.mappers[index];
unlisten.then((unlisten) => unlisten()); if (!mapper) {
}); return;
}
const offset = mapper.pos;
const colors = ledStripStore.colors.slice(offset, offset + localProps.config.len);
setColors(colors);
}); });
// get led strip sample points // get led strip sample points

View File

@ -12,21 +12,18 @@ import { LedStripConfig, LedStripPixelMapper } from '../models/led-strip-config'
import { ledStripStore } from '../stores/led-strip.store'; import { ledStripStore } from '../stores/led-strip.store';
import { invoke } from '@tauri-apps/api'; import { invoke } from '@tauri-apps/api';
import { LedStripConfigurationContext } from '../contexts/led-strip-configuration.context'; import { LedStripConfigurationContext } from '../contexts/led-strip-configuration.context';
import background from '../assets/transparent-grid-background.svg?url';
const SorterItem: Component<{ strip: LedStripConfig; mapper: LedStripPixelMapper }> = ( const SorterItem: Component<{ strip: LedStripConfig; mapper: LedStripPixelMapper }> = (
props, props,
) => { ) => {
const [fullLeds, setFullLeds] = createSignal<string[]>([]); const [fullLeds, setFullLeds] = createSignal<Array<string | null>>([]);
const [dragging, setDragging] = createSignal<boolean>(false); const [dragging, setDragging] = createSignal<boolean>(false);
const [dragStart, setDragStart] = createSignal<{ x: number; y: number } | null>(null); const [dragStart, setDragStart] = createSignal<{ x: number; y: number } | null>(null);
const [dragCurr, setDragCurr] = createSignal<{ x: number; y: number } | null>(null); const [dragCurr, setDragCurr] = createSignal<{ x: number; y: number } | null>(null);
const [dragStartIndex, setDragStartIndex] = createSignal<number>(0); const [dragStartIndex, setDragStartIndex] = createSignal<number>(0);
const [, { setSelectedStripPart }] = useContext(LedStripConfigurationContext); const [, { setSelectedStripPart }] = useContext(LedStripConfigurationContext);
const totalLedCount = createMemo(() => {
return ledStripStore.strips.reduce((acc, strip) => acc + strip.len, 0);
});
const move = (targetStart: number) => { const move = (targetStart: number) => {
if (targetStart === props.mapper.start) { if (targetStart === props.mapper.start) {
return; return;
@ -70,7 +67,8 @@ const SorterItem: Component<{ strip: LedStripConfig; mapper: LedStripPixelMapper
} }
setDragCurr({ x: ev.clientX, y: ev.clientY }); setDragCurr({ x: ev.clientX, y: ev.clientY });
const cellWidth = (ev.currentTarget as HTMLDivElement).clientWidth / totalLedCount(); const cellWidth =
(ev.currentTarget as HTMLDivElement).clientWidth / ledStripStore.totalLedCount;
const diff = ev.clientX - dragStart()!.x; const diff = ev.clientX - dragStart()!.x;
const moved = Math.round(diff / cellWidth); const moved = Math.round(diff / cellWidth);
if (moved === 0) { if (moved === 0) {
@ -85,9 +83,13 @@ const SorterItem: Component<{ strip: LedStripConfig; mapper: LedStripPixelMapper
// update fullLeds // update fullLeds
createEffect(() => { createEffect(() => {
const fullLeds = new Array(totalLedCount()).fill('rgba(255,255,255,0.5)'); const fullLeds = new Array(ledStripStore.totalLedCount).fill(null);
for (let i = props.mapper.start, j = 0; i < props.mapper.end; i++, j++) { for (
let i = props.mapper.start, j = props.mapper.pos;
i < props.mapper.end;
i++, j++
) {
fullLeds[i] = ledStripStore.colors[j]; fullLeds[i] = ledStripStore.colors[j];
} }
setFullLeds(fullLeds); setFullLeds(fullLeds);
@ -112,11 +114,12 @@ const SorterItem: Component<{ strip: LedStripConfig; mapper: LedStripPixelMapper
{(it) => ( {(it) => (
<div <div
class="flex-auto flex h-full w-full justify-center items-center relative" class="flex-auto flex h-full w-full justify-center items-center relative"
title={it} title={it ?? ''}
> >
<div <div
class="absolute top-1/2 -translate-y-1/2 h-2.5 w-2.5 rounded-full ring-1 ring-stone-300" class="absolute top-1/2 -translate-y-1/2 h-2.5 w-2.5 rounded-full ring-1 ring-stone-100"
style={{ background: it }} classList={{ 'ring-stone-300/50': !it }}
style={{ background: it ?? 'transparent' }}
/> />
</div> </div>
)} )}
@ -129,8 +132,7 @@ const SorterResult: Component = () => {
const [fullLeds, setFullLeds] = createSignal<string[]>([]); const [fullLeds, setFullLeds] = createSignal<string[]>([]);
createEffect(() => { createEffect(() => {
const totalLedCount = Math.max(0, ...ledStripStore.mappers.map((m) => m.end)); const fullLeds = new Array(ledStripStore.totalLedCount).fill('rgba(255,255,255,0.1)');
const fullLeds = new Array(totalLedCount).fill('rgba(255,255,255,0.5)');
ledStripStore.mappers.forEach((mapper) => { ledStripStore.mappers.forEach((mapper) => {
for (let i = mapper.start, j = 0; i < mapper.end; i++, j++) { for (let i = mapper.start, j = 0; i < mapper.end; i++, j++) {
@ -165,7 +167,12 @@ export const LedStripPartsSorter: Component = () => {
const context = createContext(); const context = createContext();
return ( return (
<div class="select-none overflow-hidden"> <div
class="select-none overflow-hidden"
style={{
'background-image': `url(${background})`,
}}
>
<SorterResult /> <SorterResult />
<For each={ledStripStore.strips}> <For each={ledStripStore.strips}>
{(strip, index) => ( {(strip, index) => (

View File

@ -3,6 +3,7 @@ import { Borders } from '../constants/border';
export type LedStripPixelMapper = { export type LedStripPixelMapper = {
start: number; start: number;
end: number; end: number;
pos: number;
}; };
export type LedStripConfigContainer = { export type LedStripConfigContainer = {

View File

@ -1,11 +1,12 @@
import { createStore } from 'solid-js/store'; import { createStore } from 'solid-js/store';
import { DisplayConfig } from '../models/display-config';
import { LedStripConfig, LedStripPixelMapper } from '../models/led-strip-config'; import { LedStripConfig, LedStripPixelMapper } from '../models/led-strip-config';
export const [ledStripStore, setLedStripStore] = createStore({ export const [ledStripStore, setLedStripStore] = createStore({
displays: new Array<DisplayConfig>(),
strips: new Array<LedStripConfig>(), strips: new Array<LedStripConfig>(),
mappers: new Array<LedStripPixelMapper>(), mappers: new Array<LedStripPixelMapper>(),
colors: new Uint8ClampedArray(), colors: new Array<string>(),
sortedColors: new Array<string>(), sortedColors: new Uint8ClampedArray(),
get totalLedCount() {
return Math.max(0, ...ledStripStore.mappers.map((m) => Math.max(m.start, m.end)));
},
}); });