Skip to content

Commit 29f260e

Browse files
authored
feat: introduce offset (#297)
1 parent 24cb8ea commit 29f260e

File tree

4 files changed

+64
-2
lines changed

4 files changed

+64
-2
lines changed

README.md

+16
Original file line numberDiff line numberDiff line change
@@ -73,9 +73,25 @@ const s = new MagicString(someCode, {
7373
],
7474
// mark source as ignore in DevTools, see below #Bundling
7575
ignoreList: false,
76+
// adjust the incoming position - see below
77+
offset: 0,
7678
});
7779
```
7880

81+
## Properties
82+
83+
### s.offset
84+
85+
Sets the offset property to adjust the incoming position for the following APIs: `slice`, `update`, `overwrite`, `appendLeft`, `prependLeft`, `appendRight`, `prependRight`, `move`, `reset`, and `remove`.
86+
87+
Example usage:
88+
89+
```ts
90+
const s = new MagicString('hello world', { offset: 0 });
91+
s.offset = 6;
92+
s.slice() === 'world';
93+
```
94+
7995
## Methods
8096

8197
### s.addSourcemapLocation( index )

src/MagicString.js

+27-2
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,7 @@ export default class MagicString {
3535
storedNames: { writable: true, value: {} },
3636
indentStr: { writable: true, value: undefined },
3737
ignoreList: { writable: true, value: options.ignoreList },
38+
offset: { writable: true, value: options.offset || 0 },
3839
});
3940

4041
if (DEBUG) {
@@ -57,6 +58,8 @@ export default class MagicString {
5758
}
5859

5960
appendLeft(index, content) {
61+
index = index + this.offset;
62+
6063
if (typeof content !== 'string') throw new TypeError('inserted content must be a string');
6164

6265
if (DEBUG) this.stats.time('appendLeft');
@@ -76,6 +79,8 @@ export default class MagicString {
7679
}
7780

7881
appendRight(index, content) {
82+
index = index + this.offset;
83+
7984
if (typeof content !== 'string') throw new TypeError('inserted content must be a string');
8085

8186
if (DEBUG) this.stats.time('appendRight');
@@ -95,7 +100,7 @@ export default class MagicString {
95100
}
96101

97102
clone() {
98-
const cloned = new MagicString(this.original, { filename: this.filename });
103+
const cloned = new MagicString(this.original, { filename: this.filename, offset: this.offset });
99104

100105
let originalChunk = this.firstChunk;
101106
let clonedChunk = (cloned.firstChunk = cloned.lastSearchedChunk = originalChunk.clone());
@@ -312,6 +317,10 @@ export default class MagicString {
312317
}
313318

314319
move(start, end, index) {
320+
start = start + this.offset;
321+
end = end + this.offset;
322+
index = index + this.offset;
323+
315324
if (index >= start && index <= end) throw new Error('Cannot move a selection inside itself');
316325

317326
if (DEBUG) this.stats.time('move');
@@ -358,6 +367,9 @@ export default class MagicString {
358367
}
359368

360369
update(start, end, content, options) {
370+
start = start + this.offset;
371+
end = end + this.offset;
372+
361373
if (typeof content !== 'string') throw new TypeError('replacement content must be a string');
362374

363375
if (this.original.length !== 0) {
@@ -433,6 +445,8 @@ export default class MagicString {
433445
}
434446

435447
prependLeft(index, content) {
448+
index = index + this.offset;
449+
436450
if (typeof content !== 'string') throw new TypeError('inserted content must be a string');
437451

438452
if (DEBUG) this.stats.time('insertRight');
@@ -452,6 +466,8 @@ export default class MagicString {
452466
}
453467

454468
prependRight(index, content) {
469+
index = index + this.offset;
470+
455471
if (typeof content !== 'string') throw new TypeError('inserted content must be a string');
456472

457473
if (DEBUG) this.stats.time('insertRight');
@@ -471,6 +487,9 @@ export default class MagicString {
471487
}
472488

473489
remove(start, end) {
490+
start = start + this.offset;
491+
end = end + this.offset;
492+
474493
if (this.original.length !== 0) {
475494
while (start < 0) start += this.original.length;
476495
while (end < 0) end += this.original.length;
@@ -501,6 +520,9 @@ export default class MagicString {
501520
}
502521

503522
reset(start, end) {
523+
start = start + this.offset;
524+
end = end + this.offset;
525+
504526
if (this.original.length !== 0) {
505527
while (start < 0) start += this.original.length;
506528
while (end < 0) end += this.original.length;
@@ -569,7 +591,10 @@ export default class MagicString {
569591
return this.intro + lineStr;
570592
}
571593

572-
slice(start = 0, end = this.original.length) {
594+
slice(start = 0, end = this.original.length - this.offset) {
595+
start = start + this.offset;
596+
end = end + this.offset;
597+
573598
if (this.original.length !== 0) {
574599
while (start < 0) start += this.original.length;
575600
while (end < 0) end += this.original.length;

src/index.d.ts

+3
Original file line numberDiff line numberDiff line change
@@ -107,6 +107,7 @@ export type ExclusionRange = [number, number];
107107
export interface MagicStringOptions {
108108
filename?: string;
109109
indentExclusionRanges?: ExclusionRange | Array<ExclusionRange>;
110+
offset?: number;
110111
}
111112

112113
export interface IndentOptions {
@@ -283,4 +284,6 @@ export default class MagicString {
283284
* Returns the generated string.
284285
*/
285286
toString(): string;
287+
288+
offset: number;
286289
}

test/MagicString.test.js

+18
Original file line numberDiff line numberDiff line change
@@ -1902,5 +1902,23 @@ describe('MagicString', () => {
19021902
message: 'MagicString.prototype.replaceAll called with a non-global RegExp argument',
19031903
});
19041904
});
1905+
1906+
it('with offset', () => {
1907+
const s = new MagicString('hello world', { offset: 6 });
1908+
assert.equal(s.slice(0, 5), 'world');
1909+
assert.equal(s.remove(0, 5).toString(), 'hello ');
1910+
assert.equal(s.prependLeft(0, 'w').toString(), 'hello w');
1911+
assert.equal(s.appendLeft(0, 'o').toString(), 'hello wo');
1912+
assert.equal(s.prependRight(0, 'r').toString(), 'hello wor');
1913+
assert.equal(s.appendRight(0, 'l').toString(), 'hello worl');
1914+
assert.equal(s.reset(4, 5).toString(), 'hello world');
1915+
assert.equal(s.update(0, 5, 'd').toString(), 'hello world');
1916+
assert.equal(s.overwrite(0, 5, 'rld').toString(), 'hello world');
1917+
1918+
s.offset = 1;
1919+
const s1 = s.clone();
1920+
assert.strictEqual(s1.slice(), 'ello world');
1921+
assert.equal(s1.move(0, 1, 2).slice(0), 'elo world');
1922+
});
19051923
});
19061924
});

0 commit comments

Comments
 (0)