2008-01-15 Stephane Delcroix <sdelcroix@novell.com>
[mono.git] / mcs / class / corlib / System.Collections / CaseInsensitiveHashCodeProvider.cs
1 //
2 // System.Collections.CaseInsensitiveHashCodeProvider.cs
3 //
4 // Authors:
5 //   Sergey Chaban (serge@wildwestsoftware.com)
6 //   Andreas Nahr (ClassDevelopment@A-SoftTech.com)
7 //   Sebastien Pouliot  <sebastien@ximian.com>
8 //
9 // Copyright (C) 2004-2005 Novell, Inc (http://www.novell.com)
10 //
11 // Permission is hereby granted, free of charge, to any person obtaining
12 // a copy of this software and associated documentation files (the
13 // "Software"), to deal in the Software without restriction, including
14 // without limitation the rights to use, copy, modify, merge, publish,
15 // distribute, sublicense, and/or sell copies of the Software, and to
16 // permit persons to whom the Software is furnished to do so, subject to
17 // the following conditions:
18 // 
19 // The above copyright notice and this permission notice shall be
20 // included in all copies or substantial portions of the Software.
21 // 
22 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
23 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
24 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
25 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
26 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
27 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
28 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
29 //
30
31 using System.Globalization;
32 using System.Runtime.InteropServices;
33
34 namespace System.Collections
35 {
36         [Serializable]
37 #if NET_2_0
38         [ComVisible(true)]
39         [Obsolete ("Please use StringComparer instead.")]
40 #endif
41         public class CaseInsensitiveHashCodeProvider : IHashCodeProvider
42         {
43                 static readonly CaseInsensitiveHashCodeProvider singletonInvariant = new CaseInsensitiveHashCodeProvider (
44                         CultureInfo.InvariantCulture);
45                 static CaseInsensitiveHashCodeProvider singleton;
46                 static readonly object sync = new object ();
47
48                 TextInfo m_text; // must match MS name for serialization
49
50                 // Public instance constructor
51                 public CaseInsensitiveHashCodeProvider ()
52                 {
53                         CultureInfo culture = CultureInfo.CurrentCulture;
54                         if (culture.LCID != CultureInfo.InvariantCulture.LCID)
55                                 m_text = CultureInfo.CurrentCulture.TextInfo;
56                 }
57
58                 public CaseInsensitiveHashCodeProvider (CultureInfo culture)
59                 {
60                         if (culture == null)
61                                 throw new ArgumentNullException ("culture");
62                         if (culture.LCID != CultureInfo.InvariantCulture.LCID)
63                                 m_text = culture.TextInfo;
64                 }
65
66                 //
67                 // Public static properties
68                 //
69
70                 public static CaseInsensitiveHashCodeProvider Default {
71                         get {
72                                 // MS actually constructs a new instance on each call, for
73                                 // performance reasons we're only constructing a new instance
74                                 // if the CurrentCulture changes
75                                 lock (sync) {
76                                         if (singleton == null) {
77                                                 singleton = new CaseInsensitiveHashCodeProvider ();
78                                         } else if (singleton.m_text == null) {
79                                                 if (CultureInfo.CurrentCulture.LCID != CultureInfo.InvariantCulture.LCID)
80                                                         singleton = new CaseInsensitiveHashCodeProvider ();
81                                         } else if (singleton.m_text.LCID != CultureInfo.CurrentCulture.LCID) {
82                                                 singleton = new CaseInsensitiveHashCodeProvider ();
83                                         }
84                                         return singleton;
85                                 }
86                         }
87                 }
88
89 #if NET_1_1
90                 public
91 #else
92                 internal
93 #endif
94                 static CaseInsensitiveHashCodeProvider DefaultInvariant {
95                         get {
96                                 return singletonInvariant;
97                         }
98                 }
99
100                 //
101                 // IHashCodeProvider
102                 //
103
104                 public int GetHashCode (object obj)
105                 {
106                         if (obj == null)
107                                 throw new ArgumentNullException ("obj");
108
109                         string str = obj as string;
110
111                         if (str == null)
112                                 return obj.GetHashCode ();
113
114                         int h = 0;
115                         char c;
116
117                         if ((m_text != null) && (m_text.LCID != CultureInfo.InvariantCulture.LCID)) {
118                                 str = m_text.ToLower (str);
119                                 for (int i = 0; i < str.Length; i++) {
120                                         c = str [i];
121                                         h = h * 31 + c;
122                                 }
123                         } else {
124                                 for (int i = 0; i < str.Length; i++) {
125                                         c = Char.ToLowerInvariant (str [i]);
126                                         h = h * 31 + c;
127                                 }
128                         }
129                         return h;
130                 }
131         }
132 }