| #![allow(clippy::type_complexity)] |
|
|
| use std::io::{Read, Write}; |
|
|
| pub mod cli; |
| pub mod model; |
| pub mod ort_backend; |
| pub mod yolo_result; |
| pub use crate::cli::Args; |
| pub use crate::model::YOLOv8; |
| pub use crate::ort_backend::{Batch, OrtBackend, OrtConfig, OrtEP, YOLOTask}; |
| pub use crate::yolo_result::{Bbox, Embedding, Point2, YOLOResult}; |
|
|
| pub fn non_max_suppression( |
| xs: &mut Vec<(Bbox, Option<Vec<Point2>>, Option<Vec<f32>>)>, |
| iou_threshold: f32, |
| ) { |
| xs.sort_by(|b1, b2| b2.0.confidence().partial_cmp(&b1.0.confidence()).unwrap()); |
|
|
| let mut current_index = 0; |
| for index in 0..xs.len() { |
| let mut drop = false; |
| for prev_index in 0..current_index { |
| let iou = xs[prev_index].0.iou(&xs[index].0); |
| if iou > iou_threshold { |
| drop = true; |
| break; |
| } |
| } |
| if !drop { |
| xs.swap(current_index, index); |
| current_index += 1; |
| } |
| } |
| xs.truncate(current_index); |
| } |
|
|
| pub fn gen_time_string(delimiter: &str) -> String { |
| let offset = chrono::FixedOffset::east_opt(8 * 60 * 60).unwrap(); |
| let t_now = chrono::Utc::now().with_timezone(&offset); |
| let fmt = format!( |
| "%Y{}%m{}%d{}%H{}%M{}%S{}%f", |
| delimiter, delimiter, delimiter, delimiter, delimiter, delimiter |
| ); |
| t_now.format(&fmt).to_string() |
| } |
|
|
| pub const SKELETON: [(usize, usize); 16] = [ |
| (0, 1), |
| (0, 2), |
| (1, 3), |
| (2, 4), |
| (5, 6), |
| (5, 11), |
| (6, 12), |
| (11, 12), |
| (5, 7), |
| (6, 8), |
| (7, 9), |
| (8, 10), |
| (11, 13), |
| (12, 14), |
| (13, 15), |
| (14, 16), |
| ]; |
|
|
| pub fn check_font(font: &str) -> rusttype::Font<'static> { |
| |
|
|
| |
| let font_path_config = match dirs::config_dir() { |
| Some(mut d) => { |
| d.push("Ultralytics"); |
| d.push(font); |
| d |
| } |
| None => panic!("Unsupported operating system. Now support Linux, MacOS, Windows."), |
| }; |
|
|
| |
| let font_path_current = std::path::PathBuf::from(font); |
|
|
| |
| let font_path = if font_path_config.exists() { |
| font_path_config |
| } else if font_path_current.exists() { |
| font_path_current |
| } else { |
| println!("Downloading font..."); |
| let source_url = "https://ultralytics.com/assets/Arial.ttf"; |
| let resp = ureq::get(source_url) |
| .timeout(std::time::Duration::from_secs(500)) |
| .call() |
| .unwrap_or_else(|err| panic!("> Failed to download font: {source_url}: {err:?}")); |
|
|
| |
| let mut buffer = vec![]; |
| let total_size = resp |
| .header("Content-Length") |
| .and_then(|s| s.parse::<u64>().ok()) |
| .unwrap(); |
| let _reader = resp |
| .into_reader() |
| .take(total_size) |
| .read_to_end(&mut buffer) |
| .unwrap(); |
|
|
| |
| let _path = std::fs::File::create(font).unwrap(); |
| let mut writer = std::io::BufWriter::new(_path); |
| writer.write_all(&buffer).unwrap(); |
| println!("Font saved at: {:?}", font_path_current.display()); |
| font_path_current |
| }; |
|
|
| |
| let buffer = std::fs::read(font_path).unwrap(); |
| rusttype::Font::try_from_vec(buffer).unwrap() |
| } |
|
|