// Copyright 2012-2014 The Rust Project Developers. See the COPYRIGHT // file at the top-level directory of this distribution and at // http://rust-lang.org/COPYRIGHT. // // Licensed under the Apache License, Version 2.0 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. // // ignore-lexer-test FIXME #15679 //! Unicode-intensive string manipulations. //! //! This module provides functionality to `str` that requires the Unicode methods provided by the //! unicode parts of the CharExt trait. use self::GraphemeState::*; use core::prelude::*; use core::char; use core::cmp; use core::iter::{Filter, AdditiveIterator}; use core::mem; use core::num::Int; use core::slice; use core::str::Split; use u_char::CharExt as UCharExt; // conflicts with core::prelude::CharExt use tables::grapheme::GraphemeCat; /// An iterator over the words of a string, separated by a sequence of whitespace #[stable] pub struct Words<'a> { inner: Filter<&'a str, Split<'a, fn(char) -> bool>, fn(&&str) -> bool>, } /// Methods for Unicode string slices #[allow(missing_docs)] // docs in libcollections pub trait UnicodeStr { fn graphemes<'a>(&'a self, is_extended: bool) -> Graphemes<'a>; fn grapheme_indices<'a>(&'a self, is_extended: bool) -> GraphemeIndices<'a>; fn words<'a>(&'a self) -> Words<'a>; fn is_whitespace(&self) -> bool; fn is_alphanumeric(&self) -> bool; fn width(&self, is_cjk: bool) -> uint; fn trim<'a>(&'a self) -> &'a str; fn trim_left<'a>(&'a self) -> &'a str; fn trim_right<'a>(&'a self) -> &'a str; } impl UnicodeStr for str { #[inline] fn graphemes(&self, is_extended: bool) -> Graphemes { Graphemes { string: self, extended: is_extended, cat: None, catb: None } } #[inline] fn grapheme_indices(&self, is_extended: bool) -> GraphemeIndices { GraphemeIndices { start_offset: self.as_ptr() as uint, iter: self.graphemes(is_extended) } } #[inline] fn words(&self) -> Words { fn is_not_empty(s: &&str) -> bool { !s.is_empty() } let is_not_empty: fn(&&str) -> bool = is_not_empty; // coerce to fn pointer fn is_whitespace(c: char) -> bool { c.is_whitespace() } let is_whitespace: fn(char) -> bool = is_whitespace; // coerce to fn pointer Words { inner: self.split(is_whitespace).filter(is_not_empty) } } #[inline] fn is_whitespace(&self) -> bool { self.chars().all(|c| c.is_whitespace()) } #[inline] fn is_alphanumeric(&self) -> bool { self.chars().all(|c| c.is_alphanumeric()) } #[inline] fn width(&self, is_cjk: bool) -> uint { self.chars().map(|c| c.width(is_cjk).unwrap_or(0)).sum() } #[inline] fn trim(&self) -> &str { self.trim_left().trim_right() } #[inline] fn trim_left(&self) -> &str { self.trim_left_matches(|&: c: char| c.is_whitespace()) } #[inline] fn trim_right(&self) -> &str { self.trim_right_matches(|&: c: char| c.is_whitespace()) } } /// External iterator for grapheme clusters and byte offsets. #[derive(Clone)] pub struct GraphemeIndices<'a> { start_offset: uint, iter: Graphemes<'a>, } impl<'a> Iterator for GraphemeIndices<'a> { type Item = (uint, &'a str); #[inline] fn next(&mut self) -> Option<(uint, &'a str)> { self.iter.next().map(|s| (s.as_ptr() as uint - self.start_offset, s)) } #[inline] fn size_hint(&self) -> (uint, Option) { self.iter.size_hint() } } impl<'a> DoubleEndedIterator for GraphemeIndices<'a> { #[inline] fn next_back(&mut self) -> Option<(uint, &'a str)> { self.iter.next_back().map(|s| (s.as_ptr() as uint - self.start_offset, s)) } } /// External iterator for a string's /// [grapheme clusters](http://www.unicode.org/reports/tr29/#Grapheme_Cluster_Boundaries). #[derive(Clone)] pub struct Graphemes<'a> { string: &'a str, extended: bool, cat: Option, catb: Option, } // state machine for cluster boundary rules #[derive(PartialEq,Eq)] enum GraphemeState { Start, FindExtend, HangulL, HangulLV, HangulLVT, Regional, } impl<'a> Iterator for Graphemes<'a> { type Item = &'a str; #[inline] fn size_hint(&self) -> (uint, Option) { let slen = self.string.len(); (cmp::min(slen, 1u), Some(slen)) } #[inline] fn next(&mut self) -> Option<&'a str> { use tables::grapheme as gr; if self.string.len() == 0 { return None; } let mut take_curr = true; let mut idx = 0; let mut state = Start; let mut cat = gr::GC_Any; for (curr, ch) in self.string.char_indices() { idx = curr; // retrieve cached category, if any // We do this because most of the time we would end up // looking up each character twice. cat = match self.cat { None => gr::grapheme_category(ch), _ => self.cat.take().unwrap() }; if match cat { gr::GC_Extend => true, gr::GC_SpacingMark if self.extended => true, _ => false } { state = FindExtend; // rule GB9/GB9a continue; } state = match state { Start if '\r' == ch => { let slen = self.string.len(); let nidx = idx + 1; if nidx != slen && self.string.char_at(nidx) == '\n' { idx = nidx; // rule GB3 } break; // rule GB4 } Start => match cat { gr::GC_Control => break, gr::GC_L => HangulL, gr::GC_LV | gr::GC_V => HangulLV, gr::GC_LVT | gr::GC_T => HangulLVT, gr::GC_RegionalIndicator => Regional, _ => FindExtend }, FindExtend => { // found non-extending when looking for extending take_curr = false; break; }, HangulL => match cat { // rule GB6: L x (L|V|LV|LVT) gr::GC_L => continue, gr::GC_LV | gr::GC_V => HangulLV, gr::GC_LVT => HangulLVT, _ => { take_curr = false; break; } }, HangulLV => match cat { // rule GB7: (LV|V) x (V|T) gr::GC_V => continue, gr::GC_T => HangulLVT, _ => { take_curr = false; break; } }, HangulLVT => match cat { // rule GB8: (LVT|T) x T gr::GC_T => continue, _ => { take_curr = false; break; } }, Regional => match cat { // rule GB8a gr::GC_RegionalIndicator => continue, _ => { take_curr = false; break; } } } } self.cat = if take_curr { idx = self.string.char_range_at(idx).next; None } else { Some(cat) }; let retstr = self.string.slice_to(idx); self.string = self.string.slice_from(idx); Some(retstr) } } impl<'a> DoubleEndedIterator for Graphemes<'a> { #[inline] fn next_back(&mut self) -> Option<&'a str> { use tables::grapheme as gr; if self.string.len() == 0 { return None; } let mut take_curr = true; let mut idx = self.string.len(); let mut previdx = idx; let mut state = Start; let mut cat = gr::GC_Any; for (curr, ch) in self.string.char_indices().rev() { previdx = idx; idx = curr; // cached category, if any cat = match self.catb { None => gr::grapheme_category(ch), _ => self.catb.take().unwrap() }; // a matching state machine that runs *backwards* across an input string // note that this has some implications for the Hangul matching, since // we now need to know what the rightward letter is: // // Right to left, we have: // L x L // V x (L|V|LV) // T x (V|T|LV|LVT) // HangulL means the letter to the right is L // HangulLV means the letter to the right is V // HangulLVT means the letter to the right is T state = match state { Start if '\n' == ch => { if idx > 0 && '\r' == self.string.char_at_reverse(idx) { idx -= 1; // rule GB3 } break; // rule GB4 }, Start | FindExtend => match cat { gr::GC_Extend => FindExtend, gr::GC_SpacingMark if self.extended => FindExtend, gr::GC_L | gr::GC_LV | gr::GC_LVT => HangulL, gr::GC_V => HangulLV, gr::GC_T => HangulLVT, gr::GC_RegionalIndicator => Regional, gr::GC_Control => { take_curr = Start == state; break; }, _ => break }, HangulL => match cat { // char to right is an L gr::GC_L => continue, // L x L is the only legal match _ => { take_curr = false; break; } }, HangulLV => match cat { // char to right is a V gr::GC_V => continue, // V x V, right char is still V gr::GC_L | gr::GC_LV => HangulL, // (L|V) x V, right char is now L _ => { take_curr = false; break; } }, HangulLVT => match cat { // char to right is a T gr::GC_T => continue, // T x T, right char is still T gr::GC_V => HangulLV, // V x T, right char is now V gr::GC_LV | gr::GC_LVT => HangulL, // (LV|LVT) x T, right char is now L _ => { take_curr = false; break; } }, Regional => match cat { // rule GB8a gr::GC_RegionalIndicator => continue, _ => { take_curr = false; break; } } } } self.catb = if take_curr { None } else { idx = previdx; Some(cat) }; let retstr = self.string.slice_from(idx); self.string = self.string.slice_to(idx); Some(retstr) } } // https://tools.ietf.org/html/rfc3629 static UTF8_CHAR_WIDTH: [u8; 256] = [ 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, // 0x1F 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, // 0x3F 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, // 0x5F 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, // 0x7F 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, // 0x9F 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, // 0xBF 0,0,2,2,2,2,2,2,2,2,2,2,2,2,2,2, 2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2, // 0xDF 3,3,3,3,3,3,3,3,3,3,3,3,3,3,3,3, // 0xEF 4,4,4,4,4,0,0,0,0,0,0,0,0,0,0,0, // 0xFF ]; /// Given a first byte, determine how many bytes are in this UTF-8 character #[inline] pub fn utf8_char_width(b: u8) -> uint { return UTF8_CHAR_WIDTH[b as uint] as uint; } /// Determines if a vector of `u16` contains valid UTF-16 pub fn is_utf16(v: &[u16]) -> bool { let mut it = v.iter(); macro_rules! next { ($ret:expr) => { match it.next() { Some(u) => *u, None => return $ret } } } loop { let u = next!(true); match char::from_u32(u as u32) { Some(_) => {} None => { let u2 = next!(false); if u < 0xD7FF || u > 0xDBFF || u2 < 0xDC00 || u2 > 0xDFFF { return false; } } } } } /// An iterator that decodes UTF-16 encoded codepoints from a vector /// of `u16`s. #[derive(Clone)] pub struct Utf16Items<'a> { iter: slice::Iter<'a, u16> } /// The possibilities for values decoded from a `u16` stream. #[derive(PartialEq, Eq, Clone, Show)] pub enum Utf16Item { /// A valid codepoint. ScalarValue(char), /// An invalid surrogate without its pair. LoneSurrogate(u16) } impl Copy for Utf16Item {} impl Utf16Item { /// Convert `self` to a `char`, taking `LoneSurrogate`s to the /// replacement character (U+FFFD). #[inline] pub fn to_char_lossy(&self) -> char { match *self { Utf16Item::ScalarValue(c) => c, Utf16Item::LoneSurrogate(_) => '\u{FFFD}' } } } impl<'a> Iterator for Utf16Items<'a> { type Item = Utf16Item; fn next(&mut self) -> Option { let u = match self.iter.next() { Some(u) => *u, None => return None }; if u < 0xD800 || 0xDFFF < u { // not a surrogate Some(Utf16Item::ScalarValue(unsafe {mem::transmute(u as u32)})) } else if u >= 0xDC00 { // a trailing surrogate Some(Utf16Item::LoneSurrogate(u)) } else { // preserve state for rewinding. let old = self.iter; let u2 = match self.iter.next() { Some(u2) => *u2, // eof None => return Some(Utf16Item::LoneSurrogate(u)) }; if u2 < 0xDC00 || u2 > 0xDFFF { // not a trailing surrogate so we're not a valid // surrogate pair, so rewind to redecode u2 next time. self.iter = old; return Some(Utf16Item::LoneSurrogate(u)) } // all ok, so lets decode it. let c = (((u - 0xD800) as u32) << 10 | (u2 - 0xDC00) as u32) + 0x1_0000; Some(Utf16Item::ScalarValue(unsafe {mem::transmute(c)})) } } #[inline] fn size_hint(&self) -> (uint, Option) { let (low, high) = self.iter.size_hint(); // we could be entirely valid surrogates (2 elements per // char), or entirely non-surrogates (1 element per char) (low / 2, high) } } /// Create an iterator over the UTF-16 encoded codepoints in `v`, /// returning invalid surrogates as `LoneSurrogate`s. /// /// # Example /// /// ```rust /// use unicode::str::Utf16Item::{ScalarValue, LoneSurrogate}; /// /// // 𝄞music /// let v = [0xD834, 0xDD1E, 0x006d, 0x0075, /// 0x0073, 0xDD1E, 0x0069, 0x0063, /// 0xD834]; /// /// assert_eq!(unicode::str::utf16_items(&v).collect::>(), /// vec![ScalarValue('𝄞'), /// ScalarValue('m'), ScalarValue('u'), ScalarValue('s'), /// LoneSurrogate(0xDD1E), /// ScalarValue('i'), ScalarValue('c'), /// LoneSurrogate(0xD834)]); /// ``` pub fn utf16_items<'a>(v: &'a [u16]) -> Utf16Items<'a> { Utf16Items { iter : v.iter() } } /// Iterator adaptor for encoding `char`s to UTF-16. #[derive(Clone)] pub struct Utf16Encoder { chars: I, extra: u16 } impl Utf16Encoder { /// Create an UTF-16 encoder from any `char` iterator. pub fn new(chars: I) -> Utf16Encoder where I: Iterator { Utf16Encoder { chars: chars, extra: 0 } } } impl Iterator for Utf16Encoder where I: Iterator { type Item = u16; #[inline] fn next(&mut self) -> Option { if self.extra != 0 { let tmp = self.extra; self.extra = 0; return Some(tmp); } let mut buf = [0u16; 2]; self.chars.next().map(|ch| { let n = CharExt::encode_utf16(ch, buf.as_mut_slice()).unwrap_or(0); if n == 2 { self.extra = buf[1]; } buf[0] }) } #[inline] fn size_hint(&self) -> (uint, Option) { let (low, high) = self.chars.size_hint(); // every char gets either one u16 or two u16, // so this iterator is between 1 or 2 times as // long as the underlying iterator. (low, high.and_then(|n| n.checked_mul(2))) } } impl<'a> Iterator for Words<'a> { type Item = &'a str; fn next(&mut self) -> Option<&'a str> { self.inner.next() } } impl<'a> DoubleEndedIterator for Words<'a> { fn next_back(&mut self) -> Option<&'a str> { self.inner.next_back() } }