You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							97 lines
						
					
					
						
							3.3 KiB
						
					
					
				
			
		
		
	
	
							97 lines
						
					
					
						
							3.3 KiB
						
					
					
				| 'use strict';
 | |
| require('./es6.regexp.exec');
 | |
| var redefine = require('./_redefine');
 | |
| var hide = require('./_hide');
 | |
| var fails = require('./_fails');
 | |
| var defined = require('./_defined');
 | |
| var wks = require('./_wks');
 | |
| var regexpExec = require('./_regexp-exec');
 | |
| 
 | |
| var SPECIES = wks('species');
 | |
| 
 | |
| var REPLACE_SUPPORTS_NAMED_GROUPS = !fails(function () {
 | |
|   // #replace needs built-in support for named groups.
 | |
|   // #match works fine because it just return the exec results, even if it has
 | |
|   // a "grops" property.
 | |
|   var re = /./;
 | |
|   re.exec = function () {
 | |
|     var result = [];
 | |
|     result.groups = { a: '7' };
 | |
|     return result;
 | |
|   };
 | |
|   return ''.replace(re, '$<a>') !== '7';
 | |
| });
 | |
| 
 | |
| var SPLIT_WORKS_WITH_OVERWRITTEN_EXEC = (function () {
 | |
|   // Chrome 51 has a buggy "split" implementation when RegExp#exec !== nativeExec
 | |
|   var re = /(?:)/;
 | |
|   var originalExec = re.exec;
 | |
|   re.exec = function () { return originalExec.apply(this, arguments); };
 | |
|   var result = 'ab'.split(re);
 | |
|   return result.length === 2 && result[0] === 'a' && result[1] === 'b';
 | |
| })();
 | |
| 
 | |
| module.exports = function (KEY, length, exec) {
 | |
|   var SYMBOL = wks(KEY);
 | |
| 
 | |
|   var DELEGATES_TO_SYMBOL = !fails(function () {
 | |
|     // String methods call symbol-named RegEp methods
 | |
|     var O = {};
 | |
|     O[SYMBOL] = function () { return 7; };
 | |
|     return ''[KEY](O) != 7;
 | |
|   });
 | |
| 
 | |
|   var DELEGATES_TO_EXEC = DELEGATES_TO_SYMBOL ? !fails(function () {
 | |
|     // Symbol-named RegExp methods call .exec
 | |
|     var execCalled = false;
 | |
|     var re = /a/;
 | |
|     re.exec = function () { execCalled = true; return null; };
 | |
|     if (KEY === 'split') {
 | |
|       // RegExp[@@split] doesn't call the regex's exec method, but first creates
 | |
|       // a new one. We need to return the patched regex when creating the new one.
 | |
|       re.constructor = {};
 | |
|       re.constructor[SPECIES] = function () { return re; };
 | |
|     }
 | |
|     re[SYMBOL]('');
 | |
|     return !execCalled;
 | |
|   }) : undefined;
 | |
| 
 | |
|   if (
 | |
|     !DELEGATES_TO_SYMBOL ||
 | |
|     !DELEGATES_TO_EXEC ||
 | |
|     (KEY === 'replace' && !REPLACE_SUPPORTS_NAMED_GROUPS) ||
 | |
|     (KEY === 'split' && !SPLIT_WORKS_WITH_OVERWRITTEN_EXEC)
 | |
|   ) {
 | |
|     var nativeRegExpMethod = /./[SYMBOL];
 | |
|     var fns = exec(
 | |
|       defined,
 | |
|       SYMBOL,
 | |
|       ''[KEY],
 | |
|       function maybeCallNative(nativeMethod, regexp, str, arg2, forceStringMethod) {
 | |
|         if (regexp.exec === regexpExec) {
 | |
|           if (DELEGATES_TO_SYMBOL && !forceStringMethod) {
 | |
|             // The native String method already delegates to @@method (this
 | |
|             // polyfilled function), leasing to infinite recursion.
 | |
|             // We avoid it by directly calling the native @@method method.
 | |
|             return { done: true, value: nativeRegExpMethod.call(regexp, str, arg2) };
 | |
|           }
 | |
|           return { done: true, value: nativeMethod.call(str, regexp, arg2) };
 | |
|         }
 | |
|         return { done: false };
 | |
|       }
 | |
|     );
 | |
|     var strfn = fns[0];
 | |
|     var rxfn = fns[1];
 | |
| 
 | |
|     redefine(String.prototype, KEY, strfn);
 | |
|     hide(RegExp.prototype, SYMBOL, length == 2
 | |
|       // 21.2.5.8 RegExp.prototype[@@replace](string, replaceValue)
 | |
|       // 21.2.5.11 RegExp.prototype[@@split](string, limit)
 | |
|       ? function (string, arg) { return rxfn.call(string, this, arg); }
 | |
|       // 21.2.5.6 RegExp.prototype[@@match](string)
 | |
|       // 21.2.5.9 RegExp.prototype[@@search](string)
 | |
|       : function (string) { return rxfn.call(string, this); }
 | |
|     );
 | |
|   }
 | |
| };
 |