2 // Copied from System.Web.HttpUtility and marked internal
5 // Patrik Torstensson (Patrik.Torstensson@labs2.com)
6 // Wictor Wilén (decode/encode functions) (wictor@ibizkit.se)
7 // Tim Coleman (tim@timcoleman.com)
8 // Gonzalo Paniagua Javier (gonzalo@ximian.com)
9 // Marek Safar (marek.safar@gmail.com)
11 // Copyright (C) 2005-2009 Novell, Inc (http://www.novell.com)
12 // Copyright (C) 2014 Xamarin Inc (http://www.xamarin.com)
14 // Permission is hereby granted, free of charge, to any person obtaining
15 // a copy of this software and associated documentation files (the
16 // "Software"), to deal in the Software without restriction, including
17 // without limitation the rights to use, copy, modify, merge, publish,
18 // distribute, sublicense, and/or sell copies of the Software, and to
19 // permit persons to whom the Software is furnished to do so, subject to
20 // the following conditions:
22 // The above copyright notice and this permission notice shall be
23 // included in all copies or substantial portions of the Software.
25 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
26 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
27 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
28 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
29 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
30 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
31 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
34 using System.Collections;
35 using System.Collections.Generic;
36 using System.Collections.Specialized;
37 using System.Globalization;
39 using System.Security.Permissions;
42 namespace System.Net {
44 static class HttpUtility
46 sealed class HttpQSCollection : NameValueCollection
48 public override string ToString ()
53 StringBuilder sb = new StringBuilder ();
54 string [] keys = AllKeys;
55 for (int i = 0; i < count; i++) {
56 sb.AppendFormat ("{0}={1}&", keys [i], this [keys [i]]);
60 return sb.ToString ();
65 static readonly long[] entities = new long[] {
66 (long)'A' << 56 | (long)'E' << 48 | (long)'l' << 40 | (long)'i' << 32 | (long)'g' << 24,
67 (long)'A' << 56 | (long)'a' << 48 | (long)'c' << 40 | (long)'u' << 32 | (long)'t' << 24 | (long)'e' << 16,
68 (long)'A' << 56 | (long)'c' << 48 | (long)'i' << 40 | (long)'r' << 32 | (long)'c' << 24,
69 (long)'A' << 56 | (long)'g' << 48 | (long)'r' << 40 | (long)'a' << 32 | (long)'v' << 24 | (long)'e' << 16,
70 (long)'A' << 56 | (long)'l' << 48 | (long)'p' << 40 | (long)'h' << 32 | (long)'a' << 24,
71 (long)'A' << 56 | (long)'r' << 48 | (long)'i' << 40 | (long)'n' << 32 | (long)'g' << 24,
72 (long)'A' << 56 | (long)'t' << 48 | (long)'i' << 40 | (long)'l' << 32 | (long)'d' << 24 | (long)'e' << 16,
73 (long)'A' << 56 | (long)'u' << 48 | (long)'m' << 40 | (long)'l' << 32,
74 (long)'B' << 56 | (long)'e' << 48 | (long)'t' << 40 | (long)'a' << 32,
75 (long)'C' << 56 | (long)'c' << 48 | (long)'e' << 40 | (long)'d' << 32 | (long)'i' << 24 | (long)'l' << 16,
76 (long)'C' << 56 | (long)'h' << 48 | (long)'i' << 40,
77 (long)'D' << 56 | (long)'a' << 48 | (long)'g' << 40 | (long)'g' << 32 | (long)'e' << 24 | (long)'r' << 16,
78 (long)'D' << 56 | (long)'e' << 48 | (long)'l' << 40 | (long)'t' << 32 | (long)'a' << 24,
79 (long)'E' << 56 | (long)'T' << 48 | (long)'H' << 40,
80 (long)'E' << 56 | (long)'a' << 48 | (long)'c' << 40 | (long)'u' << 32 | (long)'t' << 24 | (long)'e' << 16,
81 (long)'E' << 56 | (long)'c' << 48 | (long)'i' << 40 | (long)'r' << 32 | (long)'c' << 24,
82 (long)'E' << 56 | (long)'g' << 48 | (long)'r' << 40 | (long)'a' << 32 | (long)'v' << 24 | (long)'e' << 16,
83 (long)'E' << 56 | (long)'p' << 48 | (long)'s' << 40 | (long)'i' << 32 | (long)'l' << 24 | (long)'o' << 16 | (long)'n' << 8,
84 (long)'E' << 56 | (long)'t' << 48 | (long)'a' << 40,
85 (long)'E' << 56 | (long)'u' << 48 | (long)'m' << 40 | (long)'l' << 32,
86 (long)'G' << 56 | (long)'a' << 48 | (long)'m' << 40 | (long)'m' << 32 | (long)'a' << 24,
87 (long)'I' << 56 | (long)'a' << 48 | (long)'c' << 40 | (long)'u' << 32 | (long)'t' << 24 | (long)'e' << 16,
88 (long)'I' << 56 | (long)'c' << 48 | (long)'i' << 40 | (long)'r' << 32 | (long)'c' << 24,
89 (long)'I' << 56 | (long)'g' << 48 | (long)'r' << 40 | (long)'a' << 32 | (long)'v' << 24 | (long)'e' << 16,
90 (long)'I' << 56 | (long)'o' << 48 | (long)'t' << 40 | (long)'a' << 32,
91 (long)'I' << 56 | (long)'u' << 48 | (long)'m' << 40 | (long)'l' << 32,
92 (long)'K' << 56 | (long)'a' << 48 | (long)'p' << 40 | (long)'p' << 32 | (long)'a' << 24,
93 (long)'L' << 56 | (long)'a' << 48 | (long)'m' << 40 | (long)'b' << 32 | (long)'d' << 24 | (long)'a' << 16,
94 (long)'M' << 56 | (long)'u' << 48,
95 (long)'N' << 56 | (long)'t' << 48 | (long)'i' << 40 | (long)'l' << 32 | (long)'d' << 24 | (long)'e' << 16,
96 (long)'N' << 56 | (long)'u' << 48,
97 (long)'O' << 56 | (long)'E' << 48 | (long)'l' << 40 | (long)'i' << 32 | (long)'g' << 24,
98 (long)'O' << 56 | (long)'a' << 48 | (long)'c' << 40 | (long)'u' << 32 | (long)'t' << 24 | (long)'e' << 16,
99 (long)'O' << 56 | (long)'c' << 48 | (long)'i' << 40 | (long)'r' << 32 | (long)'c' << 24,
100 (long)'O' << 56 | (long)'g' << 48 | (long)'r' << 40 | (long)'a' << 32 | (long)'v' << 24 | (long)'e' << 16,
101 (long)'O' << 56 | (long)'m' << 48 | (long)'e' << 40 | (long)'g' << 32 | (long)'a' << 24,
102 (long)'O' << 56 | (long)'m' << 48 | (long)'i' << 40 | (long)'c' << 32 | (long)'r' << 24 | (long)'o' << 16 | (long)'n' << 8,
103 (long)'O' << 56 | (long)'s' << 48 | (long)'l' << 40 | (long)'a' << 32 | (long)'s' << 24 | (long)'h' << 16,
104 (long)'O' << 56 | (long)'t' << 48 | (long)'i' << 40 | (long)'l' << 32 | (long)'d' << 24 | (long)'e' << 16,
105 (long)'O' << 56 | (long)'u' << 48 | (long)'m' << 40 | (long)'l' << 32,
106 (long)'P' << 56 | (long)'h' << 48 | (long)'i' << 40,
107 (long)'P' << 56 | (long)'i' << 48,
108 (long)'P' << 56 | (long)'r' << 48 | (long)'i' << 40 | (long)'m' << 32 | (long)'e' << 24,
109 (long)'P' << 56 | (long)'s' << 48 | (long)'i' << 40,
110 (long)'R' << 56 | (long)'h' << 48 | (long)'o' << 40,
111 (long)'S' << 56 | (long)'c' << 48 | (long)'a' << 40 | (long)'r' << 32 | (long)'o' << 24 | (long)'n' << 16,
112 (long)'S' << 56 | (long)'i' << 48 | (long)'g' << 40 | (long)'m' << 32 | (long)'a' << 24,
113 (long)'T' << 56 | (long)'H' << 48 | (long)'O' << 40 | (long)'R' << 32 | (long)'N' << 24,
114 (long)'T' << 56 | (long)'a' << 48 | (long)'u' << 40,
115 (long)'T' << 56 | (long)'h' << 48 | (long)'e' << 40 | (long)'t' << 32 | (long)'a' << 24,
116 (long)'U' << 56 | (long)'a' << 48 | (long)'c' << 40 | (long)'u' << 32 | (long)'t' << 24 | (long)'e' << 16,
117 (long)'U' << 56 | (long)'c' << 48 | (long)'i' << 40 | (long)'r' << 32 | (long)'c' << 24,
118 (long)'U' << 56 | (long)'g' << 48 | (long)'r' << 40 | (long)'a' << 32 | (long)'v' << 24 | (long)'e' << 16,
119 (long)'U' << 56 | (long)'p' << 48 | (long)'s' << 40 | (long)'i' << 32 | (long)'l' << 24 | (long)'o' << 16 | (long)'n' << 8,
120 (long)'U' << 56 | (long)'u' << 48 | (long)'m' << 40 | (long)'l' << 32,
121 (long)'X' << 56 | (long)'i' << 48,
122 (long)'Y' << 56 | (long)'a' << 48 | (long)'c' << 40 | (long)'u' << 32 | (long)'t' << 24 | (long)'e' << 16,
123 (long)'Y' << 56 | (long)'u' << 48 | (long)'m' << 40 | (long)'l' << 32,
124 (long)'Z' << 56 | (long)'e' << 48 | (long)'t' << 40 | (long)'a' << 32,
125 (long)'a' << 56 | (long)'a' << 48 | (long)'c' << 40 | (long)'u' << 32 | (long)'t' << 24 | (long)'e' << 16,
126 (long)'a' << 56 | (long)'c' << 48 | (long)'i' << 40 | (long)'r' << 32 | (long)'c' << 24,
127 (long)'a' << 56 | (long)'c' << 48 | (long)'u' << 40 | (long)'t' << 32 | (long)'e' << 24,
128 (long)'a' << 56 | (long)'e' << 48 | (long)'l' << 40 | (long)'i' << 32 | (long)'g' << 24,
129 (long)'a' << 56 | (long)'g' << 48 | (long)'r' << 40 | (long)'a' << 32 | (long)'v' << 24 | (long)'e' << 16,
130 (long)'a' << 56 | (long)'l' << 48 | (long)'e' << 40 | (long)'f' << 32 | (long)'s' << 24 | (long)'y' << 16 | (long)'m' << 8,
131 (long)'a' << 56 | (long)'l' << 48 | (long)'p' << 40 | (long)'h' << 32 | (long)'a' << 24,
132 (long)'a' << 56 | (long)'m' << 48 | (long)'p' << 40,
133 (long)'a' << 56 | (long)'n' << 48 | (long)'d' << 40,
134 (long)'a' << 56 | (long)'n' << 48 | (long)'g' << 40,
135 (long)'a' << 56 | (long)'p' << 48 | (long)'o' << 40 | (long)'s' << 32,
136 (long)'a' << 56 | (long)'r' << 48 | (long)'i' << 40 | (long)'n' << 32 | (long)'g' << 24,
137 (long)'a' << 56 | (long)'s' << 48 | (long)'y' << 40 | (long)'m' << 32 | (long)'p' << 24,
138 (long)'a' << 56 | (long)'t' << 48 | (long)'i' << 40 | (long)'l' << 32 | (long)'d' << 24 | (long)'e' << 16,
139 (long)'a' << 56 | (long)'u' << 48 | (long)'m' << 40 | (long)'l' << 32,
140 (long)'b' << 56 | (long)'d' << 48 | (long)'q' << 40 | (long)'u' << 32 | (long)'o' << 24,
141 (long)'b' << 56 | (long)'e' << 48 | (long)'t' << 40 | (long)'a' << 32,
142 (long)'b' << 56 | (long)'r' << 48 | (long)'v' << 40 | (long)'b' << 32 | (long)'a' << 24 | (long)'r' << 16,
143 (long)'b' << 56 | (long)'u' << 48 | (long)'l' << 40 | (long)'l' << 32,
144 (long)'c' << 56 | (long)'a' << 48 | (long)'p' << 40,
145 (long)'c' << 56 | (long)'c' << 48 | (long)'e' << 40 | (long)'d' << 32 | (long)'i' << 24 | (long)'l' << 16,
146 (long)'c' << 56 | (long)'e' << 48 | (long)'d' << 40 | (long)'i' << 32 | (long)'l' << 24,
147 (long)'c' << 56 | (long)'e' << 48 | (long)'n' << 40 | (long)'t' << 32,
148 (long)'c' << 56 | (long)'h' << 48 | (long)'i' << 40,
149 (long)'c' << 56 | (long)'i' << 48 | (long)'r' << 40 | (long)'c' << 32,
150 (long)'c' << 56 | (long)'l' << 48 | (long)'u' << 40 | (long)'b' << 32 | (long)'s' << 24,
151 (long)'c' << 56 | (long)'o' << 48 | (long)'n' << 40 | (long)'g' << 32,
152 (long)'c' << 56 | (long)'o' << 48 | (long)'p' << 40 | (long)'y' << 32,
153 (long)'c' << 56 | (long)'r' << 48 | (long)'a' << 40 | (long)'r' << 32 | (long)'r' << 24,
154 (long)'c' << 56 | (long)'u' << 48 | (long)'p' << 40,
155 (long)'c' << 56 | (long)'u' << 48 | (long)'r' << 40 | (long)'r' << 32 | (long)'e' << 24 | (long)'n' << 16,
156 (long)'d' << 56 | (long)'A' << 48 | (long)'r' << 40 | (long)'r' << 32,
157 (long)'d' << 56 | (long)'a' << 48 | (long)'g' << 40 | (long)'g' << 32 | (long)'e' << 24 | (long)'r' << 16,
158 (long)'d' << 56 | (long)'a' << 48 | (long)'r' << 40 | (long)'r' << 32,
159 (long)'d' << 56 | (long)'e' << 48 | (long)'g' << 40,
160 (long)'d' << 56 | (long)'e' << 48 | (long)'l' << 40 | (long)'t' << 32 | (long)'a' << 24,
161 (long)'d' << 56 | (long)'i' << 48 | (long)'a' << 40 | (long)'m' << 32 | (long)'s' << 24,
162 (long)'d' << 56 | (long)'i' << 48 | (long)'v' << 40 | (long)'i' << 32 | (long)'d' << 24 | (long)'e' << 16,
163 (long)'e' << 56 | (long)'a' << 48 | (long)'c' << 40 | (long)'u' << 32 | (long)'t' << 24 | (long)'e' << 16,
164 (long)'e' << 56 | (long)'c' << 48 | (long)'i' << 40 | (long)'r' << 32 | (long)'c' << 24,
165 (long)'e' << 56 | (long)'g' << 48 | (long)'r' << 40 | (long)'a' << 32 | (long)'v' << 24 | (long)'e' << 16,
166 (long)'e' << 56 | (long)'m' << 48 | (long)'p' << 40 | (long)'t' << 32 | (long)'y' << 24,
167 (long)'e' << 56 | (long)'m' << 48 | (long)'s' << 40 | (long)'p' << 32,
168 (long)'e' << 56 | (long)'n' << 48 | (long)'s' << 40 | (long)'p' << 32,
169 (long)'e' << 56 | (long)'p' << 48 | (long)'s' << 40 | (long)'i' << 32 | (long)'l' << 24 | (long)'o' << 16 | (long)'n' << 8,
170 (long)'e' << 56 | (long)'q' << 48 | (long)'u' << 40 | (long)'i' << 32 | (long)'v' << 24,
171 (long)'e' << 56 | (long)'t' << 48 | (long)'a' << 40,
172 (long)'e' << 56 | (long)'t' << 48 | (long)'h' << 40,
173 (long)'e' << 56 | (long)'u' << 48 | (long)'m' << 40 | (long)'l' << 32,
174 (long)'e' << 56 | (long)'u' << 48 | (long)'r' << 40 | (long)'o' << 32,
175 (long)'e' << 56 | (long)'x' << 48 | (long)'i' << 40 | (long)'s' << 32 | (long)'t' << 24,
176 (long)'f' << 56 | (long)'n' << 48 | (long)'o' << 40 | (long)'f' << 32,
177 (long)'f' << 56 | (long)'o' << 48 | (long)'r' << 40 | (long)'a' << 32 | (long)'l' << 24 | (long)'l' << 16,
178 (long)'f' << 56 | (long)'r' << 48 | (long)'a' << 40 | (long)'c' << 32 | (long)'1' << 24 | (long)'2' << 16,
179 (long)'f' << 56 | (long)'r' << 48 | (long)'a' << 40 | (long)'c' << 32 | (long)'1' << 24 | (long)'4' << 16,
180 (long)'f' << 56 | (long)'r' << 48 | (long)'a' << 40 | (long)'c' << 32 | (long)'3' << 24 | (long)'4' << 16,
181 (long)'f' << 56 | (long)'r' << 48 | (long)'a' << 40 | (long)'s' << 32 | (long)'l' << 24,
182 (long)'g' << 56 | (long)'a' << 48 | (long)'m' << 40 | (long)'m' << 32 | (long)'a' << 24,
183 (long)'g' << 56 | (long)'e' << 48,
184 (long)'g' << 56 | (long)'t' << 48,
185 (long)'h' << 56 | (long)'A' << 48 | (long)'r' << 40 | (long)'r' << 32,
186 (long)'h' << 56 | (long)'a' << 48 | (long)'r' << 40 | (long)'r' << 32,
187 (long)'h' << 56 | (long)'e' << 48 | (long)'a' << 40 | (long)'r' << 32 | (long)'t' << 24 | (long)'s' << 16,
188 (long)'h' << 56 | (long)'e' << 48 | (long)'l' << 40 | (long)'l' << 32 | (long)'i' << 24 | (long)'p' << 16,
189 (long)'i' << 56 | (long)'a' << 48 | (long)'c' << 40 | (long)'u' << 32 | (long)'t' << 24 | (long)'e' << 16,
190 (long)'i' << 56 | (long)'c' << 48 | (long)'i' << 40 | (long)'r' << 32 | (long)'c' << 24,
191 (long)'i' << 56 | (long)'e' << 48 | (long)'x' << 40 | (long)'c' << 32 | (long)'l' << 24,
192 (long)'i' << 56 | (long)'g' << 48 | (long)'r' << 40 | (long)'a' << 32 | (long)'v' << 24 | (long)'e' << 16,
193 (long)'i' << 56 | (long)'m' << 48 | (long)'a' << 40 | (long)'g' << 32 | (long)'e' << 24,
194 (long)'i' << 56 | (long)'n' << 48 | (long)'f' << 40 | (long)'i' << 32 | (long)'n' << 24,
195 (long)'i' << 56 | (long)'n' << 48 | (long)'t' << 40,
196 (long)'i' << 56 | (long)'o' << 48 | (long)'t' << 40 | (long)'a' << 32,
197 (long)'i' << 56 | (long)'q' << 48 | (long)'u' << 40 | (long)'e' << 32 | (long)'s' << 24 | (long)'t' << 16,
198 (long)'i' << 56 | (long)'s' << 48 | (long)'i' << 40 | (long)'n' << 32,
199 (long)'i' << 56 | (long)'u' << 48 | (long)'m' << 40 | (long)'l' << 32,
200 (long)'k' << 56 | (long)'a' << 48 | (long)'p' << 40 | (long)'p' << 32 | (long)'a' << 24,
201 (long)'l' << 56 | (long)'A' << 48 | (long)'r' << 40 | (long)'r' << 32,
202 (long)'l' << 56 | (long)'a' << 48 | (long)'m' << 40 | (long)'b' << 32 | (long)'d' << 24 | (long)'a' << 16,
203 (long)'l' << 56 | (long)'a' << 48 | (long)'n' << 40 | (long)'g' << 32,
204 (long)'l' << 56 | (long)'a' << 48 | (long)'q' << 40 | (long)'u' << 32 | (long)'o' << 24,
205 (long)'l' << 56 | (long)'a' << 48 | (long)'r' << 40 | (long)'r' << 32,
206 (long)'l' << 56 | (long)'c' << 48 | (long)'e' << 40 | (long)'i' << 32 | (long)'l' << 24,
207 (long)'l' << 56 | (long)'d' << 48 | (long)'q' << 40 | (long)'u' << 32 | (long)'o' << 24,
208 (long)'l' << 56 | (long)'e' << 48,
209 (long)'l' << 56 | (long)'f' << 48 | (long)'l' << 40 | (long)'o' << 32 | (long)'o' << 24 | (long)'r' << 16,
210 (long)'l' << 56 | (long)'o' << 48 | (long)'w' << 40 | (long)'a' << 32 | (long)'s' << 24 | (long)'t' << 16,
211 (long)'l' << 56 | (long)'o' << 48 | (long)'z' << 40,
212 (long)'l' << 56 | (long)'r' << 48 | (long)'m' << 40,
213 (long)'l' << 56 | (long)'s' << 48 | (long)'a' << 40 | (long)'q' << 32 | (long)'u' << 24 | (long)'o' << 16,
214 (long)'l' << 56 | (long)'s' << 48 | (long)'q' << 40 | (long)'u' << 32 | (long)'o' << 24,
215 (long)'l' << 56 | (long)'t' << 48,
216 (long)'m' << 56 | (long)'a' << 48 | (long)'c' << 40 | (long)'r' << 32,
217 (long)'m' << 56 | (long)'d' << 48 | (long)'a' << 40 | (long)'s' << 32 | (long)'h' << 24,
218 (long)'m' << 56 | (long)'i' << 48 | (long)'c' << 40 | (long)'r' << 32 | (long)'o' << 24,
219 (long)'m' << 56 | (long)'i' << 48 | (long)'d' << 40 | (long)'d' << 32 | (long)'o' << 24 | (long)'t' << 16,
220 (long)'m' << 56 | (long)'i' << 48 | (long)'n' << 40 | (long)'u' << 32 | (long)'s' << 24,
221 (long)'m' << 56 | (long)'u' << 48,
222 (long)'n' << 56 | (long)'a' << 48 | (long)'b' << 40 | (long)'l' << 32 | (long)'a' << 24,
223 (long)'n' << 56 | (long)'b' << 48 | (long)'s' << 40 | (long)'p' << 32,
224 (long)'n' << 56 | (long)'d' << 48 | (long)'a' << 40 | (long)'s' << 32 | (long)'h' << 24,
225 (long)'n' << 56 | (long)'e' << 48,
226 (long)'n' << 56 | (long)'i' << 48,
227 (long)'n' << 56 | (long)'o' << 48 | (long)'t' << 40,
228 (long)'n' << 56 | (long)'o' << 48 | (long)'t' << 40 | (long)'i' << 32 | (long)'n' << 24,
229 (long)'n' << 56 | (long)'s' << 48 | (long)'u' << 40 | (long)'b' << 32,
230 (long)'n' << 56 | (long)'t' << 48 | (long)'i' << 40 | (long)'l' << 32 | (long)'d' << 24 | (long)'e' << 16,
231 (long)'n' << 56 | (long)'u' << 48,
232 (long)'o' << 56 | (long)'a' << 48 | (long)'c' << 40 | (long)'u' << 32 | (long)'t' << 24 | (long)'e' << 16,
233 (long)'o' << 56 | (long)'c' << 48 | (long)'i' << 40 | (long)'r' << 32 | (long)'c' << 24,
234 (long)'o' << 56 | (long)'e' << 48 | (long)'l' << 40 | (long)'i' << 32 | (long)'g' << 24,
235 (long)'o' << 56 | (long)'g' << 48 | (long)'r' << 40 | (long)'a' << 32 | (long)'v' << 24 | (long)'e' << 16,
236 (long)'o' << 56 | (long)'l' << 48 | (long)'i' << 40 | (long)'n' << 32 | (long)'e' << 24,
237 (long)'o' << 56 | (long)'m' << 48 | (long)'e' << 40 | (long)'g' << 32 | (long)'a' << 24,
238 (long)'o' << 56 | (long)'m' << 48 | (long)'i' << 40 | (long)'c' << 32 | (long)'r' << 24 | (long)'o' << 16 | (long)'n' << 8,
239 (long)'o' << 56 | (long)'p' << 48 | (long)'l' << 40 | (long)'u' << 32 | (long)'s' << 24,
240 (long)'o' << 56 | (long)'r' << 48,
241 (long)'o' << 56 | (long)'r' << 48 | (long)'d' << 40 | (long)'f' << 32,
242 (long)'o' << 56 | (long)'r' << 48 | (long)'d' << 40 | (long)'m' << 32,
243 (long)'o' << 56 | (long)'s' << 48 | (long)'l' << 40 | (long)'a' << 32 | (long)'s' << 24 | (long)'h' << 16,
244 (long)'o' << 56 | (long)'t' << 48 | (long)'i' << 40 | (long)'l' << 32 | (long)'d' << 24 | (long)'e' << 16,
245 (long)'o' << 56 | (long)'t' << 48 | (long)'i' << 40 | (long)'m' << 32 | (long)'e' << 24 | (long)'s' << 16,
246 (long)'o' << 56 | (long)'u' << 48 | (long)'m' << 40 | (long)'l' << 32,
247 (long)'p' << 56 | (long)'a' << 48 | (long)'r' << 40 | (long)'a' << 32,
248 (long)'p' << 56 | (long)'a' << 48 | (long)'r' << 40 | (long)'t' << 32,
249 (long)'p' << 56 | (long)'e' << 48 | (long)'r' << 40 | (long)'m' << 32 | (long)'i' << 24 | (long)'l' << 16,
250 (long)'p' << 56 | (long)'e' << 48 | (long)'r' << 40 | (long)'p' << 32,
251 (long)'p' << 56 | (long)'h' << 48 | (long)'i' << 40,
252 (long)'p' << 56 | (long)'i' << 48,
253 (long)'p' << 56 | (long)'i' << 48 | (long)'v' << 40,
254 (long)'p' << 56 | (long)'l' << 48 | (long)'u' << 40 | (long)'s' << 32 | (long)'m' << 24 | (long)'n' << 16,
255 (long)'p' << 56 | (long)'o' << 48 | (long)'u' << 40 | (long)'n' << 32 | (long)'d' << 24,
256 (long)'p' << 56 | (long)'r' << 48 | (long)'i' << 40 | (long)'m' << 32 | (long)'e' << 24,
257 (long)'p' << 56 | (long)'r' << 48 | (long)'o' << 40 | (long)'d' << 32,
258 (long)'p' << 56 | (long)'r' << 48 | (long)'o' << 40 | (long)'p' << 32,
259 (long)'p' << 56 | (long)'s' << 48 | (long)'i' << 40,
260 (long)'q' << 56 | (long)'u' << 48 | (long)'o' << 40 | (long)'t' << 32,
261 (long)'r' << 56 | (long)'A' << 48 | (long)'r' << 40 | (long)'r' << 32,
262 (long)'r' << 56 | (long)'a' << 48 | (long)'d' << 40 | (long)'i' << 32 | (long)'c' << 24,
263 (long)'r' << 56 | (long)'a' << 48 | (long)'n' << 40 | (long)'g' << 32,
264 (long)'r' << 56 | (long)'a' << 48 | (long)'q' << 40 | (long)'u' << 32 | (long)'o' << 24,
265 (long)'r' << 56 | (long)'a' << 48 | (long)'r' << 40 | (long)'r' << 32,
266 (long)'r' << 56 | (long)'c' << 48 | (long)'e' << 40 | (long)'i' << 32 | (long)'l' << 24,
267 (long)'r' << 56 | (long)'d' << 48 | (long)'q' << 40 | (long)'u' << 32 | (long)'o' << 24,
268 (long)'r' << 56 | (long)'e' << 48 | (long)'a' << 40 | (long)'l' << 32,
269 (long)'r' << 56 | (long)'e' << 48 | (long)'g' << 40,
270 (long)'r' << 56 | (long)'f' << 48 | (long)'l' << 40 | (long)'o' << 32 | (long)'o' << 24 | (long)'r' << 16,
271 (long)'r' << 56 | (long)'h' << 48 | (long)'o' << 40,
272 (long)'r' << 56 | (long)'l' << 48 | (long)'m' << 40,
273 (long)'r' << 56 | (long)'s' << 48 | (long)'a' << 40 | (long)'q' << 32 | (long)'u' << 24 | (long)'o' << 16,
274 (long)'r' << 56 | (long)'s' << 48 | (long)'q' << 40 | (long)'u' << 32 | (long)'o' << 24,
275 (long)'s' << 56 | (long)'b' << 48 | (long)'q' << 40 | (long)'u' << 32 | (long)'o' << 24,
276 (long)'s' << 56 | (long)'c' << 48 | (long)'a' << 40 | (long)'r' << 32 | (long)'o' << 24 | (long)'n' << 16,
277 (long)'s' << 56 | (long)'d' << 48 | (long)'o' << 40 | (long)'t' << 32,
278 (long)'s' << 56 | (long)'e' << 48 | (long)'c' << 40 | (long)'t' << 32,
279 (long)'s' << 56 | (long)'h' << 48 | (long)'y' << 40,
280 (long)'s' << 56 | (long)'i' << 48 | (long)'g' << 40 | (long)'m' << 32 | (long)'a' << 24,
281 (long)'s' << 56 | (long)'i' << 48 | (long)'g' << 40 | (long)'m' << 32 | (long)'a' << 24 | (long)'f' << 16,
282 (long)'s' << 56 | (long)'i' << 48 | (long)'m' << 40,
283 (long)'s' << 56 | (long)'p' << 48 | (long)'a' << 40 | (long)'d' << 32 | (long)'e' << 24 | (long)'s' << 16,
284 (long)'s' << 56 | (long)'u' << 48 | (long)'b' << 40,
285 (long)'s' << 56 | (long)'u' << 48 | (long)'b' << 40 | (long)'e' << 32,
286 (long)'s' << 56 | (long)'u' << 48 | (long)'m' << 40,
287 (long)'s' << 56 | (long)'u' << 48 | (long)'p' << 40,
288 (long)'s' << 56 | (long)'u' << 48 | (long)'p' << 40 | (long)'1' << 32,
289 (long)'s' << 56 | (long)'u' << 48 | (long)'p' << 40 | (long)'2' << 32,
290 (long)'s' << 56 | (long)'u' << 48 | (long)'p' << 40 | (long)'3' << 32,
291 (long)'s' << 56 | (long)'u' << 48 | (long)'p' << 40 | (long)'e' << 32,
292 (long)'s' << 56 | (long)'z' << 48 | (long)'l' << 40 | (long)'i' << 32 | (long)'g' << 24,
293 (long)'t' << 56 | (long)'a' << 48 | (long)'u' << 40,
294 (long)'t' << 56 | (long)'h' << 48 | (long)'e' << 40 | (long)'r' << 32 | (long)'e' << 24 | (long)'4' << 16,
295 (long)'t' << 56 | (long)'h' << 48 | (long)'e' << 40 | (long)'t' << 32 | (long)'a' << 24,
296 (long)'t' << 56 | (long)'h' << 48 | (long)'e' << 40 | (long)'t' << 32 | (long)'a' << 24 | (long)'s' << 16 | (long)'y' << 8 | (long)'m' << 0,
297 (long)'t' << 56 | (long)'h' << 48 | (long)'i' << 40 | (long)'n' << 32 | (long)'s' << 24 | (long)'p' << 16,
298 (long)'t' << 56 | (long)'h' << 48 | (long)'o' << 40 | (long)'r' << 32 | (long)'n' << 24,
299 (long)'t' << 56 | (long)'i' << 48 | (long)'l' << 40 | (long)'d' << 32 | (long)'e' << 24,
300 (long)'t' << 56 | (long)'i' << 48 | (long)'m' << 40 | (long)'e' << 32 | (long)'s' << 24,
301 (long)'t' << 56 | (long)'r' << 48 | (long)'a' << 40 | (long)'d' << 32 | (long)'e' << 24,
302 (long)'u' << 56 | (long)'A' << 48 | (long)'r' << 40 | (long)'r' << 32,
303 (long)'u' << 56 | (long)'a' << 48 | (long)'c' << 40 | (long)'u' << 32 | (long)'t' << 24 | (long)'e' << 16,
304 (long)'u' << 56 | (long)'a' << 48 | (long)'r' << 40 | (long)'r' << 32,
305 (long)'u' << 56 | (long)'c' << 48 | (long)'i' << 40 | (long)'r' << 32 | (long)'c' << 24,
306 (long)'u' << 56 | (long)'g' << 48 | (long)'r' << 40 | (long)'a' << 32 | (long)'v' << 24 | (long)'e' << 16,
307 (long)'u' << 56 | (long)'m' << 48 | (long)'l' << 40,
308 (long)'u' << 56 | (long)'p' << 48 | (long)'s' << 40 | (long)'i' << 32 | (long)'h' << 24,
309 (long)'u' << 56 | (long)'p' << 48 | (long)'s' << 40 | (long)'i' << 32 | (long)'l' << 24 | (long)'o' << 16 | (long)'n' << 8,
310 (long)'u' << 56 | (long)'u' << 48 | (long)'m' << 40 | (long)'l' << 32,
311 (long)'w' << 56 | (long)'e' << 48 | (long)'i' << 40 | (long)'e' << 32 | (long)'r' << 24 | (long)'p' << 16,
312 (long)'x' << 56 | (long)'i' << 48,
313 (long)'y' << 56 | (long)'a' << 48 | (long)'c' << 40 | (long)'u' << 32 | (long)'t' << 24 | (long)'e' << 16,
314 (long)'y' << 56 | (long)'e' << 48 | (long)'n' << 40,
315 (long)'y' << 56 | (long)'u' << 48 | (long)'m' << 40 | (long)'l' << 32,
316 (long)'z' << 56 | (long)'e' << 48 | (long)'t' << 40 | (long)'a' << 32,
317 (long)'z' << 56 | (long)'w' << 48 | (long)'j' << 40,
318 (long)'z' << 56 | (long)'w' << 48 | (long)'n' << 40 | (long)'j' << 32
321 static readonly char[] entities_values = new char[] {
577 static readonly char[] hexChars = new [] {
598 public static void HtmlAttributeEncode (string s, TextWriter output)
600 output.Write(HtmlAttributeEncode(s));
603 public static string HtmlAttributeEncode (string s)
608 bool needEncode = false;
609 for (int i = 0; i < s.Length; i++) {
610 if (s [i] == '&' || s [i] == '"' || s [i] == '<') {
619 StringBuilder output = new StringBuilder ();
621 for (int i = 0; i < len; i++)
624 output.Append ("&");
627 output.Append (""");
630 output.Append ("<");
633 output.Append (s [i]);
637 return output.ToString();
640 public static string UrlDecode (string str)
642 return UrlDecode(str, Encoding.UTF8);
645 static char [] GetChars (MemoryStream b, Encoding e)
647 return e.GetChars (b.GetBuffer (), 0, (int) b.Length);
650 static void WriteCharBytes (IList buf, char ch, Encoding e)
653 foreach (byte b in e.GetBytes (new char[] { ch }))
659 public static string UrlDecode (string s, Encoding e)
664 if (s.IndexOf ('%') == -1 && s.IndexOf ('+') == -1)
671 var bytes = new List <byte> ();
675 for (int i = 0; i < len; i++) {
677 if (ch == '%' && i + 2 < len && s [i + 1] != '%') {
678 if (s [i + 1] == 'u' && i + 5 < len) {
679 // unicode hex sequence
680 xchar = GetChar (s, i + 2, 4);
682 WriteCharBytes (bytes, (char)xchar, e);
685 WriteCharBytes (bytes, '%', e);
686 } else if ((xchar = GetChar (s, i + 1, 2)) != -1) {
687 WriteCharBytes (bytes, (char)xchar, e);
690 WriteCharBytes (bytes, '%', e);
696 WriteCharBytes (bytes, ' ', e);
698 WriteCharBytes (bytes, ch, e);
701 byte[] buf = bytes.ToArray ();
703 return e.GetString (buf);
707 public static string UrlDecode (byte [] bytes, Encoding e)
712 return UrlDecode (bytes, 0, bytes.Length, e);
715 static int GetInt (byte b)
718 if (c >= '0' && c <= '9')
721 if (c >= 'a' && c <= 'f')
724 if (c >= 'A' && c <= 'F')
730 static int GetChar (byte [] bytes, int offset, int length)
733 int end = length + offset;
734 for (int i = offset; i < end; i++) {
735 int current = GetInt (bytes [i]);
738 value = (value << 4) + current;
744 static int GetChar (string str, int offset, int length)
747 int end = length + offset;
748 for (int i = offset; i < end; i++) {
753 int current = GetInt ((byte) c);
756 val = (val << 4) + current;
762 public static string UrlDecode (byte [] bytes, int offset, int count, Encoding e)
770 throw new ArgumentNullException ("bytes");
772 if (offset < 0 || offset > bytes.Length)
773 throw new ArgumentOutOfRangeException ("offset");
775 if (count < 0 || offset + count > bytes.Length)
776 throw new ArgumentOutOfRangeException ("count");
778 StringBuilder output = new StringBuilder ();
779 MemoryStream acc = new MemoryStream ();
781 int end = count + offset;
783 for (int i = offset; i < end; i++) {
784 if (bytes [i] == '%' && i + 2 < count && bytes [i + 1] != '%') {
785 if (bytes [i + 1] == (byte) 'u' && i + 5 < end) {
786 if (acc.Length > 0) {
787 output.Append (GetChars (acc, e));
790 xchar = GetChar (bytes, i + 2, 4);
792 output.Append ((char) xchar);
796 } else if ((xchar = GetChar (bytes, i + 1, 2)) != -1) {
797 acc.WriteByte ((byte) xchar);
803 if (acc.Length > 0) {
804 output.Append (GetChars (acc, e));
808 if (bytes [i] == '+') {
811 output.Append ((char) bytes [i]);
815 if (acc.Length > 0) {
816 output.Append (GetChars (acc, e));
820 return output.ToString ();
823 public static byte [] UrlDecodeToBytes (byte [] bytes)
828 return UrlDecodeToBytes (bytes, 0, bytes.Length);
831 public static byte [] UrlDecodeToBytes (string str)
833 return UrlDecodeToBytes (str, Encoding.UTF8);
836 public static byte [] UrlDecodeToBytes (string str, Encoding e)
842 throw new ArgumentNullException ("e");
844 return UrlDecodeToBytes (e.GetBytes (str));
847 public static byte [] UrlDecodeToBytes (byte [] bytes, int offset, int count)
854 int len = bytes.Length;
855 if (offset < 0 || offset >= len)
856 throw new ArgumentOutOfRangeException("offset");
858 if (count < 0 || offset > len - count)
859 throw new ArgumentOutOfRangeException("count");
861 MemoryStream result = new MemoryStream ();
862 int end = offset + count;
863 for (int i = offset; i < end; i++){
864 char c = (char) bytes [i];
867 } else if (c == '%' && i < end - 2) {
868 int xchar = GetChar (bytes, i + 1, 2);
874 result.WriteByte ((byte) c);
877 return result.ToArray ();
880 public static string UrlEncode(string str)
882 return UrlEncode(str, Encoding.UTF8);
885 public static string UrlEncode (string s, Encoding Enc)
887 if (string.IsNullOrEmpty (s))
890 bool needEncode = false;
892 for (int i = 0; i < len; i++) {
894 if ((c < '0') || (c < 'A' && c > '9') || (c > 'Z' && c < 'a') || (c > 'z')) {
906 // avoided GetByteCount call
907 byte [] bytes = new byte[Enc.GetMaxByteCount(s.Length)];
908 int realLen = Enc.GetBytes (s, 0, s.Length, bytes, 0);
909 return Encoding.ASCII.GetString (UrlEncodeToBytes (bytes, 0, realLen));
912 public static string UrlEncode (byte [] bytes)
917 if (bytes.Length == 0)
920 return Encoding.ASCII.GetString (UrlEncodeToBytes (bytes, 0, bytes.Length));
923 public static string UrlEncode (byte [] bytes, int offset, int count)
928 if (bytes.Length == 0)
931 return Encoding.ASCII.GetString (UrlEncodeToBytes (bytes, offset, count));
934 public static byte [] UrlEncodeToBytes (string str)
936 return UrlEncodeToBytes (str, Encoding.UTF8);
939 public static byte [] UrlEncodeToBytes (string str, Encoding e)
947 byte [] bytes = e.GetBytes (str);
948 return UrlEncodeToBytes (bytes, 0, bytes.Length);
951 public static byte [] UrlEncodeToBytes (byte [] bytes)
956 if (bytes.Length == 0)
959 return UrlEncodeToBytes (bytes, 0, bytes.Length);
962 static bool NotEncoded (char c)
964 return (c == '!' || c == '(' || c == ')' || c == '*' || c == '-' || c == '.' || c == '_');
967 static void UrlEncodeChar (char c, Stream result, bool isUnicode) {
969 //FIXME: what happens when there is an internal error?
971 // throw new ArgumentOutOfRangeException ("c", c, "c must be less than 256");
975 result.WriteByte ((byte)'%');
976 result.WriteByte ((byte)'u');
978 result.WriteByte ((byte)hexChars [idx]);
979 idx = (i >> 8) & 0x0F;
980 result.WriteByte ((byte)hexChars [idx]);
981 idx = (i >> 4) & 0x0F;
982 result.WriteByte ((byte)hexChars [idx]);
984 result.WriteByte ((byte)hexChars [idx]);
988 if (c > ' ' && NotEncoded (c)) {
989 result.WriteByte ((byte)c);
993 result.WriteByte ((byte)'+');
997 (c < 'A' && c > '9') ||
998 (c > 'Z' && c < 'a') ||
1000 if (isUnicode && c > 127) {
1001 result.WriteByte ((byte)'%');
1002 result.WriteByte ((byte)'u');
1003 result.WriteByte ((byte)'0');
1004 result.WriteByte ((byte)'0');
1007 result.WriteByte ((byte)'%');
1009 int idx = ((int) c) >> 4;
1010 result.WriteByte ((byte)hexChars [idx]);
1011 idx = ((int) c) & 0x0F;
1012 result.WriteByte ((byte)hexChars [idx]);
1015 result.WriteByte ((byte)c);
1018 public static byte [] UrlEncodeToBytes (byte [] bytes, int offset, int count)
1023 int len = bytes.Length;
1025 return new byte [0];
1027 if (offset < 0 || offset >= len)
1028 throw new ArgumentOutOfRangeException("offset");
1030 if (count < 0 || count > len - offset)
1031 throw new ArgumentOutOfRangeException("count");
1033 MemoryStream result = new MemoryStream (count);
1034 int end = offset + count;
1035 for (int i = offset; i < end; i++)
1036 UrlEncodeChar ((char)bytes [i], result, false);
1038 return result.ToArray();
1041 public static string UrlEncodeUnicode (string str)
1046 return Encoding.ASCII.GetString (UrlEncodeUnicodeToBytes (str));
1049 public static byte [] UrlEncodeUnicodeToBytes (string str)
1055 return new byte [0];
1057 MemoryStream result = new MemoryStream (str.Length);
1058 foreach (char c in str){
1059 UrlEncodeChar (c, result, true);
1061 return result.ToArray ();
1064 static bool TryConvertKeyToEntity (string key, out char value)
1066 var token = CalculateKeyValue (key);
1072 var idx = Array.BinarySearch (entities, token);
1078 value = entities_values [idx];
1082 static long CalculateKeyValue (string s)
1088 for (int i = 0; i < s.Length; ++i) {
1090 if (ch > 'z' || ch < '0')
1093 key |= ch << ((7 - i) * 8);
1100 /// Decodes an HTML-encoded string and returns the decoded string.
1102 /// <param name="s">The HTML string to decode. </param>
1103 /// <returns>The decoded text.</returns>
1104 public static string HtmlDecode (string s)
1107 throw new ArgumentNullException ("s");
1109 if (s.IndexOf ('&') == -1)
1112 StringBuilder entity = new StringBuilder ();
1113 StringBuilder output = new StringBuilder ();
1116 // 1 -> right after '&'
1117 // 2 -> between '&' and ';' but no '#'
1118 // 3 -> '#' found after '&' and getting numbers
1121 int digit_start = 0;
1122 bool hex_number = false;
1124 for (int i = 0; i < len; i++) {
1138 if (digit_start > 0) {
1139 entity.Append (s, digit_start, i - digit_start);
1143 output.Append (entity.ToString ());
1145 entity.Append ('&');
1153 output.Append (entity.ToString ());
1172 string key = entity.ToString ();
1176 if (key.Length > 1) {
1177 var skey = key.Substring (1, key.Length - 2);
1178 if (TryConvertKeyToEntity (skey, out c)) {
1184 output.Append (key);
1190 if (number < 0x10000) {
1191 output.Append ((char)number);
1193 output.Append ((char)(0xd800 + ((number - 0x10000) >> 10)));
1194 output.Append ((char)(0xdc00 + ((number - 0x10000) & 0x3ff)));
1202 if (c == 'x' || c == 'X' && !hex_number) {
1208 if (Char.IsDigit (c)) {
1209 if (digit_start == 0)
1212 number = number * (hex_number ? 16 : 10) + ((int)c - '0');
1217 if (c >= 'a' && c <= 'f') {
1218 number = number * 16 + 10 + ((int)c - 'a');
1221 if (c >= 'A' && c <= 'F') {
1222 number = number * 16 + 10 + ((int)c - 'A');
1228 if (digit_start > 0) {
1229 entity.Append (s, digit_start, i - digit_start);
1238 if (entity.Length > 0) {
1239 output.Append (entity);
1240 } else if (digit_start > 0) {
1241 output.Append (s, digit_start, s.Length - digit_start);
1243 return output.ToString ();
1247 /// Decodes an HTML-encoded string and sends the resulting output to a TextWriter output stream.
1249 /// <param name="s">The HTML string to decode</param>
1250 /// <param name="output">The TextWriter output stream containing the decoded string. </param>
1251 public static void HtmlDecode(string s, TextWriter output)
1254 output.Write (HtmlDecode (s));
1258 /// HTML-encodes a string and returns the encoded string.
1260 /// <param name="s">The text string to encode. </param>
1261 /// <returns>The HTML-encoded text.</returns>
1262 public static string HtmlEncode (string s)
1267 bool needEncode = false;
1269 for (i = 0; i < s.Length; i++) {
1294 StringBuilder output = new StringBuilder (s, 0, i, s.Length * 2);
1296 for (; i < s.Length; i++) {
1300 output.Append ("&");
1303 output.Append (">");
1306 output.Append ("<");
1309 output.Append (""");
1312 output.Append ("'");
1315 // MS starts encoding with &# from 160 and stops at 255.
1316 // We don't do that. One reason is the 65308/65310 unicode
1317 // characters that look like '<' and '>'.
1319 if (c > 159 && c < 256) {
1323 output.Append ("&#");
1324 if (char.IsSurrogate (c))
1325 output.Append (char.ConvertToUtf32 (s, i++));
1327 output.Append (((int) c).ToString (CultureInfo.InvariantCulture));
1329 output.Append (";");
1337 return output.ToString ();
1341 /// HTML-encodes a string and sends the resulting output to a TextWriter output stream.
1343 /// <param name="s">The string to encode. </param>
1344 /// <param name="output">The TextWriter output stream containing the encoded string. </param>
1345 public static void HtmlEncode(string s, TextWriter output)
1348 output.Write (HtmlEncode (s));
1351 public static string UrlPathEncode (string s)
1353 if (s == null || s.Length == 0)
1356 MemoryStream result = new MemoryStream ();
1357 int length = s.Length;
1358 for (int i = 0; i < length; i++) {
1359 UrlPathEncodeChar (s [i], result);
1361 return Encoding.ASCII.GetString (result.ToArray ());
1364 static void UrlPathEncodeChar (char c, Stream result)
1366 if (c < 33 || c > 126) {
1367 byte [] bIn = Encoding.UTF8.GetBytes (c.ToString ());
1368 for (int i = 0; i < bIn.Length; i++) {
1369 result.WriteByte ((byte) '%');
1370 int idx = ((int) bIn [i]) >> 4;
1371 result.WriteByte ((byte) hexChars [idx]);
1372 idx = ((int) bIn [i]) & 0x0F;
1373 result.WriteByte ((byte) hexChars [idx]);
1376 else if (c == ' ') {
1377 result.WriteByte ((byte) '%');
1378 result.WriteByte ((byte) '2');
1379 result.WriteByte ((byte) '0');
1382 result.WriteByte ((byte) c);
1385 public static NameValueCollection ParseQueryString (string query)
1387 return ParseQueryString (query, Encoding.UTF8);
1390 public static NameValueCollection ParseQueryString (string query, Encoding encoding)
1393 throw new ArgumentNullException ("query");
1394 if (encoding == null)
1395 throw new ArgumentNullException ("encoding");
1396 if (query.Length == 0 || (query.Length == 1 && query[0] == '?'))
1397 return new NameValueCollection ();
1398 if (query[0] == '?')
1399 query = query.Substring (1);
1401 NameValueCollection result = new HttpQSCollection ();
1402 ParseQueryString (query, encoding, result);
1406 internal static void ParseQueryString (string query, Encoding encoding, NameValueCollection result)
1408 if (query.Length == 0)
1411 string decoded = HtmlDecode (query);
1412 int decodedLength = decoded.Length;
1415 while (namePos <= decodedLength) {
1416 int valuePos = -1, valueEnd = -1;
1417 for (int q = namePos; q < decodedLength; q++) {
1418 if (valuePos == -1 && decoded [q] == '=') {
1420 } else if (decoded [q] == '&') {
1428 if (decoded [namePos] == '?')
1433 if (valuePos == -1) {
1437 name = UrlDecode (decoded.Substring (namePos, valuePos - namePos - 1), encoding);
1441 valueEnd = decoded.Length;
1443 namePos = valueEnd + 1;
1445 value = UrlDecode (decoded.Substring (valuePos, valueEnd - valuePos), encoding);
1447 result.Add (name, value);
1452 #endregion // Methods