From a6681c8943c113c9273aa2cd016845a29fecf508 Mon Sep 17 00:00:00 2001
From: Kushagra Gour <chinchang457@gmail.com>
Date: Sat, 3 Nov 2018 12:42:21 +0530
Subject: [PATCH] integrate lingui for i18n. fixes #170

---
 .babelrc                   |  13 +-
 package.json               |   9 +-
 preact.config.js           |   5 +
 src/components/Console.jsx |   2 +
 src/components/app.jsx     | 458 +++++++++++++++++++------------------
 yarn.lock                  | 362 +++++++++++++++++++++++++++--
 6 files changed, 603 insertions(+), 246 deletions(-)

diff --git a/.babelrc b/.babelrc
index 156eaae..43925d1 100644
--- a/.babelrc
+++ b/.babelrc
@@ -1,9 +1,8 @@
 {
-  "env": {
-    "test": {
-      "presets": [
-        ["preact-cli/babel", { "modules": "commonjs" }]
-      ]
-    }
-  }
+	"presets": ["preact-cli/babel"],
+	"env": {
+		"test": {
+			"presets": [["preact-cli/babel", { "modules": "commonjs" }]]
+		}
+	}
 }
diff --git a/package.json b/package.json
index dd62ceb..4552c80 100644
--- a/package.json
+++ b/package.json
@@ -9,7 +9,10 @@
 		"dev": "preact watch --template src/index.html --https --no-prerender",
 		"lint": "eslint src",
 		"test": "jest",
-		"precommit": "lint-staged"
+		"precommit": "lint-staged",
+		"add-locale": "lingui add-locale",
+		"extract": "lingui extract",
+		"compile": "lingui compile"
 	},
 	"eslintConfig": {
 		"extends": "eslint-config-synacor"
@@ -28,6 +31,9 @@
 		]
 	},
 	"devDependencies": {
+		"@lingui/cli": "^2.7.0",
+		"@lingui/macro": "^2.7.0",
+		"babel-core": "^6.26.3",
 		"babel-eslint": "^7.2.3",
 		"babel-minify": "^0.2.0",
 		"eslint": "^4.9.0",
@@ -53,6 +59,7 @@
 	},
 	"dependencies": {
 		"@emmetio/codemirror-plugin": "^0.5.4",
+		"@lingui/react": "^2.7.0",
 		"code-blast-codemirror": "chinchang/code-blast-codemirror#web-maker",
 		"codemirror": "^5.37.0",
 		"copy-webpack-plugin": "^4.5.1",
diff --git a/preact.config.js b/preact.config.js
index 7b03ade..d6da789 100644
--- a/preact.config.js
+++ b/preact.config.js
@@ -21,6 +21,11 @@ export default function(config, env, helpers) {
 	htmlWebpackPlugin.plugin.options.preload = false;
 	htmlWebpackPlugin.plugin.options.favicon = false;
 
+	// Required for lingui-macros
+	let { rule } = helpers.getLoadersByName(config, 'babel-loader')[0];
+	let babelConfig = rule.options;
+	babelConfig.plugins.push('macros');
+
 	if (env.isProd) {
 		config.devtool = false; // disable sourcemaps
 
diff --git a/src/components/Console.jsx b/src/components/Console.jsx
index a044c52..07e93b0 100644
--- a/src/components/Console.jsx
+++ b/src/components/Console.jsx
@@ -3,6 +3,8 @@ import { PureComponent } from 'preact-compat';
 
 import { Inspector, chromeDark } from 'react-inspector';
 
+import { Trans } from '@lingui/macro';
+
 class LogRow extends Component {
 	shouldComponentUpdate() {
 		return false;
diff --git a/src/components/app.jsx b/src/components/app.jsx
index 183747b..1da95e6 100644
--- a/src/components/app.jsx
+++ b/src/components/app.jsx
@@ -64,11 +64,13 @@ import {
 	SHOW_KEYBOARD_SHORTCUTS_EVENT
 } from '../commands';
 import { commandPaletteService } from '../commandPaletteService';
-
+// import hiTranslations from '../locales/hi/messages';
+import { I18nProvider } from '@lingui/react';
+const hiTranslations = {};
 if (module.hot) {
 	require('preact/debug');
 }
-
+const catalogs = { hi: hiTranslations };
 const LocalStorageKeys = {
 	LOGIN_AND_SAVE_MESSAGE_SEEN: 'loginAndsaveMessageSeen',
 	ASKED_TO_IMPORT_CREATIONS: 'askedToImportCreations'
@@ -1396,245 +1398,255 @@ export default class App extends Component {
 
 	render() {
 		return (
-			<div class={this.getRootClasses()}>
-				<div class="main-container">
-					<MainHeader
-						externalLibCount={this.state.externalLibCount}
-						openBtnHandler={this.openBtnClickHandler.bind(this)}
-						newBtnHandler={this.newBtnClickHandler.bind(this)}
-						saveBtnHandler={this.saveBtnClickHandler.bind(this)}
-						loginBtnHandler={this.loginBtnClickHandler.bind(this)}
-						profileBtnHandler={this.profileBtnClickHandler.bind(this)}
-						addLibraryBtnHandler={this.openAddLibrary.bind(this)}
-						runBtnClickHandler={this.runBtnClickHandler.bind(this)}
-						isFetchingItems={this.state.isFetchingItems}
-						isSaving={this.state.isSaving}
-						title={this.state.currentItem.title}
-						titleInputBlurHandler={this.titleInputBlurHandler.bind(this)}
-						user={this.state.user}
-						unsavedEditCount={this.state.unsavedEditCount}
-						isFileMode={this.state.currentItem && this.state.currentItem.files}
-					/>
-					{this.state.currentItem && this.state.currentItem.files ? (
-						<ContentWrapFiles
-							currentItem={this.state.currentItem}
-							onCodeChange={this.onCodeChange.bind(this)}
-							onCodeSettingsChange={this.onCodeSettingsChange.bind(this)}
-							onCodeModeChange={this.onCodeModeChange.bind(this)}
-							onRef={comp => (this.contentWrap = comp)}
-							prefs={this.state.prefs}
-							onEditorFocus={this.editorFocusHandler.bind(this)}
-							onSplitUpdate={this.splitUpdateHandler.bind(this)}
-							onAddFile={this.addFileHandler.bind(this)}
-							onRemoveFile={this.removeFileHandler.bind(this)}
-							onRenameFile={this.renameFileHandler.bind(this)}
-							onFileDrop={this.fileDropHandler.bind(this)}
-							onFolderSelect={this.folderSelectHandler.bind(this)}
-							onPrettifyBtnClick={this.prettifyHandler.bind(this)}
+			<I18nProvider language="hi" catalogs={catalogs}>
+				<div class={this.getRootClasses()}>
+					<div class="main-container">
+						<MainHeader
+							externalLibCount={this.state.externalLibCount}
+							openBtnHandler={this.openBtnClickHandler.bind(this)}
+							newBtnHandler={this.newBtnClickHandler.bind(this)}
+							saveBtnHandler={this.saveBtnClickHandler.bind(this)}
+							loginBtnHandler={this.loginBtnClickHandler.bind(this)}
+							profileBtnHandler={this.profileBtnClickHandler.bind(this)}
+							addLibraryBtnHandler={this.openAddLibrary.bind(this)}
+							runBtnClickHandler={this.runBtnClickHandler.bind(this)}
+							isFetchingItems={this.state.isFetchingItems}
+							isSaving={this.state.isSaving}
+							title={this.state.currentItem.title}
+							titleInputBlurHandler={this.titleInputBlurHandler.bind(this)}
+							user={this.state.user}
+							unsavedEditCount={this.state.unsavedEditCount}
+							isFileMode={
+								this.state.currentItem && this.state.currentItem.files
+							}
 						/>
-					) : (
-						<ContentWrap
-							currentLayoutMode={this.state.currentLayoutMode}
-							currentItem={this.state.currentItem}
-							onCodeChange={this.onCodeChange.bind(this)}
-							onCodeSettingsChange={this.onCodeSettingsChange.bind(this)}
-							onCodeModeChange={this.onCodeModeChange.bind(this)}
-							onRef={comp => (this.contentWrap = comp)}
+						{this.state.currentItem && this.state.currentItem.files ? (
+							<ContentWrapFiles
+								currentItem={this.state.currentItem}
+								onCodeChange={this.onCodeChange.bind(this)}
+								onCodeSettingsChange={this.onCodeSettingsChange.bind(this)}
+								onCodeModeChange={this.onCodeModeChange.bind(this)}
+								onRef={comp => (this.contentWrap = comp)}
+								prefs={this.state.prefs}
+								onEditorFocus={this.editorFocusHandler.bind(this)}
+								onSplitUpdate={this.splitUpdateHandler.bind(this)}
+								onAddFile={this.addFileHandler.bind(this)}
+								onRemoveFile={this.removeFileHandler.bind(this)}
+								onRenameFile={this.renameFileHandler.bind(this)}
+								onFileDrop={this.fileDropHandler.bind(this)}
+								onFolderSelect={this.folderSelectHandler.bind(this)}
+								onPrettifyBtnClick={this.prettifyHandler.bind(this)}
+							/>
+						) : (
+							<ContentWrap
+								currentLayoutMode={this.state.currentLayoutMode}
+								currentItem={this.state.currentItem}
+								onCodeChange={this.onCodeChange.bind(this)}
+								onCodeSettingsChange={this.onCodeSettingsChange.bind(this)}
+								onCodeModeChange={this.onCodeModeChange.bind(this)}
+								onRef={comp => (this.contentWrap = comp)}
+								prefs={this.state.prefs}
+								onEditorFocus={this.editorFocusHandler.bind(this)}
+								onSplitUpdate={this.splitUpdateHandler.bind(this)}
+							/>
+						)}
+
+						<Footer
 							prefs={this.state.prefs}
-							onEditorFocus={this.editorFocusHandler.bind(this)}
-							onSplitUpdate={this.splitUpdateHandler.bind(this)}
+							layoutBtnClickHandler={this.layoutBtnClickHandler.bind(this)}
+							helpBtnClickHandler={() =>
+								this.setState({ isHelpModalOpen: true })
+							}
+							settingsBtnClickHandler={this.openSettings.bind(this)}
+							notificationsBtnClickHandler={this.notificationsBtnClickHandler.bind(
+								this
+							)}
+							supportDeveloperBtnClickHandler={this.supportDeveloperBtnClickHandler.bind(
+								this
+							)}
+							detachedPreviewBtnHandler={this.detachedPreviewBtnHandler.bind(
+								this
+							)}
+							codepenBtnClickHandler={this.codepenBtnClickHandler.bind(this)}
+							saveHtmlBtnClickHandler={this.saveHtmlBtnClickHandler.bind(this)}
+							keyboardShortcutsBtnClickHandler={this.openKeyboardShortcuts.bind(
+								this
+							)}
+							screenshotBtnClickHandler={this.screenshotBtnClickHandler.bind(
+								this
+							)}
+							onJs13KHelpBtnClick={this.js13KHelpBtnClickHandler.bind(this)}
+							onJs13KDownloadBtnClick={this.js13KDownloadBtnClickHandler.bind(
+								this
+							)}
+							hasUnseenChangelog={this.state.hasUnseenChangelog}
+							codeSize={this.state.codeSize}
 						/>
-					)}
+					</div>
 
-					<Footer
-						prefs={this.state.prefs}
-						layoutBtnClickHandler={this.layoutBtnClickHandler.bind(this)}
-						helpBtnClickHandler={() => this.setState({ isHelpModalOpen: true })}
-						settingsBtnClickHandler={this.openSettings.bind(this)}
-						notificationsBtnClickHandler={this.notificationsBtnClickHandler.bind(
+					<SavedItemPane
+						items={this.state.savedItems}
+						isOpen={this.state.isSavedItemPaneOpen}
+						closeHandler={this.closeSavedItemsPane.bind(this)}
+						itemClickHandler={this.itemClickHandler.bind(this)}
+						itemRemoveBtnClickHandler={this.itemRemoveBtnClickHandler.bind(
 							this
 						)}
-						supportDeveloperBtnClickHandler={this.supportDeveloperBtnClickHandler.bind(
-							this
-						)}
-						detachedPreviewBtnHandler={this.detachedPreviewBtnHandler.bind(
-							this
-						)}
-						codepenBtnClickHandler={this.codepenBtnClickHandler.bind(this)}
-						saveHtmlBtnClickHandler={this.saveHtmlBtnClickHandler.bind(this)}
-						keyboardShortcutsBtnClickHandler={this.openKeyboardShortcuts.bind(
-							this
-						)}
-						screenshotBtnClickHandler={this.screenshotBtnClickHandler.bind(
-							this
-						)}
-						onJs13KHelpBtnClick={this.js13KHelpBtnClickHandler.bind(this)}
-						onJs13KDownloadBtnClick={this.js13KDownloadBtnClickHandler.bind(
-							this
-						)}
-						hasUnseenChangelog={this.state.hasUnseenChangelog}
-						codeSize={this.state.codeSize}
+						itemForkBtnClickHandler={this.itemForkBtnClickHandler.bind(this)}
+						exportBtnClickHandler={this.exportBtnClickHandler.bind(this)}
 					/>
-				</div>
 
-				<SavedItemPane
-					items={this.state.savedItems}
-					isOpen={this.state.isSavedItemPaneOpen}
-					closeHandler={this.closeSavedItemsPane.bind(this)}
-					itemClickHandler={this.itemClickHandler.bind(this)}
-					itemRemoveBtnClickHandler={this.itemRemoveBtnClickHandler.bind(this)}
-					itemForkBtnClickHandler={this.itemForkBtnClickHandler.bind(this)}
-					exportBtnClickHandler={this.exportBtnClickHandler.bind(this)}
-				/>
+					<Alerts />
 
-				<Alerts />
+					<form
+						style="display:none;"
+						action="https://codepen.io/pen/define"
+						method="POST"
+						target="_blank"
+						id="js-codepen-form"
+					>
+						<input
+							type="hidden"
+							name="data"
+							value="{&quot;title&quot;: &quot;New Pen!&quot;, &quot;html&quot;: &quot;<div>Hello, World!</div>&quot;}"
+						/>
+					</form>
 
-				<form
-					style="display:none;"
-					action="https://codepen.io/pen/define"
-					method="POST"
-					target="_blank"
-					id="js-codepen-form"
-				>
-					<input
-						type="hidden"
-						name="data"
-						value="{&quot;title&quot;: &quot;New Pen!&quot;, &quot;html&quot;: &quot;<div>Hello, World!</div>&quot;}"
-					/>
-				</form>
-
-				<Modal
-					show={this.state.isAddLibraryModalOpen}
-					closeHandler={() => this.setState({ isAddLibraryModalOpen: false })}
-				>
-					<AddLibrary
-						js={
-							this.state.currentItem.externalLibs
-								? this.state.currentItem.externalLibs.js
-								: ''
+					<Modal
+						show={this.state.isAddLibraryModalOpen}
+						closeHandler={() => this.setState({ isAddLibraryModalOpen: false })}
+					>
+						<AddLibrary
+							js={
+								this.state.currentItem.externalLibs
+									? this.state.currentItem.externalLibs.js
+									: ''
+							}
+							css={
+								this.state.currentItem.externalLibs
+									? this.state.currentItem.externalLibs.css
+									: ''
+							}
+							onChange={this.onExternalLibChange.bind(this)}
+						/>
+					</Modal>
+					<Modal
+						show={this.state.isNotificationsModalOpen}
+						closeHandler={() =>
+							this.setState({ isNotificationsModalOpen: false })
 						}
-						css={
-							this.state.currentItem.externalLibs
-								? this.state.currentItem.externalLibs.css
-								: ''
-						}
-						onChange={this.onExternalLibChange.bind(this)}
-					/>
-				</Modal>
-				<Modal
-					show={this.state.isNotificationsModalOpen}
-					closeHandler={() =>
-						this.setState({ isNotificationsModalOpen: false })
-					}
-				>
-					<Notifications
+					>
+						<Notifications
+							onSupportBtnClick={this.openSupportDeveloperModal.bind(this)}
+						/>
+					</Modal>
+					<Modal
+						extraClasses="modal--settings"
+						show={this.state.isSettingsModalOpen}
+						closeHandler={() => this.setState({ isSettingsModalOpen: false })}
+					>
+						<Settings
+							prefs={this.state.prefs}
+							onChange={this.updateSetting.bind(this)}
+						/>
+					</Modal>
+					<Modal
+						extraClasses="login-modal"
+						show={this.state.isLoginModalOpen}
+						closeHandler={() => this.setState({ isLoginModalOpen: false })}
+					>
+						<Login />
+					</Modal>
+					<Modal
+						show={this.state.isProfileModalOpen}
+						closeHandler={() => this.setState({ isProfileModalOpen: false })}
+					>
+						<Profile
+							user={this.state.user}
+							logoutBtnHandler={this.logout.bind(this)}
+						/>
+					</Modal>
+					<HelpModal
+						show={this.state.isHelpModalOpen}
+						closeHandler={() => this.setState({ isHelpModalOpen: false })}
 						onSupportBtnClick={this.openSupportDeveloperModal.bind(this)}
+						version={version}
 					/>
-				</Modal>
-				<Modal
-					extraClasses="modal--settings"
-					show={this.state.isSettingsModalOpen}
-					closeHandler={() => this.setState({ isSettingsModalOpen: false })}
-				>
-					<Settings
-						prefs={this.state.prefs}
-						onChange={this.updateSetting.bind(this)}
+					<SupportDeveloperModal
+						show={this.state.isSupportDeveloperModalOpen}
+						closeHandler={() =>
+							this.setState({ isSupportDeveloperModalOpen: false })
+						}
 					/>
-				</Modal>
-				<Modal
-					extraClasses="login-modal"
-					show={this.state.isLoginModalOpen}
-					closeHandler={() => this.setState({ isLoginModalOpen: false })}
-				>
-					<Login />
-				</Modal>
-				<Modal
-					show={this.state.isProfileModalOpen}
-					closeHandler={() => this.setState({ isProfileModalOpen: false })}
-				>
-					<Profile
-						user={this.state.user}
-						logoutBtnHandler={this.logout.bind(this)}
+					<KeyboardShortcutsModal
+						show={this.state.isKeyboardShortcutsModalOpen}
+						closeHandler={() =>
+							this.setState({ isKeyboardShortcutsModalOpen: false })
+						}
 					/>
-				</Modal>
-				<HelpModal
-					show={this.state.isHelpModalOpen}
-					closeHandler={() => this.setState({ isHelpModalOpen: false })}
-					onSupportBtnClick={this.openSupportDeveloperModal.bind(this)}
-					version={version}
-				/>
-				<SupportDeveloperModal
-					show={this.state.isSupportDeveloperModalOpen}
-					closeHandler={() =>
-						this.setState({ isSupportDeveloperModalOpen: false })
-					}
-				/>
-				<KeyboardShortcutsModal
-					show={this.state.isKeyboardShortcutsModalOpen}
-					closeHandler={() =>
-						this.setState({ isKeyboardShortcutsModalOpen: false })
-					}
-				/>
-				<AskToImportModal
-					show={this.state.isAskToImportModalOpen}
-					closeHandler={() => this.setState({ isAskToImportModalOpen: false })}
-					oldSavedCreationsCount={this.oldSavedCreationsCount}
-					importBtnClickHandler={this.importCreationsAndSettingsIntoApp.bind(
-						this
-					)}
-					dontAskBtnClickHandler={this.dontAskToImportAnymore.bind(this)}
-				/>
-
-				<OnboardingModal
-					show={this.state.isOnboardModalOpen}
-					closeHandler={() => this.setState({ isOnboardModalOpen: false })}
-				/>
-
-				<Js13KModal
-					show={this.state.isJs13KModalOpen}
-					closeHandler={() => this.setState({ isJs13KModalOpen: false })}
-				/>
-
-				<CreateNewModal
-					show={this.state.isCreateNewModalOpen}
-					closeHandler={() => this.setState({ isCreateNewModalOpen: false })}
-					onBlankTemplateSelect={this.blankTemplateSelectHandler.bind(this)}
-					onBlankFileTemplateSelect={this.blankFileTemplateSelectHandler.bind(
-						this
-					)}
-					onTemplateSelect={this.templateSelectHandler.bind(this)}
-				/>
-
-				<CommandPalette
-					show={this.state.isCommandPaletteOpen}
-					closeHandler={() => this.setState({ isCommandPaletteOpen: false })}
-					files={linearizeFiles(this.state.currentItem.files || [])}
-					isCommandMode={this.state.isCommandPaletteInCommandMode}
-					closeHandler={() => this.setState({ isCommandPaletteOpen: false })}
-				/>
-
-				<Portal into="body">
-					<div
-						class="modal-overlay"
-						onClick={this.modalOverlayClickHandler.bind(this)}
+					<AskToImportModal
+						show={this.state.isAskToImportModalOpen}
+						closeHandler={() =>
+							this.setState({ isAskToImportModalOpen: false })
+						}
+						oldSavedCreationsCount={this.oldSavedCreationsCount}
+						importBtnClickHandler={this.importCreationsAndSettingsIntoApp.bind(
+							this
+						)}
+						dontAskBtnClickHandler={this.dontAskToImportAnymore.bind(this)}
 					/>
-				</Portal>
 
-				<Icons />
-				<form
-					style="display:none;"
-					action="https://codepen.io/pen/define"
-					method="POST"
-					target="_blank"
-					id="codepenForm"
-				>
-					<input
-						type="hidden"
-						name="data"
-						value="{&quot;title&quot;: &quot;New Pen!&quot;, &quot;html&quot;: &quot;<div>Hello, World!</div>&quot;}"
+					<OnboardingModal
+						show={this.state.isOnboardModalOpen}
+						closeHandler={() => this.setState({ isOnboardModalOpen: false })}
 					/>
-				</form>
-			</div>
+
+					<Js13KModal
+						show={this.state.isJs13KModalOpen}
+						closeHandler={() => this.setState({ isJs13KModalOpen: false })}
+					/>
+
+					<CreateNewModal
+						show={this.state.isCreateNewModalOpen}
+						closeHandler={() => this.setState({ isCreateNewModalOpen: false })}
+						onBlankTemplateSelect={this.blankTemplateSelectHandler.bind(this)}
+						onBlankFileTemplateSelect={this.blankFileTemplateSelectHandler.bind(
+							this
+						)}
+						onTemplateSelect={this.templateSelectHandler.bind(this)}
+					/>
+
+					<CommandPalette
+						show={this.state.isCommandPaletteOpen}
+						closeHandler={() => this.setState({ isCommandPaletteOpen: false })}
+						files={linearizeFiles(this.state.currentItem.files || [])}
+						isCommandMode={this.state.isCommandPaletteInCommandMode}
+						closeHandler={() => this.setState({ isCommandPaletteOpen: false })}
+					/>
+
+					<Portal into="body">
+						<div
+							class="modal-overlay"
+							onClick={this.modalOverlayClickHandler.bind(this)}
+						/>
+					</Portal>
+
+					<Icons />
+					<form
+						style="display:none;"
+						action="https://codepen.io/pen/define"
+						method="POST"
+						target="_blank"
+						id="codepenForm"
+					>
+						<input
+							type="hidden"
+							name="data"
+							value="{&quot;title&quot;: &quot;New Pen!&quot;, &quot;html&quot;: &quot;<div>Hello, World!</div>&quot;}"
+						/>
+					</form>
+				</div>
+			</I18nProvider>
 		);
 	}
 }
diff --git a/yarn.lock b/yarn.lock
index b816618..86c8269 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -334,6 +334,87 @@
   version "0.2.8"
   resolved "https://registry.yarnpkg.com/@firebase/webchannel-wrapper/-/webchannel-wrapper-0.2.8.tgz#38a936b60b898a1ad0f3719543ff1a1031f60f8b"
 
+"@lingui/babel-plugin-extract-messages@2.7.0":
+  version "2.7.0"
+  resolved "https://registry.yarnpkg.com/@lingui/babel-plugin-extract-messages/-/babel-plugin-extract-messages-2.7.0.tgz#cdff7944716edf79bc251d21385c3f3ba7214197"
+  dependencies:
+    "@lingui/conf" "2.7.0"
+    babel-generator "^6.26.1"
+
+"@lingui/babel-plugin-transform-js@2.7.0":
+  version "2.7.0"
+  resolved "https://registry.yarnpkg.com/@lingui/babel-plugin-transform-js/-/babel-plugin-transform-js-2.7.0.tgz#ca1a28906af28cb91e858b4bda08754c99075158"
+
+"@lingui/babel-plugin-transform-react@2.7.0":
+  version "2.7.0"
+  resolved "https://registry.yarnpkg.com/@lingui/babel-plugin-transform-react/-/babel-plugin-transform-react-2.7.0.tgz#c2ad1fc070f09526f8994b2d9fbc973e0c40fefa"
+
+"@lingui/cli@^2.7.0":
+  version "2.7.0"
+  resolved "https://registry.yarnpkg.com/@lingui/cli/-/cli-2.7.0.tgz#04137f770f148f17eb78bb6580e91eac7697b955"
+  dependencies:
+    "@lingui/babel-plugin-extract-messages" "2.7.0"
+    "@lingui/babel-plugin-transform-js" "2.7.0"
+    "@lingui/babel-plugin-transform-react" "2.7.0"
+    "@lingui/conf" "2.7.0"
+    babel-generator "^6.26.1"
+    babel-plugin-syntax-jsx "^6.18.0"
+    babel-runtime "^6.26.0"
+    babel-types "^6.26.0"
+    babylon "^6.18.0"
+    bcp-47 "^1.0.4"
+    chalk "^2.3.0"
+    cli-table "^0.3.1"
+    commander "^2.17.1"
+    date-fns "^1.29.0"
+    fuzzaldrin "^2.1.0"
+    glob "^7.1.2"
+    inquirer "^6.2.0"
+    make-plural "^4.1.1"
+    messageformat-parser "^2.0.0"
+    mkdirp "^0.5.1"
+    opencollective "^1.0.3"
+    ora "^3.0.0"
+    pofile "^1.0.11"
+    pseudolocale "^1.1.0"
+    ramda "^0.25.0"
+    typescript "^2.9.2"
+
+"@lingui/conf@2.7.0":
+  version "2.7.0"
+  resolved "https://registry.yarnpkg.com/@lingui/conf/-/conf-2.7.0.tgz#c3822896a34acbc66614bec60b99453baa722ad8"
+  dependencies:
+    chalk "^2.3.0"
+    cosmiconfig "^5.0.6"
+    jest-regex-util "^23.3.0"
+    jest-validate "^23.5.0"
+    pkg-conf "^2.1.0"
+
+"@lingui/core@2.7.0":
+  version "2.7.0"
+  resolved "https://registry.yarnpkg.com/@lingui/core/-/core-2.7.0.tgz#f187401248f88da07ca592a8ae3888bdf8619db9"
+  dependencies:
+    babel-runtime "^6.26.0"
+    make-plural "^4.1.1"
+    messageformat-parser "^2.0.0"
+
+"@lingui/macro@^2.7.0":
+  version "2.7.0"
+  resolved "https://registry.yarnpkg.com/@lingui/macro/-/macro-2.7.0.tgz#84ffdad6c056b56cb96ea4f1253bbb5f917ac821"
+  dependencies:
+    "@lingui/babel-plugin-transform-react" "2.7.0"
+    babel-plugin-macros "^2.2.0"
+
+"@lingui/react@^2.7.0":
+  version "2.7.0"
+  resolved "https://registry.yarnpkg.com/@lingui/react/-/react-2.7.0.tgz#36a2e905ab6a7f39798dedee2bd4a2b8e98335a9"
+  dependencies:
+    "@lingui/core" "2.7.0"
+    babel-runtime "^6.26.0"
+    hash-sum "^1.0.2"
+    hoist-non-react-statics "3.0.1"
+    prop-types "^15.6.2"
+
 "@samverschueren/stream-to-observable@^0.3.0":
   version "0.3.0"
   resolved "https://registry.yarnpkg.com/@samverschueren/stream-to-observable/-/stream-to-observable-0.3.0.tgz#ecdf48d532c58ea477acfcab80348424f8d0662f"
@@ -489,7 +570,7 @@ ansi-cyan@^0.1.1:
   dependencies:
     ansi-wrap "0.1.0"
 
-ansi-escapes@^1.0.0:
+ansi-escapes@^1.0.0, ansi-escapes@^1.1.0:
   version "1.4.0"
   resolved "https://registry.yarnpkg.com/ansi-escapes/-/ansi-escapes-1.4.0.tgz#d3a8a83b319aa67793662b13e761c7911422306e"
 
@@ -796,7 +877,7 @@ babel-code-frame@^6.22.0, babel-code-frame@^6.26.0:
     esutils "^2.0.2"
     js-tokens "^3.0.2"
 
-babel-core@^6.0.0, babel-core@^6.24.1, babel-core@^6.26.0:
+babel-core@^6.0.0, babel-core@^6.24.1, babel-core@^6.26.0, babel-core@^6.26.3:
   version "6.26.3"
   resolved "https://registry.yarnpkg.com/babel-core/-/babel-core-6.26.3.tgz#b2e2f09e342d0f0c88e2f02e067794125e75c207"
   dependencies:
@@ -840,7 +921,7 @@ babel-eslint@^8.1.2:
     eslint-scope "~3.7.1"
     eslint-visitor-keys "^1.0.0"
 
-babel-generator@^6.18.0, babel-generator@^6.26.0:
+babel-generator@^6.18.0, babel-generator@^6.26.0, babel-generator@^6.26.1:
   version "6.26.1"
   resolved "https://registry.yarnpkg.com/babel-generator/-/babel-generator-6.26.1.tgz#1844408d3b8f0d35a404ea7ac180f087a601bd90"
   dependencies:
@@ -1047,6 +1128,13 @@ babel-plugin-jsx-pragmatic@^1.0.2:
   dependencies:
     babel-plugin-syntax-jsx "^6.0.0"
 
+babel-plugin-macros@^2.2.0:
+  version "2.4.2"
+  resolved "https://registry.yarnpkg.com/babel-plugin-macros/-/babel-plugin-macros-2.4.2.tgz#21b1a2e82e2130403c5ff785cba6548e9b644b28"
+  dependencies:
+    cosmiconfig "^5.0.5"
+    resolve "^1.8.1"
+
 babel-plugin-minify-builtins@^0.2.0:
   version "0.2.0"
   resolved "https://registry.yarnpkg.com/babel-plugin-minify-builtins/-/babel-plugin-minify-builtins-0.2.0.tgz#317f824b0907210b6348671bb040ca072e2e0c82"
@@ -1136,9 +1224,9 @@ babel-plugin-syntax-export-extensions@^6.8.0:
   version "6.13.0"
   resolved "https://registry.yarnpkg.com/babel-plugin-syntax-export-extensions/-/babel-plugin-syntax-export-extensions-6.13.0.tgz#70a1484f0f9089a4e84ad44bac353c95b9b12721"
 
-babel-plugin-syntax-jsx@^6.0.0, babel-plugin-syntax-jsx@^6.8.0:
+babel-plugin-syntax-jsx@^6.0.0, babel-plugin-syntax-jsx@^6.18.0, babel-plugin-syntax-jsx@^6.8.0:
   version "6.18.0"
-  resolved "https://registry.yarnpkg.com/babel-plugin-syntax-jsx/-/babel-plugin-syntax-jsx-6.18.0.tgz#0af32a9a6e13ca7a3fd5069e62d7b0f58d0d8946"
+  resolved "http://registry.npmjs.org/babel-plugin-syntax-jsx/-/babel-plugin-syntax-jsx-6.18.0.tgz#0af32a9a6e13ca7a3fd5069e62d7b0f58d0d8946"
 
 babel-plugin-syntax-object-rest-spread@^6.13.0, babel-plugin-syntax-object-rest-spread@^6.8.0:
   version "6.13.0"
@@ -1448,6 +1536,14 @@ babel-plugin-transform-undefined-to-void@^6.8.3:
   version "6.9.4"
   resolved "https://registry.yarnpkg.com/babel-plugin-transform-undefined-to-void/-/babel-plugin-transform-undefined-to-void-6.9.4.tgz#be241ca81404030678b748717322b89d0c8fe280"
 
+babel-polyfill@6.23.0:
+  version "6.23.0"
+  resolved "https://registry.yarnpkg.com/babel-polyfill/-/babel-polyfill-6.23.0.tgz#8364ca62df8eafb830499f699177466c3b03499d"
+  dependencies:
+    babel-runtime "^6.22.0"
+    core-js "^2.4.0"
+    regenerator-runtime "^0.10.0"
+
 babel-polyfill@^6.2.0:
   version "6.26.0"
   resolved "https://registry.yarnpkg.com/babel-polyfill/-/babel-polyfill-6.26.0.tgz#379937abc67d7895970adc621f284cd966cf2153"
@@ -1614,6 +1710,14 @@ batch@0.6.1:
   version "0.6.1"
   resolved "https://registry.yarnpkg.com/batch/-/batch-0.6.1.tgz#dc34314f4e679318093fc760272525f94bf25c16"
 
+bcp-47@^1.0.4:
+  version "1.0.4"
+  resolved "https://registry.yarnpkg.com/bcp-47/-/bcp-47-1.0.4.tgz#8bee5d1f384fa12b3378b1a4a90dc8418764759d"
+  dependencies:
+    is-alphabetical "^1.0.0"
+    is-alphanumerical "^1.0.0"
+    is-decimal "^1.0.0"
+
 bcrypt-pbkdf@^1.0.0:
   version "1.0.1"
   resolved "https://registry.yarnpkg.com/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.1.tgz#63bc5dcb61331b92bc05fd528953c33462a06f8d"
@@ -2049,9 +2153,9 @@ center-align@^0.1.1:
     align-text "^0.1.3"
     lazy-cache "^1.0.3"
 
-chalk@^1.0.0, chalk@^1.1.1, chalk@^1.1.3:
+chalk@1.1.3, chalk@^1.0.0, chalk@^1.1.1, chalk@^1.1.3:
   version "1.1.3"
-  resolved "https://registry.yarnpkg.com/chalk/-/chalk-1.1.3.tgz#a8115c55e4a702fe4d150abd3872822a7e09fc98"
+  resolved "http://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz#a8115c55e4a702fe4d150abd3872822a7e09fc98"
   dependencies:
     ansi-styles "^2.2.1"
     escape-string-regexp "^1.0.2"
@@ -2059,7 +2163,7 @@ chalk@^1.0.0, chalk@^1.1.1, chalk@^1.1.3:
     strip-ansi "^3.0.0"
     supports-color "^2.0.0"
 
-chalk@^2.0.0, chalk@^2.0.1, chalk@^2.1.0, chalk@^2.3.1, chalk@^2.4.1:
+chalk@^2.0.0, chalk@^2.0.1, chalk@^2.1.0, chalk@^2.3.0, chalk@^2.3.1, chalk@^2.4.1:
   version "2.4.1"
   resolved "https://registry.yarnpkg.com/chalk/-/chalk-2.4.1.tgz#18c49ab16a037b6eb0152cc83e3471338215b66e"
   dependencies:
@@ -2071,6 +2175,10 @@ chardet@^0.4.0:
   version "0.4.2"
   resolved "https://registry.yarnpkg.com/chardet/-/chardet-0.4.2.tgz#b5473b33dc97c424e5d98dc87d55d4d8a29c8bf2"
 
+chardet@^0.7.0:
+  version "0.7.0"
+  resolved "https://registry.yarnpkg.com/chardet/-/chardet-0.7.0.tgz#90094849f0937f2eedc2425d0d28a9e5f0cbad9e"
+
 chokidar@^2.0.0, chokidar@^2.0.2:
   version "2.0.3"
   resolved "https://registry.yarnpkg.com/chokidar/-/chokidar-2.0.3.tgz#dcbd4f6cbb2a55b4799ba8a840ac527e5f4b1176"
@@ -2149,10 +2257,16 @@ cli-spinners@^0.1.2:
   version "0.1.2"
   resolved "https://registry.yarnpkg.com/cli-spinners/-/cli-spinners-0.1.2.tgz#bb764d88e185fb9e1e6a2a1f19772318f605e31c"
 
-cli-spinners@^1.0.1:
+cli-spinners@^1.0.1, cli-spinners@^1.1.0:
   version "1.3.1"
   resolved "https://registry.yarnpkg.com/cli-spinners/-/cli-spinners-1.3.1.tgz#002c1990912d0d59580c93bd36c056de99e4259a"
 
+cli-table@^0.3.1:
+  version "0.3.1"
+  resolved "https://registry.yarnpkg.com/cli-table/-/cli-table-0.3.1.tgz#f53b05266a8b1a0b934b3d0821e6e2dc5914ae23"
+  dependencies:
+    colors "1.0.3"
+
 cli-truncate@^0.2.1:
   version "0.2.1"
   resolved "https://registry.yarnpkg.com/cli-truncate/-/cli-truncate-0.2.1.tgz#9f15cfbb0705005369216c626ac7d05ab90dd574"
@@ -2285,6 +2399,10 @@ colormin@^1.0.5:
     css-color-names "0.0.4"
     has "^1.0.1"
 
+colors@1.0.3:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/colors/-/colors-1.0.3.tgz#0433f44d809680fdeb60ed260f1b0c262e82a40b"
+
 colors@~1.1.2:
   version "1.1.2"
   resolved "https://registry.yarnpkg.com/colors/-/colors-1.1.2.tgz#168a4701756b6a7f51a12ce0c97bfa28c084ed63"
@@ -2303,6 +2421,10 @@ command-exists@^1.2.2:
   version "1.2.6"
   resolved "https://registry.yarnpkg.com/command-exists/-/command-exists-1.2.6.tgz#577f8e5feb0cb0f159cd557a51a9be1bdd76e09e"
 
+commander@*, commander@^2.17.1:
+  version "2.19.0"
+  resolved "https://registry.yarnpkg.com/commander/-/commander-2.19.0.tgz#f6198aa84e5b83c46054b94ddedbfed5ee9ff12a"
+
 commander@2.15.x, commander@^2.14.1, commander@^2.9.0, commander@~2.15.0:
   version "2.15.1"
   resolved "https://registry.yarnpkg.com/commander/-/commander-2.15.1.tgz#df46e867d0fc2aec66a34662b406a9ccafff5b0f"
@@ -2495,6 +2617,14 @@ cosmiconfig@^5.0.2:
     js-yaml "^3.9.0"
     parse-json "^4.0.0"
 
+cosmiconfig@^5.0.5, cosmiconfig@^5.0.6:
+  version "5.0.6"
+  resolved "https://registry.yarnpkg.com/cosmiconfig/-/cosmiconfig-5.0.6.tgz#dca6cf680a0bd03589aff684700858c81abeeb39"
+  dependencies:
+    is-directory "^0.3.1"
+    js-yaml "^3.9.0"
+    parse-json "^4.0.0"
+
 create-ecdh@^4.0.0:
   version "4.0.3"
   resolved "https://registry.yarnpkg.com/create-ecdh/-/create-ecdh-4.0.3.tgz#c9111b6f33045c4697f144787f9254cdc77c45ff"
@@ -2704,7 +2834,7 @@ dashdash@^1.12.0:
   dependencies:
     assert-plus "^1.0.0"
 
-date-fns@^1.27.2:
+date-fns@^1.27.2, date-fns@^1.29.0:
   version "1.29.0"
   resolved "https://registry.yarnpkg.com/date-fns/-/date-fns-1.29.0.tgz#12e609cdcb935127311d04d33334e2960a2a54e6"
 
@@ -2818,7 +2948,7 @@ default-require-extensions@^2.0.0:
   dependencies:
     strip-bom "^3.0.0"
 
-defaults@^1.0.0:
+defaults@^1.0.0, defaults@^1.0.3:
   version "1.0.3"
   resolved "https://registry.yarnpkg.com/defaults/-/defaults-1.0.3.tgz#c656051e9817d9ff08ed881477f3fe4019f3ef7d"
   dependencies:
@@ -3627,14 +3757,22 @@ extend@3.0.1, extend@^3.0.0, extend@^3.0.1, extend@~3.0.1:
   version "3.0.1"
   resolved "https://registry.yarnpkg.com/extend/-/extend-3.0.1.tgz#a755ea7bc1adfcc5a31ce7e762dbaadc5e636444"
 
-external-editor@^2.0.4:
+external-editor@^2.0.1, external-editor@^2.0.4:
   version "2.2.0"
-  resolved "https://registry.yarnpkg.com/external-editor/-/external-editor-2.2.0.tgz#045511cfd8d133f3846673d1047c154e214ad3d5"
+  resolved "http://registry.npmjs.org/external-editor/-/external-editor-2.2.0.tgz#045511cfd8d133f3846673d1047c154e214ad3d5"
   dependencies:
     chardet "^0.4.0"
     iconv-lite "^0.4.17"
     tmp "^0.0.33"
 
+external-editor@^3.0.0:
+  version "3.0.3"
+  resolved "https://registry.yarnpkg.com/external-editor/-/external-editor-3.0.3.tgz#5866db29a97826dbe4bf3afd24070ead9ea43a27"
+  dependencies:
+    chardet "^0.7.0"
+    iconv-lite "^0.4.24"
+    tmp "^0.0.33"
+
 extglob@^0.3.1:
   version "0.3.2"
   resolved "https://registry.yarnpkg.com/extglob/-/extglob-0.3.2.tgz#2e18ff3d2f49ab2765cec9023f011daa8d8349a1"
@@ -4037,6 +4175,10 @@ functional-red-black-tree@^1.0.1:
   version "1.0.1"
   resolved "https://registry.yarnpkg.com/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz#1b0ab3bd553b2a0d6399d29c0e3ea0b252078327"
 
+fuzzaldrin@^2.1.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/fuzzaldrin/-/fuzzaldrin-2.1.0.tgz#90204c3e2fdaa6941bb28d16645d418063a90e9b"
+
 gauge@~2.7.3:
   version "2.7.4"
   resolved "https://registry.yarnpkg.com/gauge/-/gauge-2.7.4.tgz#2c03405c7538c39d7eb37b317022e325fb018bf7"
@@ -4606,6 +4748,10 @@ hash-base@^3.0.0:
     inherits "^2.0.1"
     safe-buffer "^5.0.1"
 
+hash-sum@^1.0.2:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/hash-sum/-/hash-sum-1.0.2.tgz#33b40777754c6432573c120cc3808bbd10d47f04"
+
 hash.js@^1.0.0, hash.js@^1.0.3:
   version "1.1.4"
   resolved "https://registry.yarnpkg.com/hash.js/-/hash.js-1.1.4.tgz#8b50e1f35d51bd01e5ed9ece4dbe3549ccfa0a3c"
@@ -4625,6 +4771,12 @@ hmac-drbg@^1.0.0:
     minimalistic-assert "^1.0.0"
     minimalistic-crypto-utils "^1.0.1"
 
+hoist-non-react-statics@3.0.1:
+  version "3.0.1"
+  resolved "https://registry.yarnpkg.com/hoist-non-react-statics/-/hoist-non-react-statics-3.0.1.tgz#fba3e7df0210eb9447757ca1a7cb607162f0a364"
+  dependencies:
+    react-is "^16.3.2"
+
 home-or-tmp@^2.0.0:
   version "2.0.0"
   resolved "https://registry.yarnpkg.com/home-or-tmp/-/home-or-tmp-2.0.0.tgz#e36c3f2d2cae7d746a857e38d18d5f32a7882db8"
@@ -4774,6 +4926,12 @@ iconv-lite@^0.4.17, iconv-lite@^0.4.4, iconv-lite@~0.4.13:
   dependencies:
     safer-buffer ">= 2.1.2 < 3"
 
+iconv-lite@^0.4.24:
+  version "0.4.24"
+  resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.24.tgz#2022b4b25fbddc21d2f524974a474aafe733908b"
+  dependencies:
+    safer-buffer ">= 2.1.2 < 3"
+
 icss-replace-symbols@^1.0.2, icss-replace-symbols@^1.1.0:
   version "1.1.0"
   resolved "https://registry.yarnpkg.com/icss-replace-symbols/-/icss-replace-symbols-1.1.0.tgz#06ea6f83679a7749e386cfe1fe812ae5db223ded"
@@ -4886,6 +5044,24 @@ ini@^1.3.4, ini@~1.3.0:
   version "1.3.5"
   resolved "https://registry.yarnpkg.com/ini/-/ini-1.3.5.tgz#eee25f56db1c9ec6085e0c22778083f596abf927"
 
+inquirer@3.0.6:
+  version "3.0.6"
+  resolved "https://registry.yarnpkg.com/inquirer/-/inquirer-3.0.6.tgz#e04aaa9d05b7a3cb9b0f407d04375f0447190347"
+  dependencies:
+    ansi-escapes "^1.1.0"
+    chalk "^1.0.0"
+    cli-cursor "^2.1.0"
+    cli-width "^2.0.0"
+    external-editor "^2.0.1"
+    figures "^2.0.0"
+    lodash "^4.3.0"
+    mute-stream "0.0.7"
+    run-async "^2.2.0"
+    rx "^4.1.0"
+    string-width "^2.0.0"
+    strip-ansi "^3.0.0"
+    through "^2.3.6"
+
 inquirer@^3.0.6, inquirer@^3.3.0:
   version "3.3.0"
   resolved "https://registry.yarnpkg.com/inquirer/-/inquirer-3.3.0.tgz#9dd2f2ad765dcab1ff0443b491442a20ba227dc9"
@@ -4905,6 +5081,24 @@ inquirer@^3.0.6, inquirer@^3.3.0:
     strip-ansi "^4.0.0"
     through "^2.3.6"
 
+inquirer@^6.2.0:
+  version "6.2.0"
+  resolved "https://registry.yarnpkg.com/inquirer/-/inquirer-6.2.0.tgz#51adcd776f661369dc1e894859c2560a224abdd8"
+  dependencies:
+    ansi-escapes "^3.0.0"
+    chalk "^2.0.0"
+    cli-cursor "^2.1.0"
+    cli-width "^2.0.0"
+    external-editor "^3.0.0"
+    figures "^2.0.0"
+    lodash "^4.17.10"
+    mute-stream "0.0.7"
+    run-async "^2.2.0"
+    rxjs "^6.1.0"
+    string-width "^2.1.0"
+    strip-ansi "^4.0.0"
+    through "^2.3.6"
+
 internal-ip@1.2.0:
   version "1.2.0"
   resolved "https://registry.yarnpkg.com/internal-ip/-/internal-ip-1.2.0.tgz#ae9fbf93b984878785d50a8de1b356956058cf5c"
@@ -4970,6 +5164,17 @@ is-accessor-descriptor@^1.0.0:
   dependencies:
     kind-of "^6.0.0"
 
+is-alphabetical@^1.0.0:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/is-alphabetical/-/is-alphabetical-1.0.2.tgz#1fa6e49213cb7885b75d15862fb3f3d96c884f41"
+
+is-alphanumerical@^1.0.0:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/is-alphanumerical/-/is-alphanumerical-1.0.2.tgz#1138e9ae5040158dc6ff76b820acd6b7a181fd40"
+  dependencies:
+    is-alphabetical "^1.0.0"
+    is-decimal "^1.0.0"
+
 is-arrayish@^0.2.1:
   version "0.2.1"
   resolved "https://registry.yarnpkg.com/is-arrayish/-/is-arrayish-0.2.1.tgz#77c99840527aa8ecb1a8ba697b80645a7a926a9d"
@@ -5020,6 +5225,10 @@ is-date-object@^1.0.1:
   version "1.0.1"
   resolved "https://registry.yarnpkg.com/is-date-object/-/is-date-object-1.0.1.tgz#9aa20eb6aeebbff77fbd33e74ca01b33581d3a16"
 
+is-decimal@^1.0.0:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/is-decimal/-/is-decimal-1.0.2.tgz#894662d6a8709d307f3a276ca4339c8fa5dff0ff"
+
 is-descriptor@^0.1.0:
   version "0.1.6"
   resolved "https://registry.yarnpkg.com/is-descriptor/-/is-descriptor-0.1.6.tgz#366d8240dde487ca51823b1ab9f07a10a78251ca"
@@ -5547,6 +5756,10 @@ jest-regex-util@^21.2.0:
   version "21.2.0"
   resolved "https://registry.yarnpkg.com/jest-regex-util/-/jest-regex-util-21.2.0.tgz#1b1e33e63143babc3e0f2e6c9b5ba1eb34b2d530"
 
+jest-regex-util@^23.3.0:
+  version "23.3.0"
+  resolved "https://registry.yarnpkg.com/jest-regex-util/-/jest-regex-util-23.3.0.tgz#5f86729547c2785c4002ceaa8f849fe8ca471bc5"
+
 jest-resolve-dependencies@^21.2.0:
   version "21.2.0"
   resolved "https://registry.yarnpkg.com/jest-resolve-dependencies/-/jest-resolve-dependencies-21.2.0.tgz#9e231e371e1a736a1ad4e4b9a843bc72bfe03d09"
@@ -5639,6 +5852,15 @@ jest-validate@^23.0.0:
     leven "^2.1.0"
     pretty-format "^23.0.1"
 
+jest-validate@^23.5.0:
+  version "23.6.0"
+  resolved "https://registry.yarnpkg.com/jest-validate/-/jest-validate-23.6.0.tgz#36761f99d1ed33fcd425b4e4c5595d62b6597474"
+  dependencies:
+    chalk "^2.0.1"
+    jest-get-type "^22.1.0"
+    leven "^2.1.0"
+    pretty-format "^23.6.0"
+
 jest@^21.2.1:
   version "21.2.1"
   resolved "https://registry.yarnpkg.com/jest/-/jest-21.2.1.tgz#c964e0b47383768a1438e3ccf3c3d470327604e1"
@@ -5962,6 +6184,15 @@ load-json-file@^2.0.0:
     pify "^2.0.0"
     strip-bom "^3.0.0"
 
+load-json-file@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/load-json-file/-/load-json-file-4.0.0.tgz#2f5f45ab91e33216234fd53adab668eb4ec0993b"
+  dependencies:
+    graceful-fs "^4.1.2"
+    parse-json "^4.0.0"
+    pify "^3.0.0"
+    strip-bom "^3.0.0"
+
 loader-runner@^2.3.0:
   version "2.3.0"
   resolved "https://registry.yarnpkg.com/loader-runner/-/loader-runner-2.3.0.tgz#f482aea82d543e07921700d5a46ef26fdac6b8a2"
@@ -6227,6 +6458,12 @@ make-iterator@^1.0.0:
   dependencies:
     kind-of "^6.0.2"
 
+make-plural@^4.1.1:
+  version "4.3.0"
+  resolved "https://registry.yarnpkg.com/make-plural/-/make-plural-4.3.0.tgz#f23de08efdb0cac2e0c9ba9f315b0dff6b4c2735"
+  optionalDependencies:
+    minimist "^1.2.0"
+
 makeerror@1.0.x:
   version "1.0.11"
   resolved "https://registry.yarnpkg.com/makeerror/-/makeerror-1.0.11.tgz#e01a5c9109f2af79660e4e8b9587790184f5a96c"
@@ -6318,6 +6555,10 @@ merge@^1.1.3:
   version "1.2.0"
   resolved "https://registry.yarnpkg.com/merge/-/merge-1.2.0.tgz#7531e39d4949c281a66b8c5a6e0265e8b05894da"
 
+messageformat-parser@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/messageformat-parser/-/messageformat-parser-2.0.0.tgz#e37efa0cb07b6784e5f3adc089cbf266208c01c7"
+
 methods@~1.1.2:
   version "1.1.2"
   resolved "https://registry.yarnpkg.com/methods/-/methods-1.1.2.tgz#5529a4d67654134edcc5266656835b0f851afcee"
@@ -6426,7 +6667,7 @@ minimist@0.0.8:
   version "0.0.8"
   resolved "https://registry.yarnpkg.com/minimist/-/minimist-0.0.8.tgz#857fcabfc3397d2625b8228262e86aa7a011b05d"
 
-minimist@^1.1.0, minimist@^1.1.1, minimist@^1.1.3, minimist@^1.2.0:
+minimist@1.2.0, minimist@^1.1.0, minimist@^1.1.1, minimist@^1.1.3, minimist@^1.2.0:
   version "1.2.0"
   resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.0.tgz#a35008b20f41383eec1fb914f4cd5df79a264284"
 
@@ -6566,6 +6807,13 @@ no-case@^2.2.0:
   dependencies:
     lower-case "^1.1.1"
 
+node-fetch@1.6.3:
+  version "1.6.3"
+  resolved "http://registry.npmjs.org/node-fetch/-/node-fetch-1.6.3.tgz#dc234edd6489982d58e8f0db4f695029abcd8c04"
+  dependencies:
+    encoding "^0.1.11"
+    is-stream "^1.0.1"
+
 node-fetch@^1.0.1, node-fetch@^1.7.1:
   version "1.7.3"
   resolved "https://registry.yarnpkg.com/node-fetch/-/node-fetch-1.7.3.tgz#980f6f72d85211a5347c6b2bc18c5b84c3eb47ef"
@@ -6874,6 +7122,24 @@ onetime@^2.0.0:
   dependencies:
     mimic-fn "^1.0.0"
 
+opencollective@^1.0.3:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/opencollective/-/opencollective-1.0.3.tgz#aee6372bc28144583690c3ca8daecfc120dd0ef1"
+  dependencies:
+    babel-polyfill "6.23.0"
+    chalk "1.1.3"
+    inquirer "3.0.6"
+    minimist "1.2.0"
+    node-fetch "1.6.3"
+    opn "4.0.2"
+
+opn@4.0.2:
+  version "4.0.2"
+  resolved "http://registry.npmjs.org/opn/-/opn-4.0.2.tgz#7abc22e644dff63b0a96d5ab7f2790c0f01abc95"
+  dependencies:
+    object-assign "^4.0.1"
+    pinkie-promise "^2.0.0"
+
 opn@^5.1.0:
   version "5.3.0"
   resolved "https://registry.yarnpkg.com/opn/-/opn-5.3.0.tgz#64871565c863875f052cfdf53d3e3cb5adb53b1c"
@@ -6920,6 +7186,17 @@ ora@^1.2.0:
     cli-spinners "^1.0.1"
     log-symbols "^2.1.0"
 
+ora@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/ora/-/ora-3.0.0.tgz#8179e3525b9aafd99242d63cc206fd64732741d0"
+  dependencies:
+    chalk "^2.3.1"
+    cli-cursor "^2.1.0"
+    cli-spinners "^1.1.0"
+    log-symbols "^2.2.0"
+    strip-ansi "^4.0.0"
+    wcwidth "^1.0.1"
+
 orchestrator@^0.3.0:
   version "0.3.8"
   resolved "https://registry.yarnpkg.com/orchestrator/-/orchestrator-0.3.8.tgz#14e7e9e2764f7315fbac184e506c7aa6df94ad7e"
@@ -7219,6 +7496,13 @@ pinkie@^2.0.0:
   version "2.0.4"
   resolved "https://registry.yarnpkg.com/pinkie/-/pinkie-2.0.4.tgz#72556b80cfa0d48a974e80e77248e80ed4f7f870"
 
+pkg-conf@^2.1.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/pkg-conf/-/pkg-conf-2.1.0.tgz#2126514ca6f2abfebd168596df18ba57867f0058"
+  dependencies:
+    find-up "^2.0.0"
+    load-json-file "^4.0.0"
+
 pkg-dir@^2.0.0:
   version "2.0.0"
   resolved "https://registry.yarnpkg.com/pkg-dir/-/pkg-dir-2.0.0.tgz#f6d5d1109e19d63edf428e0bd57e12777615334b"
@@ -7254,6 +7538,10 @@ pluralize@^7.0.0:
   version "7.0.0"
   resolved "https://registry.yarnpkg.com/pluralize/-/pluralize-7.0.0.tgz#298b89df8b93b0221dbf421ad2b1b1ea23fc6777"
 
+pofile@^1.0.11:
+  version "1.0.11"
+  resolved "https://registry.yarnpkg.com/pofile/-/pofile-1.0.11.tgz#35aff58c17491d127a07336d5522ebc9df57c954"
+
 portfinder@^1.0.9:
   version "1.0.13"
   resolved "https://registry.yarnpkg.com/portfinder/-/portfinder-1.0.13.tgz#bb32ecd87c27104ae6ee44b5a3ccbf0ebb1aede9"
@@ -7710,6 +7998,13 @@ pretty-format@^23.0.1:
     ansi-regex "^3.0.0"
     ansi-styles "^3.2.0"
 
+pretty-format@^23.6.0:
+  version "23.6.0"
+  resolved "https://registry.yarnpkg.com/pretty-format/-/pretty-format-23.6.0.tgz#5eaac8eeb6b33b987b7fe6097ea6a8a146ab5760"
+  dependencies:
+    ansi-regex "^3.0.0"
+    ansi-styles "^3.2.0"
+
 pretty-format@^3.5.1:
   version "3.8.0"
   resolved "https://registry.yarnpkg.com/pretty-format/-/pretty-format-3.8.0.tgz#bfbed56d5e9a776645f4b1ff7aa1a3ac4fa3c385"
@@ -7776,6 +8071,13 @@ prop-types@^15.5.8, prop-types@^15.6.1:
     loose-envify "^1.3.1"
     object-assign "^4.1.1"
 
+prop-types@^15.6.2:
+  version "15.6.2"
+  resolved "https://registry.yarnpkg.com/prop-types/-/prop-types-15.6.2.tgz#05d5ca77b4453e985d60fc7ff8c859094a497102"
+  dependencies:
+    loose-envify "^1.3.1"
+    object-assign "^4.1.1"
+
 protobufjs@^5.0.0:
   version "5.0.3"
   resolved "https://registry.yarnpkg.com/protobufjs/-/protobufjs-5.0.3.tgz#e4dfe9fb67c90b2630d15868249bcc4961467a17"
@@ -7802,6 +8104,12 @@ ps-tree@^1.0.1:
   dependencies:
     event-stream "~3.3.0"
 
+pseudolocale@^1.1.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/pseudolocale/-/pseudolocale-1.1.0.tgz#f333f229433d2c586ec384d021e81f0cf7ca8dc7"
+  dependencies:
+    commander "*"
+
 pseudomap@^1.0.2:
   version "1.0.2"
   resolved "https://registry.yarnpkg.com/pseudomap/-/pseudomap-1.0.2.tgz#f052a28da70e618917ef0a8ac34c1ae5a68286b3"
@@ -7936,6 +8244,10 @@ react-inspector@^2.3.0:
     babel-runtime "^6.26.0"
     is-dom "^1.0.9"
 
+react-is@^16.3.2:
+  version "16.6.0"
+  resolved "https://registry.yarnpkg.com/react-is/-/react-is-16.6.0.tgz#456645144581a6e99f6816ae2bd24ee94bdd0c01"
+
 read-all-stream@^3.0.0:
   version "3.1.0"
   resolved "https://registry.yarnpkg.com/read-all-stream/-/read-all-stream-3.1.0.tgz#35c3e177f2078ef789ee4bfafa4373074eaef4fa"
@@ -8054,7 +8366,7 @@ regenerate@^1.2.1:
   version "1.4.0"
   resolved "https://registry.yarnpkg.com/regenerate/-/regenerate-1.4.0.tgz#4a856ec4b56e4077c557589cae85e7a4c8869a11"
 
-regenerator-runtime@^0.10.5:
+regenerator-runtime@^0.10.0, regenerator-runtime@^0.10.5:
   version "0.10.5"
   resolved "https://registry.yarnpkg.com/regenerator-runtime/-/regenerator-runtime-0.10.5.tgz#336c3efc1220adcedda2c9fab67b5a7955a33658"
 
@@ -8278,6 +8590,12 @@ resolve@^1.1.6, resolve@^1.1.7:
   dependencies:
     path-parse "^1.0.5"
 
+resolve@^1.8.1:
+  version "1.8.1"
+  resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.8.1.tgz#82f1ec19a423ac1fbd080b0bab06ba36e84a7a26"
+  dependencies:
+    path-parse "^1.0.5"
+
 restore-cursor@^1.0.1:
   version "1.0.1"
   resolved "https://registry.yarnpkg.com/restore-cursor/-/restore-cursor-1.0.1.tgz#34661f46886327fed2991479152252df92daa541"
@@ -8349,6 +8667,10 @@ rx-lite@*, rx-lite@^4.0.8:
   version "4.0.8"
   resolved "https://registry.yarnpkg.com/rx-lite/-/rx-lite-4.0.8.tgz#0b1e11af8bc44836f04a6407e92da42467b79444"
 
+rx@^4.1.0:
+  version "4.1.0"
+  resolved "https://registry.yarnpkg.com/rx/-/rx-4.1.0.tgz#a5f13ff79ef3b740fe30aa803fb09f98805d4782"
+
 rxjs@^6.1.0:
   version "6.2.1"
   resolved "https://registry.yarnpkg.com/rxjs/-/rxjs-6.2.1.tgz#246cebec189a6cbc143a3ef9f62d6f4c91813ca1"
@@ -9362,6 +9684,10 @@ typedarray@^0.0.6:
   version "0.0.6"
   resolved "https://registry.yarnpkg.com/typedarray/-/typedarray-0.0.6.tgz#867ac74e3864187b1d3d47d996a78ec5c8830777"
 
+typescript@^2.9.2:
+  version "2.9.2"
+  resolved "https://registry.yarnpkg.com/typescript/-/typescript-2.9.2.tgz#1cbf61d05d6b96269244eb6a3bce4bd914e0f00c"
+
 ua-parser-js@^0.7.18:
   version "0.7.18"
   resolved "https://registry.yarnpkg.com/ua-parser-js/-/ua-parser-js-0.7.18.tgz#a7bfd92f56edfb117083b69e31d2aa8882d4b1ed"
@@ -9806,6 +10132,12 @@ wbuf@^1.1.0, wbuf@^1.7.2:
   dependencies:
     minimalistic-assert "^1.0.0"
 
+wcwidth@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/wcwidth/-/wcwidth-1.0.1.tgz#f0b0dcf915bc5ff1528afadb2c0e17b532da2fe8"
+  dependencies:
+    defaults "^1.0.3"
+
 webidl-conversions@^3.0.0:
   version "3.0.1"
   resolved "https://registry.yarnpkg.com/webidl-conversions/-/webidl-conversions-3.0.1.tgz#24534275e2a7bc6be7bc86611cc16ae0a5654871"