Skip to content

Commit 1944d9f

Browse files
Implement babel 7 support (#763)
* Add option to switch between babel versions * Fix build * Fix babel 7 build * Don't opt in to babel 7 by default * Drop babel-loader * Add integration tests for babel 6 / 7 * It's late * Add test for working typescript transform * Add test for TSX rendering * Support serving TSX files from ASP.NET bundle transforms * Remove out of date IIS classic mode transform * Support serving TSX files on .NET Core * Fix build * Drop unused using
1 parent 790e7e8 commit 1944d9f

29 files changed

+5927
-1318
lines changed

build.proj

+5
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,10 @@ of patent rights can be found in the PATENTS file in the same directory.
5252
WorkingDirectory="src/React.Core"
5353
Command="npm install"
5454
/>
55+
<Exec
56+
WorkingDirectory="src/React.Core/Resources/babel-legacy"
57+
Command="npm install"
58+
/>
5559
<Exec
5660
WorkingDirectory="src/React.Sample.Webpack.CoreMvc"
5761
Command="npm install"
@@ -93,6 +97,7 @@ of patent rights can be found in the PATENTS file in the same directory.
9397

9498
<Target Name="Build" DependsOnTargets="RestorePackages;UpdateVersion">
9599
<Exec WorkingDirectory="src/React.Core" Command="node_modules/.bin/webpack" />
100+
<Exec WorkingDirectory="src/React.Core/Resources/babel-legacy" Command="node_modules/.bin/webpack" />
96101
<MSBuild Projects="$(SolutionFile)" Targets="Rebuild" Properties="Configuration=Release;Platform=Any CPU;NoWarn=1607,7035,1701;Version=$(VersionString)" />
97102
<Exec WorkingDirectory="src/React.Sample.Webpack.CoreMvc" Command="node_modules/.bin/webpack" />
98103
</Target>

src/React.AspNet.Middleware/BabelFileOptions.cs

+14-3
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
/*
1+
/*
22
* Copyright (c) Facebook, Inc. and its affiliates.
33
*
44
* This source code is licensed under the MIT license found in the
@@ -7,8 +7,10 @@
77
using System.Collections.Generic;
88
#if OWIN
99
using Microsoft.Owin.StaticFiles;
10+
using Microsoft.Owin.StaticFiles.ContentTypes;
1011
#else
1112
using Microsoft.AspNetCore.Builder;
13+
using Microsoft.AspNetCore.StaticFiles;
1214
#endif
1315

1416
#if OWIN
@@ -37,8 +39,17 @@ public class BabelFileOptions
3739
/// </summary>
3840
public BabelFileOptions()
3941
{
40-
Extensions = new[] { ".jsx" };
41-
StaticFileOptions = new StaticFileOptions();
42+
Extensions = new[] { ".jsx", ".tsx" };
43+
StaticFileOptions = new StaticFileOptions() { ContentTypeProvider = new JsxContentTypeProvider() } ;
44+
}
45+
46+
private class JsxContentTypeProvider : IContentTypeProvider
47+
{
48+
public bool TryGetContentType(string subpath, out string contentType)
49+
{
50+
contentType = "text/javascript";
51+
return true;
52+
}
4253
}
4354
}
4455
}

src/React.Core/Babel.cs

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
/*
1+
/*
22
* Copyright (c) Facebook, Inc. and its affiliates.
33
*
44
* This source code is licensed under the MIT license found in the
@@ -74,7 +74,7 @@ public Babel(IReactEnvironment environment, ICache cache, IFileSystem fileSystem
7474
_fileSystem = fileSystem;
7575
_fileCacheHash = fileCacheHash;
7676
_config = siteConfig;
77-
_babelConfig = siteConfig.BabelConfig.Serialize();
77+
_babelConfig = siteConfig.BabelConfig.Serialize(_config.BabelVersion);
7878
}
7979

8080
/// <summary>

src/React.Core/BabelConfig.cs

+26-18
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
1-
using System.Collections.Generic;
1+
using System;
2+
using System.Collections.Generic;
23
using Newtonsoft.Json;
34
using Newtonsoft.Json.Serialization;
5+
using React.Exceptions;
46

57
namespace React
68
{
@@ -22,28 +24,34 @@ public class BabelConfig
2224
/// </summary>
2325
public ISet<string> Presets { get; set; }
2426

25-
/// <summary>
26-
/// Creates a new instance of <see cref="BabelConfig" />.
27-
/// </summary>
28-
public BabelConfig()
29-
{
30-
// Use es2015-no-commonjs by default so Babel doesn't prepend "use strict" to the start of the
31-
// output. This messes with the top-level "this", as we're not actually using JavaScript modules
32-
// in ReactJS.NET yet.
33-
Presets = new HashSet<string> { "es2015-no-commonjs", "stage-1", "react" };
34-
Plugins = new HashSet<string>();
35-
}
36-
3727
/// <summary>
3828
/// Serializes this Babel configuration into the format required for Babel.
3929
/// </summary>
4030
/// <returns></returns>
41-
public string Serialize()
31+
public string Serialize(string babelVersion)
4232
{
43-
return JsonConvert.SerializeObject(this, new JsonSerializerSettings
44-
{
45-
ContractResolver = new CamelCasePropertyNamesContractResolver(),
46-
});
33+
ISet<string> defaultPresets = babelVersion == BabelVersions.Babel7
34+
? new HashSet<string> { "typescript", "react" }
35+
: babelVersion == BabelVersions.Babel6 || babelVersion == null
36+
? new HashSet<string> { "es2015-no-commonjs", "stage-1", "react" }
37+
: throw new ArgumentException(nameof(babelVersion));
38+
39+
ISet<string> defaultPlugins = babelVersion == BabelVersions.Babel7
40+
? new HashSet<string> { "proposal-class-properties", "proposal-object-rest-spread" }
41+
: babelVersion == BabelVersions.Babel6 || babelVersion == null
42+
? new HashSet<string>()
43+
: throw new ArgumentException(nameof(babelVersion));
44+
45+
return JsonConvert.SerializeObject(
46+
new BabelConfig
47+
{
48+
Plugins = Plugins ?? defaultPlugins,
49+
Presets = Presets ?? defaultPresets,
50+
},
51+
new JsonSerializerSettings
52+
{
53+
ContractResolver = new CamelCasePropertyNamesContractResolver(),
54+
});
4755
}
4856
}
4957
}

src/React.Core/BabelVersions.cs

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
using System;
2+
using System.Collections.Generic;
3+
using System.Linq;
4+
using System.Text;
5+
6+
namespace React
7+
{
8+
/// <summary>
9+
/// Public string values for babel versions
10+
/// </summary>
11+
public static class BabelVersions
12+
{
13+
/// <summary>
14+
/// Babel 6
15+
/// </summary>
16+
public static readonly string Babel6 = "babel-6";
17+
18+
/// <summary>
19+
/// Babel 7
20+
/// </summary>
21+
public static readonly string Babel7 = "babel-7";
22+
}
23+
}

src/React.Core/IReactSiteConfiguration.cs

+11
Original file line numberDiff line numberDiff line change
@@ -158,6 +158,17 @@ public interface IReactSiteConfiguration
158158
/// <returns>The configuration, for chaining</returns>
159159
IReactSiteConfiguration SetBabelConfig(BabelConfig value);
160160

161+
/// <summary>
162+
/// Gets or sets the Babel version to use. Supports "babel@6 or babel@7".
163+
/// </summary>
164+
string BabelVersion { get; set; }
165+
166+
/// <summary>
167+
/// Sets the Babel configuration to use.
168+
/// </summary>
169+
/// <returns>The configuration, for chaining</returns>
170+
IReactSiteConfiguration SetBabelVersion(string value);
171+
161172
/// <summary>
162173
/// Gets or sets whether to use the debug version of React. This is slower, but gives
163174
/// useful debugging tips.

src/React.Core/React.Core.csproj

+9-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
<Project Sdk="Microsoft.NET.Sdk">
1+
<Project Sdk="Microsoft.NET.Sdk">
22

33
<PropertyGroup>
44
<Description>ReactJS and Babel tools for .NET. Important: This package does not do much on its own; you probably want an integration package (like React.Web.Mvc4) as well. Please refer to project site (http://reactjs.net/) for more details, usage examples and sample code.</Description>
@@ -21,13 +21,21 @@
2121

2222
<ItemGroup>
2323
<Compile Remove="node_modules\**" />
24+
<Compile Remove="Resources\babel-legacy\node_modules\**" />
2425
<EmbeddedResource Remove="node_modules\**" />
26+
<EmbeddedResource Remove="Resources\babel-legacy\node_modules\**" />
2527
<None Remove="node_modules\**" />
28+
<None Remove="Resources\babel-legacy\node_modules\**" />
29+
</ItemGroup>
30+
31+
<ItemGroup>
32+
<None Remove="Resources\babel-legacy.generated.min.js" />
2633
</ItemGroup>
2734

2835
<ItemGroup>
2936
<Compile Include="..\SharedAssemblyInfo.cs" />
3037
<Compile Include="..\SharedAssemblyVersionInfo.cs" />
38+
<EmbeddedResource Include="Resources\babel-legacy.generated.min.js" />
3139
<EmbeddedResource Include="Resources\shims.js;Resources\react.generated.js;Resources\react.generated.min.js;Resources\babel.generated.min.js" />
3240
</ItemGroup>
3341

src/React.Core/ReactEnvironment.cs

+5-1
Original file line numberDiff line numberDiff line change
@@ -480,7 +480,11 @@ private void EnsureBabelLoaded(IJsEngine engine)
480480
#else
481481
var assembly = typeof(ReactEnvironment).GetTypeInfo().Assembly;
482482
#endif
483-
const string resourceName = "React.Core.Resources.babel.generated.min.js";
483+
string resourceName = _config.BabelVersion == BabelVersions.Babel7
484+
? "React.Core.Resources.babel.generated.min.js"
485+
: _config.BabelVersion == BabelVersions.Babel6 || _config.BabelVersion == null
486+
? "React.Core.Resources.babel-legacy.generated.min.js"
487+
: throw new ReactConfigurationException("BabelVersion was not null, but did not contain a valid value.");
484488

485489
if (_config.AllowJavaScriptPrecompilation
486490
&& engine.TryExecuteResourceWithPrecompilation(_cache, resourceName, assembly))

src/React.Core/ReactSiteConfiguration.cs

+15
Original file line numberDiff line numberDiff line change
@@ -280,6 +280,21 @@ public IReactSiteConfiguration SetBabelConfig(BabelConfig value)
280280
return this;
281281
}
282282

283+
/// <summary>
284+
/// Gets or sets the Babel version to use. Supports "babel@6 or babel@7".
285+
/// </summary>
286+
public string BabelVersion { get; set; }
287+
288+
/// <summary>
289+
/// Sets the Babel version to use.
290+
/// </summary>
291+
/// <returns>The configuration, for chaining</returns>
292+
public IReactSiteConfiguration SetBabelVersion(string value)
293+
{
294+
BabelVersion = value;
295+
return this;
296+
}
297+
283298
/// <summary>
284299
/// Gets or sets whether to use the debug version of React. This is slower, but gives
285300
/// useful debugging tips.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
/**
2+
* Copyright (c) Facebook, Inc. and its affiliates.
3+
*
4+
* This source code is licensed under the MIT license found in the
5+
* LICENSE file in the root directory of this source tree.
6+
*/
7+
8+
import {transform as babelTransform, version as babelVersion} from 'babel-standalone';
9+
10+
export function ReactNET_transform(input, babelConfig, filename) {
11+
babelConfig = {
12+
...JSON.parse(babelConfig),
13+
ast: false,
14+
filename,
15+
}
16+
try {
17+
return babelTransform(input, babelConfig).code;
18+
} catch (ex) {
19+
// Parsing stack is extremely long and not very useful, so just rethrow the message.
20+
throw new Error(ex.message);
21+
}
22+
}
23+
24+
export function ReactNET_transform_sourcemap(input, babelConfig, filename) {
25+
babelConfig = {
26+
...JSON.parse(babelConfig),
27+
ast: false,
28+
filename,
29+
sourceMaps: true,
30+
};
31+
try {
32+
var result = babelTransform(input, babelConfig);
33+
return JSON.stringify({
34+
babelVersion,
35+
code: result.code,
36+
sourceMap: result.map
37+
});
38+
} catch (ex) {
39+
// Parsing stack is extremely long and not very useful, so just rethrow the message.
40+
throw new Error(ex.message);
41+
}
42+
}

0 commit comments

Comments
 (0)