fix(aio): preserve newlines when copying code
Before4f37f8643
, we were using `innerText` to retrieved the code content for copying. This preserved the text layout (including newlines), but suffered from other issues (browser support, performance). With4f37f8643
we switched to `textContent`, which works well except in the following case: When `prettify` formats the code to have line numbers, it removes the newlines and uses `<li>` elements instead. This affects `textContent`. This commit fixes this by keeping a reference of the code as text and using that for copying. Fixes #17659
This commit is contained in:
parent
414c7e956b
commit
87206e1986
|
@ -218,7 +218,29 @@ describe('CodeComponent', () => {
|
||||||
const copierService: CopierService = TestBed.get(CopierService);
|
const copierService: CopierService = TestBed.get(CopierService);
|
||||||
const spy = spyOn(copierService, 'copyText');
|
const spy = spyOn(copierService, 'copyText');
|
||||||
getButton().click();
|
getButton().click();
|
||||||
expect(spy.calls.argsFor(0)[0]).toEqual(oneLineCode, 'after click');
|
expect(spy.calls.argsFor(0)[0]).toBe(oneLineCode, 'after click');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should preserve newlines in the copied code', () => {
|
||||||
|
const copierService: CopierService = TestBed.get(CopierService);
|
||||||
|
const spy = spyOn(copierService, 'copyText');
|
||||||
|
const expectedCode = smallMultiLineCode.trim().replace(/</g, '<').replace(/>/g, '>');
|
||||||
|
let actualCode;
|
||||||
|
|
||||||
|
hostComponent.code = smallMultiLineCode;
|
||||||
|
|
||||||
|
[false, true, 42].forEach(linenums => {
|
||||||
|
hostComponent.linenums = linenums;
|
||||||
|
fixture.detectChanges();
|
||||||
|
codeComponent.ngOnChanges();
|
||||||
|
getButton().click();
|
||||||
|
actualCode = spy.calls.mostRecent().args[0];
|
||||||
|
|
||||||
|
expect(actualCode).toBe(expectedCode, `when linenums=${linenums}`);
|
||||||
|
expect(actualCode.match(/\r?\n/g).length).toBe(5);
|
||||||
|
|
||||||
|
spy.calls.reset();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should display a message when copy succeeds', () => {
|
it('should display a message when copy succeeds', () => {
|
||||||
|
|
|
@ -51,6 +51,11 @@ export class CodeComponent implements OnChanges {
|
||||||
@Input()
|
@Input()
|
||||||
code: string;
|
code: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The code to be copied when clicking the copy button, this should not be HTML encoded
|
||||||
|
*/
|
||||||
|
private codeText: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* set to true if the copy button is not to be shown
|
* set to true if the copy button is not to be shown
|
||||||
*/
|
*/
|
||||||
|
@ -116,6 +121,7 @@ export class CodeComponent implements OnChanges {
|
||||||
const linenums = this.getLinenums();
|
const linenums = this.getLinenums();
|
||||||
|
|
||||||
this.setCodeHtml(this.code); // start with unformatted code
|
this.setCodeHtml(this.code); // start with unformatted code
|
||||||
|
this.codeText = this.getCodeText(); // store the unformatted code as text (for copying)
|
||||||
this.pretty.formatCode(this.code, this.language, linenums).subscribe(
|
this.pretty.formatCode(this.code, this.language, linenums).subscribe(
|
||||||
formattedCode => this.setCodeHtml(formattedCode),
|
formattedCode => this.setCodeHtml(formattedCode),
|
||||||
err => { /* ignore failure to format */ }
|
err => { /* ignore failure to format */ }
|
||||||
|
@ -128,9 +134,15 @@ export class CodeComponent implements OnChanges {
|
||||||
this.codeContainer.nativeElement.innerHTML = formattedCode;
|
this.codeContainer.nativeElement.innerHTML = formattedCode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private getCodeText() {
|
||||||
|
// `prettify` may remove newlines, e.g. when `linenums` are on. Retrieve the content of the
|
||||||
|
// container as text, before prettifying it.
|
||||||
|
// We take the textContent because we don't want it to be HTML encoded.
|
||||||
|
return this.codeContainer.nativeElement.textContent;
|
||||||
|
}
|
||||||
|
|
||||||
doCopy() {
|
doCopy() {
|
||||||
// We take the textContent because we don't want it to be HTML encoded
|
const code = this.codeText;
|
||||||
const code = this.codeContainer.nativeElement.textContent.trim();
|
|
||||||
if (this.copier.copyText(code)) {
|
if (this.copier.copyText(code)) {
|
||||||
this.logger.log('Copied code to clipboard:', code);
|
this.logger.log('Copied code to clipboard:', code);
|
||||||
// success snackbar alert
|
// success snackbar alert
|
||||||
|
|
Loading…
Reference in New Issue