// Copyright 2012 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. // xfail-fast #[legacy_modes]; extern mod std; use std::map::*; enum cat_type { tuxedo, tabby, tortoiseshell } impl cat_type : cmp::Eq { pure fn eq(&self, other: &cat_type) -> bool { ((*self) as uint) == ((*other) as uint) } pure fn ne(&self, other: &cat_type) -> bool { !(*self).eq(other) } } // Very silly -- this just returns the value of the name field // for any int value that's less than the meows field // ok: T should be in scope when resolving the trait ref for map struct cat { // Yes, you can have negative meows priv mut meows : int, mut how_hungry : int, name : T, } impl cat { fn speak() { self.meow(); } fn eat() -> bool { if self.how_hungry > 0 { error!("OM NOM NOM"); self.how_hungry -= 2; return true; } else { error!("Not hungry!"); return false; } } } impl cat : StdMap { pure fn size() -> uint { self.meows as uint } fn insert(+k: int, +_v: T) -> bool { self.meows += k; true } pure fn contains_key(+k: int) -> bool { k <= self.meows } pure fn contains_key_ref(k: &int) -> bool { self.contains_key(*k) } pure fn get(+k:int) -> T { match self.find(k) { Some(v) => { v } None => { die!(~"epic fail"); } } } pure fn find(+k:int) -> Option { if k <= self.meows { Some(self.name) } else { None } } fn update_with_key(+key: int, +val: T, ff: fn(+k: int, +v0: T, +v1: T) -> T) -> bool { match self.find(key) { None => return self.insert(key, val), Some(copy orig) => return self.insert(key, ff(key, orig, val)) } } fn update(+key: int, +val: T, ff: fn(+v0: T, +v1: T) -> T) -> bool { match self.find(key) { None => return self.insert(key, val), Some(copy orig) => return self.insert(key, ff(orig, val)) } } fn remove(+k:int) -> bool { match self.find(k) { Some(x) => { self.meows -= k; true } None => { false } } } pure fn each(f: fn(+v: int, +v: T) -> bool) { let mut n = int::abs(self.meows); while n > 0 { if !f(n, self.name) { break; } n -= 1; } } pure fn each_key(&&f: fn(+v: int) -> bool) { for self.each |k, _v| { if !f(k) { break; } loop;}; } pure fn each_value(&&f: fn(+v: T) -> bool) { for self.each |_k, v| { if !f(v) { break; } loop;}; } pure fn each_ref(f: fn(k: &int, v: &T) -> bool) {} pure fn each_key_ref(f: fn(k: &int) -> bool) {} pure fn each_value_ref(f: fn(k: &T) -> bool) {} fn clear() { } } priv impl cat { fn meow() { self.meows += 1; error!("Meow %d", self.meows); if self.meows % 5 == 0 { self.how_hungry += 1; } } } fn cat(in_x : int, in_y : int, in_name: T) -> cat { cat { meows: in_x, how_hungry: in_y, name: in_name } } pub fn main() { let nyan : cat<~str> = cat(0, 2, ~"nyan"); for uint::range(1u, 5u) |_i| { nyan.speak(); } assert(nyan.find(1) == Some(~"nyan")); assert(nyan.find(10) == None); let spotty : cat = cat(2, 57, tuxedo); for uint::range(0u, 6u) |_i| { spotty.speak(); } assert(spotty.size() == 8u); assert(spotty.contains_key(2)); assert(spotty.get(3) == tuxedo); }