OLD | NEW |
1 // Copyright 2012 the V8 project authors. All rights reserved. | 1 // Copyright 2012 the V8 project authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 (function(global, utils) { | 5 (function(global, utils) { |
6 | 6 |
7 %CheckIsBootstrapping(); | 7 %CheckIsBootstrapping(); |
8 | 8 |
9 // ------------------------------------------------------------------- | 9 // ------------------------------------------------------------------- |
10 // Imports | 10 // Imports |
11 | 11 |
12 var ArrayJoin; | 12 var ArrayJoin; |
| 13 var GetSubstitution; |
13 var GlobalRegExp = global.RegExp; | 14 var GlobalRegExp = global.RegExp; |
14 var GlobalString = global.String; | 15 var GlobalString = global.String; |
15 var IsRegExp; | 16 var IsRegExp; |
16 var MaxSimple; | 17 var MaxSimple; |
17 var MinSimple; | 18 var MinSimple; |
18 var RegExpInitialize; | 19 var RegExpInitialize; |
19 var matchSymbol = utils.ImportNow("match_symbol"); | 20 var matchSymbol = utils.ImportNow("match_symbol"); |
20 var replaceSymbol = utils.ImportNow("replace_symbol"); | 21 var replaceSymbol = utils.ImportNow("replace_symbol"); |
21 var searchSymbol = utils.ImportNow("search_symbol"); | 22 var searchSymbol = utils.ImportNow("search_symbol"); |
22 var splitSymbol = utils.ImportNow("split_symbol"); | 23 var splitSymbol = utils.ImportNow("split_symbol"); |
23 | 24 |
24 utils.Import(function(from) { | 25 utils.Import(function(from) { |
25 ArrayJoin = from.ArrayJoin; | 26 ArrayJoin = from.ArrayJoin; |
| 27 GetSubstitution = from.GetSubstitution; |
26 IsRegExp = from.IsRegExp; | 28 IsRegExp = from.IsRegExp; |
27 MaxSimple = from.MaxSimple; | 29 MaxSimple = from.MaxSimple; |
28 MinSimple = from.MinSimple; | 30 MinSimple = from.MinSimple; |
29 RegExpInitialize = from.RegExpInitialize; | 31 RegExpInitialize = from.RegExpInitialize; |
30 }); | 32 }); |
31 | 33 |
32 //------------------------------------------------------------------- | 34 //------------------------------------------------------------------- |
33 | 35 |
34 // ECMA-262, section 15.5.4.6 | 36 // ECMA-262, section 15.5.4.6 |
35 function StringConcat(other /* and more */) { // length == 1 | 37 function StringConcat(other /* and more */) { // length == 1 |
(...skipping 64 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
100 | 102 |
101 var subject = TO_STRING(this); | 103 var subject = TO_STRING(this); |
102 | 104 |
103 // Equivalent to RegExpCreate (ES#sec-regexpcreate) | 105 // Equivalent to RegExpCreate (ES#sec-regexpcreate) |
104 var regexp = %_NewObject(GlobalRegExp, GlobalRegExp); | 106 var regexp = %_NewObject(GlobalRegExp, GlobalRegExp); |
105 RegExpInitialize(regexp, pattern); | 107 RegExpInitialize(regexp, pattern); |
106 return regexp[matchSymbol](subject); | 108 return regexp[matchSymbol](subject); |
107 } | 109 } |
108 | 110 |
109 | 111 |
110 // This has the same size as the RegExpLastMatchInfo array, and can be used | |
111 // for functions that expect that structure to be returned. It is used when | |
112 // the needle is a string rather than a regexp. In this case we can't update | |
113 // lastMatchArray without erroneously affecting the properties on the global | |
114 // RegExp object. | |
115 var reusableMatchInfo = [2, "", "", -1, -1]; | |
116 | |
117 | |
118 // ES6, section 21.1.3.14 | 112 // ES6, section 21.1.3.14 |
119 function StringReplace(search, replace) { | 113 function StringReplace(search, replace) { |
120 CHECK_OBJECT_COERCIBLE(this, "String.prototype.replace"); | 114 CHECK_OBJECT_COERCIBLE(this, "String.prototype.replace"); |
121 | 115 |
122 // Decision tree for dispatch | 116 // Decision tree for dispatch |
123 // .. regexp search (in src/js/regexp.js, RegExpReplace) | 117 // .. regexp search (in src/js/regexp.js, RegExpReplace) |
124 // .... string replace | 118 // .... string replace |
125 // ...... non-global search | 119 // ...... non-global search |
126 // ........ empty string replace | 120 // ........ empty string replace |
127 // ........ non-empty string replace (with $-expansion) | 121 // ........ non-empty string replace (with $-expansion) |
(...skipping 31 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
159 var start = %StringIndexOf(subject, search, 0); | 153 var start = %StringIndexOf(subject, search, 0); |
160 if (start < 0) return subject; | 154 if (start < 0) return subject; |
161 var end = start + search.length; | 155 var end = start + search.length; |
162 | 156 |
163 var result = %_SubString(subject, 0, start); | 157 var result = %_SubString(subject, 0, start); |
164 | 158 |
165 // Compute the string to replace with. | 159 // Compute the string to replace with. |
166 if (IS_CALLABLE(replace)) { | 160 if (IS_CALLABLE(replace)) { |
167 result += replace(search, start, subject); | 161 result += replace(search, start, subject); |
168 } else { | 162 } else { |
169 reusableMatchInfo[CAPTURE0] = start; | 163 // In this case, we don't have any capture groups and can get away with |
170 reusableMatchInfo[CAPTURE1] = end; | 164 // faking the captures object by simply setting its length to 1. |
171 result = ExpandReplacement(TO_STRING(replace), | 165 const captures = { length: 1 }; |
172 subject, | 166 const matched = %_SubString(subject, start, end); |
173 reusableMatchInfo, | 167 result += GetSubstitution(matched, subject, start, captures, |
174 result); | 168 TO_STRING(replace)); |
175 } | 169 } |
176 | 170 |
177 return result + %_SubString(subject, end, subject.length); | 171 return result + %_SubString(subject, end, subject.length); |
178 } | 172 } |
179 | 173 |
180 | 174 |
181 // Expand the $-expressions in the string and return a new string with | |
182 // the result. | |
183 function ExpandReplacement(string, subject, matchInfo, result) { | |
184 var length = string.length; | |
185 var next = %StringIndexOf(string, '$', 0); | |
186 if (next < 0) { | |
187 if (length > 0) result += string; | |
188 return result; | |
189 } | |
190 | |
191 if (next > 0) result += %_SubString(string, 0, next); | |
192 | |
193 while (true) { | |
194 var expansion = '$'; | |
195 var position = next + 1; | |
196 if (position < length) { | |
197 var peek = %_StringCharCodeAt(string, position); | |
198 if (peek == 36) { // $$ | |
199 ++position; | |
200 result += '$'; | |
201 } else if (peek == 38) { // $& - match | |
202 ++position; | |
203 result += | |
204 %_SubString(subject, matchInfo[CAPTURE0], matchInfo[CAPTURE1]); | |
205 } else if (peek == 96) { // $` - prefix | |
206 ++position; | |
207 result += %_SubString(subject, 0, matchInfo[CAPTURE0]); | |
208 } else if (peek == 39) { // $' - suffix | |
209 ++position; | |
210 result += %_SubString(subject, matchInfo[CAPTURE1], subject.length); | |
211 } else if (peek >= 48 && peek <= 57) { | |
212 // Valid indices are $1 .. $9, $01 .. $09 and $10 .. $99 | |
213 var scaled_index = (peek - 48) << 1; | |
214 var advance = 1; | |
215 var number_of_captures = NUMBER_OF_CAPTURES(matchInfo); | |
216 if (position + 1 < string.length) { | |
217 var next = %_StringCharCodeAt(string, position + 1); | |
218 if (next >= 48 && next <= 57) { | |
219 var new_scaled_index = scaled_index * 10 + ((next - 48) << 1); | |
220 if (new_scaled_index < number_of_captures) { | |
221 scaled_index = new_scaled_index; | |
222 advance = 2; | |
223 } | |
224 } | |
225 } | |
226 if (scaled_index != 0 && scaled_index < number_of_captures) { | |
227 var start = matchInfo[CAPTURE(scaled_index)]; | |
228 if (start >= 0) { | |
229 result += | |
230 %_SubString(subject, start, matchInfo[CAPTURE(scaled_index + 1)]); | |
231 } | |
232 position += advance; | |
233 } else { | |
234 result += '$'; | |
235 } | |
236 } else { | |
237 result += '$'; | |
238 } | |
239 } else { | |
240 result += '$'; | |
241 } | |
242 | |
243 // Go the the next $ in the string. | |
244 next = %StringIndexOf(string, '$', position); | |
245 | |
246 // Return if there are no more $ characters in the string. If we | |
247 // haven't reached the end, we need to append the suffix. | |
248 if (next < 0) { | |
249 if (position < length) { | |
250 result += %_SubString(string, position, length); | |
251 } | |
252 return result; | |
253 } | |
254 | |
255 // Append substring between the previous and the next $ character. | |
256 if (next > position) { | |
257 result += %_SubString(string, position, next); | |
258 } | |
259 } | |
260 return result; | |
261 } | |
262 | |
263 | |
264 // ES6 21.1.3.15. | 175 // ES6 21.1.3.15. |
265 function StringSearch(pattern) { | 176 function StringSearch(pattern) { |
266 CHECK_OBJECT_COERCIBLE(this, "String.prototype.search"); | 177 CHECK_OBJECT_COERCIBLE(this, "String.prototype.search"); |
267 | 178 |
268 if (!IS_NULL_OR_UNDEFINED(pattern)) { | 179 if (!IS_NULL_OR_UNDEFINED(pattern)) { |
269 var searcher = pattern[searchSymbol]; | 180 var searcher = pattern[searchSymbol]; |
270 if (!IS_UNDEFINED(searcher)) { | 181 if (!IS_UNDEFINED(searcher)) { |
271 return %_Call(searcher, pattern, this); | 182 return %_Call(searcher, pattern, this); |
272 } | 183 } |
273 } | 184 } |
(...skipping 455 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
729 "small", StringSmall, | 640 "small", StringSmall, |
730 "strike", StringStrike, | 641 "strike", StringStrike, |
731 "sub", StringSub, | 642 "sub", StringSub, |
732 "sup", StringSup | 643 "sup", StringSup |
733 ]); | 644 ]); |
734 | 645 |
735 // ------------------------------------------------------------------- | 646 // ------------------------------------------------------------------- |
736 // Exports | 647 // Exports |
737 | 648 |
738 utils.Export(function(to) { | 649 utils.Export(function(to) { |
739 to.ExpandReplacement = ExpandReplacement; | |
740 to.StringIndexOf = StringIndexOf; | 650 to.StringIndexOf = StringIndexOf; |
741 to.StringLastIndexOf = StringLastIndexOf; | 651 to.StringLastIndexOf = StringLastIndexOf; |
742 to.StringMatch = StringMatchJS; | 652 to.StringMatch = StringMatchJS; |
743 to.StringReplace = StringReplace; | 653 to.StringReplace = StringReplace; |
744 to.StringSlice = StringSlice; | 654 to.StringSlice = StringSlice; |
745 to.StringSplit = StringSplitJS; | 655 to.StringSplit = StringSplitJS; |
746 to.StringSubstr = StringSubstr; | 656 to.StringSubstr = StringSubstr; |
747 to.StringSubstring = StringSubstring; | 657 to.StringSubstring = StringSubstring; |
748 }); | 658 }); |
749 | 659 |
750 }) | 660 }) |
OLD | NEW |