edit.js 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201
  1. (function (factory) {
  2. if (typeof module === "object" && typeof module.exports === "object") {
  3. var v = factory(require, exports);
  4. if (v !== undefined) module.exports = v;
  5. }
  6. else if (typeof define === "function" && define.amd) {
  7. define(["require", "exports", "./format", "./parser"], factory);
  8. }
  9. })(function (require, exports) {
  10. /*---------------------------------------------------------------------------------------------
  11. * Copyright (c) Microsoft Corporation. All rights reserved.
  12. * Licensed under the MIT License. See License.txt in the project root for license information.
  13. *--------------------------------------------------------------------------------------------*/
  14. 'use strict';
  15. Object.defineProperty(exports, "__esModule", { value: true });
  16. exports.isWS = exports.applyEdit = exports.setProperty = exports.removeProperty = void 0;
  17. const format_1 = require("./format");
  18. const parser_1 = require("./parser");
  19. function removeProperty(text, path, options) {
  20. return setProperty(text, path, void 0, options);
  21. }
  22. exports.removeProperty = removeProperty;
  23. function setProperty(text, originalPath, value, options) {
  24. const path = originalPath.slice();
  25. const errors = [];
  26. const root = (0, parser_1.parseTree)(text, errors);
  27. let parent = void 0;
  28. let lastSegment = void 0;
  29. while (path.length > 0) {
  30. lastSegment = path.pop();
  31. parent = (0, parser_1.findNodeAtLocation)(root, path);
  32. if (parent === void 0 && value !== void 0) {
  33. if (typeof lastSegment === 'string') {
  34. value = { [lastSegment]: value };
  35. }
  36. else {
  37. value = [value];
  38. }
  39. }
  40. else {
  41. break;
  42. }
  43. }
  44. if (!parent) {
  45. // empty document
  46. if (value === void 0) { // delete
  47. throw new Error('Can not delete in empty document');
  48. }
  49. return withFormatting(text, { offset: root ? root.offset : 0, length: root ? root.length : 0, content: JSON.stringify(value) }, options);
  50. }
  51. else if (parent.type === 'object' && typeof lastSegment === 'string' && Array.isArray(parent.children)) {
  52. const existing = (0, parser_1.findNodeAtLocation)(parent, [lastSegment]);
  53. if (existing !== void 0) {
  54. if (value === void 0) { // delete
  55. if (!existing.parent) {
  56. throw new Error('Malformed AST');
  57. }
  58. const propertyIndex = parent.children.indexOf(existing.parent);
  59. let removeBegin;
  60. let removeEnd = existing.parent.offset + existing.parent.length;
  61. if (propertyIndex > 0) {
  62. // remove the comma of the previous node
  63. let previous = parent.children[propertyIndex - 1];
  64. removeBegin = previous.offset + previous.length;
  65. }
  66. else {
  67. removeBegin = parent.offset + 1;
  68. if (parent.children.length > 1) {
  69. // remove the comma of the next node
  70. let next = parent.children[1];
  71. removeEnd = next.offset;
  72. }
  73. }
  74. return withFormatting(text, { offset: removeBegin, length: removeEnd - removeBegin, content: '' }, options);
  75. }
  76. else {
  77. // set value of existing property
  78. return withFormatting(text, { offset: existing.offset, length: existing.length, content: JSON.stringify(value) }, options);
  79. }
  80. }
  81. else {
  82. if (value === void 0) { // delete
  83. return []; // property does not exist, nothing to do
  84. }
  85. const newProperty = `${JSON.stringify(lastSegment)}: ${JSON.stringify(value)}`;
  86. const index = options.getInsertionIndex ? options.getInsertionIndex(parent.children.map(p => p.children[0].value)) : parent.children.length;
  87. let edit;
  88. if (index > 0) {
  89. let previous = parent.children[index - 1];
  90. edit = { offset: previous.offset + previous.length, length: 0, content: ',' + newProperty };
  91. }
  92. else if (parent.children.length === 0) {
  93. edit = { offset: parent.offset + 1, length: 0, content: newProperty };
  94. }
  95. else {
  96. edit = { offset: parent.offset + 1, length: 0, content: newProperty + ',' };
  97. }
  98. return withFormatting(text, edit, options);
  99. }
  100. }
  101. else if (parent.type === 'array' && typeof lastSegment === 'number' && Array.isArray(parent.children)) {
  102. const insertIndex = lastSegment;
  103. if (insertIndex === -1) {
  104. // Insert
  105. const newProperty = `${JSON.stringify(value)}`;
  106. let edit;
  107. if (parent.children.length === 0) {
  108. edit = { offset: parent.offset + 1, length: 0, content: newProperty };
  109. }
  110. else {
  111. const previous = parent.children[parent.children.length - 1];
  112. edit = { offset: previous.offset + previous.length, length: 0, content: ',' + newProperty };
  113. }
  114. return withFormatting(text, edit, options);
  115. }
  116. else if (value === void 0 && parent.children.length >= 0) {
  117. // Removal
  118. const removalIndex = lastSegment;
  119. const toRemove = parent.children[removalIndex];
  120. let edit;
  121. if (parent.children.length === 1) {
  122. // only item
  123. edit = { offset: parent.offset + 1, length: parent.length - 2, content: '' };
  124. }
  125. else if (parent.children.length - 1 === removalIndex) {
  126. // last item
  127. let previous = parent.children[removalIndex - 1];
  128. let offset = previous.offset + previous.length;
  129. let parentEndOffset = parent.offset + parent.length;
  130. edit = { offset, length: parentEndOffset - 2 - offset, content: '' };
  131. }
  132. else {
  133. edit = { offset: toRemove.offset, length: parent.children[removalIndex + 1].offset - toRemove.offset, content: '' };
  134. }
  135. return withFormatting(text, edit, options);
  136. }
  137. else if (value !== void 0) {
  138. let edit;
  139. const newProperty = `${JSON.stringify(value)}`;
  140. if (!options.isArrayInsertion && parent.children.length > lastSegment) {
  141. const toModify = parent.children[lastSegment];
  142. edit = { offset: toModify.offset, length: toModify.length, content: newProperty };
  143. }
  144. else if (parent.children.length === 0 || lastSegment === 0) {
  145. edit = { offset: parent.offset + 1, length: 0, content: parent.children.length === 0 ? newProperty : newProperty + ',' };
  146. }
  147. else {
  148. const index = lastSegment > parent.children.length ? parent.children.length : lastSegment;
  149. const previous = parent.children[index - 1];
  150. edit = { offset: previous.offset + previous.length, length: 0, content: ',' + newProperty };
  151. }
  152. return withFormatting(text, edit, options);
  153. }
  154. else {
  155. throw new Error(`Can not ${value === void 0 ? 'remove' : (options.isArrayInsertion ? 'insert' : 'modify')} Array index ${insertIndex} as length is not sufficient`);
  156. }
  157. }
  158. else {
  159. throw new Error(`Can not add ${typeof lastSegment !== 'number' ? 'index' : 'property'} to parent of type ${parent.type}`);
  160. }
  161. }
  162. exports.setProperty = setProperty;
  163. function withFormatting(text, edit, options) {
  164. if (!options.formattingOptions) {
  165. return [edit];
  166. }
  167. // apply the edit
  168. let newText = applyEdit(text, edit);
  169. // format the new text
  170. let begin = edit.offset;
  171. let end = edit.offset + edit.content.length;
  172. if (edit.length === 0 || edit.content.length === 0) { // insert or remove
  173. while (begin > 0 && !(0, format_1.isEOL)(newText, begin - 1)) {
  174. begin--;
  175. }
  176. while (end < newText.length && !(0, format_1.isEOL)(newText, end)) {
  177. end++;
  178. }
  179. }
  180. const edits = (0, format_1.format)(newText, { offset: begin, length: end - begin }, { ...options.formattingOptions, keepLines: false });
  181. // apply the formatting edits and track the begin and end offsets of the changes
  182. for (let i = edits.length - 1; i >= 0; i--) {
  183. const edit = edits[i];
  184. newText = applyEdit(newText, edit);
  185. begin = Math.min(begin, edit.offset);
  186. end = Math.max(end, edit.offset + edit.length);
  187. end += edit.content.length - edit.length;
  188. }
  189. // create a single edit with all changes
  190. const editLength = text.length - (newText.length - end) - begin;
  191. return [{ offset: begin, length: editLength, content: newText.substring(begin, end) }];
  192. }
  193. function applyEdit(text, edit) {
  194. return text.substring(0, edit.offset) + edit.content + text.substring(edit.offset + edit.length);
  195. }
  196. exports.applyEdit = applyEdit;
  197. function isWS(text, offset) {
  198. return '\r\n \t'.indexOf(text.charAt(offset)) !== -1;
  199. }
  200. exports.isWS = isWS;
  201. });