The base architecture for code-contracts analysis
[mono.git] / mcs / class / Mono.CodeContracts / Mono.CodeContracts.Static.DataStructures / LispListExtensions.cs
1 // 
2 // LispListExtensions.cs
3 // 
4 // Authors:
5 //      Alexander Chebaturkin (chebaturkin@gmail.com)
6 // 
7 // Copyright (C) 2011 Alexander Chebaturkin
8 // 
9 // Permission is hereby granted, free of charge, to any person obtaining
10 // a copy of this software and associated documentation files (the
11 // "Software"), to deal in the Software without restriction, including
12 // without limitation the rights to use, copy, modify, merge, publish,
13 // distribute, sublicense, and/or sell copies of the Software, and to
14 // permit persons to whom the Software is furnished to do so, subject to
15 // the following conditions:
16 // 
17 // The above copyright notice and this permission notice shall be
18 // included in all copies or substantial portions of the Software.
19 //  
20 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
21 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 
22 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
23 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
24 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
25 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
26 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
27 // 
28
29 using System;
30 using System.Collections.Generic;
31
32 namespace Mono.CodeContracts.Static.DataStructures {
33         static class LispListExtensions {
34                 public static LispList<T> Cons<T> (this LispList<T> rest, T elem)
35                 {
36                         return LispList<T>.Cons (elem, rest);
37                 }
38
39                 public static LispList<T> Append<T> (this LispList<T> list, LispList<T> append)
40                 {
41                         if (list == null)
42                                 return append;
43                         if (append == null)
44                                 return list;
45
46                         return Cons (list.Tail.Append (append), list.Head);
47                 }
48
49                 public static LispList<T> Where<T> (this LispList<T> list, Predicate<T> keep)
50                 {
51                         if (list == null)
52                                 return null;
53                         LispList<T> rest = list.Tail.Where (keep);
54                         if (!keep (list.Head))
55                                 return rest;
56
57                         if (rest == list.Tail)
58                                 return list;
59
60                         return Cons (rest, list.Head);
61                 }
62
63                 public static void Apply<T> (this LispList<T> list, Action<T> action)
64                 {
65                         LispList<T>.Apply (list, action);
66                 }
67
68                 public static IEnumerable<T> AsEnumerable<T> (this LispList<T> list)
69                 {
70                         return LispList<T>.PrivateGetEnumerable (list);
71                 }
72
73                 public static bool Any<T> (this LispList<T> list, Predicate<T> predicate)
74                 {
75                         if (list == null)
76                                 return false;
77
78                         if (predicate (list.Head))
79                                 return true;
80
81                         return list.Tail.Any (predicate);
82                 }
83
84                 public static int Length<T> (this LispList<T> list)
85                 {
86                         return LispList<T>.LengthOf (list);
87                 }
88
89                 public static bool IsEmpty<T> (this LispList<T> list)
90                 {
91                         return list == null;
92                 }
93
94                 public static LispList<S> Select<T, S> (this LispList<T> list, Func<T, S> selector)
95                 {
96                         return LispList<T>.Select (list, selector);
97                 }
98
99                 public static T Last<T> (this LispList<T> list)
100                 {
101                         if (list == null)
102                                 return default(T);
103
104                         while (LispList<T>.LengthOf (list) > 1)
105                                 list = list.Tail;
106
107                         return list.Head;
108                 }
109
110                 public static LispList<T> Coerce<S, T> (this LispList<S> list)
111                         where S : T
112                 {
113                         return list.Select (l => (T) l);
114                 }
115         }
116 }