-
Notifications
You must be signed in to change notification settings - Fork 12.6k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Emit does not preserve empty lines #843
Comments
Adding some background info... The reason the new compiler removes all blank lines is that this is the only thing we can do consistently. Blank line preservation is always going to be a hit or miss thing when it comes to constructs that are subject to rewriting, e.g. class and module declarations. We face exactly the same issues with comment preservation. So, while I'm sympathetic to resolving this, it's not an easy thing to do. @NoelAbrahams I'm curious what issues you see when debugging? |
@ahejlsberg @NoelAbrahams I created a prototype emitted in the original CodePlex project that actually did a terrific job with comment and newline preservation. Even on constructs that got heavily rewritten (like arrow-functions to function expressions) it would almost always do what you intuitively expected it to do. Line preservation mostly a function of just reusing the line breaks when preserving old code, and using relative spacing when doing transformations. i.e. if you have:
Then when you're emitting the IIFE for 'N' you say "we should keep the trivia between the module we're rewriting and the previous syntactic element". Here is the before/after for how my emitter prototype worked that preserved comments/newlines with high levels of fidelity: https://typescript.codeplex.com/SourceControl/latest#tests/Fidelity/emitter2/ecmascript5/Parser.ts You can see a lot of examples here as well: |
The only feature i didn't implement was 'alignment'. i.e. if you had code that was aligned in some way in the original (very common with parameter declarations), we'd want the emitted code to preserve that as well. But it would have been very trivial to do. That said, conversion of contructs from TS to JS did try to preserve indentation. You can see that here: Note how the statements are properly indendented (even when they spread over multiple lines) even after we convert the nested modules and class into IIFEs |
@ahejlsberg, there are no significant issues when debugging in the browser. It just makes it easer to navigate the JavaScript code and locate lines for setting breakpoints when there is an exact correspondence with the actual TypeScript source code. I could personally live without the empty lines, but since TS has gone to such lengths to preserve and emit _beautiful_ JavaScript (:smile:) it seems only natural that this is implemented. |
@ahejlsberg @NoelAbrahams There is one issue that exists with debugging in the browser that is slightly related to this conversation. When using setter/getter chains (like with jquery) or chaining promises, the new line feeds are lost during translation. That being said, it is a huge pain point when working with Arrow functions. As an example:
Becomes:
Using Chrome and sourceMaps, the breakpoints are still skipped. |
@mtraynham, actually I think the problem you highlight is a slightly different one. In previous versions the body of an inline function was always emitted on new lines: // TS
var x = () => 'foo';
// JS - old
var x = function () {
return 'foo';
};
// JS - new
var x = function () { return 'foo'; }; I too have found this to be a problem - having to occasionally go back and create a |
@NoelAbrahams Ahh yes, I have been using this exact same temp solution... I wasn't sure if this was an appropriate bug to contribute this problem to (erasure of line feeds), or should I open another? I created #2259 to the separate issue. |
As an engineering director exploring moving our javascript development community to typescript the new line capability would really help. One of the primary appeal of typescript was maintaining the readability and structure of the code created in typescript generatedJavascript. Great to see comments are kept in the recent update and addition of the “--removeComments" command line directive. |
I would also like this for a similar reason as @timjmartel - when developers see the emitted JS looks nice they are less resistant to adopt. Preserving (at least vertical) whitespace makes the code look less like it was generated by a machine and more like idomatic JS code written by a human. If our team ever decided to abandon TS and instead continue with the transpiled JS it would be much easier to adopt the emitted JS sources if they had human-friendly whitespace. |
In regard to empty lines, would it be possible - for now - to have them emitted, even if they are hit or miss? Such experimental feature could be asked with a “--keepEmptyLines" option. It would be not so much for having nice JS, but for having more readable JS. In regard to chained function calls, would it be possible to have one call for line, to let users set breakpoints? Again, this feature could be asked with a "--oneCallForLine" option, if it is another "hit or miss" thing. Thanks for your attention. |
Actually, chained function calls could be split by a source code beautifier, therefore it doesn't make sense to embed such feature in TypeScript. |
This shouldn't be that hard, shouldn't there just be an option in tsconfig.json
? |
But I am not seeing this as a compiler option: https://www.typescriptlang.org/docs/handbook/compiler-options.html |
I just realized. One good reason to not keep whitespace is that it will really help prevent you from accidentally editing the .js instead of the .ts. I guess one thing to do would be to write out the .js files as readonly/execute only. So maybe this is a non-issue. That being said, I don't think tsc writes out .js files as readonly/execute only, is there a way to configure tsc to do this? |
No, not at the moment. Feel free to open up a separate issue for that though. |
@DanielRosenwasser you're saying if we want to preserve whitespace that we should open a separate issue? Is this issue not enough? Nevermind LOL more than a month later I realize you were saying to open a separate issue for read/write/execution permissions on transpiled files :) |
It would be nice to have this. |
+1 |
2 similar comments
+1 |
+1 |
The crowd wants |
Reason this is important is that TypeScript is supposed to "degrade gracefully" to JS. Right now it can't preserve new lines making the JS a bit dense to read, especially if you write TypeScript, but you are supposed to deliver JS to some place else. |
+1 Temporary hackUse esformatter to add linebreaks. With the following configuration file: {
"lineBreak": {
"before": {
"FunctionDeclaration": ">=2",
"FunctionDeclarationOpeningBrace": 0,
"FunctionDeclarationClosingBrace": 1,
"MethodDefinition": ">=2",
"ClassDeclaration": ">=2"
},
"after": {
"FunctionDeclaration": ">=2",
"FunctionDeclarationOpeningBrace": 1,
"MethodDefinitionClosingBrace": ">=2",
"ClassClosingBrace": ">=2"
}
}
} |
@mtraynham Your example:
with the latest compiler produces this JS:
(see TS PlayGround https://goo.gl/JViurr ) A lot has changed since TS Version 1.1 (the version of TypeScript for which this issue was created). Maybe this issue can be closed? @ahejlsberg ? |
@valera-rozuvan I had opened #2259 instead. My issue was related to line feeds, but not the exact same issue described by this bug. #2259 was closed a while back (May 2015). |
@valera-rozuvan depending on the size of your project. For my use cases where I transpile 10-ish files of 100-1000 LOC, it doesn't introduce any noticeable delay. |
Any solutions here yet? I run in this trouble too... |
I was part way into trying to fix this in the compiler itself when my teammate @emadum reminded me that tsc can preserve comments. Here's a little gulp pipeline that seems to do a fairly decent job of preserving newlines:
I think a more clever comment would prevent some false positives, but it seems to work well for us today. I've only tested this on a relatively small file, but the performance overhead there was minimal. hth |
I ended up using your idea as a base, and eventually expanded upon it until it became an npm module: I credited your post in the Readme, hopefully you don't mind :) |
Is there a way I could contribute to fixing this? If I could get some guidance around what an acceptable fix might look like, I'd be willing to take a stab at it. |
problem: in compilation tsc removes blank lines from the file this causes problem in debugging and mapping with IDEs microsoft/TypeScript#843 Signed-off-by: Pranam Lashkari <[email protected]> Change-Id: Ie2df89ba09597a34aca9221bf082b1f2df3c7d07
problem: in compilation tsc removes blank lines from the file this causes problem in debugging and mapping with IDEs microsoft/TypeScript#843 Signed-off-by: Pranam Lashkari <[email protected]> Change-Id: Ie2df89ba09597a34aca9221bf082b1f2df3c7d07
@CarabusX I took slightly minimalistic approach, but very similar to yours:
however it doesn't work well with imports |
I tried many suggestions from this thread but none worked for me. After a while, however, I managed to come up with my own solution. Before I share the code, I should note that this code relies on Prettier for formatting to get the JS code consistent with the TS code. I already use Prettier in all my projects, so I just copied the config I am using anyway. One final note, as my use case for this was the Docusaurus website for https://github.com/sapphiredev, I have also decided to publish this conversion and formatting as an NPM package: https://www.npmjs.com/package/@sapphire/docusaurus-plugin-ts2esm2cjs Now on to the code. This transpiles TS into ESM and then further converts that ESM into CJS. const { runTransform } = require('esm-to-cjs');
const ts = require('typescript');
const prettier = require('prettier');
const sapphirePrettierConfig = require('@sapphire/prettier-config');
const documentationPrettierConfig = {
...sapphirePrettierConfig,
tabWidth: 2,
useTabs: false,
printWidth: 120,
parser: 'babel'
};
/** @type {import('ts').CompilerOptions} */
const tsCompilerOptions = {
module: ts.ModuleKind.ESNext,
newLine: ts.NewLineKind.LineFeed,
moduleResolution: ts.ModuleResolutionKind.NodeJs,
target: ts.ScriptTarget.ESNext,
removeComments: false,
esModuleInterop: true,
pretty: true
};
/**
* Transpiles input TypeScript code to ESM code.
* @param {string} code The code to transpile
* @returns {{ outputText: string; diagnostics: unknown[]; sourceMapText?: string }} Input code transpiled to ESM
*/
const tsToEsm = (code) => ts.transpileModule(code, { reportDiagnostics: false, compilerOptions: tsCompilerOptions });
/**
* Transforms input ESM code to CJS code.
* @param {string} code The code to transform
* @returns Input code transformed to CommonJS
*/
const esmToCjs = (code) => runTransform(code, { quote: 'single' });
/**
* Escaped new lines in code with block comments so they can be restored by {@link restoreNewLines}
* @param {string} code The code to escape new lines in
* @returns The same code but with new lines escaped using block comments
*/
const escapeNewLines = (code) => code.replace(/\n\n/g, '\n/* :newline: */');
/**
* Reverses {@link escapeNewLines} and restores new lines
* @param {string} code The code with escaped new lines
* @returns The same code with new lines restored
*/
const restoreNewLines = (code) => code.replace(/\/\* :newline: \*\//g, '\n');
/**
* Formats the code using Prettier
* @param {string} code The code to prettier format
* @returns Prettier formatted code
*/
const prettierFormatCode = (code) => prettier.format(code, documentationPrettierConfig).slice(0, -1);
/**
* The example code that we'll be transpiling and formatting.
* Of note here is that it also has a type only import that will be removed during transpilation,
* and it needs to be accounted for with the restoration of new lines.
*/
const codeStr = `import { Command, Args } from '@sapphire/framework';
import type { Message } from 'discord.js';
export class EchoCommand extends Command {
public constructor(context: Command.Context, options: Command.Options) {
super(context, {
...options,
aliases: ['parrot', 'copy'],
description: 'Replies with the text you provide'
});
}
public async messageRun(message: Message, args: Args) {
// ...
}
}`;
const emptyLineEscapedTsCode = escapeNewLines(codeStr);
const esmCode = tsToEsm(emptyLineEscapedTsCode).outputText;
const cjsCode = esmToCjs(esmCode); |
I would also like to see this implemented, with a few specific reasons in mind:
This PR, discussed in the 3.9 release notes, show code actions preserving newlines; why can't the emitted code do something similar? |
Are there any updates for almost 10 years? |
I'm changing this to Won't Fix. Reasons:
|
Does anyone know of a good option for this? |
It depends on your exact needs. There are several suggestions in this issue and a list of other transpilers is as follows. Note that this list hasn't been sanitized for the specific requirement:
|
Hi,
TS Version: 1.1
Given
We used to get
In the new compiler the line break is missing
(Both compilers removed the first empty line, but the new compiler has gone a step further.)
This can affect the experience when debugging JavaScript in the browser.
The text was updated successfully, but these errors were encountered: