[XRegExp](http://xregexp.com/) 3.0.0 ==================================== XRegExp provides augmented and extensible JavaScript regular expressions. You get new syntax, flags, and methods beyond what browsers support natively. XRegExp is also a regex utility belt with tools to make your client-side grepping simpler and more powerful, while freeing you from worrying about pesky cross-browser inconsistencies and things like manually manipulating `lastIndex` or slicing strings when tokenizing. XRegExp supports all native ES6 regular expression syntax. It supports Internet Explorer 5.5+, Firefox 1.5+, Chrome, Safari 3+, and Opera 11+. You can also use it with Node.js, or as a RequireJS module. The base library is about 4.25 KB, minified and gzipped. ## Performance XRegExp regexes compile to native `RegExp` objects, and therefore perform just as fast as native regular expressions. There is a tiny extra cost when compiling a pattern for the first time. ## Usage examples ```js // Using named capture and flag x (free-spacing and line comments) var date = XRegExp('(? [0-9]{4} ) -? # year \n\ (? [0-9]{2} ) -? # month \n\ (? [0-9]{2} ) # day ', 'x'); // XRegExp.exec gives you named backreferences on the match result var match = XRegExp.exec('2015-02-22', date); match.year; // -> '2015' // It also includes optional pos and sticky arguments var pos = 3, result = []; while (match = XRegExp.exec('<1><2><3><4>5<6>', /<(\d+)>/, pos, 'sticky')) { result.push(match[1]); pos = match.index + match[0].length; } // result -> ['2', '3', '4'] // XRegExp.replace allows named backreferences in replacements XRegExp.replace('2015-02-22', date, '${month}/${day}/${year}'); // -> '02/22/2015' XRegExp.replace('2015-02-22', date, function(match) { return match.month + '/' + match.day + '/' + match.year; }); // -> '02/22/2015' // In fact, XRegExps compile to RegExps and work perfectly with native methods date.test('2015-02-22'); // -> true // The *only* caveat is that named captures must be referenced using numbered backreferences '2015-02-22'.replace(date, '$2/$3/$1'); // -> '02/22/2015' // If you want, you can extend native methods so you don't have to worry about this. // Doing so also fixes numerous browser bugs in the native methods XRegExp.install('natives'); '2015-02-22'.replace(date, '${month}/${day}/${year}'); // -> '02/22/2015' '2015-02-22'.replace(date, function(match) { return match.month + '/' + match.day + '/' + match.year; }); // -> '02/22/2015' date.exec('2015-02-22').year; // -> '2015' // Extract every other digit from a string using XRegExp.forEach var evens = []; XRegExp.forEach('1a2345', /\d/, function(match, i) { if (i % 2) evens.push(+match[0]); }); // evens -> [2, 4] // Get numbers within tags using XRegExp.matchChain XRegExp.matchChain('1 2 3 4 a 56', [ XRegExp('(?is).*?'), /\d+/ ]); // -> ['2', '4', '56'] // You can also pass forward and return specific backreferences var html = 'XRegExp' + 'Google'; XRegExp.matchChain(html, [ {regex: //i, backref: 1}, {regex: XRegExp('(?i)^https?://(?[^/?#]+)'), backref: 'domain'} ]); // -> ['xregexp.com', 'www.google.com'] // Merge strings and regexes into a single pattern, safely rewriting backreferences XRegExp.union(['a+b*c', /(dog)\1/, /(cat)\1/], 'i'); // -> /a\+b\*c|(dog)\1|(cat)\2/i ``` These examples should give you the flavor of what's possible, but XRegExp has more syntax, flags, methods, options, and browser fixes that aren't shown here. You can even augment XRegExp's regular expression syntax with addons (see below) or write your own. See [xregexp.com](http://xregexp.com/) for more details. ## Addons You can either load addons individually, or bundle all addons together with XRegExp by loading `xregexp-all.js`. XRegExp's [npm](http://npmjs.org/) package uses `xregexp-all.js`, so addons are always available when XRegExp is installed using npm. ### Unicode In browsers, first include the Unicode Base script and then one or more of the addons for Unicode blocks, categories, properties, or scripts. ```html ``` Then you can do this: ```js // Test the Unicode category L (Letter) var unicodeWord = XRegExp('^\\pL+$'); unicodeWord.test('Русский'); // -> true unicodeWord.test('日本語'); // -> true unicodeWord.test('العربية'); // -> true // Test some Unicode scripts XRegExp('^\\p{Hiragana}+$').test('ひらがな'); // -> true XRegExp('^[\\p{Latin}\\p{Common}]+$').test('Über Café.'); // -> true ``` By default, `\p{…}` and `\P{…}` support the Basic Multilingual Plane (i.e. code points up to `U+FFFF`). You can opt-in to full 21-bit Unicode support (with code points up to `U+10FFFF`) on a per-regex basis by using flag `A`. In XRegExp, this is called *astral mode*. You can automatically apply astral mode for all new regexes by running `XRegExp.install('astral')`. When in astral mode, `\p{…}` and `\P{…}` always match a full code point rather than a code unit, using surrogate pairs for code points above `U+FFFF`. ```js // Using flag A. The test string uses a surrogate pair to represent U+1F4A9 XRegExp('^\\pS$', 'A').test('\uD83D\uDCA9'); // -> true // Implicit flag A XRegExp.install('astral'); XRegExp('^\\pS$').test('\uD83D\uDCA9'); // -> true ``` Opting in to astral mode disables the use of `\p{…}` and `\P{…}` within character classes. In astral mode, use e.g. `(\pL|[0-9_])+` instead of `[\pL0-9_]+`. XRegExp uses Unicode 8.0.0. ### XRegExp.build In browsers, first include the script: ```html ``` You can then build regular expressions using named subpatterns, for readability and pattern reuse: ```js var time = XRegExp.build('(?x)^ {{hours}} ({{minutes}}) $', { hours: XRegExp.build('{{h12}} : | {{h24}}', { h12: /1[0-2]|0?[1-9]/, h24: /2[0-3]|[01][0-9]/ }), minutes: /^[0-5][0-9]$/ }); time.test('10:59'); // -> true XRegExp.exec('10:59', time).minutes; // -> '59' ``` Named subpatterns can be provided as strings or regex objects. A leading `^` and trailing unescaped `$` are stripped from subpatterns if both are present, which allows embedding independently-useful anchored patterns. `{{…}}` tokens can be quantified as a single unit. Any backreferences in the outer pattern or provided subpatterns are automatically renumbered to work correctly within the larger combined pattern. The syntax `({{name}})` works as shorthand for named capture via `(?{{name}})`. Named subpatterns cannot be embedded within character classes. See also: *[Creating Grammatical Regexes Using XRegExp.build](http://blog.stevenlevithan.com/archives/grammatical-patterns-xregexp-build)*. ### XRegExp.matchRecursive In browsers, first include the script: ```html ``` You can then match recursive constructs using XRegExp pattern strings as left and right delimiters: ```js var str = '(t((e))s)t()(ing)'; XRegExp.matchRecursive(str, '\\(', '\\)', 'g'); // -> ['t((e))s', '', 'ing'] // Extended information mode with valueNames str = 'Here is
an
example'; XRegExp.matchRecursive(str, '', '', 'gi', { valueNames: ['between', 'left', 'match', 'right'] }); /* -> [ {name: 'between', value: 'Here is ', start: 0, end: 8}, {name: 'left', value: '
', start: 8, end: 13}, {name: 'match', value: '
an
', start: 13, end: 27}, {name: 'right', value: '
', start: 27, end: 33}, {name: 'between', value: ' example', start: 33, end: 41} ] */ // Omitting unneeded parts with null valueNames, and using escapeChar str = '...{1}\\{{function(x,y){return y+x;}}'; XRegExp.matchRecursive(str, '{', '}', 'g', { valueNames: ['literal', null, 'value', null], escapeChar: '\\' }); /* -> [ {name: 'literal', value: '...', start: 0, end: 3}, {name: 'value', value: '1', start: 4, end: 5}, {name: 'literal', value: '\\{', start: 6, end: 8}, {name: 'value', value: 'function(x,y){return y+x;}', start: 9, end: 35} ] */ // Sticky mode via flag y str = '<1><<<2>>><3>4<5>'; XRegExp.matchRecursive(str, '<', '>', 'gy'); // -> ['1', '<<2>>', '3'] ``` `XRegExp.matchRecursive` throws an error if it scans past an unbalanced delimiter in the target string. ## Installation and usage In browsers: ```html ``` Or, to bundle XRegExp with all of its addons: ```html ``` Using [npm](http://npmjs.org/): ```bash npm install xregexp ``` In [Node.js](http://nodejs.org/): ```js var XRegExp = require('xregexp'); ``` The [CommonJS](http://wiki.commonjs.org/wiki/Modules)-style `require('xregexp').XRegExp` also works. In an AMD loader like [RequireJS](http://requirejs.org/): ```js require({paths: {xregexp: 'xregexp-all'}}, ['xregexp'], function(XRegExp) { console.log(XRegExp.version); }); ``` ## Changelog * Releases: [Version history](http://xregexp.com/history/). * Upcoming: [Issue tracker](https://github.com/slevithan/xregexp/issues). ## About XRegExp copyright 2007-2015 by [Steven Levithan](http://stevenlevithan.com/). Tools: Unicode range generators by [Mathias Bynens](http://mathiasbynens.be/), and adapted from his [unicode-data](https://github.com/mathiasbynens/unicode-data) project. Source file concatenator by [Bjarke Walling](http://twitter.com/walling). Tests: Uses [Jasmine](http://pivotal.github.com/jasmine/) for unit tests, and [Benchmark.js](http://benchmarkjs.com) for performance tests. Prior art: `XRegExp.build` inspired by [Lea Verou](http://lea.verou.me/)'s [RegExp.create](http://lea.verou.me/2011/03/create-complex-regexps-more-easily/). `XRegExp.union` inspired by [Ruby](http://www.ruby-lang.org/). XRegExp's syntax extensions and flags come from [Perl](http://www.perl.org/), [.NET](http://www.microsoft.com/net), etc. All code, including addons, tools, and tests, is released under the terms of the [MIT License](http://mit-license.org/). Fork me to show support, fix, and extend.