use core::cmp;
use core::iter::Filter;
mod fwd {
use crate::tables::sentence::SentenceCat;
use core::cmp;
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
enum StatePart {
Sot,
Eot,
Other,
CR,
LF,
Sep,
ATerm,
UpperLower,
ClosePlus,
SpPlus,
STerm,
}
#[derive(Debug, Clone, PartialEq, Eq)]
struct SentenceBreaksState(pub [StatePart; 4]);
const INITIAL_STATE: SentenceBreaksState = SentenceBreaksState([
StatePart::Sot,
StatePart::Sot,
StatePart::Sot,
StatePart::Sot,
]);
#[derive(Debug, Clone)]
pub struct SentenceBreaks<'a> {
pub string: &'a str,
pos: usize,
state: SentenceBreaksState,
}
impl SentenceBreaksState {
fn next(&self, cat: SentenceCat) -> SentenceBreaksState {
let &SentenceBreaksState(parts) = self;
let parts = match (parts[3], cat) {
(StatePart::ClosePlus, SentenceCat::SC_Close) => parts,
(StatePart::SpPlus, SentenceCat::SC_Sp) => parts,
_ => [
parts[1],
parts[2],
parts[3],
match cat {
SentenceCat::SC_CR => StatePart::CR,
SentenceCat::SC_LF => StatePart::LF,
SentenceCat::SC_Sep => StatePart::Sep,
SentenceCat::SC_ATerm => StatePart::ATerm,
SentenceCat::SC_Upper | SentenceCat::SC_Lower => StatePart::UpperLower,
SentenceCat::SC_Close => StatePart::ClosePlus,
SentenceCat::SC_Sp => StatePart::SpPlus,
SentenceCat::SC_STerm => StatePart::STerm,
_ => StatePart::Other,
},
],
};
SentenceBreaksState(parts)
}
fn end(&self) -> SentenceBreaksState {
let &SentenceBreaksState(parts) = self;
SentenceBreaksState([parts[1], parts[2], parts[3], StatePart::Eot])
}
fn match1(&self, part: StatePart) -> bool {
let &SentenceBreaksState(parts) = self;
part == parts[3]
}
fn match2(&self, part1: StatePart, part2: StatePart) -> bool {
let &SentenceBreaksState(parts) = self;
part1 == parts[2] && part2 == parts[3]
}
}
fn match_sb8(state: &SentenceBreaksState, ahead: &str) -> bool {
let &SentenceBreaksState(parts) = state;
let mut idx = if parts[3] == StatePart::SpPlus { 2 } else { 3 };
if parts[idx] == StatePart::ClosePlus {
idx -= 1
}
if parts[idx] == StatePart::ATerm {
use crate::tables::sentence as se;
for next_char in ahead.chars() {
match se::sentence_category(next_char).2 {
se::SC_Lower => return true,
se::SC_OLetter
| se::SC_Upper
| se::SC_Sep
| se::SC_CR
| se::SC_LF
| se::SC_STerm
| se::SC_ATerm => return false,
_ => continue,
}
}
}
false
}
fn match_sb8a(state: &SentenceBreaksState) -> bool {
let &SentenceBreaksState(parts) = state;
let mut idx = if parts[3] == StatePart::SpPlus { 2 } else { 3 };
if parts[idx] == StatePart::ClosePlus {
idx -= 1
}
parts[idx] == StatePart::STerm || parts[idx] == StatePart::ATerm
}
fn match_sb9(state: &SentenceBreaksState) -> bool {
let &SentenceBreaksState(parts) = state;
let idx = if parts[3] == StatePart::ClosePlus {
2
} else {
3
};
parts[idx] == StatePart::STerm || parts[idx] == StatePart::ATerm
}
fn match_sb11(state: &SentenceBreaksState) -> bool {
let &SentenceBreaksState(parts) = state;
let mut idx = match parts[3] {
StatePart::Sep | StatePart::CR | StatePart::LF => 2,
_ => 3,
};
if parts[idx] == StatePart::SpPlus {
idx -= 1
}
if parts[idx] == StatePart::ClosePlus {
idx -= 1
}
parts[idx] == StatePart::STerm || parts[idx] == StatePart::ATerm
}
impl<'a> Iterator for SentenceBreaks<'a> {
type Item = usize;
#[inline]
fn size_hint(&self) -> (usize, Option<usize>) {
let slen = self.string.len();
(cmp::min(slen, 2), Some(slen + 1))
}
#[inline]
fn next(&mut self) -> Option<usize> {
use crate::tables::sentence as se;
for next_char in self.string[self.pos..].chars() {
let position_before = self.pos;
let state_before = self.state.clone();
let next_cat = se::sentence_category(next_char).2;
self.pos += next_char.len_utf8();
self.state = self.state.next(next_cat);
match next_cat {
_ if state_before.match1(StatePart::Sot) => return Some(position_before),
SentenceCat::SC_LF if state_before.match1(StatePart::CR) => continue,
_ if state_before.match1(StatePart::Sep)
|| state_before.match1(StatePart::CR)
|| state_before.match1(StatePart::LF) =>
{
return Some(position_before)
}
SentenceCat::SC_Extend | SentenceCat::SC_Format => self.state = state_before,
SentenceCat::SC_Numeric if state_before.match1(StatePart::ATerm) => continue,
SentenceCat::SC_Upper
if state_before.match2(StatePart::UpperLower, StatePart::ATerm) =>
{
continue
}
_ if match_sb8(&state_before, &self.string[position_before..]) => continue,
SentenceCat::SC_SContinue | SentenceCat::SC_STerm | SentenceCat::SC_ATerm
if match_sb8a(&state_before) =>
{
continue
}
SentenceCat::SC_Close
| SentenceCat::SC_Sp
| SentenceCat::SC_Sep
| SentenceCat::SC_CR
| SentenceCat::SC_LF
if match_sb9(&state_before) =>
{
continue
}
SentenceCat::SC_Sp
| SentenceCat::SC_Sep
| SentenceCat::SC_CR
| SentenceCat::SC_LF
if match_sb8a(&state_before) =>
{
continue
}
_ if match_sb11(&state_before) => return Some(position_before),
_ => continue,
}
}
if self.state.match1(StatePart::Sot) || self.state.match1(StatePart::Eot) {
None
} else {
self.state = self.state.end();
Some(self.pos)
}
}
}
pub fn new_sentence_breaks(source: &str) -> SentenceBreaks<'_> {
SentenceBreaks {
string: source,
pos: 0,
state: INITIAL_STATE,
}
}
}
#[derive(Debug, Clone)]
pub struct UnicodeSentences<'a> {
inner: Filter<USentenceBounds<'a>, fn(&&str) -> bool>,
}
#[derive(Debug, Clone)]
pub struct USentenceBounds<'a> {
iter: fwd::SentenceBreaks<'a>,
sentence_start: Option<usize>,
}
#[derive(Debug, Clone)]
pub struct USentenceBoundIndices<'a> {
start_offset: usize,
iter: USentenceBounds<'a>,
}
#[inline]
pub fn new_sentence_bounds(source: &str) -> USentenceBounds<'_> {
USentenceBounds {
iter: fwd::new_sentence_breaks(source),
sentence_start: None,
}
}
#[inline]
pub fn new_sentence_bound_indices(source: &str) -> USentenceBoundIndices<'_> {
USentenceBoundIndices {
start_offset: source.as_ptr() as usize,
iter: new_sentence_bounds(source),
}
}
#[inline]
pub fn new_unicode_sentences(s: &str) -> UnicodeSentences<'_> {
use super::UnicodeSegmentation;
use crate::tables::util::is_alphanumeric;
fn has_alphanumeric(s: &&str) -> bool {
s.chars().any(is_alphanumeric)
}
let has_alphanumeric: fn(&&str) -> bool = has_alphanumeric;
UnicodeSentences {
inner: s.split_sentence_bounds().filter(has_alphanumeric),
}
}
impl<'a> Iterator for UnicodeSentences<'a> {
type Item = &'a str;
#[inline]
fn next(&mut self) -> Option<&'a str> {
self.inner.next()
}
#[inline]
fn size_hint(&self) -> (usize, Option<usize>) {
self.inner.size_hint()
}
}
impl<'a> Iterator for USentenceBounds<'a> {
type Item = &'a str;
#[inline]
fn size_hint(&self) -> (usize, Option<usize>) {
let (lower, upper) = self.iter.size_hint();
(cmp::max(0, lower - 1), upper.map(|u| cmp::max(0, u - 1)))
}
#[inline]
fn next(&mut self) -> Option<&'a str> {
if self.sentence_start.is_none() {
if let Some(start_pos) = self.iter.next() {
self.sentence_start = Some(start_pos)
} else {
return None;
}
}
if let Some(break_pos) = self.iter.next() {
let start_pos = self.sentence_start.unwrap();
let sentence = &self.iter.string[start_pos..break_pos];
self.sentence_start = Some(break_pos);
Some(sentence)
} else {
None
}
}
}
impl<'a> Iterator for USentenceBoundIndices<'a> {
type Item = (usize, &'a str);
#[inline]
fn next(&mut self) -> Option<(usize, &'a str)> {
self.iter
.next()
.map(|s| (s.as_ptr() as usize - self.start_offset, s))
}
#[inline]
fn size_hint(&self) -> (usize, Option<usize>) {
self.iter.size_hint()
}
}