web.url-search-params.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347
  1. 'use strict';
  2. // TODO: in core-js@4, move /modules/ dependencies to public entries for better optimization by tools like `preset-env`
  3. require('../modules/es.array.iterator');
  4. var $ = require('../internals/export');
  5. var getBuiltIn = require('../internals/get-built-in');
  6. var USE_NATIVE_URL = require('../internals/native-url');
  7. var redefine = require('../internals/redefine');
  8. var redefineAll = require('../internals/redefine-all');
  9. var setToStringTag = require('../internals/set-to-string-tag');
  10. var createIteratorConstructor = require('../internals/create-iterator-constructor');
  11. var InternalStateModule = require('../internals/internal-state');
  12. var anInstance = require('../internals/an-instance');
  13. var hasOwn = require('../internals/has');
  14. var bind = require('../internals/function-bind-context');
  15. var classof = require('../internals/classof');
  16. var anObject = require('../internals/an-object');
  17. var isObject = require('../internals/is-object');
  18. var create = require('../internals/object-create');
  19. var createPropertyDescriptor = require('../internals/create-property-descriptor');
  20. var getIterator = require('../internals/get-iterator');
  21. var getIteratorMethod = require('../internals/get-iterator-method');
  22. var wellKnownSymbol = require('../internals/well-known-symbol');
  23. var $fetch = getBuiltIn('fetch');
  24. var Headers = getBuiltIn('Headers');
  25. var ITERATOR = wellKnownSymbol('iterator');
  26. var URL_SEARCH_PARAMS = 'URLSearchParams';
  27. var URL_SEARCH_PARAMS_ITERATOR = URL_SEARCH_PARAMS + 'Iterator';
  28. var setInternalState = InternalStateModule.set;
  29. var getInternalParamsState = InternalStateModule.getterFor(URL_SEARCH_PARAMS);
  30. var getInternalIteratorState = InternalStateModule.getterFor(URL_SEARCH_PARAMS_ITERATOR);
  31. var plus = /\+/g;
  32. var sequences = Array(4);
  33. var percentSequence = function (bytes) {
  34. return sequences[bytes - 1] || (sequences[bytes - 1] = RegExp('((?:%[\\da-f]{2}){' + bytes + '})', 'gi'));
  35. };
  36. var percentDecode = function (sequence) {
  37. try {
  38. return decodeURIComponent(sequence);
  39. } catch (error) {
  40. return sequence;
  41. }
  42. };
  43. var deserialize = function (it) {
  44. var result = it.replace(plus, ' ');
  45. var bytes = 4;
  46. try {
  47. return decodeURIComponent(result);
  48. } catch (error) {
  49. while (bytes) {
  50. result = result.replace(percentSequence(bytes--), percentDecode);
  51. }
  52. return result;
  53. }
  54. };
  55. var find = /[!'()~]|%20/g;
  56. var replace = {
  57. '!': '%21',
  58. "'": '%27',
  59. '(': '%28',
  60. ')': '%29',
  61. '~': '%7E',
  62. '%20': '+'
  63. };
  64. var replacer = function (match) {
  65. return replace[match];
  66. };
  67. var serialize = function (it) {
  68. return encodeURIComponent(it).replace(find, replacer);
  69. };
  70. var parseSearchParams = function (result, query) {
  71. if (query) {
  72. var attributes = query.split('&');
  73. var index = 0;
  74. var attribute, entry;
  75. while (index < attributes.length) {
  76. attribute = attributes[index++];
  77. if (attribute.length) {
  78. entry = attribute.split('=');
  79. result.push({
  80. key: deserialize(entry.shift()),
  81. value: deserialize(entry.join('='))
  82. });
  83. }
  84. }
  85. }
  86. };
  87. var updateSearchParams = function (query) {
  88. this.entries.length = 0;
  89. parseSearchParams(this.entries, query);
  90. };
  91. var validateArgumentsLength = function (passed, required) {
  92. if (passed < required) throw TypeError('Not enough arguments');
  93. };
  94. var URLSearchParamsIterator = createIteratorConstructor(function Iterator(params, kind) {
  95. setInternalState(this, {
  96. type: URL_SEARCH_PARAMS_ITERATOR,
  97. iterator: getIterator(getInternalParamsState(params).entries),
  98. kind: kind
  99. });
  100. }, 'Iterator', function next() {
  101. var state = getInternalIteratorState(this);
  102. var kind = state.kind;
  103. var step = state.iterator.next();
  104. var entry = step.value;
  105. if (!step.done) {
  106. step.value = kind === 'keys' ? entry.key : kind === 'values' ? entry.value : [entry.key, entry.value];
  107. } return step;
  108. });
  109. // `URLSearchParams` constructor
  110. // https://url.spec.whatwg.org/#interface-urlsearchparams
  111. var URLSearchParamsConstructor = function URLSearchParams(/* init */) {
  112. anInstance(this, URLSearchParamsConstructor, URL_SEARCH_PARAMS);
  113. var init = arguments.length > 0 ? arguments[0] : undefined;
  114. var that = this;
  115. var entries = [];
  116. var iteratorMethod, iterator, next, step, entryIterator, entryNext, first, second, key;
  117. setInternalState(that, {
  118. type: URL_SEARCH_PARAMS,
  119. entries: entries,
  120. updateURL: function () { /* empty */ },
  121. updateSearchParams: updateSearchParams
  122. });
  123. if (init !== undefined) {
  124. if (isObject(init)) {
  125. iteratorMethod = getIteratorMethod(init);
  126. if (typeof iteratorMethod === 'function') {
  127. iterator = iteratorMethod.call(init);
  128. next = iterator.next;
  129. while (!(step = next.call(iterator)).done) {
  130. entryIterator = getIterator(anObject(step.value));
  131. entryNext = entryIterator.next;
  132. if (
  133. (first = entryNext.call(entryIterator)).done ||
  134. (second = entryNext.call(entryIterator)).done ||
  135. !entryNext.call(entryIterator).done
  136. ) throw TypeError('Expected sequence with length 2');
  137. entries.push({ key: first.value + '', value: second.value + '' });
  138. }
  139. } else for (key in init) if (hasOwn(init, key)) entries.push({ key: key, value: init[key] + '' });
  140. } else {
  141. parseSearchParams(entries, typeof init === 'string' ? init.charAt(0) === '?' ? init.slice(1) : init : init + '');
  142. }
  143. }
  144. };
  145. var URLSearchParamsPrototype = URLSearchParamsConstructor.prototype;
  146. redefineAll(URLSearchParamsPrototype, {
  147. // `URLSearchParams.prototype.append` method
  148. // https://url.spec.whatwg.org/#dom-urlsearchparams-append
  149. append: function append(name, value) {
  150. validateArgumentsLength(arguments.length, 2);
  151. var state = getInternalParamsState(this);
  152. state.entries.push({ key: name + '', value: value + '' });
  153. state.updateURL();
  154. },
  155. // `URLSearchParams.prototype.delete` method
  156. // https://url.spec.whatwg.org/#dom-urlsearchparams-delete
  157. 'delete': function (name) {
  158. validateArgumentsLength(arguments.length, 1);
  159. var state = getInternalParamsState(this);
  160. var entries = state.entries;
  161. var key = name + '';
  162. var index = 0;
  163. while (index < entries.length) {
  164. if (entries[index].key === key) entries.splice(index, 1);
  165. else index++;
  166. }
  167. state.updateURL();
  168. },
  169. // `URLSearchParams.prototype.get` method
  170. // https://url.spec.whatwg.org/#dom-urlsearchparams-get
  171. get: function get(name) {
  172. validateArgumentsLength(arguments.length, 1);
  173. var entries = getInternalParamsState(this).entries;
  174. var key = name + '';
  175. var index = 0;
  176. for (; index < entries.length; index++) {
  177. if (entries[index].key === key) return entries[index].value;
  178. }
  179. return null;
  180. },
  181. // `URLSearchParams.prototype.getAll` method
  182. // https://url.spec.whatwg.org/#dom-urlsearchparams-getall
  183. getAll: function getAll(name) {
  184. validateArgumentsLength(arguments.length, 1);
  185. var entries = getInternalParamsState(this).entries;
  186. var key = name + '';
  187. var result = [];
  188. var index = 0;
  189. for (; index < entries.length; index++) {
  190. if (entries[index].key === key) result.push(entries[index].value);
  191. }
  192. return result;
  193. },
  194. // `URLSearchParams.prototype.has` method
  195. // https://url.spec.whatwg.org/#dom-urlsearchparams-has
  196. has: function has(name) {
  197. validateArgumentsLength(arguments.length, 1);
  198. var entries = getInternalParamsState(this).entries;
  199. var key = name + '';
  200. var index = 0;
  201. while (index < entries.length) {
  202. if (entries[index++].key === key) return true;
  203. }
  204. return false;
  205. },
  206. // `URLSearchParams.prototype.set` method
  207. // https://url.spec.whatwg.org/#dom-urlsearchparams-set
  208. set: function set(name, value) {
  209. validateArgumentsLength(arguments.length, 1);
  210. var state = getInternalParamsState(this);
  211. var entries = state.entries;
  212. var found = false;
  213. var key = name + '';
  214. var val = value + '';
  215. var index = 0;
  216. var entry;
  217. for (; index < entries.length; index++) {
  218. entry = entries[index];
  219. if (entry.key === key) {
  220. if (found) entries.splice(index--, 1);
  221. else {
  222. found = true;
  223. entry.value = val;
  224. }
  225. }
  226. }
  227. if (!found) entries.push({ key: key, value: val });
  228. state.updateURL();
  229. },
  230. // `URLSearchParams.prototype.sort` method
  231. // https://url.spec.whatwg.org/#dom-urlsearchparams-sort
  232. sort: function sort() {
  233. var state = getInternalParamsState(this);
  234. var entries = state.entries;
  235. // Array#sort is not stable in some engines
  236. var slice = entries.slice();
  237. var entry, entriesIndex, sliceIndex;
  238. entries.length = 0;
  239. for (sliceIndex = 0; sliceIndex < slice.length; sliceIndex++) {
  240. entry = slice[sliceIndex];
  241. for (entriesIndex = 0; entriesIndex < sliceIndex; entriesIndex++) {
  242. if (entries[entriesIndex].key > entry.key) {
  243. entries.splice(entriesIndex, 0, entry);
  244. break;
  245. }
  246. }
  247. if (entriesIndex === sliceIndex) entries.push(entry);
  248. }
  249. state.updateURL();
  250. },
  251. // `URLSearchParams.prototype.forEach` method
  252. forEach: function forEach(callback /* , thisArg */) {
  253. var entries = getInternalParamsState(this).entries;
  254. var boundFunction = bind(callback, arguments.length > 1 ? arguments[1] : undefined, 3);
  255. var index = 0;
  256. var entry;
  257. while (index < entries.length) {
  258. entry = entries[index++];
  259. boundFunction(entry.value, entry.key, this);
  260. }
  261. },
  262. // `URLSearchParams.prototype.keys` method
  263. keys: function keys() {
  264. return new URLSearchParamsIterator(this, 'keys');
  265. },
  266. // `URLSearchParams.prototype.values` method
  267. values: function values() {
  268. return new URLSearchParamsIterator(this, 'values');
  269. },
  270. // `URLSearchParams.prototype.entries` method
  271. entries: function entries() {
  272. return new URLSearchParamsIterator(this, 'entries');
  273. }
  274. }, { enumerable: true });
  275. // `URLSearchParams.prototype[@@iterator]` method
  276. redefine(URLSearchParamsPrototype, ITERATOR, URLSearchParamsPrototype.entries);
  277. // `URLSearchParams.prototype.toString` method
  278. // https://url.spec.whatwg.org/#urlsearchparams-stringification-behavior
  279. redefine(URLSearchParamsPrototype, 'toString', function toString() {
  280. var entries = getInternalParamsState(this).entries;
  281. var result = [];
  282. var index = 0;
  283. var entry;
  284. while (index < entries.length) {
  285. entry = entries[index++];
  286. result.push(serialize(entry.key) + '=' + serialize(entry.value));
  287. } return result.join('&');
  288. }, { enumerable: true });
  289. setToStringTag(URLSearchParamsConstructor, URL_SEARCH_PARAMS);
  290. $({ global: true, forced: !USE_NATIVE_URL }, {
  291. URLSearchParams: URLSearchParamsConstructor
  292. });
  293. // Wrap `fetch` for correct work with polyfilled `URLSearchParams`
  294. // https://github.com/zloirock/core-js/issues/674
  295. if (!USE_NATIVE_URL && typeof $fetch == 'function' && typeof Headers == 'function') {
  296. $({ global: true, enumerable: true, forced: true }, {
  297. fetch: function fetch(input /* , init */) {
  298. var args = [input];
  299. var init, body, headers;
  300. if (arguments.length > 1) {
  301. init = arguments[1];
  302. if (isObject(init)) {
  303. body = init.body;
  304. if (classof(body) === URL_SEARCH_PARAMS) {
  305. headers = init.headers ? new Headers(init.headers) : new Headers();
  306. if (!headers.has('content-type')) {
  307. headers.set('content-type', 'application/x-www-form-urlencoded;charset=UTF-8');
  308. }
  309. init = create(init, {
  310. body: createPropertyDescriptor(0, String(body)),
  311. headers: createPropertyDescriptor(0, headers)
  312. });
  313. }
  314. }
  315. args.push(init);
  316. } return $fetch.apply(this, args);
  317. }
  318. });
  319. }
  320. module.exports = {
  321. URLSearchParams: URLSearchParamsConstructor,
  322. getState: getInternalParamsState
  323. };