// Copyright 2012-2013 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. /*! The `ToStr` trait for converting to strings */ use option::{Some, None}; use str::OwnedStr; use hashmap::HashMap; use hashmap::HashSet; use hash::Hash; use iterator::Iterator; use cmp::Eq; use vec::ImmutableVector; /// A generic trait for converting a value to a string pub trait ToStr { /// Converts the value of `self` to an owned string fn to_str(&self) -> ~str; } /// Trait for converting a type to a string, consuming it in the process. pub trait ToStrConsume { /// Consume and convert to a string. fn into_str(self) -> ~str; } impl ToStr for () { #[inline] fn to_str(&self) -> ~str { ~"()" } } impl ToStr for (A,) { #[inline] fn to_str(&self) -> ~str { match *self { (ref a,) => { fmt!("(%s,)", (*a).to_str()) } } } } impl ToStr for HashMap { #[inline] fn to_str(&self) -> ~str { let mut acc = ~"{"; let mut first = true; for (key, value) in self.iter() { if first { first = false; } else { acc.push_str(", "); } acc.push_str(key.to_str()); acc.push_str(": "); acc.push_str(value.to_str()); } acc.push_char('}'); acc } } impl ToStr for HashSet { #[inline] fn to_str(&self) -> ~str { let mut acc = ~"{"; let mut first = true; for element in self.iter() { if first { first = false; } else { acc.push_str(", "); } acc.push_str(element.to_str()); } acc.push_char('}'); acc } } impl ToStr for (A, B) { #[inline] fn to_str(&self) -> ~str { // FIXME(#4653): this causes an llvm assertion //let &(ref a, ref b) = self; match *self { (ref a, ref b) => { fmt!("(%s, %s)", (*a).to_str(), (*b).to_str()) } } } } impl ToStr for (A, B, C) { #[inline] fn to_str(&self) -> ~str { // FIXME(#4653): this causes an llvm assertion //let &(ref a, ref b, ref c) = self; match *self { (ref a, ref b, ref c) => { fmt!("(%s, %s, %s)", (*a).to_str(), (*b).to_str(), (*c).to_str() ) } } } } impl<'self,A:ToStr> ToStr for &'self [A] { #[inline] fn to_str(&self) -> ~str { let mut acc = ~"["; let mut first = true; for elt in self.iter() { if first { first = false; } else { acc.push_str(", "); } acc.push_str(elt.to_str()); } acc.push_char(']'); acc } } impl ToStr for ~[A] { #[inline] fn to_str(&self) -> ~str { let mut acc = ~"["; let mut first = true; for elt in self.iter() { if first { first = false; } else { acc.push_str(", "); } acc.push_str(elt.to_str()); } acc.push_char(']'); acc } } impl ToStr for @[A] { #[inline] fn to_str(&self) -> ~str { let mut acc = ~"["; let mut first = true; for elt in self.iter() { if first { first = false; } else { acc.push_str(", "); } acc.push_str(elt.to_str()); } acc.push_char(']'); acc } } #[cfg(test)] mod tests { use hashmap::HashMap; use hashmap::HashSet; use container::{MutableSet, MutableMap}; use super::*; #[test] fn test_simple_types() { assert_eq!(1i.to_str(), ~"1"); assert_eq!((-1i).to_str(), ~"-1"); assert_eq!(200u.to_str(), ~"200"); assert_eq!(2u8.to_str(), ~"2"); assert_eq!(true.to_str(), ~"true"); assert_eq!(false.to_str(), ~"false"); assert_eq!(().to_str(), ~"()"); assert_eq!((~"hi").to_str(), ~"hi"); assert_eq!((@"hi").to_str(), ~"hi"); } #[test] fn test_tuple_types() { assert_eq!((1, 2).to_str(), ~"(1, 2)"); assert_eq!((~"a", ~"b", false).to_str(), ~"(a, b, false)"); assert_eq!(((), ((), 100)).to_str(), ~"((), ((), 100))"); } #[test] fn test_vectors() { let x: ~[int] = ~[]; assert_eq!(x.to_str(), ~"[]"); assert_eq!((~[1]).to_str(), ~"[1]"); assert_eq!((~[1, 2, 3]).to_str(), ~"[1, 2, 3]"); assert!((~[~[], ~[1], ~[1, 1]]).to_str() == ~"[[], [1], [1, 1]]"); } struct StructWithToStrWithoutEqOrHash { value: int } impl ToStr for StructWithToStrWithoutEqOrHash { fn to_str(&self) -> ~str { fmt!("s%d", self.value) } } #[test] fn test_hashmap() { let mut table: HashMap = HashMap::new(); let empty: HashMap = HashMap::new(); table.insert(3, StructWithToStrWithoutEqOrHash { value: 4 }); table.insert(1, StructWithToStrWithoutEqOrHash { value: 2 }); let table_str = table.to_str(); assert!(table_str == ~"{1: s2, 3: s4}" || table_str == ~"{3: s4, 1: s2}"); assert_eq!(empty.to_str(), ~"{}"); } #[test] fn test_hashset() { let mut set: HashSet = HashSet::new(); let empty_set: HashSet = HashSet::new(); set.insert(1); set.insert(2); let set_str = set.to_str(); assert!(set_str == ~"{1, 2}" || set_str == ~"{2, 1}"); assert_eq!(empty_set.to_str(), ~"{}"); } }