use core::cmp::max;
use alloc::vec::Vec;
use super::char_data::BidiClass::{self, *};
use super::prepare::{IsolatingRunSequence, LevelRun, not_removed_by_x9, removed_by_x9};
use super::level::Level;
#[cfg_attr(feature = "flame_it", flamer::flame)]
pub fn resolve_weak(sequence: &IsolatingRunSequence, processing_classes: &mut [BidiClass]) {
let mut prev_class = sequence.sos;
let mut last_strong_is_al = false;
let mut et_run_indices = Vec::new();
fn id(x: LevelRun) -> LevelRun {
x
}
let mut indices = sequence.runs.iter().cloned().flat_map(
id as fn(LevelRun) -> LevelRun,
);
while let Some(i) = indices.next() {
match processing_classes[i] {
NSM => {
processing_classes[i] = match prev_class {
RLI | LRI | FSI | PDI => ON,
_ => prev_class,
};
}
EN => {
if last_strong_is_al {
processing_classes[i] = AN;
} else {
for j in &et_run_indices {
processing_classes[*j] = EN;
}
et_run_indices.clear();
}
}
AL => processing_classes[i] = R,
ES | CS => {
let next_class = indices
.clone()
.map(|j| processing_classes[j])
.find(not_removed_by_x9)
.unwrap_or(sequence.eos);
processing_classes[i] = match (prev_class, processing_classes[i], next_class) {
(EN, ES, EN) | (EN, CS, EN) => EN,
(AN, CS, AN) => AN,
(_, _, _) => ON,
}
}
ET => {
match prev_class {
EN => processing_classes[i] = EN,
_ => et_run_indices.push(i), }
}
class => {
if removed_by_x9(class) {
continue;
}
}
}
prev_class = processing_classes[i];
match prev_class {
L | R => {
last_strong_is_al = false;
}
AL => {
last_strong_is_al = true;
}
_ => {}
}
if prev_class != ET {
for j in &et_run_indices {
processing_classes[*j] = ON;
}
et_run_indices.clear();
}
}
let mut last_strong_is_l = sequence.sos == L;
for run in &sequence.runs {
for i in run.clone() {
match processing_classes[i] {
EN if last_strong_is_l => {
processing_classes[i] = L;
}
L => {
last_strong_is_l = true;
}
R | AL => {
last_strong_is_l = false;
}
_ => {}
}
}
}
}
#[cfg_attr(feature = "flame_it", flamer::flame)]
pub fn resolve_neutral(
sequence: &IsolatingRunSequence,
levels: &[Level],
processing_classes: &mut [BidiClass],
) {
let e: BidiClass = levels[sequence.runs[0].start].bidi_class();
let mut indices = sequence.runs.iter().flat_map(Clone::clone);
let mut prev_class = sequence.sos;
while let Some(mut i) = indices.next() {
let mut ni_run = Vec::new();
if is_NI(processing_classes[i]) {
ni_run.push(i);
let mut next_class;
loop {
match indices.next() {
Some(j) => {
i = j;
if removed_by_x9(processing_classes[i]) {
continue;
}
next_class = processing_classes[j];
if is_NI(next_class) {
ni_run.push(i);
} else {
break;
}
}
None => {
next_class = sequence.eos;
break;
}
};
}
let new_class = match (prev_class, next_class) {
(L, L) => L,
(R, R) | (R, AN) | (R, EN) | (AN, R) | (AN, AN) | (AN, EN) | (EN, R) |
(EN, AN) | (EN, EN) => R,
(_, _) => e,
};
for j in &ni_run {
processing_classes[*j] = new_class;
}
ni_run.clear();
}
prev_class = processing_classes[i];
}
}
#[cfg_attr(feature = "flame_it", flamer::flame)]
pub fn resolve_levels(original_classes: &[BidiClass], levels: &mut [Level]) -> Level {
let mut max_level = Level::ltr();
assert_eq!(original_classes.len(), levels.len());
for i in 0..levels.len() {
match (levels[i].is_rtl(), original_classes[i]) {
(false, AN) | (false, EN) => levels[i].raise(2).expect("Level number error"),
(false, R) | (true, L) | (true, EN) | (true, AN) => {
levels[i].raise(1).expect("Level number error")
}
(_, _) => {}
}
max_level = max(max_level, levels[i]);
}
max_level
}
#[allow(non_snake_case)]
fn is_NI(class: BidiClass) -> bool {
match class {
B | S | WS | ON | FSI | LRI | RLI | PDI => true,
_ => false,
}
}